1
0
mirror of https://github.com/chylex/Better-Controls.git synced 2025-08-18 10:31:48 +02:00

26 Commits

Author SHA1 Message Date
eaaa3ddc78 Release v1.2.0a for Minecraft 1.15.2 2021-08-22 06:46:48 +02:00
264a408396 Fix crash when installed on a server 2021-08-22 06:46:33 +02:00
e3cd6d0e63 Update README 2020-12-11 05:26:22 +01:00
348f712729 Release v1.2.0 for Minecraft 1.15.2 2020-12-11 05:26:21 +01:00
d59ac15b52 Fix tapping sneak to stop flying on ground not working when flight inertia is disabled 2020-12-11 05:24:07 +01:00
9d67cf5505 Add option to disable double tapping 'Jump' key to flight
Fixes #6
2020-12-11 05:23:56 +01:00
6e2625c737 Change flight to 'Hold to Sprint' mode at all times & fix not boost unless flying forward 2020-12-11 05:23:15 +01:00
3923a65cd8 Add option to disable flight inertia after releasing movement keys
Fixes #2
2020-12-11 05:23:12 +01:00
c1bbef8228 Add logs folder to .gitignore 2020-12-11 05:21:12 +01:00
4bc33637ac Minor code tweaks 2020-12-11 05:21:09 +01:00
d5e69c1650 Add README 2020-12-11 05:21:06 +01:00
5231da7f42 Why are empty logs committed??? 2020-11-04 03:43:58 +01:00
098ee8d7b5 Fix bad method rename that broke Esc in config screen 2020-10-30 22:24:31 +01:00
07a486b605 Backport to 1.15.x 2020-10-30 22:17:40 +01:00
72f4c6702d Port to Forge 2020-10-30 16:57:17 +01:00
2b1a032295 Release v1.1.0 for Minecraft 1.16.2+ 2020-10-27 17:14:07 +01:00
5f5c3da988 Avoid sudden FOV jump when starting to fly with disabled FOV changing 2020-10-27 17:03:15 +01:00
b511727be9 Add options for vertical flight speed boost in creative/spectator and default/sprinting modes 2020-10-27 16:37:13 +01:00
fa82145530 Add more flight speed options (5x and 7x) 2020-10-26 22:23:21 +01:00
ebd522dca7 Add option to prevent FOV changing while flying 2020-10-26 21:30:23 +01:00
90c3379d57 Add key binding to reset all toggles at once 2020-10-26 21:12:26 +01:00
04c789a50e Implement third sprinting mode (hold key to sprint) & option to change sprinting mode when flying 2020-10-26 21:12:26 +01:00
a79871b690 Release v1.0.1 for Minecraft 1.16.2+ 2020-10-19 21:12:33 +02:00
c537e037b5 Avoid wobble sprinting when resuming sprinting after hitting an obstacle, by simulating holding sprint for multiple ticks 2020-10-19 21:11:38 +02:00
c0e9c92197 Release v1.0.0 for Minecraft 1.16.2+ 2020-10-18 16:47:35 +02:00
e55cef95b3 Fix sprinting not stopping when untoggling sometimes 2020-10-18 16:02:15 +02:00
50 changed files with 442 additions and 437 deletions

4
.gitignore vendored
View File

@@ -24,6 +24,10 @@ bin/
.classpath .classpath
.project .project
# forge
logs/
# fabric # fabric
run/ run/

View File

@@ -27,12 +27,15 @@ You can also bind a key that resets all **Toggle Keybinds** at once. That makes
#### Flying #### Flying
* **Sprint Key Mode While Flying** changes how the Sprint key behaves during flight. * **Double Tap 'Jump' To Fly** can be turned off to prevent accidental flight toggling.
* **Disable Flight Inertia** stops you instantly when you stop holding movement keys.
* **Disable Field Of View Changing** prevents sprinting, potions, and other factors from changing the FOV while flying in creative and spectator mode. * **Disable Field Of View Changing** prevents sprinting, potions, and other factors from changing the FOV while flying in creative and spectator mode.
* **Fly On Ground** lets you fly while touching the ground in creative mode (and also lets you stop flying by tapping Sneak while touching the ground). * **Fly On Ground** lets you fly while touching the ground in creative mode (and also lets you stop flying by tapping Sneak while touching the ground).
* **Flight Speed Multiplier** (0.25x - 8x) changes how fast you fly in creative and spectator mode. * **Flight Speed Multiplier** (0.25x - 8x) changes how fast you fly in creative and spectator mode.
* **Vertical Speed Boost** (up to +300%) adds additional vertical speed boost while flying in creative and spectator mode. * **Vertical Speed Boost** (up to +300%) adds additional vertical speed boost while flying in creative and spectator mode.
Both speed boosts can be configured separately for sprinting, which will be active when the Sprint key is held. Unlike in vanilla, the sprinting flight boost works in all directions.
## Installation ## Installation
The following mod loaders are supported: The following mod loaders are supported:

View File

@@ -1,41 +1,100 @@
plugins { buildscript {
id 'fabric-loom' version '0.5-SNAPSHOT' repositories {
id 'maven-publish' maven { url = 'https://files.minecraftforge.net/maven' }
maven { url = "https://repo.spongepowered.org/maven" }
mavenCentral()
}
dependencies {
classpath group: 'net.minecraftforge.gradle', name: 'ForgeGradle', version: '3.+', changing: true
classpath group: 'org.spongepowered', name: 'mixingradle', version: '0.7-SNAPSHOT'
}
} }
sourceCompatibility = JavaVersion.VERSION_1_8 repositories {
targetCompatibility = JavaVersion.VERSION_1_8 maven {
url "https://maven.tterrag.com/"
}
}
archivesBaseName = project.archives_base_name apply plugin: 'net.minecraftforge.gradle'
version = project.mod_version apply plugin: 'org.spongepowered.mixin'
group = project.maven_group apply plugin: 'eclipse'
def mcversion = "1.15.2"
def forgeversion = "31.2.45"
def prefixName = 'displayName = '
def prefixVersion = 'version = '
def metaLines = file('src/main/resources/META-INF/mods.toml').readLines()
def metaName = metaLines.find { line -> line.startsWith(prefixName) }.substring(prefixName.length())[1..-2]
def metaVersion = metaLines.find { line -> line.startsWith(prefixVersion) }.substring(prefixVersion.length())[1..-2]
group = 'chylex.bettercontrols'
version = metaVersion
archivesBaseName = metaName.replaceAll('\\s', '')
sourceCompatibility = targetCompatibility = compileJava.sourceCompatibility = compileJava.targetCompatibility = '1.8'
minecraft {
mappings channel: 'snapshot', version: '20201030-1.15.1'
runs {
client {
workingDirectory file('run')
property 'mixin.env.disableRefMap', 'true'
arg '-mixin.config=bettercontrols.mixins.json'
mods {
bettercontrols {
source sourceSets.main
}
}
}
server {
workingDirectory file('run')
property 'mixin.env.disableRefMap', 'true'
arg '-mixin.config=bettercontrols.mixins.json'
mods {
bettercontrols {
source sourceSets.main
}
}
}
}
}
dependencies { dependencies {
minecraft "com.mojang:minecraft:${project.minecraft_version}" minecraft 'net.minecraftforge:forge:' + mcversion + '-' + forgeversion
mappings "net.fabricmc:yarn:${project.yarn_mappings}:v2"
modImplementation "net.fabricmc:fabric-loader:${project.loader_version}"
} }
processResources {
inputs.property "version", project.version
from(sourceSets.main.resources.srcDirs) {
include "fabric.mod.json"
expand "version": project.version
}
from(sourceSets.main.resources.srcDirs) {
exclude "fabric.mod.json"
}
}
tasks.withType(JavaCompile) {
options.encoding = "UTF-8"
}
version = 'v' + version
jar { jar {
from "LICENSE" archiveName = archivesBaseName + '-' + mcversion + '-v' + version + '.jar'
exclude "io/github/prospector/"
from('./') {
include 'LICENSE'
}
manifest {
attributes([
'Specification-Title' : 'bettercontrols',
'Specification-Version' : '1',
'Specification-Vendor' : 'chylex',
'Implementation-Title' : metaName,
'Implementation-Version' : metaVersion,
'Implementation-Vendor' : 'chylex',
'Implementation-Timestamp': new Date().format("yyyy-MM-dd'T'HH:mm:ssZ"),
'MixinConfigs' : 'bettercontrols.mixins.json'
])
}
finalizedBy('reobfJar')
} }

View File

@@ -1,15 +1,4 @@
# Done to increase the memory available to gradle. # Sets default memory used for gradle commands. Can be overridden by user or command line properties.
org.gradle.jvmargs=-Xmx1G # This is required to provide enough memory for the Minecraft decompilation process.
org.gradle.jvmargs=-Xmx3G
# Fabric Properties org.gradle.daemon=false
minecraft_version=1.15.2
yarn_mappings=1.15.2+build.17
loader_version=0.10.2+build.210
# Mod Properties
mod_version=1.2.0
maven_group=chylex.bettercontrols
archives_base_name=BetterControls-1.15.x
# https://fabricmc.net/use
# https://maven.fabricmc.net/net/fabricmc/fabric-api/fabric-api

Binary file not shown.

View File

@@ -1,5 +1,5 @@
distributionBase=GRADLE_USER_HOME distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-6.6.1-bin.zip
zipStoreBase=GRADLE_USER_HOME zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists zipStorePath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-4.10.3-bin.zip

53
gradlew vendored
View File

@@ -1,21 +1,5 @@
#!/usr/bin/env sh #!/usr/bin/env sh
#
# Copyright 2015 the original author or authors.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
############################################################################## ##############################################################################
## ##
## Gradle start up script for UN*X ## Gradle start up script for UN*X
@@ -44,7 +28,7 @@ APP_NAME="Gradle"
APP_BASE_NAME=`basename "$0"` 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. # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' DEFAULT_JVM_OPTS=""
# 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"
@@ -82,7 +66,6 @@ esac
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java 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
@@ -126,11 +109,10 @@ if $darwin; then
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
fi fi
# For Cygwin or MSYS, switch paths to Windows format before running java # For Cygwin, switch paths to Windows format before running java
if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then if $cygwin ; then
APP_HOME=`cygpath --path --mixed "$APP_HOME"` APP_HOME=`cygpath --path --mixed "$APP_HOME"`
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
JAVACMD=`cygpath --unix "$JAVACMD"` JAVACMD=`cygpath --unix "$JAVACMD"`
# We build the pattern for arguments to be converted via cygpath # We build the pattern for arguments to be converted via cygpath
@@ -156,19 +138,19 @@ if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then
else else
eval `echo args$i`="\"$arg\"" eval `echo args$i`="\"$arg\""
fi fi
i=`expr $i + 1` i=$((i+1))
done done
case $i in case $i in
0) set -- ;; (0) set -- ;;
1) set -- "$args0" ;; (1) set -- "$args0" ;;
2) set -- "$args0" "$args1" ;; (2) set -- "$args0" "$args1" ;;
3) set -- "$args0" "$args1" "$args2" ;; (3) set -- "$args0" "$args1" "$args2" ;;
4) set -- "$args0" "$args1" "$args2" "$args3" ;; (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
esac esac
fi fi
@@ -177,9 +159,14 @@ save () {
for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
echo " " echo " "
} }
APP_ARGS=`save "$@"` APP_ARGS=$(save "$@")
# Collect all arguments for the java command, following the shell quoting and substitution rules # 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" 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" "$@" exec "$JAVACMD" "$@"

