package dev.isxander.yacl.gui.controllers; import dev.isxander.yacl.api.Controller; import dev.isxander.yacl.api.NameableEnum; import dev.isxander.yacl.api.Option; import dev.isxander.yacl.api.utils.Dimension; import dev.isxander.yacl.gui.AbstractWidget; import dev.isxander.yacl.gui.YACLScreen; import net.minecraft.client.gui.screen.Screen; import net.minecraft.text.Text; import net.minecraft.util.TranslatableOption; import org.jetbrains.annotations.ApiStatus; import org.lwjgl.glfw.GLFW; import java.util.function.Function; /** * Simple controller type that displays the enum on the right. *

* Cycles forward with left click, cycles backward with right click or when shift is held * * @param enum type */ public class EnumController> implements Controller { private final Option option; private final Function valueFormatter; private final T[] availableValues; public static > Function getDefaultFormatter() { return value -> { if (value instanceof NameableEnum nameableEnum) return nameableEnum.getDisplayName(); if (value instanceof TranslatableOption translatableOption) return translatableOption.getText(); return Text.of(value.toString()); }; } /** * Constructs a cycling enum controller with a default value formatter and all values being available. * The default value formatter first searches if the * enum is a {@link NameableEnum} else, just uses {@link Enum#toString()} * * @param option bound option */ public EnumController(Option option) { this(option, getDefaultFormatter()); } /** * Constructs a cycling enum controller with all values being available. * * @param option bound option * @param valueFormatter format the enum into any {@link Text} */ public EnumController(Option option, Function valueFormatter) { this(option, valueFormatter, option.typeClass().getEnumConstants()); } /** * Constructs a cycling enum controller. * * @param option bound option * @param valueFormatter format the enum into any {@link Text} * @param availableValues all enum constants that can be cycled through */ public EnumController(Option option, Function valueFormatter, T[] availableValues) { this.option = option; this.valueFormatter = valueFormatter; this.availableValues = availableValues; } /** * {@inheritDoc} */ @Override public Option option() { return option; } /** * {@inheritDoc} */ @Override public Text formatValue() { return valueFormatter.apply(option().pendingValue()); } /** * {@inheritDoc} */ @Override public AbstractWidget provideWidget(YACLScreen screen, Dimension widgetDimension) { return new EnumControllerElement<>(this, screen, widgetDimension, availableValues); } public static class EnumControllerElement> extends ControllerWidget> { private final T[] values; public EnumControllerElement(EnumController control, YACLScreen screen, Dimension dim, T[] values) { super(control, screen, dim); this.values = values; } public void cycleValue(int increment) { int targetIdx = control.option().pendingValue().ordinal() + increment; if (targetIdx >= values.length) { targetIdx -= values.length; } else if (targetIdx < 0) { targetIdx += values.length; } control.option().requestSet(values[targetIdx]); } @Override public boolean mouseClicked(double mouseX, double mouseY, int button) { if (!isMouseOver(mouseX, mouseY) || (button != 0 && button != 1) || !isAvailable()) return false; playDownSound(); cycleValue(button == 1 || Screen.hasShiftDown() || Screen.hasControlDown() ? -1 : 1); return true; } @Override public boolean keyPressed(int keyCode, int scanCode, int modifiers) { if (!focused) return false; switch (keyCode) { case GLFW.GLFW_KEY_LEFT, GLFW.GLFW_KEY_DOWN -> cycleValue(-1); case GLFW.GLFW_KEY_RIGHT, GLFW.GLFW_KEY_UP -> cycleValue(1); case GLFW.GLFW_KEY_ENTER, GLFW.GLFW_KEY_SPACE, GLFW.GLFW_KEY_KP_ENTER -> cycleValue(Screen.hasControlDown() || Screen.hasShiftDown() ? -1 : 1); default -> { return false; } } return true; } @Override protected int getHoveredControlWidth() { return getUnhoveredControlWidth(); } } }