首次提交
This commit is contained in:
commit
9423c2dce6
45
.github/workflows/main.yml
vendored
Normal file
45
.github/workflows/main.yml
vendored
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
name: build
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: checkout repository
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
- name: validate gradle wrapper
|
||||||
|
uses: gradle/wrapper-validation-action@v1
|
||||||
|
- name: setup jdk 8.0
|
||||||
|
uses: actions/setup-java@v2
|
||||||
|
with:
|
||||||
|
distribution: adopt
|
||||||
|
java-version: 8.0
|
||||||
|
- name: make gradle wrapper executable
|
||||||
|
run: chmod +x ./gradlew
|
||||||
|
# 第一次构建
|
||||||
|
- name: build
|
||||||
|
id: build_1
|
||||||
|
run: ./gradlew build
|
||||||
|
# 第二次构建
|
||||||
|
- name: build (retry 1)
|
||||||
|
id: build_2
|
||||||
|
if: steps.build_1.outcome == 'failure'
|
||||||
|
run: ./gradlew build
|
||||||
|
# 第三次构建
|
||||||
|
- name: build (retry 2)
|
||||||
|
id: build_3
|
||||||
|
if: steps.build_2.outcome == 'failure'
|
||||||
|
run: ./gradlew build
|
||||||
|
# 第四次构建
|
||||||
|
- name: build (retry 3)
|
||||||
|
id: build_4
|
||||||
|
if: steps.build_3.outcome == 'failure'
|
||||||
|
run: ./gradlew build
|
||||||
|
# 上传构建文件
|
||||||
|
- name: capture build artifacts
|
||||||
|
uses: actions/upload-artifact@v2
|
||||||
|
with:
|
||||||
|
name: Artifacts
|
||||||
|
path: build/libs/
|
3
.gitignore
vendored
Normal file
3
.gitignore
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
.gradle
|
||||||
|
.idea
|
||||||
|
build
|
121
LICENSE
Normal file
121
LICENSE
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
Creative Commons Legal Code
|
||||||
|
|
||||||
|
CC0 1.0 Universal
|
||||||
|
|
||||||
|
CREATIVE COMMONS CORPORATION IS NOT A LAW FIRM AND DOES NOT PROVIDE
|
||||||
|
LEGAL SERVICES. DISTRIBUTION OF THIS DOCUMENT DOES NOT CREATE AN
|
||||||
|
ATTORNEY-CLIENT RELATIONSHIP. CREATIVE COMMONS PROVIDES THIS
|
||||||
|
INFORMATION ON AN "AS-IS" BASIS. CREATIVE COMMONS MAKES NO WARRANTIES
|
||||||
|
REGARDING THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS
|
||||||
|
PROVIDED HEREUNDER, AND DISCLAIMS LIABILITY FOR DAMAGES RESULTING FROM
|
||||||
|
THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS PROVIDED
|
||||||
|
HEREUNDER.
|
||||||
|
|
||||||
|
Statement of Purpose
|
||||||
|
|
||||||
|
The laws of most jurisdictions throughout the world automatically confer
|
||||||
|
exclusive Copyright and Related Rights (defined below) upon the creator
|
||||||
|
and subsequent owner(s) (each and all, an "owner") of an original work of
|
||||||
|
authorship and/or a database (each, a "Work").
|
||||||
|
|
||||||
|
Certain owners wish to permanently relinquish those rights to a Work for
|
||||||
|
the purpose of contributing to a commons of creative, cultural and
|
||||||
|
scientific works ("Commons") that the public can reliably and without fear
|
||||||
|
of later claims of infringement build upon, modify, incorporate in other
|
||||||
|
works, reuse and redistribute as freely as possible in any form whatsoever
|
||||||
|
and for any purposes, including without limitation commercial purposes.
|
||||||
|
These owners may contribute to the Commons to promote the ideal of a free
|
||||||
|
culture and the further production of creative, cultural and scientific
|
||||||
|
works, or to gain reputation or greater distribution for their Work in
|
||||||
|
part through the use and efforts of others.
|
||||||
|
|
||||||
|
For these and/or other purposes and motivations, and without any
|
||||||
|
expectation of additional consideration or compensation, the person
|
||||||
|
associating CC0 with a Work (the "Affirmer"), to the extent that he or she
|
||||||
|
is an owner of Copyright and Related Rights in the Work, voluntarily
|
||||||
|
elects to apply CC0 to the Work and publicly distribute the Work under its
|
||||||
|
terms, with knowledge of his or her Copyright and Related Rights in the
|
||||||
|
Work and the meaning and intended legal effect of CC0 on those rights.
|
||||||
|
|
||||||
|
1. Copyright and Related Rights. A Work made available under CC0 may be
|
||||||
|
protected by copyright and related or neighboring rights ("Copyright and
|
||||||
|
Related Rights"). Copyright and Related Rights include, but are not
|
||||||
|
limited to, the following:
|
||||||
|
|
||||||
|
i. the right to reproduce, adapt, distribute, perform, display,
|
||||||
|
communicate, and translate a Work;
|
||||||
|
ii. moral rights retained by the original author(s) and/or performer(s);
|
||||||
|
iii. publicity and privacy rights pertaining to a person's image or
|
||||||
|
likeness depicted in a Work;
|
||||||
|
iv. rights protecting against unfair competition in regards to a Work,
|
||||||
|
subject to the limitations in paragraph 4(a), below;
|
||||||
|
v. rights protecting the extraction, dissemination, use and reuse of data
|
||||||
|
in a Work;
|
||||||
|
vi. database rights (such as those arising under Directive 96/9/EC of the
|
||||||
|
European Parliament and of the Council of 11 March 1996 on the legal
|
||||||
|
protection of databases, and under any national implementation
|
||||||
|
thereof, including any amended or successor version of such
|
||||||
|
directive); and
|
||||||
|
vii. other similar, equivalent or corresponding rights throughout the
|
||||||
|
world based on applicable law or treaty, and any national
|
||||||
|
implementations thereof.
|
||||||
|
|
||||||
|
2. Waiver. To the greatest extent permitted by, but not in contravention
|
||||||
|
of, applicable law, Affirmer hereby overtly, fully, permanently,
|
||||||
|
irrevocably and unconditionally waives, abandons, and surrenders all of
|
||||||
|
Affirmer's Copyright and Related Rights and associated claims and causes
|
||||||
|
of action, whether now known or unknown (including existing as well as
|
||||||
|
future claims and causes of action), in the Work (i) in all territories
|
||||||
|
worldwide, (ii) for the maximum duration provided by applicable law or
|
||||||
|
treaty (including future time extensions), (iii) in any current or future
|
||||||
|
medium and for any number of copies, and (iv) for any purpose whatsoever,
|
||||||
|
including without limitation commercial, advertising or promotional
|
||||||
|
purposes (the "Waiver"). Affirmer makes the Waiver for the benefit of each
|
||||||
|
member of the public at large and to the detriment of Affirmer's heirs and
|
||||||
|
successors, fully intending that such Waiver shall not be subject to
|
||||||
|
revocation, rescission, cancellation, termination, or any other legal or
|
||||||
|
equitable action to disrupt the quiet enjoyment of the Work by the public
|
||||||
|
as contemplated by Affirmer's express Statement of Purpose.
|
||||||
|
|
||||||
|
3. Public License Fallback. Should any part of the Waiver for any reason
|
||||||
|
be judged legally invalid or ineffective under applicable law, then the
|
||||||
|
Waiver shall be preserved to the maximum extent permitted taking into
|
||||||
|
account Affirmer's express Statement of Purpose. In addition, to the
|
||||||
|
extent the Waiver is so judged Affirmer hereby grants to each affected
|
||||||
|
person a royalty-free, non transferable, non sublicensable, non exclusive,
|
||||||
|
irrevocable and unconditional license to exercise Affirmer's Copyright and
|
||||||
|
Related Rights in the Work (i) in all territories worldwide, (ii) for the
|
||||||
|
maximum duration provided by applicable law or treaty (including future
|
||||||
|
time extensions), (iii) in any current or future medium and for any number
|
||||||
|
of copies, and (iv) for any purpose whatsoever, including without
|
||||||
|
limitation commercial, advertising or promotional purposes (the
|
||||||
|
"License"). The License shall be deemed effective as of the date CC0 was
|
||||||
|
applied by Affirmer to the Work. Should any part of the License for any
|
||||||
|
reason be judged legally invalid or ineffective under applicable law, such
|
||||||
|
partial invalidity or ineffectiveness shall not invalidate the remainder
|
||||||
|
of the License, and in such case Affirmer hereby affirms that he or she
|
||||||
|
will not (i) exercise any of his or her remaining Copyright and Related
|
||||||
|
Rights in the Work or (ii) assert any associated claims and causes of
|
||||||
|
action with respect to the Work, in either case contrary to Affirmer's
|
||||||
|
express Statement of Purpose.
|
||||||
|
|
||||||
|
4. Limitations and Disclaimers.
|
||||||
|
|
||||||
|
a. No trademark or patent rights held by Affirmer are waived, abandoned,
|
||||||
|
surrendered, licensed or otherwise affected by this document.
|
||||||
|
b. Affirmer offers the Work as-is and makes no representations or
|
||||||
|
warranties of any kind concerning the Work, express, implied,
|
||||||
|
statutory or otherwise, including without limitation warranties of
|
||||||
|
title, merchantability, fitness for a particular purpose, non
|
||||||
|
infringement, or the absence of latent or other defects, accuracy, or
|
||||||
|
the present or absence of errors, whether or not discoverable, all to
|
||||||
|
the greatest extent permissible under applicable law.
|
||||||
|
c. Affirmer disclaims responsibility for clearing rights of other persons
|
||||||
|
that may apply to the Work or any use thereof, including without
|
||||||
|
limitation any person's Copyright and Related Rights in the Work.
|
||||||
|
Further, Affirmer disclaims responsibility for obtaining any necessary
|
||||||
|
consents, permissions or other rights required for any use of the
|
||||||
|
Work.
|
||||||
|
d. Affirmer understands and acknowledges that Creative Commons is not a
|
||||||
|
party to this document and has no duty or obligation with respect to
|
||||||
|
this CC0 or use of the Work.
|
19
README.md
Normal file
19
README.md
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
# TabooLib SDK
|
||||||
|
|
||||||
|
## 构建发行版本
|
||||||
|
|
||||||
|
发行版本用于正常使用, 不含 TabooLib 本体。
|
||||||
|
|
||||||
|
```
|
||||||
|
./gradlew build
|
||||||
|
```
|
||||||
|
|
||||||
|
## 构建开发版本
|
||||||
|
|
||||||
|
开发版本包含 TabooLib 本体, 用于开发者使用, 但不可运行。
|
||||||
|
|
||||||
|
```
|
||||||
|
./gradlew taboolibBuildApi -PDeleteCode
|
||||||
|
```
|
||||||
|
|
||||||
|
> 参数 -PDeleteCode 表示移除所有逻辑代码以减少体积。
|
47
build.gradle.kts
Normal file
47
build.gradle.kts
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
import io.izzel.taboolib.gradle.BUKKIT
|
||||||
|
import io.izzel.taboolib.gradle.BUKKIT_ALL
|
||||||
|
import io.izzel.taboolib.gradle.UNIVERSAL
|
||||||
|
import org.jetbrains.kotlin.gradle.tasks.KotlinCompile
|
||||||
|
|
||||||
|
plugins {
|
||||||
|
java
|
||||||
|
id("io.izzel.taboolib") version "2.0.11"
|
||||||
|
id("org.jetbrains.kotlin.jvm") version "1.8.22"
|
||||||
|
}
|
||||||
|
|
||||||
|
taboolib {
|
||||||
|
env {
|
||||||
|
// 安装模块
|
||||||
|
install(UNIVERSAL, BUKKIT_ALL)
|
||||||
|
}
|
||||||
|
version { taboolib = "6.1.2-beta10" }
|
||||||
|
}
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
mavenCentral()
|
||||||
|
maven { url = uri("https://repo.tabooproject.org/repository/releases/") }
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
compileOnly("ink.ptms.core:v12004:12004:mapped")
|
||||||
|
compileOnly("ink.ptms.core:v12004:12004:universal")
|
||||||
|
compileOnly(kotlin("stdlib"))
|
||||||
|
compileOnly(fileTree("libs"))
|
||||||
|
compileOnly("ink.ptms.adyeshach:all:2.0.0-snapshot-36")
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.withType<JavaCompile> {
|
||||||
|
options.encoding = "UTF-8"
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.withType<KotlinCompile> {
|
||||||
|
kotlinOptions {
|
||||||
|
jvmTarget = "1.8"
|
||||||
|
freeCompilerArgs = listOf("-Xjvm-default=all")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
configure<JavaPluginConvention> {
|
||||||
|
sourceCompatibility = JavaVersion.VERSION_1_8
|
||||||
|
targetCompatibility = JavaVersion.VERSION_1_8
|
||||||
|
}
|
6
gradle.properties
Normal file
6
gradle.properties
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
group=io.github.beradq.adybubbles
|
||||||
|
version=1.0.0
|
||||||
|
kotlin.incremental=true
|
||||||
|
kotlin.incremental.java=true
|
||||||
|
kotlin.caching.enabled=true
|
||||||
|
kotlin.parallel.tasks.in.project=true
|
BIN
gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
BIN
gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
Binary file not shown.
5
gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
5
gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
distributionBase=GRADLE_USER_HOME
|
||||||
|
distributionPath=wrapper/dists
|
||||||
|
distributionUrl=https\://services.gradle.org/distributions/gradle-7.0.2-bin.zip
|
||||||
|
zipStoreBase=GRADLE_USER_HOME
|
||||||
|
zipStorePath=wrapper/dists
|
172
gradlew
vendored
Normal file
172
gradlew
vendored
Normal file
@ -0,0 +1,172 @@
|
|||||||
|
#!/usr/bin/env sh
|
||||||
|
|
||||||
|
##############################################################################
|
||||||
|
##
|
||||||
|
## Gradle start up script for UN*X
|
||||||
|
##
|
||||||
|
##############################################################################
|
||||||
|
|
||||||
|
# Attempt to set APP_HOME
|
||||||
|
# Resolve links: $0 may be a link
|
||||||
|
PRG="$0"
|
||||||
|
# Need this for relative symlinks.
|
||||||
|
while [ -h "$PRG" ] ; do
|
||||||
|
ls=`ls -ld "$PRG"`
|
||||||
|
link=`expr "$ls" : '.*-> \(.*\)$'`
|
||||||
|
if expr "$link" : '/.*' > /dev/null; then
|
||||||
|
PRG="$link"
|
||||||
|
else
|
||||||
|
PRG=`dirname "$PRG"`"/$link"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
SAVED="`pwd`"
|
||||||
|
cd "`dirname \"$PRG\"`/" >/dev/null
|
||||||
|
APP_HOME="`pwd -P`"
|
||||||
|
cd "$SAVED" >/dev/null
|
||||||
|
|
||||||
|
APP_NAME="Gradle"
|
||||||
|
APP_BASE_NAME=`basename "$0"`
|
||||||
|
|
||||||
|
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||||
|
DEFAULT_JVM_OPTS=""
|
||||||
|
|
||||||
|
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||||
|
MAX_FD="maximum"
|
||||||
|
|
||||||
|
warn () {
|
||||||
|
echo "$*"
|
||||||
|
}
|
||||||
|
|
||||||
|
die () {
|
||||||
|
echo
|
||||||
|
echo "$*"
|
||||||
|
echo
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
# OS specific support (must be 'true' or 'false').
|
||||||
|
cygwin=false
|
||||||
|
msys=false
|
||||||
|
darwin=false
|
||||||
|
nonstop=false
|
||||||
|
case "`uname`" in
|
||||||
|
CYGWIN* )
|
||||||
|
cygwin=true
|
||||||
|
;;
|
||||||
|
Darwin* )
|
||||||
|
darwin=true
|
||||||
|
;;
|
||||||
|
MINGW* )
|
||||||
|
msys=true
|
||||||
|
;;
|
||||||
|
NONSTOP* )
|
||||||
|
nonstop=true
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
|
||||||
|
|
||||||
|
# Determine the Java command to use to start the JVM.
|
||||||
|
if [ -n "$JAVA_HOME" ] ; then
|
||||||
|
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||||
|
# IBM's JDK on AIX uses strange locations for the executables
|
||||||
|
JAVACMD="$JAVA_HOME/jre/sh/java"
|
||||||
|
else
|
||||||
|
JAVACMD="$JAVA_HOME/bin/java"
|
||||||
|
fi
|
||||||
|
if [ ! -x "$JAVACMD" ] ; then
|
||||||
|
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
|
||||||
|
|
||||||
|
Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
location of your Java installation."
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
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.
|
||||||
|
|
||||||
|
Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
location of your Java installation."
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Increase the maximum file descriptors if we can.
|
||||||
|
if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
|
||||||
|
MAX_FD_LIMIT=`ulimit -H -n`
|
||||||
|
if [ $? -eq 0 ] ; then
|
||||||
|
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
|
||||||
|
MAX_FD="$MAX_FD_LIMIT"
|
||||||
|
fi
|
||||||
|
ulimit -n $MAX_FD
|
||||||
|
if [ $? -ne 0 ] ; then
|
||||||
|
warn "Could not set maximum file descriptor limit: $MAX_FD"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# For Darwin, add options to specify how the application appears in the dock
|
||||||
|
if $darwin; then
|
||||||
|
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
|
||||||
|
fi
|
||||||
|
|
||||||
|
# For Cygwin, switch paths to Windows format before running java
|
||||||
|
if $cygwin ; then
|
||||||
|
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
|
||||||
|
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
|
||||||
|
JAVACMD=`cygpath --unix "$JAVACMD"`
|
||||||
|
|
||||||
|
# We build the pattern for arguments to be converted via cygpath
|
||||||
|
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
|
||||||
|
SEP=""
|
||||||
|
for dir in $ROOTDIRSRAW ; do
|
||||||
|
ROOTDIRS="$ROOTDIRS$SEP$dir"
|
||||||
|
SEP="|"
|
||||||
|
done
|
||||||
|
OURCYGPATTERN="(^($ROOTDIRS))"
|
||||||
|
# Add a user-defined pattern to the cygpath arguments
|
||||||
|
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
|
||||||
|
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
|
||||||
|
fi
|
||||||
|
# Now convert the arguments - kludge to limit ourselves to /bin/sh
|
||||||
|
i=0
|
||||||
|
for arg in "$@" ; do
|
||||||
|
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
|
||||||
|
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
|
||||||
|
|
||||||
|
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
|
||||||
|
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
|
||||||
|
else
|
||||||
|
eval `echo args$i`="\"$arg\""
|
||||||
|
fi
|
||||||
|
i=$((i+1))
|
||||||
|
done
|
||||||
|
case $i in
|
||||||
|
(0) set -- ;;
|
||||||
|
(1) set -- "$args0" ;;
|
||||||
|
(2) set -- "$args0" "$args1" ;;
|
||||||
|
(3) set -- "$args0" "$args1" "$args2" ;;
|
||||||
|
(4) set -- "$args0" "$args1" "$args2" "$args3" ;;
|
||||||
|
(5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
|
||||||
|
(6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
|
||||||
|
(7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
|
||||||
|
(8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
|
||||||
|
(9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
|
||||||
|
esac
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Escape application args
|
||||||
|
save () {
|
||||||
|
for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
|
||||||
|
echo " "
|
||||||
|
}
|
||||||
|
APP_ARGS=$(save "$@")
|
||||||
|
|
||||||
|
# Collect all arguments for the java command, following the shell quoting and substitution rules
|
||||||
|
eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
|
||||||
|
|
||||||
|
# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong
|
||||||
|
if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then
|
||||||
|
cd "$(dirname "$0")"
|
||||||
|
fi
|
||||||
|
|
||||||
|
exec "$JAVACMD" "$@"
|
84
gradlew.bat
vendored
Normal file
84
gradlew.bat
vendored
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
@if "%DEBUG%" == "" @echo off
|
||||||
|
@rem ##########################################################################
|
||||||
|
@rem
|
||||||
|
@rem Gradle startup script for Windows
|
||||||
|
@rem
|
||||||
|
@rem ##########################################################################
|
||||||
|
|
||||||
|
@rem Set local scope for the variables with windows NT shell
|
||||||
|
if "%OS%"=="Windows_NT" setlocal
|
||||||
|
|
||||||
|
set DIRNAME=%~dp0
|
||||||
|
if "%DIRNAME%" == "" set DIRNAME=.
|
||||||
|
set APP_BASE_NAME=%~n0
|
||||||
|
set APP_HOME=%DIRNAME%
|
||||||
|
|
||||||
|
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||||
|
set DEFAULT_JVM_OPTS=
|
||||||
|
|
||||||
|
@rem Find java.exe
|
||||||
|
if defined JAVA_HOME goto findJavaFromJavaHome
|
||||||
|
|
||||||
|
set JAVA_EXE=java.exe
|
||||||
|
%JAVA_EXE% -version >NUL 2>&1
|
||||||
|
if "%ERRORLEVEL%" == "0" goto init
|
||||||
|
|
||||||
|
echo.
|
||||||
|
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||||
|
echo.
|
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
echo location of your Java installation.
|
||||||
|
|
||||||
|
goto fail
|
||||||
|
|
||||||
|
:findJavaFromJavaHome
|
||||||
|
set JAVA_HOME=%JAVA_HOME:"=%
|
||||||
|
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
|
||||||
|
|
||||||
|
if exist "%JAVA_EXE%" goto init
|
||||||
|
|
||||||
|
echo.
|
||||||
|
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
|
||||||
|
echo.
|
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
echo location of your Java installation.
|
||||||
|
|
||||||
|
goto fail
|
||||||
|
|
||||||
|
:init
|
||||||
|
@rem Get command-line arguments, handling Windows variants
|
||||||
|
|
||||||
|
if not "%OS%" == "Windows_NT" goto win9xME_args
|
||||||
|
|
||||||
|
:win9xME_args
|
||||||
|
@rem Slurp the command line arguments.
|
||||||
|
set CMD_LINE_ARGS=
|
||||||
|
set _SKIP=2
|
||||||
|
|
||||||
|
:win9xME_args_slurp
|
||||||
|
if "x%~1" == "x" goto execute
|
||||||
|
|
||||||
|
set CMD_LINE_ARGS=%*
|
||||||
|
|
||||||
|
:execute
|
||||||
|
@rem Setup the command line
|
||||||
|
|
||||||
|
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
||||||
|
|
||||||
|
@rem Execute Gradle
|
||||||
|
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
|
||||||
|
|
||||||
|
:end
|
||||||
|
@rem End local scope for the variables with windows NT shell
|
||||||
|
if "%ERRORLEVEL%"=="0" goto mainEnd
|
||||||
|
|
||||||
|
:fail
|
||||||
|
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
||||||
|
rem the _cmd.exe /c_ return code!
|
||||||
|
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
|
||||||
|
exit /b 1
|
||||||
|
|
||||||
|
:mainEnd
|
||||||
|
if "%OS%"=="Windows_NT" endlocal
|
||||||
|
|
||||||
|
:omega
|
1
settings.gradle.kts
Normal file
1
settings.gradle.kts
Normal file
@ -0,0 +1 @@
|
|||||||
|
rootProject.name = "AdyBubbles"
|
12
src/main/kotlin/io/github/beradq/adybubbles/AdyBubbles.kt
Normal file
12
src/main/kotlin/io/github/beradq/adybubbles/AdyBubbles.kt
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
package io.github.beradq.adybubbles
|
||||||
|
|
||||||
|
import ink.ptms.adyeshach.core.Adyeshach
|
||||||
|
import taboolib.common.platform.Plugin
|
||||||
|
import taboolib.common.platform.function.info
|
||||||
|
|
||||||
|
object AdyBubbles : Plugin() {
|
||||||
|
val adyApi = Adyeshach.api()
|
||||||
|
override fun onEnable() {
|
||||||
|
info("Successfully running AdyBubbles!")
|
||||||
|
}
|
||||||
|
}
|
71
src/main/kotlin/io/github/beradq/adybubbles/BubblesBundle.kt
Normal file
71
src/main/kotlin/io/github/beradq/adybubbles/BubblesBundle.kt
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
package io.github.beradq.adybubbles
|
||||||
|
|
||||||
|
import ink.ptms.adyeshach.core.AdyeshachHologram
|
||||||
|
import io.github.beradq.adybubbles.AdyBubbles.adyApi
|
||||||
|
import taboolib.common.LifeCycle
|
||||||
|
import taboolib.common.platform.Awake
|
||||||
|
|
||||||
|
|
||||||
|
object BubblesBundle {
|
||||||
|
private val map: MutableMap<String, MutableList<AdyeshachHologram>> = mutableMapOf()
|
||||||
|
operator fun get(key: String): List<AdyeshachHologram>? {
|
||||||
|
return map[key]
|
||||||
|
}
|
||||||
|
|
||||||
|
fun popup(npcId: String, text: String) {
|
||||||
|
val adyEntity = adyApi.getEntityFinder().getEntityFromUniqueId(npcId) ?: error("Entity not found: $npcId")
|
||||||
|
val hologramList = map.getOrPut(npcId) { mutableListOf() }
|
||||||
|
val hologram = adyApi.getHologramHandler().createHologram(
|
||||||
|
adyEntity.getLocation().add(0.0, adyEntity.entitySize.height, 0.0), listOf(text)
|
||||||
|
)
|
||||||
|
hologramList.add(hologram)
|
||||||
|
updateLocation(npcId)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun clear(npcId: String) {
|
||||||
|
map[npcId]?.forEach { it.remove() }
|
||||||
|
map[npcId]?.clear()
|
||||||
|
}
|
||||||
|
|
||||||
|
fun deleteFirst(npcId: String) {
|
||||||
|
map[npcId]?.removeFirst()?.remove()
|
||||||
|
}
|
||||||
|
|
||||||
|
fun limit(npcId: String, limit: Int) {
|
||||||
|
if (!map.containsKey(npcId)) return
|
||||||
|
while (map[npcId]!!.size > limit) {
|
||||||
|
deleteFirst(npcId)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun updateLocation(npcId: String) {
|
||||||
|
val adyEntity = adyApi.getEntityFinder().getEntityFromUniqueId(npcId) ?: error("Entity not found: $npcId")
|
||||||
|
val hologramList = map.getOrPut(npcId) { mutableListOf() }
|
||||||
|
hologramList.reversed().withIndex().forEach {
|
||||||
|
it.value.teleport(
|
||||||
|
adyEntity.getLocation().add(
|
||||||
|
0.0,
|
||||||
|
adyEntity.entitySize.height + Config.bubblesConfig.offset + it.index * Config.bubblesConfig.lineHeight,
|
||||||
|
0.0
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Awake(LifeCycle.ACTIVE)
|
||||||
|
private fun enable() {
|
||||||
|
adyApi.getEventBus().prepareTeleport { e ->
|
||||||
|
val hologram = this[e.entity.uniqueId] ?: return@prepareTeleport true
|
||||||
|
hologram.reversed().withIndex().forEach {
|
||||||
|
it.value.teleport(
|
||||||
|
e.location.clone().add(
|
||||||
|
0.0,
|
||||||
|
e.entity.entitySize.height + Config.bubblesConfig.offset + it.index * Config.bubblesConfig.lineHeight,
|
||||||
|
0.0
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,40 @@
|
|||||||
|
package io.github.beradq.adybubbles
|
||||||
|
|
||||||
|
import org.bukkit.command.CommandSender
|
||||||
|
import org.bukkit.entity.Player
|
||||||
|
import taboolib.common.platform.command.CommandBody
|
||||||
|
import taboolib.common.platform.command.CommandHeader
|
||||||
|
import taboolib.common.platform.command.subCommand
|
||||||
|
import taboolib.platform.util.nextChatInTick
|
||||||
|
|
||||||
|
@CommandHeader("bubbles-chat")
|
||||||
|
object BubblesChatCommand {
|
||||||
|
@CommandBody
|
||||||
|
val edit = subCommand {
|
||||||
|
dynamic("NPC_ID") {
|
||||||
|
suggestion<CommandSender>(uncheck = true) { sender, _ ->
|
||||||
|
AdyBubbles.adyApi.getEntityFinder().getEntities().map { it.id }
|
||||||
|
}
|
||||||
|
execute<CommandSender> { sender, context, _ ->
|
||||||
|
val npcId = context["NPC_ID"]
|
||||||
|
val npcList = AdyBubbles.adyApi.getEntityFinder().getEntitiesFromId(npcId)
|
||||||
|
if (npcList.size == 1) {
|
||||||
|
TraitBubblesChat.edit(sender as Player, npcList.first())
|
||||||
|
} else {
|
||||||
|
sender.sendMessage("找到了${npcList.size}个NPC:")
|
||||||
|
npcList.withIndex().forEach {
|
||||||
|
sender.sendMessage("${it.index + 1}: ${it.value.uniqueId}")
|
||||||
|
}
|
||||||
|
sender.sendMessage("请通过聊天输入序号指定NPC")
|
||||||
|
(sender as Player).nextChatInTick(20 * 10, {
|
||||||
|
val npc = npcList[it.toInt() - 1]
|
||||||
|
TraitBubblesChat.edit(sender, npc)
|
||||||
|
}, {
|
||||||
|
sender.sendMessage("超时")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,78 @@
|
|||||||
|
package io.github.beradq.adybubbles
|
||||||
|
|
||||||
|
import org.bukkit.command.CommandSender
|
||||||
|
import org.bukkit.entity.Player
|
||||||
|
import taboolib.common.platform.command.*
|
||||||
|
import taboolib.platform.util.*
|
||||||
|
|
||||||
|
@CommandHeader("bubbles")
|
||||||
|
object BubblesCommand {
|
||||||
|
@CommandBody
|
||||||
|
val popup = subCommand {
|
||||||
|
dynamic("NPC_ID") {
|
||||||
|
suggestion<CommandSender>(uncheck = true) { sender, _ ->
|
||||||
|
AdyBubbles.adyApi.getEntityFinder().getEntities().map { it.id }
|
||||||
|
}
|
||||||
|
dynamic("TEXT") {
|
||||||
|
execute<CommandSender> { sender, context, _ ->
|
||||||
|
val npcId = context["NPC_ID"]
|
||||||
|
val npcList = AdyBubbles.adyApi.getEntityFinder().getEntitiesFromId(npcId)
|
||||||
|
if (npcList.size == 1) {
|
||||||
|
BubblesBundle.popup(npcList.first().uniqueId, context["TEXT"])
|
||||||
|
} else {
|
||||||
|
sender.sendMessage("找到了${npcList.size}个NPC:")
|
||||||
|
npcList.withIndex().forEach {
|
||||||
|
sender.sendMessage("${it.index + 1}: ${it.value.uniqueId}")
|
||||||
|
}
|
||||||
|
sender.sendMessage("请通过聊天输入序号指定NPC")
|
||||||
|
(sender as Player).nextChatInTick(20 * 10, {
|
||||||
|
val npcUniqueId = npcList[it.toInt() - 1].uniqueId
|
||||||
|
BubblesBundle.popup(npcUniqueId, context["TEXT"])
|
||||||
|
}, {
|
||||||
|
sender.sendMessage("超时")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@CommandBody
|
||||||
|
val clear = subCommand {
|
||||||
|
dynamic("NPC_ID") {
|
||||||
|
suggestion<CommandSender>(uncheck = true) { sender, _ ->
|
||||||
|
AdyBubbles.adyApi.getEntityFinder().getEntities().map { it.id }
|
||||||
|
}
|
||||||
|
execute<CommandSender> { sender, context, _ ->
|
||||||
|
val npcId = context["NPC_ID"]
|
||||||
|
val npcList = AdyBubbles.adyApi.getEntityFinder().getEntitiesFromId(npcId)
|
||||||
|
if (npcList.size == 1) {
|
||||||
|
BubblesBundle.clear(npcList.first().uniqueId)
|
||||||
|
} else {
|
||||||
|
sender.sendMessage("找到了${npcList.size}个NPC:")
|
||||||
|
npcList.withIndex().forEach {
|
||||||
|
sender.sendMessage("${it.index + 1}: ${it.value.uniqueId}")
|
||||||
|
}
|
||||||
|
sender.sendMessage("请通过聊天输入序号指定NPC")
|
||||||
|
(sender as Player).nextChatInTick(20 * 10, {
|
||||||
|
val npc = npcList[it.toInt() - 1]
|
||||||
|
BubblesBundle.clear(npc.uniqueId)
|
||||||
|
}, {
|
||||||
|
sender.sendMessage("超时")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@CommandBody
|
||||||
|
val reload = subCommand {
|
||||||
|
execute<CommandSender> { sender, _, _ ->
|
||||||
|
Config.reload()
|
||||||
|
sender.sendMessage("重载完成")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
68
src/main/kotlin/io/github/beradq/adybubbles/Command.kt~
Normal file
68
src/main/kotlin/io/github/beradq/adybubbles/Command.kt~
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
package io.github.beradq.adybubbles
|
||||||
|
|
||||||
|
import org.bukkit.command.CommandSender
|
||||||
|
import org.bukkit.entity.Player
|
||||||
|
import taboolib.common.platform.command.*
|
||||||
|
import taboolib.platform.util.*
|
||||||
|
|
||||||
|
@CommandHeader("bubbles")
|
||||||
|
object Command {
|
||||||
|
@CommandBody
|
||||||
|
val popup = subCommand {
|
||||||
|
dynamic("NPC_ID") {
|
||||||
|
suggestion<CommandSender>(uncheck = true) { sender, _ ->
|
||||||
|
AdyBubbles.adyApi.getEntityFinder().getEntities().map { it.id }
|
||||||
|
}
|
||||||
|
dynamic("TEXT") {
|
||||||
|
execute<CommandSender> { sender, context, _ ->
|
||||||
|
val npcId = context["NPC_ID"]
|
||||||
|
val npcList = AdyBubbles.adyApi.getEntityFinder().getEntitiesFromId(npcId)
|
||||||
|
if (npcList.size == 1) {
|
||||||
|
BubblesBundle.popup(npcList.first().uniqueId, context["TEXT"])
|
||||||
|
} else {
|
||||||
|
sender.sendMessage("找到了${npcList.size}个NPC:")
|
||||||
|
npcList.withIndex().forEach {
|
||||||
|
sender.sendMessage("${it.index + 1}: ${it.value.uniqueId}")
|
||||||
|
}
|
||||||
|
sender.sendMessage("请通过聊天输入序号指定NPC")
|
||||||
|
(sender as Player).nextChatInTick(20 * 10, {
|
||||||
|
val npcUniqueId = npcList[it.toInt() - 1].uniqueId
|
||||||
|
BubblesBundle.popup(npcUniqueId, context["TEXT"])
|
||||||
|
}, {
|
||||||
|
sender.sendMessage("超时")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@CommandBody
|
||||||
|
val edit = subCommand {
|
||||||
|
dynamic("NPC_ID") {
|
||||||
|
suggestion<CommandSender>(uncheck = true) { sender, _ ->
|
||||||
|
AdyBubbles.adyApi.getEntityFinder().getEntities().map { it.id }
|
||||||
|
}
|
||||||
|
execute<CommandSender> { sender, context, _ ->
|
||||||
|
val npcId = context["NPC_ID"]
|
||||||
|
val npcList = AdyBubbles.adyApi.getEntityFinder().getEntitiesFromId(npcId)
|
||||||
|
if (npcList.size == 1) {
|
||||||
|
TraitBubblesChat.edit(sender as Player, npcList.first())
|
||||||
|
} else {
|
||||||
|
sender.sendMessage("找到了${npcList.size}个NPC:")
|
||||||
|
npcList.withIndex().forEach {
|
||||||
|
sender.sendMessage("${it.index + 1}: ${it.value.uniqueId}")
|
||||||
|
}
|
||||||
|
sender.sendMessage("请通过聊天输入序号指定NPC")
|
||||||
|
(sender as Player).nextChatInTick(20 * 10, {
|
||||||
|
val npc = npcList[it.toInt() - 1]
|
||||||
|
TraitBubblesChat.edit(sender, npc)
|
||||||
|
}, {
|
||||||
|
sender.sendMessage("超时")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
34
src/main/kotlin/io/github/beradq/adybubbles/Config.kt
Normal file
34
src/main/kotlin/io/github/beradq/adybubbles/Config.kt
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
package io.github.beradq.adybubbles
|
||||||
|
|
||||||
|
import taboolib.common.LifeCycle
|
||||||
|
import taboolib.common.platform.Awake
|
||||||
|
import taboolib.module.configuration.Config
|
||||||
|
import taboolib.module.configuration.ConfigFile
|
||||||
|
|
||||||
|
object Config {
|
||||||
|
@Config("config.yml")
|
||||||
|
lateinit var configFile: ConfigFile
|
||||||
|
lateinit var bubblesConfig: BubblesConfig
|
||||||
|
|
||||||
|
class BubblesChatConfig(
|
||||||
|
val period: Long,
|
||||||
|
val limit: Int
|
||||||
|
)
|
||||||
|
|
||||||
|
class BubblesConfig(
|
||||||
|
val offset: Double,
|
||||||
|
val lineHeight: Double,
|
||||||
|
val chat: BubblesChatConfig
|
||||||
|
)
|
||||||
|
|
||||||
|
@Awake(LifeCycle.ENABLE)
|
||||||
|
fun reload() {
|
||||||
|
configFile.reload()
|
||||||
|
val offset = configFile["offset"] as? Double ?: 0.5
|
||||||
|
val lineHeight = configFile["line-height"] as? Double ?: 0.5
|
||||||
|
val chat = configFile.getConfigurationSection("chat")
|
||||||
|
val period = chat?.get("period") as? Long ?: (20 * 3)
|
||||||
|
val limit = chat?.get("limit") as? Int ?: 3
|
||||||
|
bubblesConfig = BubblesConfig(offset, lineHeight, BubblesChatConfig(period, limit))
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,94 @@
|
|||||||
|
package io.github.beradq.adybubbles
|
||||||
|
|
||||||
|
import ink.ptms.adyeshach.core.entity.EntityInstance
|
||||||
|
import ink.ptms.adyeshach.core.event.AdyeshachEntityVisibleEvent
|
||||||
|
import ink.ptms.adyeshach.impl.entity.trait.Trait
|
||||||
|
import ink.ptms.adyeshach.impl.util.Inputs.inputBook
|
||||||
|
import taboolib.common.LifeCycle
|
||||||
|
import org.bukkit.entity.Player
|
||||||
|
import taboolib.common.platform.Awake
|
||||||
|
import taboolib.common.platform.event.EventPriority
|
||||||
|
import taboolib.common.platform.event.SubscribeEvent
|
||||||
|
import taboolib.common.platform.function.submit
|
||||||
|
import taboolib.common.platform.service.PlatformExecutor
|
||||||
|
import taboolib.module.chat.uncolored
|
||||||
|
import java.util.concurrent.CompletableFuture
|
||||||
|
|
||||||
|
object TraitBubblesChat : Trait() {
|
||||||
|
private val bubbleLines = mutableMapOf<String, List<String>>()
|
||||||
|
private val bubbleState = mutableMapOf<String, Int>()
|
||||||
|
private val visibleEntity = mutableListOf<String>()
|
||||||
|
private var handle: PlatformExecutor.PlatformTask? = null
|
||||||
|
|
||||||
|
override fun edit(player: Player, entityInstance: EntityInstance): CompletableFuture<Void> {
|
||||||
|
val future = CompletableFuture<Void>()
|
||||||
|
val content = bubbleLines.getOrPut(entityInstance.uniqueId) { mutableListOf() }
|
||||||
|
player.inputBook(content) {
|
||||||
|
future.complete(null)
|
||||||
|
if (it.all { s -> s.isBlank() }) {
|
||||||
|
data[entityInstance.uniqueId] = null
|
||||||
|
bubbleState.remove(entityInstance.uniqueId)
|
||||||
|
bubbleLines.remove(entityInstance.uniqueId)
|
||||||
|
BubblesBundle.clear(entityInstance.uniqueId)
|
||||||
|
} else {
|
||||||
|
data[entityInstance.uniqueId] = it.uncolored()
|
||||||
|
}
|
||||||
|
update(entityInstance.uniqueId)
|
||||||
|
}
|
||||||
|
return future
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun update(entityInstanceUniqueId: String) {
|
||||||
|
if (!data.contains(entityInstanceUniqueId)) return
|
||||||
|
val lines = data.getStringList(entityInstanceUniqueId)
|
||||||
|
bubbleLines[entityInstanceUniqueId] = lines
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun fistUpdate(entityInstanceUniqueId: String) {
|
||||||
|
if (bubbleLines.containsKey(entityInstanceUniqueId)) return
|
||||||
|
update(entityInstanceUniqueId)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Awake(LifeCycle.ACTIVE)
|
||||||
|
private fun startTicking() {
|
||||||
|
handle?.cancel()
|
||||||
|
handle = submit(period = Config.bubblesConfig.chat.period, async = true) {
|
||||||
|
val ve = visibleEntity.toList()
|
||||||
|
ve.forEach {
|
||||||
|
nextBubble(it)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@SubscribeEvent(priority = EventPriority.MONITOR, ignoreCancelled = true)
|
||||||
|
private fun onVisible(e: AdyeshachEntityVisibleEvent) {
|
||||||
|
if (e.entity.isPublic()) {
|
||||||
|
fistUpdate(e.entity.uniqueId)
|
||||||
|
if (e.visible) {
|
||||||
|
if (visibleEntity.contains(e.entity.uniqueId)) return
|
||||||
|
visibleEntity.add(e.entity.uniqueId)
|
||||||
|
} else {
|
||||||
|
visibleEntity.remove(e.entity.uniqueId)
|
||||||
|
BubblesBundle.clear(e.entity.uniqueId)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun nextBubble(entityInstanceUniqueId: String) {
|
||||||
|
val bubbleLine = bubbleLines[entityInstanceUniqueId] ?: return
|
||||||
|
if (bubbleLine.isEmpty()) return
|
||||||
|
var state = bubbleState[entityInstanceUniqueId] ?: 0
|
||||||
|
if (state >= bubbleLine.size) {
|
||||||
|
state = 0
|
||||||
|
}
|
||||||
|
val currentLine = bubbleLine[state]
|
||||||
|
BubblesBundle.popup(entityInstanceUniqueId, currentLine)
|
||||||
|
BubblesBundle.limit(entityInstanceUniqueId, Config.bubblesConfig.chat.limit)
|
||||||
|
bubbleState[entityInstanceUniqueId] = state
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun id(): String {
|
||||||
|
return "bubbles-chat"
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user