22
gradlew.bat vendored
View File

@@ -1,19 +1,3 @@
@rem
@rem Copyright 2015 the original author or authors.
@rem
@rem Licensed under the Apache License, Version 2.0 (the "License");
@rem you may not use this file except in compliance with the License.
@rem You may obtain a copy of the License at
@rem
@rem https://www.apache.org/licenses/LICENSE-2.0
@rem
@rem Unless required by applicable law or agreed to in writing, software
@rem distributed under the License is distributed on an "AS IS" BASIS,
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
@rem See the License for the specific language governing permissions and
@rem limitations under the License.
@rem
@if "%DEBUG%" == "" @echo off @if "%DEBUG%" == "" @echo off
@rem ########################################################################## @rem ##########################################################################
@rem @rem
@@ -29,11 +13,8 @@ if "%DIRNAME%" == "" set DIRNAME=.
set APP_BASE_NAME=%~n0 set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME% set APP_HOME=%DIRNAME%
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. @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="-Xmx64m" "-Xms64m" set DEFAULT_JVM_OPTS=
@rem Find java.exe @rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome if defined JAVA_HOME goto findJavaFromJavaHome
@@ -84,7 +65,6 @@ set CMD_LINE_ARGS=%*
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
@rem Execute Gradle @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% "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%

View File

@@ -1,10 +0,0 @@
pluginManagement {
repositories {
jcenter()
maven {
name = 'Fabric'
url = 'https://maven.fabricmc.net/'
}
gradlePluginPortal()
}
}

View File

@@ -1,11 +1,33 @@
package chylex.bettercontrols; package chylex.bettercontrols;
import chylex.bettercontrols.config.BetterControlsConfig; import chylex.bettercontrols.config.BetterControlsConfig;
import net.fabricmc.api.ClientModInitializer; import chylex.bettercontrols.gui.BetterControlsScreen;
import net.fabricmc.loader.api.FabricLoader; import net.minecraft.client.Minecraft;
import net.minecraft.client.gui.screen.Screen;
import net.minecraftforge.api.distmarker.Dist;
import net.minecraftforge.fml.DistExecutor;
import net.minecraftforge.fml.ExtensionPoint;
import net.minecraftforge.fml.ModLoadingContext;
import net.minecraftforge.fml.common.Mod;
import net.minecraftforge.fml.loading.FMLPaths;
import net.minecraftforge.fml.network.FMLNetworkConstants;
import org.apache.commons.lang3.tuple.Pair;
public final class BetterControlsMod implements ClientModInitializer{ @Mod("bettercontrols")
public static final BetterControlsConfig config = BetterControlsConfig.load(FabricLoader.getInstance().getConfigDir().resolve("BetterControls.json")); public final class BetterControlsMod{
public static final BetterControlsConfig config = DistExecutor.safeCallWhenOn(Dist.CLIENT, () -> ClientLoader::loadConfig);
@Override public BetterControlsMod(){
public void onInitializeClient(){} ModLoadingContext.get().registerExtensionPoint(ExtensionPoint.CONFIGGUIFACTORY, () -> ClientLoader::createScreen);
ModLoadingContext.get().registerExtensionPoint(ExtensionPoint.DISPLAYTEST, () -> Pair.of(() -> FMLNetworkConstants.IGNORESERVERONLY, (a, b) -> true));
}
private static final class ClientLoader {
public static BetterControlsConfig loadConfig() {
return BetterControlsConfig.load(FMLPaths.CONFIGDIR.get().resolve("BetterControls.json"));
}
public static BetterControlsScreen createScreen(final Minecraft mc, final Screen parentScreen) {
return new BetterControlsScreen(parentScreen);
}
}
} }

View File

@@ -1,16 +0,0 @@
package chylex.bettercontrols.compatibility;
import chylex.bettercontrols.gui.BetterControlsScreen;
import io.github.prospector.modmenu.api.ConfigScreenFactory;
import io.github.prospector.modmenu.api.ModMenuApi;
public class ModMenuSupport implements ModMenuApi{
@Override
public String getModId(){
return "bettercontrols";
}
@Override
public ConfigScreenFactory<?> getModConfigScreenFactory(){
return BetterControlsScreen::new;
}
}

View File

@@ -12,14 +12,14 @@ import chylex.bettercontrols.input.ModifierKey;
import chylex.bettercontrols.input.SprintMode; import chylex.bettercontrols.input.SprintMode;
import chylex.bettercontrols.util.Key; import chylex.bettercontrols.util.Key;
import chylex.bettercontrols.util.LiteralText; import chylex.bettercontrols.util.LiteralText;
import net.minecraft.client.gui.Element; import net.minecraft.client.gui.IGuiEventListener;
import net.minecraft.client.gui.screen.Screen; import net.minecraft.client.gui.screen.Screen;
import net.minecraft.client.gui.screen.options.GameOptionsScreen; import net.minecraft.client.gui.screen.SettingsScreen;
import net.minecraft.client.gui.widget.ButtonWidget; import net.minecraft.client.gui.widget.button.Button;
import net.minecraft.client.options.KeyBinding; import net.minecraft.client.resources.I18n;
import net.minecraft.client.resource.language.I18n; import net.minecraft.client.settings.KeyBinding;
import net.minecraft.client.util.InputUtil;
import org.lwjgl.glfw.GLFW; import org.lwjgl.glfw.GLFW;
import javax.annotation.Nullable;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
import java.util.List; import java.util.List;
@@ -32,7 +32,7 @@ import static chylex.bettercontrols.gui.elements.TextWidget.CENTER;
import static chylex.bettercontrols.util.LiteralText.text; import static chylex.bettercontrols.util.LiteralText.text;
import static chylex.bettercontrols.util.Statics.OPTIONS; import static chylex.bettercontrols.util.Statics.OPTIONS;
public class BetterControlsScreen extends GameOptionsScreen{ public class BetterControlsScreen extends SettingsScreen{
public static final LiteralText TITLE = text("Better Controls"); public static final LiteralText TITLE = text("Better Controls");
private static final int BOTTOM_PADDING = 3; private static final int BOTTOM_PADDING = 3;
@@ -48,7 +48,7 @@ public class BetterControlsScreen extends GameOptionsScreen{
// Options // Options
private int generateSprintingOptions(int y, final List<Element> elements){ private int generateSprintingOptions(int y, final List<IGuiEventListener> elements){
final BetterControlsConfig cfg = BetterControlsMod.config; final BetterControlsConfig cfg = BetterControlsMod.config;
generateKeyBindingWithModifierOption(y, elements, text("Toggle Sprint"), cfg.keyToggleSprint); generateKeyBindingWithModifierOption(y, elements, text("Toggle Sprint"), cfg.keyToggleSprint);
@@ -72,7 +72,7 @@ public class BetterControlsScreen extends GameOptionsScreen{
return y; return y;
} }
private int generateSneakingOptions(int y, final List<Element> elements){ private int generateSneakingOptions(int y, final List<IGuiEventListener> elements){
final BetterControlsConfig cfg = BetterControlsMod.config; final BetterControlsConfig cfg = BetterControlsMod.config;
generateKeyBindingWithModifierOption(y, elements, text("Toggle Sneak"), cfg.keyToggleSneak); generateKeyBindingWithModifierOption(y, elements, text("Toggle Sneak"), cfg.keyToggleSneak);
@@ -87,7 +87,7 @@ public class BetterControlsScreen extends GameOptionsScreen{
} }
@SuppressWarnings({ "AutoBoxing", "AutoUnboxing" }) @SuppressWarnings({ "AutoBoxing", "AutoUnboxing" })
private int generateFlightOptions(int y, final List<Element> elements){ private int generateFlightOptions(int y, final List<IGuiEventListener> elements){
final BetterControlsConfig cfg = BetterControlsMod.config; final BetterControlsConfig cfg = BetterControlsMod.config;
final List<Option<Float>> flightSpeedOptions = Arrays.asList( final List<Option<Float>> flightSpeedOptions = Arrays.asList(
@@ -172,7 +172,7 @@ public class BetterControlsScreen extends GameOptionsScreen{
return y; return y;
} }
private int generateMiscellaneousOptions(int y, final List<Element> elements){ private int generateMiscellaneousOptions(int y, final List<IGuiEventListener> elements){
final BetterControlsConfig cfg = BetterControlsMod.config; final BetterControlsConfig cfg = BetterControlsMod.config;
generateKeyBindingWithModifierOption(y, elements, text("Toggle Walk Forwards"), cfg.keyToggleWalkForward); generateKeyBindingWithModifierOption(y, elements, text("Toggle Walk Forwards"), cfg.keyToggleWalkForward);
@@ -202,7 +202,7 @@ public class BetterControlsScreen extends GameOptionsScreen{
new Option<>(ModifierKey.ALT, text("Alt")) new Option<>(ModifierKey.ALT, text("Alt"))
); );
private void generateKeyBindingWithModifierOption(final int y, final List<Element> elements, final LiteralText text, final KeyBindingWithModifier binding){ private void generateKeyBindingWithModifierOption(final int y, final List<IGuiEventListener> elements, final LiteralText text, final KeyBindingWithModifier binding){
final CycleButtonWidget<ModifierKey> modifierButton = new CycleButtonWidget<>(col4(2), y, COL4_W, MODIFIER_OPTIONS, binding.getModifier(), binding::setModifier); final CycleButtonWidget<ModifierKey> modifierButton = new CycleButtonWidget<>(col4(2), y, COL4_W, MODIFIER_OPTIONS, binding.getModifier(), binding::setModifier);
final KeyBindingWidget bindingButton = new KeyBindingWidget(col4(3), y, COL4_W, binding, this::startEditingKeyBinding); final KeyBindingWidget bindingButton = new KeyBindingWidget(col4(3), y, COL4_W, binding, this::startEditingKeyBinding);
bindingButton.linkButtonToBoundState(modifierButton); bindingButton.linkButtonToBoundState(modifierButton);
@@ -213,7 +213,7 @@ public class BetterControlsScreen extends GameOptionsScreen{
allKeyBindings.add(bindingButton); allKeyBindings.add(bindingButton);
} }
private static void generateLeftSideText(final int y, final List<Element> elements, final LiteralText text){ private static void generateLeftSideText(final int y, final List<IGuiEventListener> elements, final LiteralText text){
elements.add(new TextWidget(col2(0), y, COL2_W - TEXT_PADDING_RIGHT, text)); elements.add(new TextWidget(col2(0), y, COL2_W - TEXT_PADDING_RIGHT, text));
} }
@@ -223,7 +223,7 @@ public class BetterControlsScreen extends GameOptionsScreen{
private KeyBindingWidget editingKeyBinding; private KeyBindingWidget editingKeyBinding;
private final List<KeyBindingWidget> allKeyBindings = new ArrayList<>(); private final List<KeyBindingWidget> allKeyBindings = new ArrayList<>();
public BetterControlsScreen(final Screen parentScreen){ public BetterControlsScreen(@Nullable final Screen parentScreen){
super(parentScreen, OPTIONS, TITLE); super(parentScreen, OPTIONS, TITLE);
} }
@@ -231,7 +231,7 @@ public class BetterControlsScreen extends GameOptionsScreen{
public void init(){ public void init(){
allKeyBindings.clear(); allKeyBindings.clear();
final List<Element> elements = new ArrayList<>(); final List<IGuiEventListener> elements = new ArrayList<>();
int y = 0; int y = 0;
elements.add(new TextWidget(0, y, ROW_WIDTH, ROW_HEIGHT, text("Sprinting"), CENTER)); elements.add(new TextWidget(0, y, ROW_WIDTH, ROW_HEIGHT, text("Sprinting"), CENTER));
@@ -246,7 +246,7 @@ public class BetterControlsScreen extends GameOptionsScreen{
elements.add(new TextWidget(0, y, ROW_WIDTH, ROW_HEIGHT, text("Miscellaneous"), CENTER)); elements.add(new TextWidget(0, y, ROW_WIDTH, ROW_HEIGHT, text("Miscellaneous"), CENTER));
y = generateMiscellaneousOptions(y + ROW_HEIGHT, elements) + TITLE_MARGIN_TOP; y = generateMiscellaneousOptions(y + ROW_HEIGHT, elements) + TITLE_MARGIN_TOP;
addButton(new ButtonWidget(width / 2 - 99, height - 29, 200, 20, I18n.translate("gui.done"), btn -> minecraft.openScreen(parent))); addButton(new Button(width / 2 - 99, height - 29, 200, 20, I18n.format("gui.done"), btn -> minecraft.displayGuiScreen(parentScreen)));
children.add(optionsWidget = new OptionListWidget(21, height - 32, width, height, elements, y - TITLE_MARGIN_TOP + BOTTOM_PADDING)); children.add(optionsWidget = new OptionListWidget(21, height - 32, width, height, elements, y - TITLE_MARGIN_TOP + BOTTOM_PADDING));
} }
@@ -259,7 +259,7 @@ public class BetterControlsScreen extends GameOptionsScreen{
public void render(final int mouseX, final int mouseY, final float delta){ public void render(final int mouseX, final int mouseY, final float delta){
renderBackground(); renderBackground();
optionsWidget.render(mouseX, mouseY, delta); optionsWidget.render(mouseX, mouseY, delta);
drawCenteredString(font, title.asFormattedString(), width / 2, 8, (255 << 16) | (255 << 8) | 255); drawCenteredString(font, title.getString(), width / 2, 8, (255 << 16) | (255 << 8) | 255);
super.render(mouseX, mouseY, delta); super.render(mouseX, mouseY, delta);
} }
@@ -303,7 +303,7 @@ public class BetterControlsScreen extends GameOptionsScreen{
private void onKeyBindingEditingFinished(){ private void onKeyBindingEditingFinished(){
editingKeyBinding = null; editingKeyBinding = null;
KeyBinding.updateKeysByCode(); KeyBinding.resetKeyBindingArrayAndHash();
for(final KeyBindingWidget widget : allKeyBindings){ for(final KeyBindingWidget widget : allKeyBindings){
widget.updateKeyBindingText(); widget.updateKeyBindingText();

View File

@@ -1,9 +1,8 @@
package chylex.bettercontrols.gui; package chylex.bettercontrols.gui;
import chylex.bettercontrols.gui.OptionListWidget.Entry; import chylex.bettercontrols.gui.OptionListWidget.Entry;
import net.minecraft.client.gui.Drawable; import net.minecraft.client.gui.IGuiEventListener;
import net.minecraft.client.gui.Element; import net.minecraft.client.gui.IRenderable;
import net.minecraft.client.gui.widget.AbstractButtonWidget; import net.minecraft.client.gui.widget.list.AbstractOptionList;
import net.minecraft.client.gui.widget.ElementListWidget;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections; import java.util.Collections;
import java.util.List; import java.util.List;
@@ -12,7 +11,7 @@ import java.util.function.Function;
import java.util.stream.Collectors; import java.util.stream.Collectors;
import static chylex.bettercontrols.util.Statics.MINECRAFT; import static chylex.bettercontrols.util.Statics.MINECRAFT;
public final class OptionListWidget extends ElementListWidget<Entry>{ public final class OptionListWidget extends AbstractOptionList<Entry>{
public static final int ROW_WIDTH = 408; public static final int ROW_WIDTH = 408;
public static final int ROW_PADDING = 2; public static final int ROW_PADDING = 2;
@@ -27,19 +26,19 @@ public final class OptionListWidget extends ElementListWidget<Entry>{
return (column * ROW_WIDTH) / 4; return (column * ROW_WIDTH) / 4;
} }
private static Offset getElementOffset(final Element element){ private static Offset getElementOffset(final IGuiEventListener element){
if (element instanceof Widget){ if (element instanceof Widget){
return new Offset(((Widget)element).getX(), ((Widget)element).getY()); return new Offset(((Widget)element).getX(), ((Widget)element).getY());
} }
else if (element instanceof AbstractButtonWidget){ else if (element instanceof net.minecraft.client.gui.widget.Widget){
return new Offset(((AbstractButtonWidget)element).x, ((AbstractButtonWidget)element).y); return new Offset(((net.minecraft.client.gui.widget.Widget)element).x, ((net.minecraft.client.gui.widget.Widget)element).y);
} }
else{ else{
return new Offset(0, 0); return new Offset(0, 0);
} }
} }
public interface Widget extends Element, Drawable{ public interface Widget extends IGuiEventListener, IRenderable{
int getX(); int getX();
int getY(); int getY();
void setX(int x); void setX(int x);
@@ -56,7 +55,7 @@ public final class OptionListWidget extends ElementListWidget<Entry>{
} }
} }
public OptionListWidget(final int top, final int bottom, final int width, final int height, final List<Element> widgets, final int innerHeight){ public OptionListWidget(final int top, final int bottom, final int width, final int height, final List<IGuiEventListener> widgets, final int innerHeight){
super(MINECRAFT, width, height, top, bottom, innerHeight); super(MINECRAFT, width, height, top, bottom, innerHeight);
addEntry(new Entry(widgets)); addEntry(new Entry(widgets));
} }
@@ -76,27 +75,27 @@ public final class OptionListWidget extends ElementListWidget<Entry>{
return (width + ROW_WIDTH) / 2 + 4; return (width + ROW_WIDTH) / 2 + 4;
} }
protected static final class Entry extends ElementListWidget.Entry<Entry>{ protected static final class Entry extends AbstractOptionList.Entry<Entry>{
private final List<Element> elements; private final List<IGuiEventListener> elements;
private final Map<Element, Offset> offsets; private final Map<IGuiEventListener, Offset> offsets;
public Entry(final List<Element> elements){ public Entry(final List<IGuiEventListener> elements){
this.elements = new ArrayList<>(elements); this.elements = new ArrayList<>(elements);
this.offsets = elements.stream().collect(Collectors.toMap(Function.identity(), OptionListWidget::getElementOffset)); this.offsets = elements.stream().collect(Collectors.toMap(Function.identity(), OptionListWidget::getElementOffset));
} }
@Override @Override
public List<? extends Element> children(){ public List<? extends IGuiEventListener> children(){
return Collections.unmodifiableList(elements); return Collections.unmodifiableList(elements);
} }
@Override @Override
public void render(final int index, final int y, final int x, final int entryWidth, final int entryHeight, final int mouseX, final int mouseY, final boolean hovered, final float tickDelta){ public void render(final int index, final int y, final int x, final int entryWidth, final int entryHeight, final int mouseX, final int mouseY, final boolean hovered, final float tickDelta){
for(final Element element : elements){ for(final IGuiEventListener element : elements){
final Offset offset = offsets.get(element); final Offset offset = offsets.get(element);
if (element instanceof AbstractButtonWidget){ if (element instanceof net.minecraft.client.gui.widget.Widget){
final AbstractButtonWidget button = (AbstractButtonWidget)element; final net.minecraft.client.gui.widget.Widget button = (net.minecraft.client.gui.widget.Widget)element;
button.x = x + offset.x; button.x = x + offset.x;
button.y = y + offset.y; button.y = y + offset.y;
} }
@@ -106,8 +105,8 @@ public final class OptionListWidget extends ElementListWidget<Entry>{
widget.setY(y + offset.y); widget.setY(y + offset.y);
} }
if (element instanceof Drawable){ if (element instanceof IRenderable){
((Drawable)element).render(mouseX, mouseY, tickDelta); ((IRenderable)element).render(mouseX, mouseY, tickDelta);
} }
} }
} }

View File

@@ -1,14 +1,14 @@
package chylex.bettercontrols.gui; package chylex.bettercontrols.gui;
import chylex.bettercontrols.mixin.AccessOptionButtonWidgetOption; import chylex.bettercontrols.mixin.AccessOptionButtonWidgetOption;
import chylex.bettercontrols.mixin.AccessScreenButtons; import chylex.bettercontrols.mixin.AccessScreenButtons;
import net.minecraft.client.gui.Element; import net.minecraft.client.gui.AccessibilityScreen;
import net.minecraft.client.gui.ParentElement; import net.minecraft.client.gui.IGuiEventListener;
import net.minecraft.client.gui.screen.options.AccessibilityScreen; import net.minecraft.client.gui.INestedGuiEventHandler;
import net.minecraft.client.gui.screen.options.ControlsOptionsScreen; import net.minecraft.client.gui.screen.ControlsScreen;
import net.minecraft.client.gui.widget.AbstractButtonWidget; import net.minecraft.client.gui.widget.button.AbstractButton;
import net.minecraft.client.gui.widget.ButtonWidget; import net.minecraft.client.gui.widget.button.Button;
import net.minecraft.client.gui.widget.OptionButtonWidget; import net.minecraft.client.gui.widget.button.OptionButton;
import net.minecraft.client.options.Option; import net.minecraft.client.settings.AbstractOption;
import java.util.List; import java.util.List;
import java.util.function.Consumer; import java.util.function.Consumer;
import static chylex.bettercontrols.util.Statics.MINECRAFT; import static chylex.bettercontrols.util.Statics.MINECRAFT;
@@ -16,15 +16,15 @@ import static chylex.bettercontrols.util.Statics.MINECRAFT;
public final class ScreenPatcher{ public final class ScreenPatcher{
private ScreenPatcher(){} private ScreenPatcher(){}
public static void onControlsScreenOpened(final ControlsOptionsScreen screen){ public static void onControlsScreenOpened(final ControlsScreen screen){
final AccessScreenButtons accessor = (AccessScreenButtons)screen; final AccessScreenButtons accessor = (AccessScreenButtons)screen;
final List<? extends Element> children = screen.children(); final List<? extends IGuiEventListener> children = screen.children();
final List<AbstractButtonWidget> buttons = accessor.getButtons(); final List<AbstractButton> buttons = accessor.getButtons();
final AbstractButtonWidget autoJump = buttons final AbstractButton autoJump = buttons
.stream() .stream()
.filter(it -> it instanceof OptionButtonWidget && ((AccessOptionButtonWidgetOption)it).getOption() == Option.AUTO_JUMP) .filter(it -> it instanceof OptionButton && ((AccessOptionButtonWidgetOption)it).getOption() == AbstractOption.AUTO_JUMP)
.findAny() .findAny()
.orElse(null); .orElse(null);
@@ -32,31 +32,31 @@ public final class ScreenPatcher{
children.remove(autoJump); children.remove(autoJump);
buttons.remove(autoJump); buttons.remove(autoJump);
accessor.callAddButton(new ButtonWidget(autoJump.x, autoJump.y, autoJump.getWidth(), 20, BetterControlsScreen.TITLE.copy().append("...").asFormattedString(), btn -> { accessor.callAddButton(new Button(autoJump.x, autoJump.y, autoJump.getWidth(), 20, BetterControlsScreen.TITLE.copy().appendText("...").getString(), btn -> {
MINECRAFT.openScreen(new BetterControlsScreen(screen)); MINECRAFT.displayGuiScreen(new BetterControlsScreen(screen));
})); }));
} }
} }
public static void onAccessibilityScreenOpened(final AccessibilityScreen screen){ public static void onAccessibilityScreenOpened(final AccessibilityScreen screen){
walkChildren(screen.children(), it -> { walkChildren(screen.children(), it -> {
if (it instanceof OptionButtonWidget){ if (it instanceof OptionButton){
final OptionButtonWidget button = (OptionButtonWidget)it; final OptionButton button = (OptionButton)it;
final Option option = ((AccessOptionButtonWidgetOption)button).getOption(); final AbstractOption option = ((AccessOptionButtonWidgetOption)button).getOption();
if (option == Option.SPRINT_TOGGLED || option == Option.SNEAK_TOGGLED){ if (option == AbstractOption.SPRINT || option == AbstractOption.SNEAK){
button.active = false; button.active = false;
} }
} }
}); });
} }
private static void walkChildren(final List<? extends Element> elements, final Consumer<Element> callback){ private static void walkChildren(final List<? extends IGuiEventListener> listeners, final Consumer<IGuiEventListener> callback){
for(final Element element : elements){ for(final IGuiEventListener listener : listeners){
callback.accept(element); callback.accept(listener);
if (element instanceof ParentElement){ if (listener instanceof INestedGuiEventHandler){
walkChildren(((ParentElement)element).children(), callback); walkChildren(((INestedGuiEventHandler)listener).children(), callback);
} }
} }
} }

View File

@@ -1,14 +1,14 @@
package chylex.bettercontrols.gui.elements; package chylex.bettercontrols.gui.elements;
import it.unimi.dsi.fastutil.booleans.BooleanConsumer; import it.unimi.dsi.fastutil.booleans.BooleanConsumer;
import net.minecraft.client.gui.widget.ButtonWidget; import net.minecraft.client.gui.widget.button.Button;
import net.minecraft.client.resource.language.I18n; import net.minecraft.client.resources.I18n;
public final class BooleanValueWidget extends ButtonWidget{ public final class BooleanValueWidget extends Button{
private final BooleanConsumer onChanged; private final BooleanConsumer onChanged;
private boolean value; private boolean value;
public BooleanValueWidget(final int x, final int y, final int width, final int height, final boolean currentValue, final BooleanConsumer onChanged){ public BooleanValueWidget(final int x, final int y, final int width, final int height, final boolean currentValue, final BooleanConsumer onChanged){
super(x, y, width, height, I18n.translate(currentValue ? "options.on" : "options.off"), ignore -> {}); super(x, y, width, height, I18n.format(currentValue ? "options.on" : "options.off"), ignore -> {});
this.value = currentValue; this.value = currentValue;
this.onChanged = onChanged; this.onChanged = onChanged;
} }
@@ -21,7 +21,7 @@ public final class BooleanValueWidget extends ButtonWidget{
public void onPress(){ public void onPress(){
super.onPress(); super.onPress();
value = !value; value = !value;
setMessage(I18n.translate(value ? "options.on" : "options.off")); setMessage(I18n.format(value ? "options.on" : "options.off"));
onChanged.accept(value); onChanged.accept(value);
} }
} }

View File

@@ -1,15 +1,15 @@
package chylex.bettercontrols.gui.elements; package chylex.bettercontrols.gui.elements;
import net.minecraft.client.gui.widget.ButtonWidget; import net.minecraft.client.gui.widget.button.Button;
import java.util.List; import java.util.List;
import java.util.function.Consumer; import java.util.function.Consumer;
public class CycleButtonWidget<T> extends ButtonWidget{ public class CycleButtonWidget<T> extends Button{
private final List<Option<T>> options; private final List<Option<T>> options;
private final Consumer<T> onChanged; private final Consumer<T> onChanged;
private T selectedValue; private T selectedValue;
public CycleButtonWidget(final int x, final int y, final int width, final int height, final List<Option<T>> options, final T selectedValue, final Consumer<T> onChanged){ public CycleButtonWidget(final int x, final int y, final int width, final int height, final List<Option<T>> options, final T selectedValue, final Consumer<T> onChanged){
super(x, y, width, height, Option.find(options, selectedValue).getText().asFormattedString(), btn -> {}); super(x, y, width, height, Option.find(options, selectedValue).getText().getString(), btn -> {});
this.options = options; this.options = options;
this.selectedValue = selectedValue; this.selectedValue = selectedValue;
this.onChanged = onChanged; this.onChanged = onChanged;
@@ -31,6 +31,6 @@ public class CycleButtonWidget<T> extends ButtonWidget{
selectedValue = newSelectedOption.getValue(); selectedValue = newSelectedOption.getValue();
onChanged.accept(selectedValue); onChanged.accept(selectedValue);
setMessage(newSelectedOption.getText().asFormattedString()); setMessage(newSelectedOption.getText().getString());
} }
} }

View File

@@ -1,10 +1,10 @@
package chylex.bettercontrols.gui.elements; package chylex.bettercontrols.gui.elements;
import net.minecraft.client.gui.widget.SliderWidget; import net.minecraft.client.gui.widget.AbstractSlider;
import net.minecraft.util.math.MathHelper; import net.minecraft.util.math.MathHelper;
import java.util.List; import java.util.List;
import java.util.function.Consumer; import java.util.function.Consumer;
public final class DiscreteValueSliderWidget<T> extends SliderWidget{ public final class DiscreteValueSliderWidget<T> extends AbstractSlider{
private final List<Option<T>> options; private final List<Option<T>> options;
private final Consumer<T> onChanged; private final Consumer<T> onChanged;
private T selectedValue; private T selectedValue;
@@ -27,7 +27,7 @@ public final class DiscreteValueSliderWidget<T> extends SliderWidget{
@Override @Override
protected void updateMessage(){ protected void updateMessage(){
setMessage(getSelectedOption().getText().asFormattedString()); setMessage(getSelectedOption().getText().getString());
} }
@Override @Override

View File

@@ -1,21 +1,23 @@
package chylex.bettercontrols.gui.elements; package chylex.bettercontrols.gui.elements;
import chylex.bettercontrols.util.Key; import chylex.bettercontrols.util.Key;
import net.minecraft.client.gui.widget.AbstractButtonWidget; import net.minecraft.client.gui.widget.button.AbstractButton;
import net.minecraft.client.gui.widget.ButtonWidget; import net.minecraft.client.gui.widget.button.Button;
import net.minecraft.client.options.KeyBinding; import net.minecraft.client.resources.I18n;
import net.minecraft.client.resource.language.I18n; import net.minecraft.client.settings.KeyBinding;
import net.minecraft.client.util.InputUtil; import net.minecraft.client.util.InputMappings;
import net.minecraft.util.Formatting; import net.minecraft.util.text.ITextComponent;
import net.minecraft.util.text.TextFormatting;
import net.minecraft.util.text.TranslationTextComponent;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
import java.util.function.Consumer; import java.util.function.Consumer;
import static chylex.bettercontrols.util.Statics.OPTIONS; import static chylex.bettercontrols.util.Statics.OPTIONS;
public final class KeyBindingWidget extends ButtonWidget{ public final class KeyBindingWidget extends Button{
private final KeyBinding binding; private final KeyBinding binding;
private final String bindingName; private final ITextComponent bindingName;
private final List<AbstractButtonWidget> linkedButtons = new ArrayList<>(1); private final List<AbstractButton> linkedButtons = new ArrayList<>(1);
private final Consumer<KeyBindingWidget> onEditingStarted; private final Consumer<KeyBindingWidget> onEditingStarted;
private boolean isEditing; private boolean isEditing;
@@ -23,7 +25,7 @@ public final class KeyBindingWidget extends ButtonWidget{
public KeyBindingWidget(final int x, final int y, final int width, final int height, final KeyBinding binding, final Consumer<KeyBindingWidget> onEditingStarted){ public KeyBindingWidget(final int x, final int y, final int width, final int height, final KeyBinding binding, final Consumer<KeyBindingWidget> onEditingStarted){
super(x, y, width, height, "", btn -> {}); super(x, y, width, height, "", btn -> {});
this.binding = binding; this.binding = binding;
this.bindingName = I18n.translate(binding.getId()); this.bindingName = new TranslationTextComponent(binding.getTranslationKey());
this.onEditingStarted = onEditingStarted; this.onEditingStarted = onEditingStarted;
updateKeyBindingText(); updateKeyBindingText();
} }
@@ -32,14 +34,14 @@ public final class KeyBindingWidget extends ButtonWidget{
this(x, y, width, 20, binding, onEditingStarted); this(x, y, width, 20, binding, onEditingStarted);
} }
public void linkButtonToBoundState(final AbstractButtonWidget button){ public void linkButtonToBoundState(final AbstractButton button){
linkedButtons.add(button); linkedButtons.add(button);
button.active = !Key.isUnbound(binding); button.active = !Key.isUnbound(binding);
} }
@Override @Override
protected String getNarrationMessage(){ protected String getNarrationMessage(){
return Key.isUnbound(binding) ? I18n.translate("narrator.controls.unbound", bindingName) : I18n.translate("narrator.controls.bound", bindingName, super.getNarrationMessage()); return Key.isUnbound(binding) ? I18n.format("narrator.controls.unbound", bindingName) : I18n.format("narrator.controls.bound", bindingName, super.getNarrationMessage());
} }
@Override @Override
@@ -49,11 +51,11 @@ public final class KeyBindingWidget extends ButtonWidget{
updateKeyBindingText(); updateKeyBindingText();
} }
public void bindAndStopEditing(final InputUtil.KeyCode key){ public void bindAndStopEditing(final InputMappings.Input key){
Key.bind(binding, key); Key.bind(binding, key);
stopEditing(); stopEditing();
for(final AbstractButtonWidget button : linkedButtons){ for(final AbstractButton button : linkedButtons){
button.active = !Key.isUnbound(binding); button.active = !Key.isUnbound(binding);
} }
} }
@@ -67,7 +69,7 @@ public final class KeyBindingWidget extends ButtonWidget{
boolean hasConflict = false; boolean hasConflict = false;
if (!Key.isUnbound(binding)){ if (!Key.isUnbound(binding)){
for(final KeyBinding other : OPTIONS.keysAll){ for(final KeyBinding other : OPTIONS.keyBindings){
if (binding != other && binding.equals(other)){ if (binding != other && binding.equals(other)){
hasConflict = true; hasConflict = true;
break; break;
@@ -76,10 +78,10 @@ public final class KeyBindingWidget extends ButtonWidget{
} }
if (isEditing){ if (isEditing){
setMessage(Formatting.WHITE + "> " + Formatting.YELLOW + binding.getLocalizedName() + Formatting.WHITE + " <"); setMessage(TextFormatting.WHITE + "> " + TextFormatting.YELLOW + Key.getBoundKeyText(binding) + TextFormatting.WHITE + " <");
} }
else if (hasConflict){ else if (hasConflict){
setMessage(Formatting.RED + Key.getBoundKeyText(binding)); setMessage(TextFormatting.RED + Key.getBoundKeyText(binding));
} }
else{ else{
setMessage(Key.isUnbound(binding) ? "(No Binding)" : Key.getBoundKeyText(binding)); setMessage(Key.isUnbound(binding) ? "(No Binding)" : Key.getBoundKeyText(binding));

View File

@@ -1,12 +1,12 @@
package chylex.bettercontrols.gui.elements; package chylex.bettercontrols.gui.elements;
import chylex.bettercontrols.gui.OptionListWidget.Widget; import chylex.bettercontrols.gui.OptionListWidget.Widget;
import chylex.bettercontrols.util.LiteralText; import chylex.bettercontrols.util.LiteralText;
import net.minecraft.client.MinecraftClient; import net.minecraft.client.gui.AbstractGui;
import net.minecraft.client.font.TextRenderer; import net.minecraft.client.gui.FontRenderer;
import net.minecraft.client.gui.DrawableHelper;
import java.util.List; import java.util.List;
import static chylex.bettercontrols.util.Statics.MINECRAFT;
public final class TextWidget extends DrawableHelper implements Widget{ public final class TextWidget extends AbstractGui implements Widget{
public static final int LEFT = 0; public static final int LEFT = 0;
public static final int CENTER = 1; public static final int CENTER = 1;
@@ -56,16 +56,16 @@ public final class TextWidget extends DrawableHelper implements Widget{
@Override @Override
public void render(final int mouseX, final int mouseY, final float delta){ public void render(final int mouseX, final int mouseY, final float delta){
final TextRenderer textRenderer = MinecraftClient.getInstance().textRenderer; final FontRenderer textRenderer = MINECRAFT.fontRenderer;
final List<String> lines = textRenderer.wrapStringToWidthAsList(text.asFormattedString(), width); final List<String> lines = textRenderer.listFormattedStringToWidth(text.getString(), width);
final int lineHeight = textRenderer.fontHeight + 1; final int lineHeight = textRenderer.FONT_HEIGHT + 1;
final int finalX = align == CENTER ? x + (width / 2) - (lines.stream().mapToInt(textRenderer::getStringWidth).max().orElse(0) / 2) : x; final int finalX = align == CENTER ? x + (width / 2) - (lines.stream().mapToInt(textRenderer::getStringWidth).max().orElse(0) / 2) : x;
final int finalY = y + (height / 2) - (lineHeight * lines.size() / 2) + 1; final int finalY = y + (height / 2) - (lineHeight * lines.size() / 2) + 1;
for(int i = 0; i < lines.size(); i++){ for(int i = 0; i < lines.size(); i++){
final String line = lines.get(i); final String line = lines.get(i);
textRenderer.drawWithShadow(line, finalX, finalY + (i * lineHeight), (255 << 16) | (255 << 8) | 255); textRenderer.drawStringWithShadow(line, finalX, finalY + (i * lineHeight), (255 << 16) | (255 << 8) | 255);
} }
} }
} }

View File

@@ -1,14 +1,14 @@
package chylex.bettercontrols.input; package chylex.bettercontrols.input;
import net.minecraft.client.options.KeyBinding; import net.minecraft.client.resources.I18n;
import net.minecraft.client.resource.language.I18n; import net.minecraft.client.settings.KeyBinding;
import net.minecraft.client.util.InputUtil.Type; import net.minecraft.client.util.InputMappings.Type;
import org.jetbrains.annotations.Nullable; import javax.annotation.Nullable;
public class KeyBindingWithModifier extends KeyBinding{ public class KeyBindingWithModifier extends KeyBinding{
public static final String CATEGORY = "key.categories.bettercontrols"; public static final String CATEGORY = "key.categories.bettercontrols";
public static boolean checkCategoryMatches(final String name){ public static boolean checkCategoryMatches(final String text){
return I18n.translate(CATEGORY).equals(name); return I18n.format(CATEGORY).equals(text);
} }
@Nullable @Nullable
@@ -28,12 +28,12 @@ public class KeyBindingWithModifier extends KeyBinding{
} }
@Override @Override
public boolean isPressed(){ public boolean isKeyDown(){
return super.isPressed() && (modifier == null || modifier.isPressed()); return super.isKeyDown() && (modifier == null || modifier.isPressed());
} }
@Override @Override
public boolean wasPressed(){ public boolean isPressed(){
return super.wasPressed() && (modifier == null || modifier.isPressed()); return super.isPressed() && (modifier == null || modifier.isPressed());
} }
} }

View File

@@ -1,6 +1,6 @@
package chylex.bettercontrols.input; package chylex.bettercontrols.input;
import chylex.bettercontrols.util.Key; import chylex.bettercontrols.util.Key;
import net.minecraft.client.options.KeyBinding; import net.minecraft.client.settings.KeyBinding;
public class ToggleTracker{ public class ToggleTracker{
protected final KeyBinding bindingToggle; protected final KeyBinding bindingToggle;

View File

@@ -1,7 +1,7 @@
package chylex.bettercontrols.input; package chylex.bettercontrols.input;
import chylex.bettercontrols.mixin.AccessKeyBindingFields; import chylex.bettercontrols.mixin.AccessKeyBindingFields;
import it.unimi.dsi.fastutil.booleans.BooleanConsumer; import it.unimi.dsi.fastutil.booleans.BooleanConsumer;
import net.minecraft.client.options.KeyBinding; import net.minecraft.client.settings.KeyBinding;
import java.util.HashSet; import java.util.HashSet;
import java.util.Set; import java.util.Set;

View File

@@ -1,14 +1,14 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import net.minecraft.client.render.Camera; import net.minecraft.client.renderer.ActiveRenderInfo;
import net.minecraft.entity.Entity; import net.minecraft.entity.Entity;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.gen.Accessor; import org.spongepowered.asm.mixin.gen.Accessor;
@Mixin(Camera.class) @Mixin(ActiveRenderInfo.class)
public interface AccessCameraFields{ public interface AccessCameraFields{
@Accessor @Accessor("renderViewEntity")
Entity getFocusedEntity(); Entity getFocusedEntity();
@Accessor @Accessor("height")
void setCameraY(float y); void setCameraY(float y);
} }

View File

@@ -1,10 +1,10 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import net.minecraft.client.network.ClientPlayerEntity; import net.minecraft.client.entity.player.ClientPlayerEntity;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.gen.Accessor; import org.spongepowered.asm.mixin.gen.Accessor;
@Mixin(ClientPlayerEntity.class) @Mixin(ClientPlayerEntity.class)
public interface AccessClientPlayerFields{ public interface AccessClientPlayerFields{
@Accessor("field_3935") @Accessor("sprintToggleTimer")
void setTicksLeftToDoubleTapSprint(int value); void setTicksLeftToDoubleTapSprint(int value);
} }

View File

@@ -1,10 +1,10 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import net.minecraft.client.gui.screen.options.ControlsListWidget.CategoryEntry; import net.minecraft.client.gui.widget.list.KeyBindingList.CategoryEntry;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.gen.Accessor; import org.spongepowered.asm.mixin.gen.Accessor;
@Mixin(CategoryEntry.class) @Mixin(CategoryEntry.class)
public interface AccessControlsListCategory{ public interface AccessControlsListCategory{
@Accessor @Accessor("labelText")
String getName(); String getText();
} }

View File

@@ -1,11 +1,11 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import net.minecraft.client.gui.screen.options.ControlsListWidget.KeyBindingEntry; import net.minecraft.client.gui.widget.list.KeyBindingList.KeyEntry;
import net.minecraft.client.options.KeyBinding; import net.minecraft.client.settings.KeyBinding;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.gen.Accessor; import org.spongepowered.asm.mixin.gen.Accessor;
@Mixin(KeyBindingEntry.class) @Mixin(KeyEntry.class)
public interface AccessControlsListKeyBinding{ public interface AccessControlsListKeyBinding{
@Accessor @Accessor("keybinding")
KeyBinding getBinding(); KeyBinding getBinding();
} }

View File

@@ -1,10 +1,10 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import net.minecraft.client.render.GameRenderer; import net.minecraft.client.renderer.GameRenderer;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.gen.Accessor; import org.spongepowered.asm.mixin.gen.Accessor;
@Mixin(GameRenderer.class) @Mixin(GameRenderer.class)
public interface AccessGameRendererFields{ public interface AccessGameRendererFields{
@Accessor @Accessor("fovModifierHand")
void setMovementFovMultiplier(float value); void setMovementFovMultiplier(float value);
} }

View File

@@ -1,12 +1,12 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import net.minecraft.client.options.KeyBinding; import net.minecraft.client.settings.KeyBinding;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.gen.Accessor; import org.spongepowered.asm.mixin.gen.Accessor;
import java.util.Map; import java.util.Map;
@Mixin(KeyBinding.class) @Mixin(KeyBinding.class)
public interface AccessKeyBindingFields{ public interface AccessKeyBindingFields{
@Accessor @Accessor("CATEGORY_ORDER")
static Map<String, Integer> getCategoryOrderMap(){ static Map<String, Integer> getCategoryOrderMap(){
throw new AssertionError(); throw new AssertionError();
} }

View File

@@ -1,11 +1,11 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import net.minecraft.client.gui.widget.OptionButtonWidget; import net.minecraft.client.gui.widget.button.OptionButton;
import net.minecraft.client.options.Option; import net.minecraft.client.settings.AbstractOption;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.gen.Accessor; import org.spongepowered.asm.mixin.gen.Accessor;
@Mixin(OptionButtonWidget.class) @Mixin(OptionButton.class)
public interface AccessOptionButtonWidgetOption{ public interface AccessOptionButtonWidgetOption{
@Accessor @Accessor("enumOptions")
Option getOption(); AbstractOption getOption();
} }

View File

@@ -5,6 +5,6 @@ import org.spongepowered.asm.mixin.gen.Accessor;
@Mixin(PlayerEntity.class) @Mixin(PlayerEntity.class)
public interface AccessPlayerFields{ public interface AccessPlayerFields{
@Accessor("field_7489") @Accessor("flyToggleTimer")
void setTicksLeftToDoubleTapFlight(int value); void setTicksLeftToDoubleTapFlight(int value);
} }

View File

@@ -1,6 +1,7 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import net.minecraft.client.gui.screen.Screen; import net.minecraft.client.gui.screen.Screen;
import net.minecraft.client.gui.widget.AbstractButtonWidget; import net.minecraft.client.gui.widget.Widget;
import net.minecraft.client.gui.widget.button.AbstractButton;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.gen.Accessor; import org.spongepowered.asm.mixin.gen.Accessor;
import org.spongepowered.asm.mixin.gen.Invoker; import org.spongepowered.asm.mixin.gen.Invoker;
@@ -9,8 +10,8 @@ import java.util.List;
@Mixin(Screen.class) @Mixin(Screen.class)
public interface AccessScreenButtons{ public interface AccessScreenButtons{
@Accessor @Accessor
List<AbstractButtonWidget> getButtons(); List<AbstractButton> getButtons();
@Invoker @Invoker
<T extends AbstractButtonWidget> T callAddButton(T button); <T extends Widget> T callAddButton(T button);
} }

View File

@@ -1,16 +1,16 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import net.minecraft.client.options.StickyKeyBinding; import net.minecraft.client.settings.ToggleableKeyBinding;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.Mutable; import org.spongepowered.asm.mixin.Mutable;
import org.spongepowered.asm.mixin.gen.Accessor; import org.spongepowered.asm.mixin.gen.Accessor;
import java.util.function.BooleanSupplier; import java.util.function.BooleanSupplier;
@Mixin(StickyKeyBinding.class) @Mixin(ToggleableKeyBinding.class)
public interface AccessStickyKeyBindingStateGetter{ public interface AccessStickyKeyBindingStateGetter{
@Accessor @Accessor("getterToggle")
BooleanSupplier getToggleGetter(); BooleanSupplier getToggleGetter();
@Accessor @Accessor("getterToggle")
@Mutable @Mutable
void setToggleGetter(final BooleanSupplier toggleGetter); void setToggleGetter(final BooleanSupplier toggleGetter);
} }

View File

@@ -1,7 +1,8 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import chylex.bettercontrols.player.PlayerTicker; import chylex.bettercontrols.player.PlayerTicker;
import net.minecraft.client.input.KeyboardInput; import net.minecraft.client.entity.player.ClientPlayerEntity;
import net.minecraft.client.network.ClientPlayerEntity; import net.minecraft.util.MovementInput;
import net.minecraft.util.MovementInputFromOptions;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.injection.At; import org.spongepowered.asm.mixin.injection.At;
import org.spongepowered.asm.mixin.injection.Inject; import org.spongepowered.asm.mixin.injection.Inject;
@@ -9,11 +10,11 @@ import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
import static chylex.bettercontrols.util.Statics.MINECRAFT; import static chylex.bettercontrols.util.Statics.MINECRAFT;
import static org.spongepowered.asm.mixin.injection.At.Shift.AFTER; import static org.spongepowered.asm.mixin.injection.At.Shift.AFTER;
@Mixin(KeyboardInput.class) @Mixin(MovementInputFromOptions.class)
public abstract class HookClientPlayerInputTick{ public abstract class HookClientPlayerInputTick{
@Inject(method = "tick(Z)V", at = @At(value = "FIELD", target = "Lnet/minecraft/client/input/KeyboardInput;pressingForward:Z", ordinal = 0, shift = AFTER)) @Inject(method = "func_225607_a_(Z)V", at = @At(value = "FIELD", target = "Lnet/minecraft/util/MovementInputFromOptions;forwardKeyDown:Z", ordinal = 0, shift = AFTER))
private void afterInputTick(final CallbackInfo info){ private void afterInputTick(final CallbackInfo info){
final KeyboardInput input = (KeyboardInput)(Object)this; final MovementInput input = (MovementInput)(Object)this;
final ClientPlayerEntity player = MINECRAFT.player; final ClientPlayerEntity player = MINECRAFT.player;
if (player != null){ if (player != null){

View File

@@ -1,8 +1,8 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import chylex.bettercontrols.player.PlayerTicker; import chylex.bettercontrols.player.PlayerTicker;
import com.mojang.authlib.GameProfile; import com.mojang.authlib.GameProfile;
import net.minecraft.client.network.AbstractClientPlayerEntity; import net.minecraft.client.entity.player.AbstractClientPlayerEntity;
import net.minecraft.client.network.ClientPlayerEntity; import net.minecraft.client.entity.player.ClientPlayerEntity;
import net.minecraft.client.world.ClientWorld; import net.minecraft.client.world.ClientWorld;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.injection.At; import org.spongepowered.asm.mixin.injection.At;
@@ -16,19 +16,19 @@ public abstract class HookClientPlayerTick extends AbstractClientPlayerEntity{
super(world, profile); super(world, profile);
} }
@Inject(method = "tickMovement()V", at = @At("HEAD")) @Inject(method = "livingTick", at = @At("HEAD"))
private void atHead(final CallbackInfo info){ private void atHead(final CallbackInfo info){
final ClientPlayerEntity player = (ClientPlayerEntity)(Object)this; final ClientPlayerEntity player = (ClientPlayerEntity)(Object)this;
PlayerTicker.get(player).atHead(player); PlayerTicker.get(player).atHead(player);
} }
@Inject(method = "tickMovement()V", at = @At(value = "INVOKE", target = "Lnet/minecraft/client/input/Input;tick(Z)V", ordinal = 0, shift = AFTER)) @Inject(method = "livingTick()V", at = @At(value = "INVOKE", target = "Lnet/minecraft/util/MovementInput;func_225607_a_(Z)V", ordinal = 0, shift = AFTER))
private void afterInputTick(final CallbackInfo info){ private void afterInputTick(final CallbackInfo info){
final ClientPlayerEntity player = (ClientPlayerEntity)(Object)this; final ClientPlayerEntity player = (ClientPlayerEntity)(Object)this;
PlayerTicker.get(player).afterInputTick(player); PlayerTicker.get(player).afterInputTick(player);
} }
@Inject(method = "tickMovement()V", at = @At(value = "INVOKE", target = "Lnet/minecraft/client/network/AbstractClientPlayerEntity;tickMovement()V", ordinal = 0, shift = AFTER)) @Inject(method = "livingTick()V", at = @At(value = "INVOKE", target = "Lnet/minecraft/client/entity/player/AbstractClientPlayerEntity;livingTick()V", ordinal = 0, shift = AFTER))
private void afterSuperCall(final CallbackInfo info){ private void afterSuperCall(final CallbackInfo info){
final ClientPlayerEntity player = (ClientPlayerEntity)(Object)this; final ClientPlayerEntity player = (ClientPlayerEntity)(Object)this;
PlayerTicker.get(player).afterSuperCall(player); PlayerTicker.get(player).afterSuperCall(player);

View File

@@ -1,33 +1,33 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import chylex.bettercontrols.BetterControlsMod; import chylex.bettercontrols.BetterControlsMod;
import chylex.bettercontrols.input.KeyBindingWithModifier; import chylex.bettercontrols.input.KeyBindingWithModifier;
import net.minecraft.client.MinecraftClient; import net.minecraft.client.Minecraft;
import net.minecraft.client.gui.screen.options.ControlsListWidget; import net.minecraft.client.gui.screen.ControlsScreen;
import net.minecraft.client.gui.screen.options.ControlsListWidget.CategoryEntry; import net.minecraft.client.gui.widget.list.AbstractOptionList;
import net.minecraft.client.gui.screen.options.ControlsListWidget.Entry; import net.minecraft.client.gui.widget.list.KeyBindingList;
import net.minecraft.client.gui.screen.options.ControlsListWidget.KeyBindingEntry; import net.minecraft.client.gui.widget.list.KeyBindingList.CategoryEntry;
import net.minecraft.client.gui.screen.options.ControlsOptionsScreen; import net.minecraft.client.gui.widget.list.KeyBindingList.Entry;
import net.minecraft.client.gui.widget.ElementListWidget; import net.minecraft.client.gui.widget.list.KeyBindingList.KeyEntry;
import org.apache.commons.lang3.ArrayUtils; import org.apache.commons.lang3.ArrayUtils;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.injection.At; import org.spongepowered.asm.mixin.injection.At;
import org.spongepowered.asm.mixin.injection.Inject; import org.spongepowered.asm.mixin.injection.Inject;
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
@Mixin(ControlsListWidget.class) @Mixin(KeyBindingList.class)
public abstract class HookControlsListWidget extends ElementListWidget<Entry>{ public abstract class HookControlsListWidget extends AbstractOptionList<Entry>{
public HookControlsListWidget(final MinecraftClient client, final int width, final int height, final int top, final int bottom, final int itemHeight){ public HookControlsListWidget(final Minecraft client, final int width, final int height, final int top, final int bottom, final int itemHeight){
super(client, width, height, top, bottom, itemHeight); super(client, width, height, top, bottom, itemHeight);
} }
@Inject(method = "<init>", at = @At("TAIL")) @Inject(method = "<init>", at = @At("TAIL"))
public void init(final ControlsOptionsScreen parent, final MinecraftClient client, final CallbackInfo ci){ public void init(final ControlsScreen parent, final Minecraft client, final CallbackInfo ci){
children().removeIf(it -> { children().removeIf(it -> {
if (it instanceof CategoryEntry && KeyBindingWithModifier.checkCategoryMatches(((AccessControlsListCategory)it).getName())){ if (it instanceof CategoryEntry && KeyBindingWithModifier.checkCategoryMatches(((AccessControlsListCategory)it).getText())){
return true; return true;
} }
if (it instanceof KeyBindingEntry && ArrayUtils.contains(BetterControlsMod.config.getAllKeyBindings(), ((AccessControlsListKeyBinding)it).getBinding())){ if (it instanceof KeyEntry && ArrayUtils.contains(BetterControlsMod.config.getAllKeyBindings(), ((AccessControlsListKeyBinding)it).getBinding())){
return true; return true;
} }

View File

@@ -1,34 +1,30 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import chylex.bettercontrols.BetterControlsMod; import chylex.bettercontrols.BetterControlsMod;
import chylex.bettercontrols.input.KeyBindingWithModifier; import chylex.bettercontrols.input.KeyBindingWithModifier;
import net.minecraft.client.options.GameOptions; import net.minecraft.client.GameSettings;
import net.minecraft.client.options.KeyBinding; import net.minecraft.client.settings.KeyBinding;
import org.apache.commons.lang3.ArrayUtils; import org.apache.commons.lang3.ArrayUtils;
import org.spongepowered.asm.mixin.Final;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.Mutable;
import org.spongepowered.asm.mixin.Shadow; import org.spongepowered.asm.mixin.Shadow;
import org.spongepowered.asm.mixin.injection.At; import org.spongepowered.asm.mixin.injection.At;
import org.spongepowered.asm.mixin.injection.Inject; import org.spongepowered.asm.mixin.injection.Inject;
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
@Mixin(GameOptions.class) @Mixin(GameSettings.class)
public abstract class HookLoadGameOptions{ public abstract class HookLoadGameOptions{
private boolean hasLoaded = false; private boolean hasLoaded = false;
@Mutable
@Final
@Shadow @Shadow
public KeyBinding[] keysAll; public KeyBinding[] keyBindings;
@Inject(method = "load()V", at = @At("HEAD")) @Inject(method = "loadOptions()V", at = @At("HEAD"))
private void load(final CallbackInfo info){ private void load(final CallbackInfo info){
if (hasLoaded){ if (hasLoaded){
return; return;
} }
hasLoaded = true; hasLoaded = true;
keysAll = ArrayUtils.addAll(keysAll, BetterControlsMod.config.getAllKeyBindings()); keyBindings = ArrayUtils.addAll(keyBindings, BetterControlsMod.config.getAllKeyBindings());
AccessKeyBindingFields.getCategoryOrderMap().put(KeyBindingWithModifier.CATEGORY, Integer.valueOf(Integer.MAX_VALUE)); AccessKeyBindingFields.getCategoryOrderMap().put(KeyBindingWithModifier.CATEGORY, Integer.valueOf(Integer.MAX_VALUE));
} }
} }

View File

@@ -1,24 +1,24 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import chylex.bettercontrols.gui.ScreenPatcher; import chylex.bettercontrols.gui.ScreenPatcher;
import net.minecraft.client.MinecraftClient; import net.minecraft.client.Minecraft;
import net.minecraft.client.gui.AccessibilityScreen;
import net.minecraft.client.gui.screen.ControlsScreen;
import net.minecraft.client.gui.screen.Screen; import net.minecraft.client.gui.screen.Screen;
import net.minecraft.client.gui.screen.options.AccessibilityScreen;
import net.minecraft.client.gui.screen.options.ControlsOptionsScreen;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.injection.At; import org.spongepowered.asm.mixin.injection.At;
import org.spongepowered.asm.mixin.injection.Inject; import org.spongepowered.asm.mixin.injection.Inject;
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
import static chylex.bettercontrols.util.Statics.MINECRAFT; import static chylex.bettercontrols.util.Statics.MINECRAFT;
@Mixin(value = MinecraftClient.class, priority = 100) @Mixin(value = Minecraft.class, priority = 100)
public abstract class HookOpenScreen{ public abstract class HookOpenScreen{
@Inject(method = "openScreen(Lnet/minecraft/client/gui/screen/Screen;)V", at = @At("TAIL")) @Inject(method = "displayGuiScreen(Lnet/minecraft/client/gui/screen/Screen;)V", at = @At("TAIL"))
private void openScreen(final Screen ignore, final CallbackInfo ci){ private void openScreen(final Screen ignore, final CallbackInfo ci){
final Screen screen = MINECRAFT.currentScreen; final Screen screen = MINECRAFT.currentScreen;
if (screen != null && !Screen.hasAltDown()){ if (screen != null && !Screen.hasAltDown()){
if (screen.getClass() == ControlsOptionsScreen.class){ if (screen.getClass() == ControlsScreen.class){
ScreenPatcher.onControlsScreenOpened((ControlsOptionsScreen)screen); ScreenPatcher.onControlsScreenOpened((ControlsScreen)screen);
} }
else if (screen.getClass() == AccessibilityScreen.class){ else if (screen.getClass() == AccessibilityScreen.class){
ScreenPatcher.onAccessibilityScreenOpened((AccessibilityScreen)screen); ScreenPatcher.onAccessibilityScreenOpened((AccessibilityScreen)screen);

View File

@@ -1,7 +1,7 @@
package chylex.bettercontrols.mixin; package chylex.bettercontrols.mixin;
import chylex.bettercontrols.input.ToggleTrackerForStickyKey; import chylex.bettercontrols.input.ToggleTrackerForStickyKey;
import net.minecraft.client.options.KeyBinding; import net.minecraft.client.settings.KeyBinding;
import net.minecraft.client.options.StickyKeyBinding; import net.minecraft.client.settings.ToggleableKeyBinding;
import org.spongepowered.asm.mixin.Final; import org.spongepowered.asm.mixin.Final;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.Shadow; import org.spongepowered.asm.mixin.Shadow;
@@ -10,11 +10,11 @@ import org.spongepowered.asm.mixin.injection.Inject;
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
import java.util.function.BooleanSupplier; import java.util.function.BooleanSupplier;
@Mixin(StickyKeyBinding.class) @Mixin(ToggleableKeyBinding.class)
public abstract class HookStickyKeyBindingState extends KeyBinding{ public abstract class HookStickyKeyBindingState extends KeyBinding{
@Shadow @Shadow
@Final @Final
private BooleanSupplier toggleGetter; private BooleanSupplier getterToggle;
public HookStickyKeyBindingState(final String translationKey, final int code, final String category){ public HookStickyKeyBindingState(final String translationKey, final int code, final String category){
super(translationKey, code, category); super(translationKey, code, category);
@@ -29,7 +29,7 @@ public abstract class HookStickyKeyBindingState extends KeyBinding{
} }
@Override @Override
public boolean isPressed(){ public boolean isKeyDown(){
return super.isPressed() || (ToggleTrackerForStickyKey.isOverrideEnabled(this) && toggleGetter.getAsBoolean()); return super.isKeyDown() || (ToggleTrackerForStickyKey.isOverrideEnabled(this) && getterToggle.getAsBoolean());
} }
} }

View File

@@ -1,7 +1,7 @@
package chylex.bettercontrols.player; package chylex.bettercontrols.player;
import chylex.bettercontrols.BetterControlsMod; import chylex.bettercontrols.BetterControlsMod;
import chylex.bettercontrols.config.BetterControlsConfig; import chylex.bettercontrols.config.BetterControlsConfig;
import net.minecraft.client.network.ClientPlayerEntity; import net.minecraft.client.entity.player.ClientPlayerEntity;
final class FlightHelper{ final class FlightHelper{
private FlightHelper(){} private FlightHelper(){}
@@ -14,11 +14,11 @@ final class FlightHelper{
} }
static boolean isFlyingCreativeOrSpectator(final ClientPlayerEntity player){ static boolean isFlyingCreativeOrSpectator(final ClientPlayerEntity player){
return player.abilities.flying && (player.isCreative() || player.isSpectator()); return player.abilities.isFlying && (player.isCreative() || player.isSpectator());
} }
static boolean shouldFlyOnGround(final ClientPlayerEntity player){ static boolean shouldFlyOnGround(final ClientPlayerEntity player){
return cfg().flyOnGroundInCreative && player.isCreative() && player.abilities.flying; return cfg().flyOnGroundInCreative && player.isCreative() && player.abilities.isFlying;
} }
static float getFlightSpeed(final ClientPlayerEntity player, final boolean boost){ static float getFlightSpeed(final ClientPlayerEntity player, final boolean boost){

View File

@@ -11,9 +11,8 @@ import chylex.bettercontrols.mixin.AccessGameRendererFields;
import chylex.bettercontrols.mixin.AccessPlayerFields; import chylex.bettercontrols.mixin.AccessPlayerFields;
import chylex.bettercontrols.mixin.AccessStickyKeyBindingStateGetter; import chylex.bettercontrols.mixin.AccessStickyKeyBindingStateGetter;
import chylex.bettercontrols.util.Key; import chylex.bettercontrols.util.Key;
import net.minecraft.client.input.Input; import net.minecraft.client.entity.player.ClientPlayerEntity;
import net.minecraft.client.input.KeyboardInput; import net.minecraft.util.MovementInput;
import net.minecraft.client.network.ClientPlayerEntity;
import java.lang.ref.WeakReference; import java.lang.ref.WeakReference;
import java.util.function.BooleanSupplier; import java.util.function.BooleanSupplier;
import static chylex.bettercontrols.util.Statics.KEY_FORWARD; import static chylex.bettercontrols.util.Statics.KEY_FORWARD;
@@ -47,8 +46,8 @@ public final class PlayerTicker{
// Logic // Logic
private final ToggleTracker toggleSprint = new ToggleTrackerForStickyKey(cfg().keyToggleSprint, KEY_SPRINT, toggled -> OPTIONS.sprintToggled = toggled); private final ToggleTracker toggleSprint = new ToggleTrackerForStickyKey(cfg().keyToggleSprint, KEY_SPRINT, toggled -> OPTIONS.toggleSprint = toggled);
private final ToggleTracker toggleSneak = new ToggleTrackerForStickyKey(cfg().keyToggleSneak, KEY_SNEAK, toggled -> OPTIONS.sneakToggled = toggled); private final ToggleTracker toggleSneak = new ToggleTrackerForStickyKey(cfg().keyToggleSneak, KEY_SNEAK, toggled -> OPTIONS.toggleCrouch = toggled);
private final ToggleTracker toggleWalkForward = new ToggleTracker(cfg().keyToggleWalkForward, KEY_FORWARD); private final ToggleTracker toggleWalkForward = new ToggleTracker(cfg().keyToggleWalkForward, KEY_FORWARD);
private final ToggleTracker toggleJump = new ToggleTracker(cfg().keyToggleJump, KEY_JUMP); private final ToggleTracker toggleJump = new ToggleTracker(cfg().keyToggleJump, KEY_JUMP);
@@ -88,7 +87,7 @@ public final class PlayerTicker{
} }
final SprintMode sprintMode = cfg().sprintMode; final SprintMode sprintMode = cfg().sprintMode;
final boolean wasSprintToggled = OPTIONS.sprintToggled; final boolean wasSprintToggled = OPTIONS.toggleSprint;
final boolean isSprintToggled = toggleSprint.tick(); final boolean isSprintToggled = toggleSprint.tick();
if (temporarySprintTimer > 0){ if (temporarySprintTimer > 0){
@@ -144,17 +143,17 @@ public final class PlayerTicker{
toggleSneak.tick(); toggleSneak.tick();
} }
public void afterInputAssignsPressingForward(final KeyboardInput input){ public void afterInputAssignsPressingForward(final MovementInput input){
if (MINECRAFT.currentScreen == null){ if (MINECRAFT.currentScreen == null){
input.pressingForward |= toggleWalkForward.tick(); input.forwardKeyDown |= toggleWalkForward.tick();
} }
} }
public void afterInputTick(final ClientPlayerEntity player){ public void afterInputTick(final ClientPlayerEntity player){
final Input input = player.input; final MovementInput input = player.movementInput;
if (MINECRAFT.currentScreen == null && !player.abilities.flying){ if (MINECRAFT.currentScreen == null && !player.abilities.isFlying){
input.jumping |= toggleJump.tick(); input.jump |= toggleJump.tick();
} }
if (FlightHelper.isFlyingCreativeOrSpectator(player)){ if (FlightHelper.isFlyingCreativeOrSpectator(player)){
@@ -166,25 +165,25 @@ public final class PlayerTicker{
player.abilities.setFlySpeed(flightSpeed); player.abilities.setFlySpeed(flightSpeed);
} }
if (Math.abs(verticalVelocity) > 1E-5F && player == MINECRAFT.getCameraEntity()){ if (Math.abs(verticalVelocity) > 1E-5F && player == MINECRAFT.getRenderViewEntity()){
int direction = 0; int direction = 0;
if (input.sneaking){ if (input.sneaking){
--direction; --direction;
} }
if (input.jumping){ if (input.jump){
++direction; ++direction;
} }
if (direction != 0){ if (direction != 0){
player.setVelocity(player.getVelocity().add(0D, flightSpeed * verticalVelocity * direction, 0D)); player.setMotion(player.getMotion().add(0D, flightSpeed * verticalVelocity * direction, 0D));
} }
} }
} }
if (cfg().resumeSprintingAfterHittingObstacle){ if (cfg().resumeSprintingAfterHittingObstacle){
if (wasHittingObstacle != player.horizontalCollision){ if (wasHittingObstacle != player.collidedHorizontally){
if (!wasHittingObstacle){ if (!wasHittingObstacle){
wasSprintingBeforeHittingObstacle = player.isSprinting() || Key.isPressed(KEY_SPRINT); wasSprintingBeforeHittingObstacle = player.isSprinting() || Key.isPressed(KEY_SPRINT);
} }
@@ -194,11 +193,11 @@ public final class PlayerTicker{
} }
// collision also stops when the player lets go of movement keys // collision also stops when the player lets go of movement keys
wasHittingObstacle = player.horizontalCollision; wasHittingObstacle = player.collidedHorizontally;
} }
} }
else{ else{
wasHittingObstacle = player.horizontalCollision; wasHittingObstacle = player.collidedHorizontally;
wasSprintingBeforeHittingObstacle = false; wasSprintingBeforeHittingObstacle = false;
} }
@@ -230,8 +229,8 @@ public final class PlayerTicker{
holdingSneakWhileTouchingGround = true; holdingSneakWhileTouchingGround = true;
} }
else if (holdingSneakWhileTouchingGround){ else if (holdingSneakWhileTouchingGround){
player.abilities.flying = false; player.abilities.isFlying = false;
player.sendAbilitiesUpdate(); player.sendPlayerAbilities();
cancelLanding = false; cancelLanding = false;
} }
} }
@@ -247,23 +246,23 @@ public final class PlayerTicker{
} }
if (FlightHelper.isFlyingCreativeOrSpectator(player) && cfg().disableFlightInertia){ if (FlightHelper.isFlyingCreativeOrSpectator(player) && cfg().disableFlightInertia){
final Input input = player.input; final MovementInput input = player.movementInput;
if (input.movementForward == 0F && input.movementSideways == 0F){ if (input.moveForward == 0F && input.moveStrafe == 0F){
player.setVelocity(player.getVelocity().multiply(0.0, 1.0, 0.0)); player.setMotion(player.getMotion().mul(0.0, 1.0, 0.0));
} }
if (!input.jumping && !input.sneaking){ if (!input.jump && !input.sneaking){
player.setVelocity(player.getVelocity().multiply(1.0, 0.0, 1.0)); player.setMotion(player.getMotion().mul(1.0, 0.0, 1.0));
} }
} }
if (player.isCreative()){ if (player.isCreative()){
if (Key.wasPressed(cfg().keyToggleFlight)){ if (Key.wasPressed(cfg().keyToggleFlight)){
final boolean isFlying = !player.abilities.flying; final boolean isFlying = !player.abilities.isFlying;
player.abilities.flying = isFlying; player.abilities.isFlying = isFlying;
player.sendAbilitiesUpdate(); player.sendPlayerAbilities();
if (isFlying){ if (isFlying){
temporaryFlyOnGroundTimer = 10; temporaryFlyOnGroundTimer = 10;
@@ -285,10 +284,10 @@ public final class PlayerTicker{
} }
if (!cfg().sneakingMovesCameraSmoothly){ if (!cfg().sneakingMovesCameraSmoothly){
final AccessCameraFields camera = (AccessCameraFields)MINECRAFT.gameRenderer.getCamera(); final AccessCameraFields camera = (AccessCameraFields)MINECRAFT.gameRenderer.getActiveRenderInfo();
if (camera.getFocusedEntity() == player){ if (camera.getFocusedEntity() == player){
camera.setCameraY(player.getStandingEyeHeight()); camera.setCameraY(player.getEyeHeight());
} }
} }
@@ -300,7 +299,7 @@ public final class PlayerTicker{
} }
if (Key.isPressed(cfg().keyOpenMenu)){ if (Key.isPressed(cfg().keyOpenMenu)){
MINECRAFT.openScreen(new BetterControlsScreen(null)); MINECRAFT.displayGuiScreen(new BetterControlsScreen(null));
} }
} }
} }

View File

@@ -1,45 +1,46 @@
package chylex.bettercontrols.util; package chylex.bettercontrols.util;
import net.minecraft.client.options.KeyBinding; import net.minecraft.client.settings.KeyBinding;
import net.minecraft.client.util.InputUtil; import net.minecraft.client.util.InputMappings;
import net.minecraft.client.util.InputMappings.Input;
public final class Key{ public final class Key{
private Key(){} private Key(){}
public static final InputUtil.KeyCode INVALID = InputUtil.UNKNOWN_KEYCODE; public static final Input INVALID = InputMappings.INPUT_INVALID;
public static boolean isUnbound(final KeyBinding binding){ public static boolean isUnbound(final KeyBinding binding){
return binding.isNotBound(); return binding.isInvalid();
} }
public static boolean isPressed(final KeyBinding binding){ public static boolean isPressed(final KeyBinding binding){
return binding.isPressed(); return binding.isKeyDown();
} }
public static boolean wasPressed(final KeyBinding binding){ public static boolean wasPressed(final KeyBinding binding){
return binding.wasPressed(); return binding.isPressed();
} }
public static String getBoundKeyText(final KeyBinding binding){ public static String getBoundKeyText(final KeyBinding binding){
return binding.getLocalizedName(); return binding.getLocalizedName();
} }
public static void bind(final KeyBinding binding, final InputUtil.KeyCode input){ public static void bind(final KeyBinding binding, final Input input){
binding.setKeyCode(input); binding.bind(input);
} }
public static String writeBinding(final KeyBinding binding){ public static String writeBinding(final KeyBinding binding){
return binding.getName(); return binding.getTranslationKey();
} }
public static void readBinding(final KeyBinding binding, final String serialized){ public static void readBinding(final KeyBinding binding, final String serialized){
bind(binding, InputUtil.fromName(serialized)); bind(binding, InputMappings.getInputByName(serialized));
} }
public static InputUtil.KeyCode inputFromMouse(final int button){ public static Input inputFromMouse(final int button){
return InputUtil.Type.MOUSE.createFromCode(button); return InputMappings.Type.MOUSE.getOrMakeInput(button);
} }
public static InputUtil.KeyCode inputFromKeyboard(final int keyCode, final int scanCode){ public static Input inputFromKeyboard(final int keyCode, final int scanCode){
return InputUtil.getKeyCode(keyCode, scanCode); return InputMappings.getInputByCode(keyCode, scanCode);
} }
} }

View File

@@ -1,11 +1,19 @@
package chylex.bettercontrols.util; package chylex.bettercontrols.util;
import net.minecraft.util.text.StringTextComponent;
public final class LiteralText extends net.minecraft.text.LiteralText{ public final class LiteralText extends StringTextComponent{
public static LiteralText text(final String text){ public static LiteralText text(final String text){
return new LiteralText(text); return new LiteralText(text);
} }
private final String msg;
public LiteralText(final String msg){ public LiteralText(final String msg){
super(msg); super(msg);
this.msg = msg;
}
public LiteralText copy(){
return new LiteralText(msg);
} }
} }

View File

@@ -1,16 +1,16 @@
package chylex.bettercontrols.util; package chylex.bettercontrols.util;
import net.minecraft.client.MinecraftClient; import net.minecraft.client.GameSettings;
import net.minecraft.client.options.GameOptions; import net.minecraft.client.Minecraft;
import net.minecraft.client.options.KeyBinding; import net.minecraft.client.settings.KeyBinding;
public final class Statics{ public final class Statics{
private Statics(){} private Statics(){}
public static final MinecraftClient MINECRAFT = MinecraftClient.getInstance(); public static final Minecraft MINECRAFT = Minecraft.getInstance();
public static final GameOptions OPTIONS = MINECRAFT.options; public static final GameSettings OPTIONS = MINECRAFT.gameSettings;
public static final KeyBinding KEY_SPRINT = OPTIONS.keySprint; public static final KeyBinding KEY_SPRINT = OPTIONS.keyBindSprint;
public static final KeyBinding KEY_SNEAK = OPTIONS.keySneak; public static final KeyBinding KEY_SNEAK = OPTIONS.keyBindSneak;
public static final KeyBinding KEY_FORWARD = OPTIONS.keyForward; public static final KeyBinding KEY_FORWARD = OPTIONS.keyBindForward;
public static final KeyBinding KEY_JUMP = OPTIONS.keyJump; public static final KeyBinding KEY_JUMP = OPTIONS.keyBindJump;
} }

View File

@@ -1,8 +0,0 @@
package io.github.prospector.modmenu.api;
import net.minecraft.client.gui.screen.Screen;
@SuppressWarnings("unused")
@FunctionalInterface
public interface ConfigScreenFactory<S extends Screen>{
S create(Screen var1);
}

View File

@@ -1,16 +0,0 @@
package io.github.prospector.modmenu.api;
import com.google.common.collect.ImmutableMap;
import java.util.Map;
@SuppressWarnings("unused")
public interface ModMenuApi{
String getModId();
default ConfigScreenFactory<?> getModConfigScreenFactory(){
return ignore -> null;
}
default Map<String, ConfigScreenFactory<?>> getProvidedConfigScreenFactories(){
return ImmutableMap.of();
}
}

View File

@@ -0,0 +1,30 @@
modLoader = "javafml"
loaderVersion = "[31,)"
authors = "chylex"
license = "MPL-2.0"
issueTrackerURL = "https://github.com/chylex/Better-Controls/issues"
[[mods]]
modId = "bettercontrols"
version = "1.2.0a"
displayName = "Better Controls"
logoFile = "icon.png"
description = '''
Adds many powerful key bindings and options to control your movement.
The features complement vanilla mechanics without giving unfair advantages, so server use should be fine.
'''
[[dependencies.bettercontrols]]
modId = "minecraft"
mandatory = true
versionRange = "[1.15.2,1.16)"
ordering = "NONE"
side = "BOTH"
[[dependencies.bettercontrols]]
modId = "forge"
mandatory = true
versionRange = "[31.2.44,)"
ordering = "NONE"
side = "BOTH"

View File

@@ -3,7 +3,7 @@
"minVersion": "0.8", "minVersion": "0.8",
"package": "chylex.bettercontrols.mixin", "package": "chylex.bettercontrols.mixin",
"compatibilityLevel": "JAVA_8", "compatibilityLevel": "JAVA_8",
"mixins": [ "client": [
"AccessCameraFields", "AccessCameraFields",
"AccessClientPlayerFields", "AccessClientPlayerFields",
"AccessControlsListCategory", "AccessControlsListCategory",

View File

@@ -1,33 +0,0 @@
{
"schemaVersion": 1,
"id": "bettercontrols",
"version": "${version}",
"name": "Better Controls",
"description": "Adds many powerful key bindings and options to control your movement.\nThe features complement vanilla mechanics without giving unfair advantages, so server use should be fine.",
"icon": "assets/bettercontrols/icon.png",
"license": "MPL-2.0",
"authors": [ "chylex" ],
"contact": {
"homepage": "",
"sources": "https://github.com/chylex/Better-Controls",
"issues": "https://github.com/chylex/Better-Controls/issues"
},
"environment": "client",
"entrypoints": {
"client": [ "chylex.bettercontrols.BetterControlsMod" ],
"modmenu": [ "chylex.bettercontrols.compatibility.ModMenuSupport" ]
},
"mixins": [{
"config": "bettercontrols.mixins.json",
"environment": "client"
}],
"depends": {
"fabricloader": ">=0.7.4",
"minecraft": "1.15.x"
}
}

View File

Before

Width:  |  Height:  |  Size: 21 KiB

After

Width:  |  Height:  |  Size: 21 KiB

View File

@@ -0,0 +1,7 @@
{
"pack": {
"description": "Better Controls",
"pack_format": 5,
"_comment": ""
}
}