From 9fbd40b1ce99b13129e65c8d77a87da8d4de651b Mon Sep 17 00:00:00 2001 From: brandon3055 Date: Fri, 26 Jan 2024 15:41:10 +1100 Subject: [PATCH] Feature/modular gui (#451) * Modular GUI Implementation. * First round of tweaks based on covers feedback. * Made requested changes * Update JEI maven --- build.gradle | 4 + gradle.properties | 1 + .../java/codechicken/lib/colour/Colour.java | 72 + .../codechicken/lib/compat/JEIPlugin.java | 37 + .../lib/gui/modular/ModularGui.java | 597 ++++++ .../lib/gui/modular/ModularGuiContainer.java | 280 +++ .../lib/gui/modular/ModularGuiScreen.java | 121 ++ .../lib/gui/modular/elements/GuiButton.java | 335 ++++ .../gui/modular/elements/GuiColourPicker.java | 232 +++ .../gui/modular/elements/GuiContextMenu.java | 135 ++ .../lib/gui/modular/elements/GuiDVD.java | 97 + .../lib/gui/modular/elements/GuiDialog.java | 241 +++ .../lib/gui/modular/elements/GuiElement.java | 491 +++++ .../gui/modular/elements/GuiEnergyBar.java | 143 ++ .../modular/elements/GuiEntityRenderer.java | 264 +++ .../modular/elements/GuiEventProvider.java | 122 ++ .../gui/modular/elements/GuiFluidTank.java | 252 +++ .../gui/modular/elements/GuiItemStack.java | 118 ++ .../lib/gui/modular/elements/GuiList.java | 221 +++ .../gui/modular/elements/GuiManipulable.java | 401 ++++ .../gui/modular/elements/GuiProgressIcon.java | 115 ++ .../gui/modular/elements/GuiRectangle.java | 165 ++ .../gui/modular/elements/GuiScrolling.java | 219 +++ .../lib/gui/modular/elements/GuiSlider.java | 266 +++ .../lib/gui/modular/elements/GuiSlots.java | 312 +++ .../lib/gui/modular/elements/GuiText.java | 268 +++ .../gui/modular/elements/GuiTextField.java | 660 +++++++ .../lib/gui/modular/elements/GuiTextList.java | 173 ++ .../lib/gui/modular/elements/GuiTexture.java | 110 ++ .../lib/gui/modular/lib/BackgroundRender.java | 34 + .../lib/gui/modular/lib/ColourState.java | 72 + .../lib/gui/modular/lib/Constraints.java | 209 ++ .../lib/gui/modular/lib/ContentElement.java | 14 + .../lib/gui/modular/lib/CursorHelper.java | 100 + .../lib/gui/modular/lib/DynamicTextures.java | 30 + .../lib/gui/modular/lib/ElementEvents.java | 272 +++ .../lib/gui/modular/lib/ForegroundRender.java | 32 + .../lib/gui/modular/lib/GuiProvider.java | 42 + .../lib/gui/modular/lib/GuiRender.java | 1728 +++++++++++++++++ .../lib/gui/modular/lib/ScissorHandler.java | 75 + .../lib/gui/modular/lib/SliderState.java | 139 ++ .../lib/gui/modular/lib/TextState.java | 56 + .../lib/gui/modular/lib/TooltipHandler.java | 105 + .../lib/container/ContainerGuiProvider.java | 33 + .../lib/container/ContainerScreenAccess.java | 22 + .../gui/modular/lib/container/DataSync.java | 48 + .../gui/modular/lib/container/SlotGroup.java | 119 ++ .../lib/gui/modular/lib/geometry/Align.java | 14 + .../lib/gui/modular/lib/geometry/Axis.java | 13 + .../gui/modular/lib/geometry/AxisConfig.java | 96 + .../lib/gui/modular/lib/geometry/Borders.java | 118 ++ .../lib/geometry/ConstrainedGeometry.java | 376 ++++ .../gui/modular/lib/geometry/Constraint.java | 154 ++ .../modular/lib/geometry/ConstraintImpl.java | 294 +++ .../gui/modular/lib/geometry/Direction.java | 40 + .../gui/modular/lib/geometry/GeoParam.java | 32 + .../lib/gui/modular/lib/geometry/GeoRef.java | 36 + .../gui/modular/lib/geometry/GuiParent.java | 190 ++ .../gui/modular/lib/geometry/Position.java | 103 + .../gui/modular/lib/geometry/Rectangle.java | 308 +++ .../lib/gui/modular/sprite/CCGuiTextures.java | 59 + .../lib/gui/modular/sprite/Material.java | 126 ++ .../gui/modular/sprite/ModAtlasHolder.java | 101 + .../codechicken/lib/internal/ClientInit.java | 12 + .../lib/internal/network/CCLNetwork.java | 6 +- .../internal/network/ClientPacketHandler.java | 5 +- .../internal/network/ServerPacketHandler.java | 22 + .../container/data/AbstractDataStore.java | 45 + .../inventory/container/data/BooleanData.java | 42 + .../inventory/container/data/ByteData.java | 42 + .../inventory/container/data/DoubleData.java | 42 + .../inventory/container/data/FloatData.java | 42 + .../inventory/container/data/FluidData.java | 53 + .../lib/inventory/container/data/IntData.java | 42 + .../inventory/container/data/LongData.java | 42 + .../inventory/container/data/ShortData.java | 42 + .../modular/ModularGuiContainerMenu.java | 321 +++ .../container/modular/ModularSlot.java | 121 ++ .../lib/render/CCRenderEventHandler.java | 2 + .../java/codechicken/lib/util/FormatUtil.java | 46 + .../resources/META-INF/accesstransformer.cfg | 24 + .../assets/codechickenlib/atlases/gui.json | 9 + .../textures/gui/cursors/drag.png | Bin 0 -> 4183 bytes .../textures/gui/cursors/resize_diag_tlbr.png | Bin 0 -> 3987 bytes .../textures/gui/cursors/resize_diag_trbl.png | Bin 0 -> 4339 bytes .../textures/gui/cursors/resize_h.png | Bin 0 -> 3325 bytes .../textures/gui/cursors/resize_v.png | Bin 0 -> 3955 bytes .../gui/dynamic/button_borderless.png | Bin 0 -> 8152 bytes .../gui/dynamic/button_borderless_pressed.png | Bin 0 -> 12424 bytes .../textures/gui/dynamic/button_highlight.png | Bin 0 -> 10447 bytes .../dynamic/button_highlight_borderless.png | Bin 0 -> 14008 bytes .../gui/dynamic/button_highlight_pressed.png | Bin 0 -> 14458 bytes .../textures/gui/dynamic/button_pressed.png | Bin 0 -> 12741 bytes .../textures/gui/dynamic/button_vanilla.png | Bin 0 -> 8373 bytes .../gui/dynamic/button_vanilla_disabled.png | Bin 0 -> 6849 bytes .../textures/gui/dynamic/gui_borderless.png | Bin 0 -> 6281 bytes .../textures/gui/dynamic/gui_vanilla.png | Bin 0 -> 6532 bytes .../textures/gui/widgets/energy_empty.png | Bin 0 -> 8061 bytes .../textures/gui/widgets/energy_full.png | Bin 0 -> 13959 bytes .../gui/widgets/progress_arrow_empty.png | Bin 0 -> 7299 bytes .../gui/widgets/progress_arrow_full.png | Bin 0 -> 6769 bytes .../textures/gui/widgets/slot.png | Bin 0 -> 2123 bytes .../textures/gui/widgets/slot_large.png | Bin 0 -> 5546 bytes 103 files changed, 12599 insertions(+), 3 deletions(-) create mode 100644 src/main/java/codechicken/lib/compat/JEIPlugin.java create mode 100644 src/main/java/codechicken/lib/gui/modular/ModularGui.java create mode 100644 src/main/java/codechicken/lib/gui/modular/ModularGuiContainer.java create mode 100644 src/main/java/codechicken/lib/gui/modular/ModularGuiScreen.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiButton.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiColourPicker.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiContextMenu.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiDVD.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiDialog.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiElement.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiEnergyBar.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiEntityRenderer.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiEventProvider.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiFluidTank.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiItemStack.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiList.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiManipulable.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiProgressIcon.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiRectangle.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiScrolling.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiSlider.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiSlots.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiText.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiTextField.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiTextList.java create mode 100644 src/main/java/codechicken/lib/gui/modular/elements/GuiTexture.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/BackgroundRender.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/ColourState.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/Constraints.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/ContentElement.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/CursorHelper.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/DynamicTextures.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/ElementEvents.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/ForegroundRender.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/GuiProvider.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/GuiRender.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/ScissorHandler.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/SliderState.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/TextState.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/TooltipHandler.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/container/ContainerGuiProvider.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/container/ContainerScreenAccess.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/container/DataSync.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/container/SlotGroup.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/Align.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/Axis.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/AxisConfig.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/Borders.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/ConstrainedGeometry.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/Constraint.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/ConstraintImpl.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/Direction.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/GeoParam.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/GeoRef.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/GuiParent.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/Position.java create mode 100644 src/main/java/codechicken/lib/gui/modular/lib/geometry/Rectangle.java create mode 100644 src/main/java/codechicken/lib/gui/modular/sprite/CCGuiTextures.java create mode 100644 src/main/java/codechicken/lib/gui/modular/sprite/Material.java create mode 100644 src/main/java/codechicken/lib/gui/modular/sprite/ModAtlasHolder.java create mode 100644 src/main/java/codechicken/lib/internal/network/ServerPacketHandler.java create mode 100644 src/main/java/codechicken/lib/inventory/container/data/AbstractDataStore.java create mode 100644 src/main/java/codechicken/lib/inventory/container/data/BooleanData.java create mode 100644 src/main/java/codechicken/lib/inventory/container/data/ByteData.java create mode 100644 src/main/java/codechicken/lib/inventory/container/data/DoubleData.java create mode 100644 src/main/java/codechicken/lib/inventory/container/data/FloatData.java create mode 100644 src/main/java/codechicken/lib/inventory/container/data/FluidData.java create mode 100644 src/main/java/codechicken/lib/inventory/container/data/IntData.java create mode 100644 src/main/java/codechicken/lib/inventory/container/data/LongData.java create mode 100644 src/main/java/codechicken/lib/inventory/container/data/ShortData.java create mode 100644 src/main/java/codechicken/lib/inventory/container/modular/ModularGuiContainerMenu.java create mode 100644 src/main/java/codechicken/lib/inventory/container/modular/ModularSlot.java create mode 100644 src/main/java/codechicken/lib/util/FormatUtil.java create mode 100644 src/main/resources/assets/codechickenlib/atlases/gui.json create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/cursors/drag.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/cursors/resize_diag_tlbr.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/cursors/resize_diag_trbl.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/cursors/resize_h.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/cursors/resize_v.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_borderless.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_borderless_pressed.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_highlight.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_highlight_borderless.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_highlight_pressed.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_pressed.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_vanilla.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_vanilla_disabled.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/dynamic/gui_borderless.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/dynamic/gui_vanilla.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/widgets/energy_empty.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/widgets/energy_full.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/widgets/progress_arrow_empty.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/widgets/progress_arrow_full.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/widgets/slot.png create mode 100644 src/main/resources/assets/codechickenlib/textures/gui/widgets/slot_large.png diff --git a/build.gradle b/build.gradle index 35fb7af0..6c3b0857 100644 --- a/build.gradle +++ b/build.gradle @@ -72,6 +72,7 @@ configurations { repositories { mavenLocal() maven { url "https://maven.covers1624.net/" } + maven { url "https://maven.blamejared.com/" } } dependencies { @@ -81,6 +82,9 @@ dependencies { testImplementation 'org.junit.jupiter:junit-jupiter-api:5.8.2' testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.8.2' + + compileOnly(fg.deobf("mezz.jei:jei-${mc_version}-common-api:${jei_version}")) + compileOnly(fg.deobf("mezz.jei:jei-${mc_version}-forge-api:${jei_version}")) } test { diff --git a/gradle.properties b/gradle.properties index 37d4aca9..6f3ff3aa 100644 --- a/gradle.properties +++ b/gradle.properties @@ -3,3 +3,4 @@ org.gradle.daemon=false mc_version=1.20.1 forge_version=47.1.65 mod_version=4.4.0 +jei_version=15.2.0.27 diff --git a/src/main/java/codechicken/lib/colour/Colour.java b/src/main/java/codechicken/lib/colour/Colour.java index 00edece3..61b22b05 100644 --- a/src/main/java/codechicken/lib/colour/Colour.java +++ b/src/main/java/codechicken/lib/colour/Colour.java @@ -134,6 +134,78 @@ public Colour set(float[] floats) { return set(floats[0], floats[1], floats[2], floats[3]); } + public Colour rF(float r) { + this.r = (byte) (255F * r); + return this; + } + + public Colour gF(float g) { + this.g = (byte) (255F * g); + return this; + } + + public Colour bF(float b) { + this.b = (byte) (255F * b); + return this; + } + + public Colour aF(float a) { + this.a = (byte) (255F * a); + return this; + } + + public Colour rF(int r) { + this.r = (byte) r; + return this; + } + + public Colour gF(int g) { + this.g = (byte) g; + return this; + } + + public Colour bF(int b) { + this.b = (byte) b; + return this; + } + + public Colour aF(int a) { + this.a = (byte) a; + return this; + } + + public float rF() { + return r / 255F; + } + + public float gF() { + return g / 255F; + } + + public float bF() { + return b / 255F; + } + + public float aF() { + return a / 255F; + } + + public int r() { + return r & 0xFF; + } + + public int g() { + return g & 0xFF; + } + + public int b() { + return b & 0xFF; + } + + public int a() { + return a & 0xFF; + } + /** * Flips a color between ABGR and RGBA. * diff --git a/src/main/java/codechicken/lib/compat/JEIPlugin.java b/src/main/java/codechicken/lib/compat/JEIPlugin.java new file mode 100644 index 00000000..c9608464 --- /dev/null +++ b/src/main/java/codechicken/lib/compat/JEIPlugin.java @@ -0,0 +1,37 @@ +package codechicken.lib.compat; + +import codechicken.lib.CodeChickenLib; +import codechicken.lib.gui.modular.ModularGuiContainer; +import mezz.jei.api.IModPlugin; +import mezz.jei.api.JeiPlugin; +import mezz.jei.api.gui.handlers.IGuiContainerHandler; +import mezz.jei.api.registration.IGuiHandlerRegistration; +import net.minecraft.client.renderer.Rect2i; +import net.minecraft.resources.ResourceLocation; + +import java.util.List; + +/** + * Created by brandon3055 on 31/12/2023 + */ +@JeiPlugin +public class JEIPlugin implements IModPlugin { + private static final ResourceLocation ID = new ResourceLocation(CodeChickenLib.MOD_ID, "jei_plugin"); + + public JEIPlugin() {} + + @Override + public ResourceLocation getPluginUid() { + return ID; + } + + @Override + public void registerGuiHandlers(IGuiHandlerRegistration registration) { + registration.addGuiContainerHandler(ModularGuiContainer.class, new IGuiContainerHandler<>() { + @Override + public List getGuiExtraAreas(ModularGuiContainer screen) { + return screen.getModularGui().getJeiExclusions().map(e -> e.getRectangle().toRect2i()).toList(); + } + }); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/ModularGui.java b/src/main/java/codechicken/lib/gui/modular/ModularGui.java new file mode 100644 index 00000000..b0a9d527 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/ModularGui.java @@ -0,0 +1,597 @@ +package codechicken.lib.gui.modular; + +import codechicken.lib.gui.modular.elements.GuiElement; +import codechicken.lib.gui.modular.lib.*; +import codechicken.lib.gui.modular.lib.container.ContainerGuiProvider; +import codechicken.lib.gui.modular.lib.geometry.Constraint; +import codechicken.lib.gui.modular.lib.geometry.GeoParam; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import net.covers1624.quack.collection.FastStream; +import net.minecraft.client.Minecraft; +import net.minecraft.client.gui.Font; +import net.minecraft.client.gui.GuiGraphics; +import net.minecraft.client.gui.screens.Screen; +import net.minecraft.client.renderer.MultiBufferSource; +import net.minecraft.network.chat.Component; +import net.minecraft.resources.ResourceLocation; +import net.minecraft.world.inventory.Slot; +import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; +import org.apache.logging.log4j.util.TriConsumer; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.function.Consumer; +import java.util.stream.Stream; + +/** + * The modular gui system is built around "Gui Elements" but those elements need to be rendered by a base parent element. That's what this class is. + * This class is essentially just a container for the root gui element. + *

+ * Created by brandon3055 on 18/08/2023 + * + * @see GuiProvider + * @see ContainerGuiProvider + */ +public class ModularGui implements GuiParent { + private static final Logger LOGGER = LogManager.getLogger(); + + private final GuiProvider provider; + private final GuiElement root; + + private boolean guiBuilt = false; + private boolean pauseScreen = false; + private boolean closeOnEscape = true; + private boolean renderBackground = true; + private boolean vanillaSlotRendering = false; + + private Font font; + private Minecraft mc; + private int screenWidth; + private int screenHeight; + private Screen screen; + private Screen parentScreen; + + private Component guiTitle = Component.empty(); + private ResourceLocation newCursor = null; + + private final Map> slotHandlers = new HashMap<>(); + private final List tickListeners = new ArrayList<>(); + private final List resizeListeners = new ArrayList<>(); + private final List closeListeners = new ArrayList<>(); + private final List> preClickListeners = new ArrayList<>(); + private final List> postClickListeners = new ArrayList<>(); + private final List> preKeyPressListeners = new ArrayList<>(); + private final List> postKeyPressListeners = new ArrayList<>(); + + private final List> jeiExclusions = new ArrayList<>(); + + /** + * @param provider The gui builder that will be used to construct this modular gui when the screen is initialized. + */ + public ModularGui(GuiProvider provider) { + this.provider = provider; + if (provider instanceof DynamicTextures textures) textures.makeTextures(DynamicTextures.DynamicTexture::guiTexturePath); + Minecraft mc = Minecraft.getInstance(); + updateScreenData(mc, mc.font, mc.getWindow().getGuiScaledWidth(), mc.getWindow().getGuiScaledHeight()); + try { + this.root = provider.createRootElement(this); + } catch (Throwable ex) { + LOGGER.error("An error occurred while constructing a modular gui", ex); + throw ex; + } + } + + public ModularGui(GuiProvider provider, Screen parentScreen) { + this(provider); + this.parentScreen = parentScreen; + } + + public void setScreen(Screen screen) { + this.screen = screen; + } + + public GuiProvider getProvider() { + return provider; + } + + //=== Modular Gui Setup ===// + + public void setGuiTitle(@NotNull Component guiTitle) { + this.guiTitle = guiTitle; + } + + @NotNull + public Component getGuiTitle() { + return guiTitle; + } + + /** + * @param pauseScreen Should a single-player game pause while this screen is open? + */ + public void setPauseScreen(boolean pauseScreen) { + this.pauseScreen = pauseScreen; + } + + public boolean isPauseScreen() { + return pauseScreen; + } + + public void setCloseOnEscape(boolean closeOnEscape) { + this.closeOnEscape = closeOnEscape; + } + + public boolean closeOnEscape() { + return closeOnEscape; + } + + /** + * Enable / disable the default screen background. (Default Enabled) + * This will be the usual darkened background when in-game, or the dirt background when not in game. + */ + public void renderScreenBackground(boolean renderBackground) { + this.renderBackground = renderBackground; + } + + public boolean renderBackground() { + return renderBackground; + } + + /** + * @return the root element to which content elements should be added. + */ + public GuiElement getRoot() { + return root instanceof ContentElement ? ((ContentElement) root).getContentElement() : root; + } + + public GuiElement getDirectRoot() { + return root; + } + + /** + * Sets up this gui to render like any other standards gui with the specified width and height. + * Meaning, the root element (usually the gui background image) will be centered on the screen, and will have the specified width and height. + *

+ * + * @param guiWidth Gui Width + * @param guiHeight Gui Height + * @see #initFullscreenGui() + */ + public ModularGui initStandardGui(int guiWidth, int guiHeight) { + root.constrain(GeoParam.WIDTH, Constraint.literal(guiWidth)); + root.constrain(GeoParam.HEIGHT, Constraint.literal(guiHeight)); + root.constrain(GeoParam.LEFT, Constraint.midPoint(get(GeoParam.LEFT), get(GeoParam.RIGHT), guiWidth / -2D)); + root.constrain(GeoParam.TOP, Constraint.midPoint(get(GeoParam.TOP), get(GeoParam.BOTTOM), guiHeight / -2D)); + return this; + } + + /** + * Sets up this gui to render as a full screen gui. + * Meaning the root element's geometry will match that of the underlying minecraft screen. + *

+ * + * @see #initStandardGui(int, int) + */ + public ModularGui initFullscreenGui() { + root.constrain(GeoParam.WIDTH, Constraint.match(get(GeoParam.WIDTH))); + root.constrain(GeoParam.HEIGHT, Constraint.match(get(GeoParam.HEIGHT))); + root.constrain(GeoParam.TOP, Constraint.match(get(GeoParam.TOP))); + root.constrain(GeoParam.LEFT, Constraint.match(get(GeoParam.LEFT))); + return this; + } + + /** + * By default, modular gui completely overrides vanillas default slot rendering. + * This ensures slots render within the depth constraint of the slot element and avoids situations where + * stacks in slots render on top of other parts of the gui. + * Meaning you can do things like hide slots by disabling the slot element, Or render elements on top of the slots. + *

+ * This method allow you to return full rendering control to vanilla if you need to for whatever reason. + */ + public void setVanillaSlotRendering(boolean vanillaSlotRendering) { + this.vanillaSlotRendering = vanillaSlotRendering; + } + + public boolean vanillaSlotRendering() { + return vanillaSlotRendering; + } + + //=== Modular Gui Passthrough Methods ===// + + /** + * Create a new {@link GuiRender} for the current render call. + * + * @param buffers BufferSource can be retried from {@link GuiGraphics} + * @return A new {@link GuiRender} for the current render call. + */ + public GuiRender createRender(MultiBufferSource.BufferSource buffers) { + return new GuiRender(mc, buffers); + } + + /** + * Primary render method for ModularGui. The screen implementing ModularGui must call this in its render method. + * Followed by the {@link #renderOverlay(GuiRender, float)} method to handle overlay rendering. + * + * @param render A new gui render call should be constructed for each frame via {@link #createRender(MultiBufferSource.BufferSource)} + */ + public void render(GuiRender render, float partialTicks) { + root.clearGeometryCache(); + double mouseX = computeMouseX(); + double mouseY = computeMouseY(); + root.render(render, mouseX, mouseY, partialTicks); + + //Ensure overlay is rendered at a depth of ether 400 or total element depth + 100 (whichever is greater) + double depth = root.getCombinedElementDepth(); + if (depth <= 300) { + render.pose().translate(0, 0, 400 - depth); + } else { + render.pose().translate(0, 0, 100); + } + } + + /** + * Handles gui overlay rendering. This is where things like tool tips are rendered. + * This should be called immediately after {@link #render(GuiRender, float)} + *

+ * The reason this is split out from {@link #render(GuiRender, float)} is to allow + * stack tool tips to override gui overlay rendering in {@link ModularGuiContainer} + * + * @param render This should be the same render instance that was passed to the previous {@link #render(GuiRender, float)} call. + * @return true if an overlay such as a tooltip is currently being drawn. + */ + public boolean renderOverlay(GuiRender render, float partialTicks) { + double mouseX = computeMouseX(); + double mouseY = computeMouseY(); + return root.renderOverlay(render, mouseX, mouseY, partialTicks, false); + } + + /** + * Primary update / tick method. Must be called from the tick method of the implementing screen. + */ + public void tick() { + newCursor = null; + double mouseX = computeMouseX(); + double mouseY = computeMouseY(); + root.updateMouseOver(mouseX, mouseY, false); + tickListeners.forEach(Runnable::run); + root.tick(mouseX, mouseY); + CursorHelper.setCursor(newCursor); + } + + /** + * Pass through for the mouseMoved event. Any screen implementing {@link ModularGui} must pass through this event. + * + * @param mouseX new mouse X position + * @param mouseY new mouse Y position + */ + public void mouseMoved(double mouseX, double mouseY) { + root.mouseMoved(mouseX, mouseY); + } + + /** + * Pass through for the mouseClicked event. Any screen implementing {@link ModularGui} must pass through this event. + * + * @param mouseX Mouse X position + * @param mouseY Mouse Y position + * @param button Mouse Button + * @return true if this event has been consumed. + */ + public boolean mouseClicked(double mouseX, double mouseY, int button) { + preClickListeners.forEach(e -> e.accept(mouseX, mouseY, button)); + boolean consumed = root.mouseClicked(mouseX, mouseY, button, false); + if (!consumed) { + postClickListeners.forEach(e -> e.accept(mouseX, mouseY, button)); + } + return consumed; + } + + /** + * Pass through for the mouseReleased event. Any screen implementing {@link ModularGui} must pass through this event. + * + * @param mouseX Mouse X position + * @param mouseY Mouse Y position + * @param button Mouse Button + * @return true if this event has been consumed. + */ + public boolean mouseReleased(double mouseX, double mouseY, int button) { + return root.mouseReleased(mouseX, mouseY, button, false); + } + + /** + * Pass through for the keyPressed event. Any screen implementing {@link ModularGui} must pass through this event. + * + * @param key the keyboard key that was pressed. + * @param scancode the system-specific scancode of the key + * @param modifiers bitfield describing which modifier keys were held down. + * @return true if this event has been consumed. + */ + public boolean keyPressed(int key, int scancode, int modifiers) { + preKeyPressListeners.forEach(e -> e.accept(key, scancode, modifiers)); + boolean consumed = root.keyPressed(key, scancode, modifiers, false); + if (!consumed) { + postKeyPressListeners.forEach(e -> e.accept(key, scancode, modifiers)); + } + return consumed; + } + + /** + * Pass through for the keyReleased event. Any screen implementing {@link ModularGui} must pass through this event. + * + * @param key the keyboard key that was released. + * @param scancode the system-specific scancode of the key + * @param modifiers bitfield describing which modifier keys were held down. + * @return true if this event has been consumed. + */ + public boolean keyReleased(int key, int scancode, int modifiers) { + return root.keyReleased(key, scancode, modifiers, false); + } + + /** + * Pass through for the charTyped event. Any screen implementing {@link ModularGui} must pass through this event. + * + * @param character The character typed. + * @param modifiers bitfield describing which modifier keys were held down. + * @return true if this event has been consumed. + */ + public boolean charTyped(char character, int modifiers) { + return root.charTyped(character, modifiers, false); + } + + /** + * Pass through for the mouseScrolled event. Any screen implementing {@link ModularGui} must pass through this event. + * + * @param mouseX Mouse X position + * @param mouseY Mouse Y position + * @param scroll Scroll direction and amount + * @return true if this event has been consumed. + */ + public boolean mouseScrolled(double mouseX, double mouseY, double scroll) { + return root.mouseScrolled(mouseX, mouseY, scroll, false); + } + + /** + * Must be called by the screen when this gui is closed. + */ + public void onGuiClose() { + CursorHelper.resetCursor(); + closeListeners.forEach(Runnable::run); + } + + //=== Basic Minecraft Stuff ===// + + protected void updateScreenData(Minecraft mc, Font font, int screenWidth, int screenHeight) { + this.mc = mc; + this.font = font; + this.screenWidth = screenWidth; + this.screenHeight = screenHeight; + } + + @Override + public void onScreenInit(Minecraft mc, Font font, int screenWidth, int screenHeight) { + updateScreenData(mc, font, screenWidth, screenHeight); + root.clearGeometryCache(); + try { + root.onScreenInit(mc, font, screenWidth, screenHeight); + if (!guiBuilt) { + guiBuilt = true; + provider.buildGui(this); + } else { + resizeListeners.forEach(Runnable::run); + } + } catch (Throwable ex) { + //Because it seems the default behavior is to just silently consume init errors... Not helpful! + LOGGER.error("An error occurred while building a modular gui", ex); + throw ex; + } + } + + @Override + public Minecraft mc() { + return mc; + } + + @Override + public Font font() { + return font; + } + + @Override + public int scaledScreenWidth() { + return screenWidth; + } + + @Override + public int scaledScreenHeight() { + return screenHeight; + } + + @Override + public ModularGui getModularGui() { + return this; + } + + /** + * Returns the Screen housing this {@link ModularGui} + * With custom ModularGui implementations this may be null. + */ + public Screen getScreen() { + return screen; + } + + @Nullable + public Screen getParentScreen() { + return parentScreen; + } + + //=== Child Elements ===// + + @Override + public List> getChildren() { + throw new UnsupportedOperationException("Child elements must be managed via the root gui element not the modular gui itself."); + } + + @Override + public void addChild(GuiElement child) { + if (root == null) { //If child is null, we are adding the root element. + child.initElement(this); + return; + } + throw new UnsupportedOperationException("Child elements must be managed via the root gui element not the modular gui itself."); + } + + @Override + public ModularGui addChild(Consumer createChild) { + throw new UnsupportedOperationException("Child elements must be managed via the root gui element not the modular gui itself."); + } + + @Override + public void adoptChild(GuiElement child) { + throw new UnsupportedOperationException("Child elements must be managed via the root gui element not the modular gui itself."); + } + + @Override + public void removeChild(GuiElement child) { + throw new UnsupportedOperationException("Child elements must be managed via the root gui element not the modular gui itself."); + } + + //=== Geometry ===// + //The geometry of the base ModularGui class should always match the underlying minecraft screen. + + @Override + public double xMin() { + return 0; + } + + @Override + public double xMax() { + return screenWidth; + } + + @Override + public double xSize() { + return screenWidth; + } + + @Override + public double yMin() { + return 0; + } + + @Override + public double yMax() { + return screenHeight; + } + + @Override + public double ySize() { + return screenHeight; + } + + //=== Other ===// + + public double computeMouseX() { + return mc.mouseHandler.xpos() * (double) mc.getWindow().getGuiScaledWidth() / (double) mc.getWindow().getScreenWidth(); + } + + public double computeMouseY() { + return mc.mouseHandler.ypos() * (double) mc.getWindow().getGuiScaledHeight() / (double) mc.getWindow().getScreenHeight(); + } + + /** + * Provides a way to later retrieve the gui element responsible for positioning and rendering a slot. + */ + public void setSlotHandler(Slot slot, GuiElement handler) { + if (slotHandlers.containsKey(slot)) throw new IllegalStateException("A gui slot can only have a single handler!"); + slotHandlers.put(slot, handler); + } + + /** + * Returns the gui element responsible for managing a gui slot. + */ + public GuiElement getSlotHandler(Slot slot) { + return slotHandlers.get(slot); + } + + /** + * Sets the current mouse cursor. + * The cursor is reset at the end of each UI tick so this must be set every tick for as long as you want your custom cursor to be active. + * */ + public void setCursor(ResourceLocation cursor) { + this.newCursor = cursor; + } + + /** + * Add an element to the list of jei exclusions. + * Use this for any elements that render outside the normal gui bounds. + * This will ensure JEI does not try to render on top of these elements. + */ + public void jeiExclude(GuiElement element) { + if (!jeiExclusions.contains(element)) { + jeiExclusions.add(element); + } + } + + /** + * Remove an element from the list of jei exclusions. + */ + public void removeJEIExclude(GuiElement element) { + jeiExclusions.remove(element); + } + + public FastStream> getJeiExclusions() { + return FastStream.of(jeiExclusions).filter(GuiElement::isEnabled); + } + + /** + * Allows you to attach a callback that will be fired at the start of each gui tick. + */ + public void onTick(Runnable onTick) { + tickListeners.add(onTick); + } + + /** + * Allows you to attach a callback that will be fired when the parent screen is resized. + */ + public void onResize(Runnable onResize) { + resizeListeners.add(onResize); + } + + public void onClose(Runnable onClose) { + closeListeners.add(onClose); + } + + /** + * Allows you to attach a callback that will be fired on mouse click, before the click is handled by the rest of the gui. + */ + public void onMouseClickPre(TriConsumer onClick) { + preClickListeners.add(onClick); + } + + /** + * Allows you to attach a callback that will be fired on mouse click, after the click has been handled by the rest of the gui. + * Will only be fired if the event was not consumed by an element. + */ + public void onMouseClickPost(TriConsumer onClick) { + postClickListeners.add(onClick); + } + + + /** + * Allows you to attach a callback that will be fired on key press, before the is handled by the rest of the gui. + */ + public void onKeyPressPre(TriConsumer preKeyPress) { + preKeyPressListeners.add(preKeyPress); + } + + /** + * Allows you to attach a callback that will be fired on key press, after it has been handled by the rest of the gui. + * Will only be fired if the event was not consumed by an element. + */ + public void onKeyPressPost(TriConsumer postKeyPress) { + postKeyPressListeners.add(postKeyPress); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/ModularGuiContainer.java b/src/main/java/codechicken/lib/gui/modular/ModularGuiContainer.java new file mode 100644 index 00000000..0063b6a6 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/ModularGuiContainer.java @@ -0,0 +1,280 @@ +package codechicken.lib.gui.modular; + +import codechicken.lib.gui.modular.elements.GuiElement; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.container.ContainerGuiProvider; +import codechicken.lib.gui.modular.lib.container.ContainerScreenAccess; +import codechicken.lib.gui.modular.lib.geometry.GeoParam; +import net.minecraft.ChatFormatting; +import net.minecraft.Util; +import net.minecraft.client.Minecraft; +import net.minecraft.client.gui.GuiGraphics; +import net.minecraft.client.gui.screens.inventory.AbstractContainerScreen; +import net.minecraft.network.chat.Component; +import net.minecraft.util.Mth; +import net.minecraft.world.entity.player.Inventory; +import net.minecraft.world.inventory.AbstractContainerMenu; +import net.minecraft.world.inventory.ClickType; +import net.minecraft.world.inventory.Slot; +import net.minecraft.world.item.ItemStack; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +/** + * Container screen implementation for {@link ModularGui}. + * + *

+ * Created by brandon3055 on 08/09/2023 + */ +public class ModularGuiContainer extends AbstractContainerScreen implements ContainerScreenAccess { + + public final ModularGui modularGui; + + public ModularGuiContainer(T containerMenu, Inventory inventory, ContainerGuiProvider provider) { + super(containerMenu, inventory, Component.empty()); + provider.setMenuAccess(this); + this.modularGui = new ModularGui(provider); + this.modularGui.setScreen(this); + } + + public ModularGui getModularGui() { + return modularGui; + } + + @NotNull + @Override + public Component getTitle() { + return modularGui.getGuiTitle(); + } + + @Override + public boolean shouldCloseOnEsc() { + return modularGui.closeOnEscape(); + } + + @Override + protected void init() { + modularGui.onScreenInit(minecraft, font, width, height); + } + + @Override + public void resize(@NotNull Minecraft minecraft, int width, int height) { + super.resize(minecraft, width, height); + modularGui.onScreenInit(minecraft, font, width, height); + } + + @Override + public void render(@NotNull GuiGraphics graphics, int mouseX, int mouseY, float partialTicks) { + GuiElement root = modularGui.getRoot(); + topPos = (int) root.getValue(GeoParam.TOP); + leftPos = (int) root.getValue(GeoParam.LEFT); + imageWidth = (int) root.getValue(GeoParam.WIDTH); + imageHeight = (int) root.getValue(GeoParam.HEIGHT); + + if (modularGui.renderBackground()) { + renderBackground(graphics); + } + GuiRender render = modularGui.createRender(graphics.bufferSource()); + modularGui.render(render, partialTicks); + + super.render(graphics, mouseX, mouseY, partialTicks); + + if (!handleFloatingItemRender(render, mouseX, mouseY) && !renderHoveredStackToolTip(render, mouseX, mouseY)) { + modularGui.renderOverlay(render, partialTicks); + } + } + + protected boolean handleFloatingItemRender(GuiRender render, int mouseX, int mouseY) { + if (modularGui.vanillaSlotRendering()) return false; + boolean ret = false; + + ItemStack stack = draggingItem.isEmpty() ? menu.getCarried() : draggingItem; + if (!stack.isEmpty()) { + int yOffset = draggingItem.isEmpty() ? 8 : 16; + String countOverride = null; + if (!draggingItem.isEmpty() && isSplittingStack) { + stack = stack.copyWithCount(Mth.ceil((float) stack.getCount() / 2.0F)); + } else if (isQuickCrafting && quickCraftSlots.size() > 1) { + stack = stack.copyWithCount(this.quickCraftingRemainder); + if (stack.isEmpty()) { + countOverride = ChatFormatting.YELLOW + "0"; + } + } + renderFloatingItem(render, stack, mouseX - 8, mouseY - yOffset, countOverride); + ret = true; + } + + if (!this.snapbackItem.isEmpty()) { + float anim = (float) (Util.getMillis() - this.snapbackTime) / 100.0F; + if (anim >= 1.0F) { + anim = 1.0F; + this.snapbackItem = ItemStack.EMPTY; + } + + int xDist = snapbackEnd.x - snapbackStartX; + int yDist = snapbackEnd.y - snapbackStartY; + int xPos = snapbackStartX + (int) ((float) xDist * anim); + int yPos = snapbackStartY + (int) ((float) yDist * anim); + renderFloatingItem(render, snapbackItem, xPos + leftPos, yPos + topPos, null); + ret = true; + } + + return ret; + } + + protected boolean renderHoveredStackToolTip(GuiRender guiGraphics, int mouseX, int mouseY) { + if (this.menu.getCarried().isEmpty() && this.hoveredSlot != null && this.hoveredSlot.hasItem()) { + GuiElement handler = modularGui.getSlotHandler(hoveredSlot); + if (handler != null && handler.blockMouseOver(handler, mouseX, mouseY)) { + return false; + } + ItemStack itemStack = this.hoveredSlot.getItem(); + guiGraphics.toolTipWithImage(this.getTooltipFromContainerItem(itemStack), itemStack.getTooltipImage(), mouseX, mouseY); + return true; + } + return false; + } + + @Override + protected void containerTick() { + modularGui.tick(); + } + + @Override + public void removed() { + super.removed(); + modularGui.onGuiClose(); + } + + //=== Input Pass-though ===// + + @Override + public void mouseMoved(double mouseX, double mouseY) { + modularGui.mouseMoved(mouseX, mouseY); + } + + @Override + public boolean mouseClicked(double mouseX, double mouseY, int button) { + return modularGui.mouseClicked(mouseX, mouseY, button) || super.mouseClicked(mouseX, mouseY, button); + } + + @Override + public boolean mouseReleased(double mouseX, double mouseY, int button) { + return modularGui.mouseReleased(mouseX, mouseY, button) || super.mouseReleased(mouseX, mouseY, button); + } + + @Override + public boolean mouseScrolled(double mouseX, double mouseY, double scroll) { + return modularGui.mouseScrolled(mouseX, mouseY, scroll) || super.mouseScrolled(mouseX, mouseY, scroll); + } + + @Override + public boolean keyPressed(int key, int scancode, int modifiers) { + return modularGui.keyPressed(key, scancode, modifiers) || super.keyPressed(key, scancode, modifiers); + } + + @Override + public boolean keyReleased(int key, int scancode, int modifiers) { + return modularGui.keyReleased(key, scancode, modifiers) || super.keyReleased(key, scancode, modifiers); + } + + @Override + public boolean charTyped(char character, int modifiers) { + return modularGui.charTyped(character, modifiers) || super.charTyped(character, modifiers); + } + + //=== AbstractContainerMenu Overrides ===// + + @Override + protected void renderBg(GuiGraphics guiGraphics, float f, int i, int j) { + } + + @Override + public void renderSlot(GuiGraphics guiGraphics, Slot slot) { + if (modularGui.vanillaSlotRendering()) super.renderSlot(guiGraphics, slot); + } + + //Modular gui friendly version of the slot render + @Override + public void renderSlot(GuiRender render, Slot slot) { + if (modularGui.vanillaSlotRendering()) return; + int slotX = slot.x + leftPos; + int slotY = slot.y + topPos; + ItemStack slotStack = slot.getItem(); + boolean dragingToSlot = false; + boolean dontRenderItem = slot == this.clickedSlot && !this.draggingItem.isEmpty() && !this.isSplittingStack; + + ItemStack carriedStack = this.menu.getCarried(); + String countString = null; + if (slot == this.clickedSlot && !this.draggingItem.isEmpty() && this.isSplittingStack && !slotStack.isEmpty()) { + slotStack = slotStack.copyWithCount(slotStack.getCount() / 2); + } else if (this.isQuickCrafting && this.quickCraftSlots.contains(slot) && !carriedStack.isEmpty()) { + if (this.quickCraftSlots.size() == 1) { + return; + } + + if (AbstractContainerMenu.canItemQuickReplace(slot, carriedStack, true) && this.menu.canDragTo(slot)) { + dragingToSlot = true; + int k = Math.min(carriedStack.getMaxStackSize(), slot.getMaxStackSize(carriedStack)); + int l = slot.getItem().isEmpty() ? 0 : slot.getItem().getCount(); + int m = AbstractContainerMenu.getQuickCraftPlaceCount(this.quickCraftSlots, this.quickCraftingType, carriedStack) + l; + if (m > k) { + m = k; + countString = ChatFormatting.YELLOW.toString() + k; + } + + slotStack = carriedStack.copyWithCount(m); + } else { + this.quickCraftSlots.remove(slot); + this.recalculateQuickCraftRemaining(); + } + } + + if (!dontRenderItem) { + if (dragingToSlot) { + //Highlights slots when doing a drag place operation. + render.fill(slotX, slotY, slotX + 16, slotY + 16, 0x80ffffff); + } + render.renderItem(slotStack, slotX, slotY, 16, slot.x + (slot.y * this.imageWidth)); //TODO May want a random that does not change if the slot is moved. + render.renderItemDecorations(slotStack, slotX, slotY, countString); + } + } + + @Override //Disable vanilla title and inventory name rendering + protected void renderLabels(GuiGraphics guiGraphics, int i, int j) { + } + + @Override + public void renderFloatingItem(GuiGraphics guiGraphics, ItemStack itemStack, int i, int j, String string) { + if (modularGui.vanillaSlotRendering()) super.renderFloatingItem(guiGraphics, itemStack, i, j, string); + } + + public void renderFloatingItem(GuiRender render, ItemStack itemStack, int x, int y, String string) { + render.pose().pushPose(); + render.pose().translate(0.0F, 0.0F, 50F); + render.renderItem(itemStack, x, y); + render.renderItemDecorations(itemStack, x, y - (this.draggingItem.isEmpty() ? 0 : 8), string); + render.pose().popPose(); + } + + @Nullable + @Override + public Slot findSlot(double mouseX, double mouseY) { + Slot slot = super.findSlot(mouseX, mouseY); + if (slot == null) return null; + GuiElement handler = modularGui.getSlotHandler(slot); + if (handler != null && (!handler.isEnabled() || !handler.isMouseOver())) { + return null; + } + return slot; + } + + @Override + protected void slotClicked(Slot slot, int i, int j, ClickType clickType) { + if (slot != null) { + GuiElement handler = modularGui.getSlotHandler(slot); + if (handler != null && !handler.isEnabled()) return; + } + super.slotClicked(slot, i, j, clickType); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/ModularGuiScreen.java b/src/main/java/codechicken/lib/gui/modular/ModularGuiScreen.java new file mode 100644 index 00000000..10fd1060 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/ModularGuiScreen.java @@ -0,0 +1,121 @@ +package codechicken.lib.gui.modular; + +import codechicken.lib.gui.modular.lib.GuiProvider; +import codechicken.lib.gui.modular.lib.GuiRender; +import net.minecraft.client.Minecraft; +import net.minecraft.client.gui.GuiGraphics; +import net.minecraft.client.gui.screens.Screen; +import net.minecraft.network.chat.Component; +import org.jetbrains.annotations.NotNull; + +/** + * A simple ModularGui screen implementation. + * This is simply a wrapper for a {@link ModularGui} that takes a {@link GuiProvider} + * This should be suitable for most basic gui screens. + *

+ * Created by brandon3055 on 19/08/2023 + */ +public class ModularGuiScreen extends Screen { + + protected final ModularGui modularGui; + + public ModularGuiScreen(GuiProvider provider) { + super(Component.empty()); + this.modularGui = new ModularGui(provider); + this.modularGui.setScreen(this); + } + + public ModularGuiScreen(GuiProvider builder, Screen parentScreen) { + super(Component.empty()); + this.modularGui = new ModularGui(builder, parentScreen); + this.modularGui.setScreen(this); + } + + public ModularGui getModularGui() { + return modularGui; + } + + @NotNull + @Override + public Component getTitle() { + return modularGui.getGuiTitle(); + } + + @Override + public boolean isPauseScreen() { + return modularGui.isPauseScreen(); + } + + @Override + public boolean shouldCloseOnEsc() { + return modularGui.closeOnEscape(); + } + + @Override + protected void init() { + modularGui.onScreenInit(minecraft, font, width, height); + } + + @Override + public void resize(@NotNull Minecraft minecraft, int width, int height) { + super.resize(minecraft, width, height); + modularGui.onScreenInit(minecraft, font, width, height); + } + + @Override + public void render(@NotNull GuiGraphics graphics, int mouseX, int mouseY, float partialTicks) { + if (modularGui.renderBackground()) { + renderBackground(graphics); + } + GuiRender render = modularGui.createRender(graphics.bufferSource()); + modularGui.render(render, partialTicks); + modularGui.renderOverlay(render, partialTicks); + } + + @Override + public void tick() { + modularGui.tick(); + } + + @Override + public void removed() { + modularGui.onGuiClose(); + } + + //=== Input Pass-though ===// + + @Override + public void mouseMoved(double mouseX, double mouseY) { + modularGui.mouseMoved(mouseX, mouseY); + } + + @Override + public boolean mouseClicked(double mouseX, double mouseY, int button) { + return modularGui.mouseClicked(mouseX, mouseY, button) || super.mouseClicked(mouseX, mouseY, button); + } + + @Override + public boolean mouseReleased(double mouseX, double mouseY, int button) { + return modularGui.mouseReleased(mouseX, mouseY, button) || super.mouseReleased(mouseX, mouseY, button); + } + + @Override + public boolean mouseScrolled(double mouseX, double mouseY, double scroll) { + return modularGui.mouseScrolled(mouseX, mouseY, scroll) || super.mouseScrolled(mouseX, mouseY, scroll); + } + + @Override + public boolean keyPressed(int key, int scancode, int modifiers) { + return modularGui.keyPressed(key, scancode, modifiers) || super.keyPressed(key, scancode, modifiers); + } + + @Override + public boolean keyReleased(int key, int scancode, int modifiers) { + return modularGui.keyReleased(key, scancode, modifiers) || super.keyReleased(key, scancode, modifiers); + } + + @Override + public boolean charTyped(char character, int modifiers) { + return modularGui.charTyped(character, modifiers) || super.charTyped(character, modifiers); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiButton.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiButton.java new file mode 100644 index 00000000..b8df4d2b --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiButton.java @@ -0,0 +1,335 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.Constraints; +import codechicken.lib.gui.modular.lib.geometry.Constraint; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.gui.modular.sprite.CCGuiTextures; +import net.minecraft.client.resources.sounds.SimpleSoundInstance; +import net.minecraft.core.Holder; +import net.minecraft.network.chat.Component; +import net.minecraft.sounds.SoundEvent; +import net.minecraft.sounds.SoundEvents; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.HashMap; +import java.util.Map; +import java.util.function.Function; +import java.util.function.Supplier; + +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * Created by brandon3055 on 28/08/2023 + */ +public class GuiButton extends GuiElement { + public static final int LEFT_CLICK = 0; + public static final int RIGHT_CLICK = 1; + public static final int MIDDLE_CLICK = 2; + + private final Map onClick = new HashMap<>(); + private final Map onPress = new HashMap<>(); + private boolean pressed = false; + private Holder pressSound = SoundEvents.UI_BUTTON_CLICK; + private Holder releaseSound = null; + private Supplier disabled = () -> false; + private Supplier toggleState; + private GuiText label = null; + + /** + * In its default state this is a blank, invisible element that can fire callbacks when pressed. + * To make an actual usable button, ether use one of the builtin static create methods, + * Or add your own elements to make this button look and function in a way that meets your needs. + * + * @param parent parent {@link GuiParent}. + */ + public GuiButton(@NotNull GuiParent parent) { + super(parent); + } + + /** + * Creates a new gui button that looks and acts exactly like a standard vanilla button. + */ + public static GuiButton vanilla(@NotNull GuiParent parent, @Nullable Component label, Runnable onClick) { + return vanilla(parent, label).onClick(onClick); + } + + /** + * Creates a new gui button that looks and acts exactly like a standard vanilla button. + */ + public static GuiButton vanilla(@NotNull GuiParent parent, @Nullable Component label) { + GuiButton button = new GuiButton(parent); + GuiTexture texture = new GuiTexture(button, CCGuiTextures.getter(() -> button.toggleState() ? "dynamic/button_highlight" : "dynamic/button_vanilla")); + texture.dynamicTexture(); + GuiRectangle highlight = new GuiRectangle(button).border(() -> button.hoverTime() > 0 ? 0xFFFFFFFF : 0); + + Constraints.bind(texture, button); + Constraints.bind(highlight, button); + + if (label != null) { + button.setLabel(new GuiText(button, label)); + Constraints.bind(button.getLabel(), button, 0, 2, 0, 2); + } + + return button; + } + + /** + * Creates a vanilla button with a "press" animation. + */ + public static GuiButton vanillaAnimated(@NotNull GuiParent parent, Component label, Runnable onPress) { + return vanillaAnimated(parent, label == null ? null : () -> label, onPress); + } + + /** + * Creates a vanilla button with a "press" animation. + */ + public static GuiButton vanillaAnimated(@NotNull GuiParent parent, @Nullable Supplier label, Runnable onPress) { + return vanillaAnimated(parent, label).onPress(onPress); + } + + //TODO Could use a quad-sliced texture for this. + + /** + * Creates a vanilla button with a "press" animation. + */ + public static GuiButton vanillaAnimated(@NotNull GuiParent parent, Component label) { + return vanillaAnimated(parent, label == null ? null : () -> label); + } + + /** + * Creates a vanilla button with a "press" animation. + */ + public static GuiButton vanillaAnimated(@NotNull GuiParent parent, @Nullable Supplier label) { + GuiButton button = new GuiButton(parent); + GuiTexture texture = new GuiTexture(button, CCGuiTextures.getter(() -> button.toggleState() || button.isPressed() ? "dynamic/button_pressed" : "dynamic/button_vanilla")); + texture.dynamicTexture(); + GuiRectangle highlight = new GuiRectangle(button).border(() -> button.isMouseOver() ? 0xFFFFFFFF : 0); + + Constraints.bind(texture, button); + Constraints.bind(highlight, button); + + if (label != null) { + button.setLabel(new GuiText(button, label) + .constrain(TOP, Constraint.relative(button.get(TOP), () -> button.isPressed() ? -0.5D : 0.5D).precise()) + .constrain(LEFT, Constraint.relative(button.get(LEFT), () -> button.isPressed() ? 1.5D : 2.5D).precise()) + .constrain(WIDTH, Constraint.relative(button.get(WIDTH), -4)) + .constrain(HEIGHT, Constraint.match(button.get(HEIGHT))) + ); + } + + return button; + } + + /** + * Super simple button that is just a coloured rectangle with a label. + */ + public static GuiButton flatColourButton(@NotNull GuiParent parent, @Nullable Supplier label, Function buttonColour) { + return flatColourButton(parent, label, buttonColour, null); + } + + /** + * Super simple button that is just a coloured rectangle with a label. + */ + public static GuiButton flatColourButton(@NotNull GuiParent parent, @Nullable Supplier label, Function buttonColour, @Nullable Function borderColour) { + GuiButton button = new GuiButton(parent); + GuiRectangle background = new GuiRectangle(button) + .fill(() -> buttonColour.apply(button.isMouseOver() || button.toggleState() || button.isPressed())) + .border(borderColour == null ? null : () -> borderColour.apply(button.isMouseOver() || button.toggleState() || button.isPressed())); + Constraints.bind(background, button); + + if (label != null) { + GuiText text = new GuiText(button, label); + button.setLabel(text); + Constraints.bind(text, button, 0, 2, 0, 2); + } + + return button; + } + + /** + * When creating buttons with labels, use this method to store a reference to the label in the button fore easy retrival later. + * + * @param label The button label. + */ + public GuiButton setLabel(GuiText label) { + this.label = label; + return this; + } + + /** + * @return The buttons label element, If it has one. + */ + public GuiText getLabel() { + return label; + } + + /** + * This event is fired immediately when this button is left-clicked. + * This is the logic used by most vanilla gui buttons. + * + * @see #onPress(Runnable) + */ + public GuiButton onClick(Runnable onClick) { + return onClick(onClick, LEFT_CLICK); + } + + /** + * This event is fired immediately when this button is clicked with the specified mouse button. + * This is the logic used by most vanilla gui buttons. + * Note: You can apply one listener per mouse button. + * + * @see #onPress(Runnable, int) + */ + public GuiButton onClick(Runnable onClick, int mouseButton) { + this.onClick.put(mouseButton, onClick); + return this; + } + + /** + * This event is fired when the button is pressed and then released using the left mosue button. + * The event is only fired if the cursor is still over the button when left click is released. + * This is the standard logic for most buttons in the world, but not vanillas. + * Note: You can apply one listener per mouse button. + * + * @see #onPress(Runnable, int) + */ + public GuiButton onPress(Runnable onPress) { + return onPress(onPress, LEFT_CLICK); + } + + /** + * This event is fired when the button is pressed and then released using the specified mouse button. + * The event is only fired if the cursor is still over the button when left click is released. + * This is the standard logic for most buttons in the world, but not vanillas. + * + * @see #onPress(Runnable, int) + */ + public GuiButton onPress(Runnable onPress, int mouseButton) { + this.onPress.put(mouseButton, onPress); + return this; + } + + /** + * Allows set the disabled status of this button + * Note: This is not the same as {@link #setEnabled(boolean)} the "enabled" allows you to completely disable an element. + * This "disabled" status is specific to {@link GuiButton}, + * When disabled via this method a button is still visible but greyed out / not clickable. + */ + public GuiButton setDisabled(boolean disabled) { + this.disabled = () -> disabled; + return this; + } + + /** + * Allows you to install a suppler that controls the disabled state of this button. + * Note: This is not the same as {@link #setEnabled(Supplier)} the "enabled" allows you to completely disable an element. + * This "disabled" status is specific to {@link GuiButton}, + * When disabled via this method a button is still visible but greyed out / not clickable. + */ + public GuiButton setDisabled(Supplier disabled) { + this.disabled = disabled; + return this; + } + + /** + * Allows this button to be used as a toggle or radio button. + * This method allows you to install a suppler that controls the current "selected / toggled" state. + * + * @param toggleState supplier that indicates weather or not this button should currently render as pressed/selected. + */ + public GuiButton setToggleMode(@Nullable Supplier toggleState) { + this.toggleState = toggleState; + return this; + } + + /** + * @return the "disabled" status. + * @see #setDisabled(boolean) + * @see #setDisabled(Supplier) + */ + public boolean isDisabled() { + return disabled.get(); + } + + /** + * @return true if this button is currently pressed by the user (left click held down on button) + */ + public boolean isPressed() { + return pressed && hoverTime() > 0; + } + + public boolean toggleState() { + return toggleState != null && toggleState.get(); + } + + /** + * Sets the sound to be played when this button is pressed. + */ + public GuiButton setPressSound(Holder pressSound) { + this.pressSound = pressSound; + return this; + } + + /** + * Sets the sound to be played when this button is released. + */ + public GuiButton setReleaseSound(Holder releaseSound) { + this.releaseSound = releaseSound; + return this; + } + + public Holder getPressSound() { + return pressSound; + } + + public Holder getReleaseSound() { + return releaseSound; + } + + @Override + public boolean mouseClicked(double mouseX, double mouseY, int button) { + if (!isMouseOver() || isDisabled()) return false; + Runnable onClick = this.onClick.get(button); + Runnable onPress = this.onPress.get(button); + if (onClick == null && onPress == null) return false; + pressed = true; + hoverTime = 1; + + boolean consume = false; + if (onClick != null) { + onClick.run(); + consume = true; + } + if (onPress != null) { + consume = true; + } + + if (getPressSound() != null) { + mc().getSoundManager().play(SimpleSoundInstance.forUI(getPressSound(), 1F)); + } + return consume; + } + + @Override + public boolean mouseReleased(double mouseX, double mouseY, int button, boolean consumed) { + consumed = super.mouseReleased(mouseX, mouseY, button, consumed); + if (!pressed) return consumed; + Runnable onClick = this.onClick.get(button); + Runnable onPress = this.onPress.get(button); + if (onClick == null && onPress == null) return consumed; + hoverTime = 1; + + if (!isDisabled() && isMouseOver()) { + if (pressed && onPress != null) { + onPress.run(); + consumed = true; + } + if (getReleaseSound() != null && (toggleState == null || !toggleState.get())) { + mc().getSoundManager().play(SimpleSoundInstance.forUI(getReleaseSound(), 1F)); + } + } + pressed = false; + return consumed; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiColourPicker.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiColourPicker.java new file mode 100644 index 00000000..c576a7e8 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiColourPicker.java @@ -0,0 +1,232 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.colour.Colour; +import codechicken.lib.gui.modular.lib.*; +import codechicken.lib.gui.modular.lib.geometry.Axis; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import net.minecraft.client.gui.screens.Screen; +import net.minecraft.network.chat.Component; +import org.jetbrains.annotations.NotNull; + +import java.util.function.Supplier; + +import static codechicken.lib.gui.modular.lib.geometry.Constraint.*; +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * Created by brandon3055 on 17/11/2023 + */ +public class GuiColourPicker extends GuiManipulable { + + private ColourState colourState = ColourState.create(); + private GuiButton okButton; + private GuiButton cancelButton; + + public GuiColourPicker(@NotNull GuiParent parent) { + super(parent); + } + + public static GuiColourPicker create(GuiParent guiParent, ColourState colourState) { + return create(guiParent, colourState, true); + } + + public static GuiColourPicker create(GuiParent guiParent, ColourState colourState, boolean hasAlpha) { + Colour initialColour = colourState.getColour(); + GuiColourPicker picker = new GuiColourPicker(guiParent.getModularGui().getRoot()); + picker.setOpaque(true); + picker.setColourState(colourState); + Constraints.size(picker, 80, hasAlpha ? 80 : 68); + + GuiRectangle background = GuiRectangle.toolTipBackground(picker.getContentElement()); + Constraints.bind(background, picker.getContentElement()); + + var hexField = GuiTextField.create(background, 0xFF000000, 0xFF505050, 0xe0e0e0); + hexField.field() + .setTextState(picker.getTextState()) + .setMaxLength(hasAlpha ? 8 : 6) + .setFilter(s -> s.isEmpty() || validHex(s)); + hexField.container() + .setOpaque(true) + .constrain(HEIGHT, literal(12)) + .constrain(TOP, relative(background.get(TOP), 4)) + .constrain(LEFT, relative(background.get(LEFT), 4)) + .constrain(RIGHT, relative(background.get(RIGHT), -4)); + + SliderBG slider = makeSlider(background, 0xFFFF0000, picker.sliderStateRed()) + .constrain(TOP, relative(hexField.container().get(BOTTOM), 2)); + + slider = makeSlider(background, 0xFF00FF00, picker.sliderStateGreen()) + .constrain(TOP, relative(slider.get(BOTTOM), 1)); + + slider = makeSlider(background, 0xFF0000FF, picker.sliderStateBlue()) + .constrain(TOP, relative(slider.get(BOTTOM), 1)); + + if (hasAlpha) { + slider = makeSlider(background, 0xFFFFFFFF, picker.sliderStateAlpha()) + .constrain(TOP, relative(slider.get(BOTTOM), 1)); + } else { + colourState.set(colourState.getColour().aF(0)); + } + + ColourPreview preview = new ColourPreview(background, () -> hasAlpha ? colourState.get() : (colourState.get() | 0xFF000000)) + .setOpaque(true) + .constrain(HEIGHT, literal(6)) + .constrain(TOP, relative(slider.get(BOTTOM), 2)) + .constrain(LEFT, relative(background.get(LEFT), 4)) + .constrain(RIGHT, relative(background.get(RIGHT), -4)); + + picker.cancelButton = GuiButton.flatColourButton(background, () -> Component.translatable("gui.cancel"), e -> 0xFF000000, e -> e ? 0xFF777777 : 0xFF555555) + .setOpaque(true) + .onPress(() -> { + colourState.set(initialColour); + picker.getParent().removeChild(picker); + }) + .constrain(HEIGHT, literal(10)) + .constrain(TOP, relative(preview.get(BOTTOM), 2)) + .constrain(LEFT, midPoint(background.get(LEFT), background.get(RIGHT), -4)) + .constrain(RIGHT, relative(background.get(RIGHT), -4)); + + picker.okButton = GuiButton.flatColourButton(background, () -> Component.translatable("gui.ok"), e -> 0xFF000000, e -> e ? 0xFF777777 : 0xFF555555) + .setOpaque(true) + .onPress(() -> picker.getParent().removeChild(picker)) + .constrain(HEIGHT, literal(10)) + .constrain(TOP, relative(preview.get(BOTTOM), 2)) + .constrain(LEFT, relative(background.get(LEFT), 4)) + .constrain(RIGHT, dynamic(() -> picker.cancelButton.isEnabled() ? picker.cancelButton.xMin() - 2 : background.xMax() - 4)); + + return picker; + } + + public static SliderBG makeSlider(GuiElement background, int colour, SliderState state) { + SliderBG slideBG = new SliderBG(background, 0xFF505050, 0x30FFFFFF) + .setOpaque(true) + .constrain(HEIGHT, literal(9)) + .constrain(LEFT, relative(background.get(LEFT), 4)) + .constrain(RIGHT, relative(background.get(RIGHT), -4)); + + GuiSlider slider = new GuiSlider(slideBG, Axis.X) + .setSliderState(state); + Constraints.bind(slider, slideBG, 0, 1, 0, 1); + slideBG.slider = slider; + + GuiRectangle handle = new GuiRectangle(slider) + .fill(colour) + .border(0xFF000000) + .borderWidth(0.5) + .constrain(WIDTH, literal(4)); + + slider.installSlider(handle) + .bindSliderWidth(); + + return slideBG; + } + + private static boolean validHex(String value) { + try { + Integer.parseUnsignedInt(value, 16); + return true; + } catch (Throwable ignored) { + return false; + } + } + + public static class SliderBG extends GuiElement implements BackgroundRender { + public int colour; + public int highlight; + public GuiSlider slider; + public boolean pressed = false; + + public SliderBG(@NotNull GuiParent parent, int colour, int highlight) { + super(parent); + this.colour = colour; + this.highlight = highlight; + } + + @Override + public boolean mouseClicked(double mouseX, double mouseY, int button, boolean consumed) { + pressed = button == 0; + return super.mouseClicked(mouseX, mouseY, button, consumed); + } + + @Override + public boolean mouseReleased(double mouseX, double mouseY, int button, boolean consumed) { + if (button == 0) pressed = false; + return super.mouseReleased(mouseX, mouseY, button, consumed); + } + + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + render.fill(xMin(), yMin(), xMin() + 1, yMax(), colour); + render.fill(xMax() - 1, yMin(), xMax(), yMax(), colour); + render.fill(xMin() + 1, yCenter() - 0.5, xMax() - 1, yCenter() + 0.5, colour); + + if ((isMouseOver() && !pressed) || slider.isDragging()) { + render.rect(getRectangle(), highlight); + } + } + } + + public static class ColourPreview extends GuiElement implements BackgroundRender { + private final Supplier colour; + public int colourA = 0xFF999999; + public int colourB = 0xFF666666; + + public ColourPreview(@NotNull GuiParent parent, Supplier colour) { + super(parent); + this.colour = colour; + } + + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + render.pushScissorRect(xMin(), yMin(), xSize(), ySize()); + for (int x = 0; xMin() + (x * 2) < xMax(); x++) { + for (int y = 0; yMin() + (y * 2) < yMax(); y++) { + int col = (y & 1) == 0 ? ((x & 1) == 0 ? colourA : colourB) : ((x & 1) == 0 ? colourB : colourA); + render.rect(xMin() + (x * 2), yMin() + (y * 2), 2, 2, col); + } + } + render.popScissor(); + render.rect(getRectangle(), colour.get()); + } + } + + public GuiColourPicker setColourState(ColourState colourState) { + this.colourState = colourState; + return this; + } + + public ColourState getState() { + return colourState; + } + + public SliderState sliderStateAlpha() { + return SliderState.forSlider(() -> (double) colourState.getColour().aF(), e -> colourState.set(colourState.getColour().aF(e.floatValue())), () -> -1D / (Screen.hasShiftDown() ? 16 : 64)); + } + + public SliderState sliderStateRed() { + return SliderState.forSlider(() -> (double) colourState.getColour().rF(), e -> colourState.set(colourState.getColour().rF(e.floatValue())), () -> -1D / (Screen.hasShiftDown() ? 16 : 64)); + } + + public SliderState sliderStateGreen() { + return SliderState.forSlider(() -> (double) colourState.getColour().gF(), e -> colourState.set(colourState.getColour().gF(e.floatValue())), () -> -1D / (Screen.hasShiftDown() ? 16 : 64)); + } + + public SliderState sliderStateBlue() { + return SliderState.forSlider(() -> (double) colourState.getColour().bF(), e -> colourState.set(colourState.getColour().bF(e.floatValue())), () -> -1D / (Screen.hasShiftDown() ? 16 : 64)); + } + + public TextState getTextState() { + return TextState.create(colourState::getHexColour, colourState::setHexColour); + } + + public GuiButton getOkButton() { + return okButton; + } + + /** + * If cancel button is disabled, ok button will automatically resize. + */ + public GuiButton getCancelButton() { + return cancelButton; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiContextMenu.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiContextMenu.java new file mode 100644 index 00000000..77c2b343 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiContextMenu.java @@ -0,0 +1,135 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.ModularGui; +import codechicken.lib.gui.modular.lib.Constraints; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import net.covers1624.quack.collection.FastStream; +import net.minecraft.network.chat.Component; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.function.BiFunction; +import java.util.function.Supplier; + +import static codechicken.lib.gui.modular.lib.geometry.Constraint.*; +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * Context menus get added to the root element when they are created so as long as no new elements are added after the menu is opened, + * the menu will always be on top. + * It will also automatically close when an option is selected, or when the user clicks outside the context menu. + *

+ * Created by brandon3055 on 21/11/2023 + */ +public class GuiContextMenu extends GuiElement { + + private BiFunction, GuiButton> buttonBuilder = (menu, label) -> GuiButton.flatColourButton(menu, label, hover -> hover ? 0xFF475b6a : 0xFF151515).constrain(HEIGHT, literal(12)); + private final Map, Runnable> options = new HashMap<>(); + private final Map, Supplier>> tooltips = new HashMap<>(); + private final List buttons = new ArrayList<>(); + private boolean closeOnItemClicked = true; + private boolean closeOnOutsideClick = true; + + public GuiContextMenu(ModularGui gui) { + super(gui.getRoot()); + } + + public static GuiContextMenu tooltipStyleMenu(GuiParent parent) { + GuiContextMenu menu = new GuiContextMenu(parent.getModularGui()); + Constraints.bind(GuiRectangle.toolTipBackground(menu), menu); + return menu; + } + + public GuiContextMenu setCloseOnItemClicked(boolean closeOnItemClicked) { + this.closeOnItemClicked = closeOnItemClicked; + return this; + } + + public GuiContextMenu setCloseOnOutsideClick(boolean closeOnOutsideClick) { + this.closeOnOutsideClick = closeOnOutsideClick; + return this; + } + + /** + * Only height should be constrained, with will be set automatically to accommodate the provided label. + */ + public GuiContextMenu setButtonBuilder(BiFunction, GuiButton> buttonBuilder) { + this.buttonBuilder = buttonBuilder; + return this; + } + + public GuiContextMenu addOption(Supplier label, Runnable action) { + options.put(label, action); + rebuildButtons(); + return this; + } + + public GuiContextMenu addOption(Supplier label, Supplier> tooltip, Runnable action) { + options.put(label, action); + tooltips.put(label, tooltip); + rebuildButtons(); + return this; + } + + public GuiContextMenu addOption(Supplier label, List tooltip, Runnable action) { + return addOption(label, () -> tooltip, action); + } + + public GuiContextMenu addOption(Supplier label, Runnable action, Component... tooltip) { + return addOption(label, () -> List.of(tooltip), action); + } + + private void rebuildButtons() { + buttons.forEach(this::removeChild); + buttons.clear(); + + //Menu options can be dynamic so the width constraint needs to be dynamic. + //This is probably a little expensive, but its only while a context menu is open. + constrain(WIDTH, dynamic(() -> FastStream.of(options.keySet()).map(Supplier::get).intSum(font()::width) + 6D + 4D)); + + double height = 3; + for (Supplier label : options.keySet()) { + Runnable action = options.get(label); + GuiButton button = buttonBuilder.apply(this, label) + .onPress(action) + .constrain(TOP, relative(get(TOP), height)) + .constrain(LEFT, relative(get(LEFT), 3)) + .constrain(RIGHT, relative(get(RIGHT), -3)); + if (tooltips.containsKey(label)) { + button.setTooltip(tooltips.get(label)); + } + button.getLabel().setScroll(false); + buttons.add(button); + height += button.ySize(); + } + constrain(HEIGHT, literal(height + 3)); + } + + @Override + public boolean mouseClicked(double mouseX, double mouseY, int button, boolean consumed) { + consumed = super.mouseClicked(mouseX, mouseY, button, consumed); + if (isMouseOver() || consumed) { + if (consumed && closeOnItemClicked) { + close(); + } + return true; + } else if (closeOnOutsideClick) { + close(); + return true; + } + + return consumed; + } + + public void close() { + getParent().removeChild(this); + } + + public GuiContextMenu setNormalizedPos(double x, double y) { + constrain(LEFT, dynamic(() -> Math.min(Math.max(x, 0), scaledScreenWidth() - xSize()))); + constrain(TOP, dynamic(() -> Math.min(Math.max(y, 0), scaledScreenHeight() - ySize()))); + return this; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiDVD.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiDVD.java new file mode 100644 index 00000000..a3195c62 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiDVD.java @@ -0,0 +1,97 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.ContentElement; +import codechicken.lib.math.MathHelper; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.geometry.Constraint; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.gui.modular.lib.geometry.Rectangle; +import org.jetbrains.annotations.NotNull; +import org.joml.Vector2d; + +import java.util.Random; +import java.util.function.Consumer; + +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * Just for fun! + * Created by brandon3055 on 10/09/2023 + */ +public class GuiDVD extends GuiElement implements ContentElement> { + private static final Random randy = new Random(); + + private final GuiElement movingElement; + private double xOffset = 0; + private double yOffset = 0; + private Vector2d velocity = null; + private int bounce = 0; + private Consumer onBounce = bounce -> { + }; + + public GuiDVD(@NotNull GuiParent parent) { + super(parent); + this.movingElement = new GuiElement<>(this) + .constrain(TOP, Constraint.relative(get(TOP), () -> yOffset)) + .constrain(LEFT, Constraint.relative(get(LEFT), () -> xOffset)) + .constrain(WIDTH, Constraint.match(get(WIDTH))) + .constrain(HEIGHT, Constraint.match(get(HEIGHT))); + } + + @Override + public GuiElement getContentElement() { + return movingElement; + } + + public void start() { + if (velocity == null) { + velocity = new Vector2d(randy.nextBoolean() ? 1 : -1, randy.nextBoolean() ? 1 : -1); + velocity.normalize(); + } else { + velocity = null; + xOffset = yOffset = 0; + bounce = 0; + } + } + + public void onBounce(Consumer onBounce) { + this.onBounce = onBounce; + } + + @Override + public void tick(double mouseX, double mouseY) { + super.tick(mouseX, mouseY); + } + + @Override + public void render(GuiRender render, double mouseX, double mouseY, float partialTicks) { + super.render(render, mouseX, mouseY, partialTicks); + if (velocity == null) return; + + double speed = 5 * partialTicks; + xOffset += velocity.x * speed; + yOffset += velocity.y * speed; + Rectangle rect = movingElement.getRectangle(); + + int bounces = 0; + if ((velocity.y < 0 && rect.y() < 0) || (velocity.y > 0 && rect.yMax() > scaledScreenHeight())) { + velocity.y *= -1; + onBounce.accept(bounce++); + bounces++; + } + + if ((velocity.x < 0 && rect.x() < 0) || (velocity.x > 0 && rect.xMax() > scaledScreenWidth())) { + velocity.x *= -1; + onBounce.accept(bounce++); + bounces++; + } + + if (bounce > 0) { + velocity.y += -0.05 + (randy.nextGaussian() * 0.1); + velocity.x += -0.05 + (randy.nextGaussian() * 0.1); + velocity.x = velocity.x > 0 ? MathHelper.clip(velocity.x, 0.4, 0.6) : MathHelper.clip(velocity.x, -0.4, -0.6); + velocity.y = velocity.y > 0 ? MathHelper.clip(velocity.y, 0.4, 0.6) : MathHelper.clip(velocity.y, -0.4, -0.6); + velocity.normalize(); + } + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiDialog.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiDialog.java new file mode 100644 index 00000000..064e1f9f --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiDialog.java @@ -0,0 +1,241 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.ModularGui; +import codechicken.lib.gui.modular.lib.Constraints; +import codechicken.lib.gui.modular.lib.geometry.Constraint; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import net.covers1624.quack.collection.FastStream; +import net.minecraft.network.chat.Component; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.HashMap; +import java.util.Map; +import java.util.function.BiFunction; +import java.util.function.Function; + +import static codechicken.lib.gui.modular.lib.geometry.Constraint.*; +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * This class is designed to assist with the easy creation of a number of standard dialog windows. + *

+ * Created by brandon3055 on 14/12/2023 + */ +public class GuiDialog extends GuiElement { + + private boolean blockKeyInput = true; + private boolean blockMouseInput = true; + + protected GuiDialog(@NotNull GuiParent parent) { + super(parent); + } + + /** + * Option dialog builder with pre-configured background and button builders. + * + * @param parent Can be any gui element (Will just be used to get the root element) + * @param title Sets a separate title that will be displayed above the main dialog text. (Optional) + * @param dialogText The main dialog text. + * @param width The dialog width, (Height will automatically adjust based on content.) + * @param options The list of options for this dialog. + */ + public static GuiDialog optionsDialog(@NotNull GuiParent parent, @Nullable Component title, Component dialogText, int width, Option... options) { + return optionsDialog(parent, title, dialogText, GuiRectangle::toolTipBackground, GuiDialog::defaultButton, width, options); + } + + /** + * Option dialog builder with pre-configured background and button builders. + * + * @param parent Can be any gui element (Will just be used to get the root element) + * @param dialogText The main dialog text. + * @param width The dialog width, (Height will automatically adjust based on content.) + * @param options The list of options for this dialog. + */ + public static GuiDialog optionsDialog(@NotNull GuiParent parent, Component dialogText, int width, Option... options) { + return optionsDialog(parent, null, dialogText, width, options); + } + + /** + * Creates a simple info dialog for displaying information to the user. + * The dialog has a single "Ok" button that will close the dialog + * + * @param parent Can be any gui element (Will just be used to get the root element) + * @param title Sets a separate title that will be displayed above the main dialog text. (Optional) + * @param dialogText The main dialog text. + * @param width The dialog width, (Height will automatically adjust based on content.) + */ + public static GuiDialog infoDialog(@NotNull GuiParent parent, @Nullable Component title, Component dialogText, int width, @Nullable Runnable okAction) { + return optionsDialog(parent, title, dialogText, width, neutral(Component.translatable("gui.ok"), okAction)); + } + + /** + * Creates a simple info dialog for displaying information to the user. + * The dialog has a single "Ok" button that will close the dialog + * + * @param parent Can be any gui element (Will just be used to get the root element) + * @param title Sets a separate title that will be displayed above the main dialog text. (Optional) + * @param dialogText The main dialog text. + * @param width The dialog width, (Height will automatically adjust based on content.) + */ + public static GuiDialog infoDialog(@NotNull GuiParent parent, @Nullable Component title, Component dialogText, int width) { + return infoDialog(parent, title, dialogText, width, null); + } + + /** + * Creates a simple info dialog for displaying information to the user. + * The dialog has a single "Ok" button that will close the dialog + * + * @param parent Can be any gui element (Will just be used to get the root element) + * @param dialogText The main dialog text. + * @param width The dialog width, (Height will automatically adjust based on content.) + */ + public static GuiDialog infoDialog(@NotNull GuiParent parent, Component dialogText, int width) { + return infoDialog(parent, null, dialogText, width); + } + + /** + * Create a green "Primary" button option. + */ + public static Option primary(Component text, @Nullable Runnable action) { + return new Option(text, action, hovered -> hovered ? 0xFF44AA44 : 0xFF118811); + } + + /** + * Create a grey "Neutral" button option. + */ + public static Option neutral(Component text, @Nullable Runnable action) { + return new Option(text, action, hovered -> hovered ? 0xFF909090 : 0xFF505050); + } + + /** + * Create a red "Caution" button option. + */ + public static Option caution(Component text, @Nullable Runnable action) { + return new Option(text, action, hovered -> hovered ? 0xFFAA4444 : 0xFF881111); + } + + /** + * @param blockKeyInput Prevent keyboard inputs from being sent to the rest of the gui while this dialog is open. + * Default: true. + */ + public GuiDialog setBlockKeyInput(boolean blockKeyInput) { + this.blockKeyInput = blockKeyInput; + return this; + } + + /** + * @param blockMouseInput Prevent mouse inputs from being sent to the rest of the gui while this dialog is open. + * Default: true. + */ + public GuiDialog setBlockMouseInput(boolean blockMouseInput) { + this.blockMouseInput = blockMouseInput; + return this; + } + + public void close() { + getParent().removeChild(this); + } + + @Override + public boolean keyPressed(int key, int scancode, int modifiers) { + return blockKeyInput; + } + + @Override + public boolean mouseClicked(double mouseX, double mouseY, int button) { + return blockMouseInput; + } + + /** + * This is the core dialog builder method. + * It takes a title text component and a map of Component>Runnable map that is used to define the options. + * Dialog will automatically be centered on the screen. + * + * @param parent Can be any gui element (Will just be used to get the root element) + * @param title Sets a separate title that will be displayed above the main dialog text. (Optional) + * @param dialogText The main dialog text. + * @param backgroundBuilder A function that is used to create the background of the dialog. + * @param buttonBuilder A function that is used to create the dialog buttons. + * @param width The dialog width, (Height will automatically adjust based on content.) + * @param options The list of options for this dialog. + */ + public static GuiDialog optionsDialog(@NotNull GuiParent parent, @Nullable Component title, Component dialogText, Function> backgroundBuilder, BiFunction buttonBuilder, int width, Option... options) { + if (options.length == 0) throw new IllegalStateException("Can not create gui dialog with no options!"); + ModularGui gui = parent.getModularGui(); + + GuiDialog dialog = new GuiDialog(gui.getRoot()) + .constrain(WIDTH, literal(width)) + .setOpaque(true); + Constraints.bind(backgroundBuilder.apply(dialog), dialog); + + Constraint left = relative(dialog.get(LEFT), 5); + Constraint right = relative(dialog.get(RIGHT), -5); + + if (title != null) { + GuiText titleText = new GuiText(dialog, title) + .setWrap(true) + .constrain(TOP, relative(dialog.get(TOP), 5)) + .constrain(LEFT, left) + .constrain(RIGHT, right) + .autoHeight(); + + GuiText bodyText = new GuiText(dialog, dialogText) + .setWrap(true) + .constrain(TOP, relative(titleText.get(BOTTOM), 5)) + .constrain(LEFT, left) + .constrain(RIGHT, right) + .autoHeight(); + dialog.constrain(HEIGHT, dynamic(() -> 5 + titleText.ySize() + 5 + bodyText.ySize() + 5 + 14 + 5)); + } else { + GuiText bodyText = new GuiText(dialog, dialogText) + .setWrap(true) + .constrain(TOP, relative(dialog.get(TOP), 5)) + .constrain(LEFT, left) + .constrain(RIGHT, right) + .autoHeight(); + dialog.constrain(HEIGHT, dynamic(() -> 5 + bodyText.ySize() + 5 + 14 + 5)); + } + + double totalWidth = FastStream.of(options).doubleSum(e -> dialog.font().width(e.text)); + double pos = 5; + int spacing = 2; + for (Option option : options) { + double fraction = dialog.font().width(option.text) / totalWidth; + double opWidth = (dialog.xSize() - 10 - ((options.length - 1) * spacing)) * fraction; + buttonBuilder.apply(dialog, option) + .constrain(BOTTOM, relative(dialog.get(BOTTOM), -5)) + .constrain(HEIGHT, literal(14)) + .constrain(LEFT, relative(dialog.get(LEFT), pos).precise()) + .constrain(WIDTH, literal(opWidth).precise()); + pos += opWidth + spacing; + } + + dialog.constrain(TOP, midPoint(gui.get(TOP), gui.get(BOTTOM), () -> -(dialog.ySize() / 2D))); + dialog.constrain(LEFT, midPoint(gui.get(LEFT), gui.get(RIGHT), -(width / 2D))); + return dialog; + } + + private static GuiButton defaultButton(GuiDialog dialog, Option option) { + GuiButton button = new GuiButton(dialog); + + GuiRectangle background = new GuiRectangle(button) + .fill(() -> option.colour.apply(button.isMouseOver())); + Constraints.bind(background, button); + + GuiText text = new GuiText(button, option.text()); + button.setLabel(text); + Constraints.bind(text, button, 0, 2, 0, 2); + + button.onPress(() -> { + if (option.action != null) { + option.action.run(); + } + dialog.close(); + }); + + return button; + } + + public record Option(Component text, @Nullable Runnable action, Function colour) {} +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiElement.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiElement.java new file mode 100644 index 00000000..8942e51f --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiElement.java @@ -0,0 +1,491 @@ +package codechicken.lib.gui.modular.elements; + +import com.google.common.collect.Lists; +import com.mojang.blaze3d.vertex.PoseStack; +import codechicken.lib.gui.modular.ModularGui; +import codechicken.lib.gui.modular.lib.*; +import codechicken.lib.gui.modular.lib.geometry.ConstrainedGeometry; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.gui.modular.lib.geometry.Position; +import codechicken.lib.gui.modular.lib.geometry.Rectangle; +import net.covers1624.quack.util.SneakyUtils; +import net.minecraft.client.Minecraft; +import net.minecraft.client.gui.Font; +import net.minecraft.client.gui.GuiGraphics; +import net.minecraft.network.chat.Component; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.security.InvalidParameterException; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; +import java.util.function.Supplier; + +/** + * This is the Base class for all gui elements in Modular Gui Version 3. + *

+ * In v2 this vas a massive monolithic class that had way too much crammed into it. + * The primary goals of v3 are the following: + * - Build a new, Extremely flexible system for handling element geometry, including relative positions, anchoring, etc. + * This was archived using the new Geometry system. For details see {@link GuiParent} and {@link ConstrainedGeometry} + * - Implement a system to properly handle element z offsets. + * This was archived by giving all elements a 'depth' property which defines an elements size on the z axis. + * This is then used to properly layer elements and child elements when they are rendered. + * - Switch everything over to the new RenderType system. (This is mostly handled behind the scenes. You don't need to mess with it when creating a GUI) + * - Consolidate all the various rendering helper methods into one convenient utility class. + * The new {@link GuiGraphics} system showed me a good way to implement this. + * - Reduce the amount of ambiguity when building GUIs. (Whether I succeeded here is up for debate xD) + * - Cut out a lot of random bloat that was never used in v2. + *

+ *

+ * Created by brandon3055 on 04/07/2023 + */ +public class GuiElement> extends ConstrainedGeometry implements ElementEvents, TooltipHandler { + + @NotNull + private GuiParent parent; + + private final List> addedQueue = new ArrayList<>(); + private final List> removeQueue = new ArrayList<>(); + private final List> childElements = new ArrayList<>(); + private final List> childElementsUnmodifiable = Collections.unmodifiableList(childElements); + public boolean initialized = false; + + private Minecraft mc; + private Font font; + private int screenWidth; + private int screenHeight; + + protected int hoverTime = 0; + private int hoverTextDelay = 10; + private boolean isMouseOver = false; + private boolean opaque = false; + private boolean removed = true; + private boolean zStacking = true; + private Supplier enabled = () -> true; + private Supplier enableToolTip = () -> true; + private Supplier> toolTip = null; + private Rectangle renderCull = Rectangle.create(Position.create(0, 0), () -> (double) screenWidth, () -> (double) screenHeight); + + /** + * @param parent parent {@link GuiParent}. + */ + public GuiElement(@NotNull GuiParent parent) { + this.parent = parent; + this.parent.addChild(this); + } + + @NotNull + @Override + public GuiParent getParent() { + return parent; + } + + //=== Child Element Handling ===// + + @Override + public List> getChildren() { + return childElementsUnmodifiable; + } + + /** + * In Modular GUI v3, The add child method is primarily for internal use, + * Child elements are automatically added to their parent on construction. + * + * @param child The child element to be added. + */ + @Override + public void addChild(GuiElement child) { + if (!initialized) throw new IllegalStateException("Attempted to add a child to an element before that element has been initialised!"); + if (child == this) throw new InvalidParameterException("Attempted to add element to itself as a child element."); + if (child.getParent() != this) throw new UnsupportedOperationException("Attempted to add an already initialized element to a different parent element."); + if (removeQueue.contains(child)) { + removeQueue.remove(child); + if (!childElements.contains(child)) { + addedQueue.add(child); + } + child.initElement(this); + } else if (!childElements.contains(child)) { + addedQueue.add(child); + child.initElement(this); + } + } + + protected void applyQueuedChildUpdates() { + if (!removeQueue.isEmpty()) { + childElements.removeAll(removeQueue); + removeQueue.clear(); + } + + if (!addedQueue.isEmpty()) { + childElements.addAll(addedQueue); + addedQueue.clear(); + } + } + + /** + * Called immediately after an element is added to its parent, use to initialize the child element. + */ + public void initElement(GuiParent parent) { + removed = false; + updateScreenData(parent.mc(), parent.font(), parent.scaledScreenWidth(), parent.scaledScreenHeight()); + initialized = true; + } + + @Override + public void adoptChild(GuiElement child) { + child.getParent().removeChild(child); + child.parent = this; + addChild(child); + } + + @Override + public void removeChild(GuiElement child) { + if (childElements.contains(child)) { + child.removed = true; + removeQueue.add(child); + } + addedQueue.remove(child); + } + + @Override + public boolean isDescendantOf(GuiElement ancestor) { + return ancestor == parent || parent.isDescendantOf(ancestor); + } + + //=== Minecraft Properties / Initialisation ===// + //TODO I can probably just pass these calls all the way up to the root parent... + + @Override + public Minecraft mc() { + return mc; + } + + @Override + public Font font() { + return font; + } + + @Override + public int scaledScreenWidth() { + return screenWidth; + } + + @Override + public int scaledScreenHeight() { + return screenHeight; + } + + @Override + public ModularGui getModularGui() { + return getParent().getModularGui(); + } + + @Override + public void onScreenInit(Minecraft mc, Font font, int screenWidth, int screenHeight) { + updateScreenData(mc, font, screenWidth, screenHeight); + super.onScreenInit(mc, font, screenWidth, screenHeight); + } + + protected void updateScreenData(Minecraft mc, Font font, int screenWidth, int screenHeight) { + this.mc = mc; + this.font = font; + this.screenWidth = screenWidth; + this.screenHeight = screenHeight; + } + + //=== Element Status ===// + + public T setEnabled(boolean enabled) { + this.enabled = () -> enabled; + return SneakyUtils.unsafeCast(this); + } + + public T setEnabled(@Nullable Supplier enabled) { + this.enabled = enabled; + return SneakyUtils.unsafeCast(this); + } + + public boolean isEnabled() { + return !removed && enabled.get(); + } + + public boolean isRemoved() { + return removed; + } + + public T setEnableToolTip(Supplier enableToolTip) { + this.enableToolTip = enableToolTip; + return SneakyUtils.unsafeCast(this); + } + + @Override + public boolean blockMouseOver(GuiElement element, double mouseX, double mouseY) { + return getParent().blockMouseOver(element, mouseX, mouseY); + } + + @Override + public boolean blockMouseEvents() { + return isMouseOver() && isOpaque(); + } + + /** + * @return True if the cursor is within the bounds of this element, and there is no opaque element above this one obstructing the cursor. + */ + public boolean isMouseOver() { + return isMouseOver; + } + + public boolean isOpaque() { + return opaque; + } + + /** + * If an element is marked as opaque it will consume mouseOver updates, thereby preventing elements bellow from accepting mouseOver input. + * Also prevents mouse events within this element from being passed to elements bellow. + */ + public T setOpaque(boolean opaque) { + this.opaque = opaque; + return SneakyUtils.unsafeCast(this); + } + + /** + * @return the amount of time the cursor has spent inside this element's bounds, + * resets to zero when the cursor leaves this element's bounds. + */ + public int hoverTime() { + return hoverTime; + } + + @Override + public String toString() { + return getClass().getSimpleName() + "{" + + "geometry=" + getRectangle() + + '}'; + } + + /** + * Add this element to the list of jei exclusions. + * Use this for any elements that render outside the normal gui bounds. + * This will ensure JEI does not try to render on top of these elements. + */ + public T jeiExclude() { + getModularGui().jeiExclude(this); + return SneakyUtils.unsafeCast(this); + } + + /** + * Remove this element from the list of jei exclusions. + */ + public T removeJEIExclude() { + getModularGui().removeJEIExclude(this); + return SneakyUtils.unsafeCast(this); + } + + //=== Render / Update ===// + + /** + * Any child elements completely outside this rectangle will not be rendered at all. + * By default, this is set to the screen bounds (meaning the minecraft window) + * Setting this to null will disable culling. + */ + public T setRenderCull(@Nullable Rectangle renderCull) { + this.renderCull = renderCull; + return SneakyUtils.unsafeCast(this); + } + + /** + * Allows you to disable child z-stacking, Meaning all child elements will be rendered at the same z-level + * rather than being stacked. (Not Recursive, children their sub elements with stacking) + *

+ * This can be useful when rendering a lot of high z depth elements such as ItemStacks. + * As long as you know for sure none of the elements intersect, it should be safe to disable stacking. + * + * @param zStacking Enable z stacking (default true) + */ + public T setZStacking(boolean zStacking) { + this.zStacking = zStacking; + return SneakyUtils.unsafeCast(this); + } + + public boolean zStacking() { + return zStacking; + } + + /** + * Returns the depth of this element plus all of its children (recursively) + * Note: You should almost never need to override this! Depth of background and / or foreground content + * should be specified via {@link BackgroundRender#getBackgroundDepth()} and {@link ForegroundRender#getForegroundDepth()} + * + * @return The depth (z height) of this element plus all of its children. + */ + public double getCombinedElementDepth() { + double depth = 0; + if (this instanceof BackgroundRender bgr) depth += bgr.getBackgroundDepth(); + if (this instanceof ForegroundRender fgr) depth += fgr.getForegroundDepth(); + + double childDepth = 0; + for (GuiElement child : childElements) { + if (!child.isEnabled()) continue; + if (zStacking) { + childDepth += child.getCombinedElementDepth(); + } else { + childDepth = Math.max(childDepth, child.getCombinedElementDepth()); + } + } + + return depth + childDepth; + } + + /** + * This is the main render method that handles rendering this element and any child elements it may have. + * This method almost never needs to be overridden, instead when creating custom elements with custom rendering, + * your element should implement {@link BackgroundRender} and / or {@link ForegroundRender} in or order to implement + * its rendering. + *

+ * Note: After the render is complete, the poseStack's z pos will be offset by the total depth of this element and its children. + * This is intended behavior, + * + * @param render Contains gui context information as well as essential render methods/utils including the PoseStack. + * @param mouseX Current mouse X position + * @param mouseY Current mouse Y position + * @param partialTicks Partial render ticks + */ + public void render(GuiRender render, double mouseX, double mouseY, float partialTicks) { + applyQueuedChildUpdates(); + if (this instanceof BackgroundRender bgr) { + double depth = bgr.getBackgroundDepth(); + bgr.renderBackground(render, mouseX, mouseY, partialTicks); + if (depth > 0) { + render.pose().translate(0, 0, depth); + } + } + + double maxDepth = 0; + for (GuiElement child : childElements) { + if (child.isEnabled()) { + boolean rendered = renderChild(child, render, mouseX, mouseY, partialTicks); + //If z-stacking is disabled, we need to undo the z offset that was applied by the child element. + if (!zStacking && rendered) { + double depth = child.getCombinedElementDepth(); + maxDepth = Math.max(maxDepth, depth); + render.pose().translate(0, 0, -depth); + } + } + } + + if (!zStacking) { + //Now we need to apply the z offset of the tallest child. + render.pose().translate(0, 0, maxDepth); + } + + if (this instanceof ForegroundRender fgr) { + double depth = fgr.getForegroundDepth(); + fgr.renderForeground(render, mouseX, mouseY, partialTicks); + if (depth > 0) { + render.pose().translate(0, 0, depth); + } + } + } + + protected boolean renderChild(GuiElement child, GuiRender render, double mouseX, double mouseY, float partialTicks) { + if (renderCull != null && !renderCull.intersects(child.getRectangle())) return false; + child.render(render, mouseX, mouseY, partialTicks); + return true; + } + + /** + * Used to render overlay's such as hover text. Anything rendered in this method will be rendered on top of everything else on the screen. + * Only one overlay should be rendered at a time, When an element renders content via the overlay method it must return true to indicate the render call has been 'consumed' + * If the render call has already been consumed (Check via the consumed boolean) then this element should avoid rendering its overlay. + *

+ * When rendering overlay content, always use the {@link PoseStack} available via the provided {@link GuiRender} + * This stack will already have the correct Z translation to ensure the overlay renders above everything else on the screen. + *

+ * To check if the cursor is over this element, use 'render.hoveredElement() == this' + * {@link #isMouseOver()} Will also work, but may be problematic when multiple, stacked elements have overlay content. + * + * @param render Contains gui context information as well as essential render methods/utils including the PoseStack. + * @param mouseX Current mouse X position + * @param mouseY Current mouse Y position + * @param partialTicks Partial render ticks + * @param consumed Will be true if the overlay render call has already been consumed by another element. + * @return true if the render call has been consumed. + */ + public boolean renderOverlay(GuiRender render, double mouseX, double mouseY, float partialTicks, boolean consumed) { + for (GuiElement child : Lists.reverse(getChildren())) { + if (child.isEnabled()) { + consumed |= child.renderOverlay(render, mouseX, mouseY, partialTicks, consumed); + } + } + return consumed || (showToolTip() && renderTooltip(render, mouseX, mouseY)); + } + + private boolean showToolTip() { + return isMouseOver() && enableToolTip.get() && hoverTime() >= getTooltipDelay(); + } + + /** + * Called every tick to update the element. Note this is called regardless of weather or not the element is actually enabled. + * + * @param mouseX Current mouse X position + * @param mouseY Current mouse Y position + */ + public void tick(double mouseX, double mouseY) { + if (isMouseOver()) { + hoverTime++; + } else { + hoverTime = 0; + } + + for (GuiElement childElement : childElements) { + childElement.tick(mouseX, mouseY); + } + } + + /** + * Called at the start of each tick to update the 'mouseOver' state of each element. + * If the cursor is over an element that is marked as opaque, the update will be consumed. + * This ensures no elements below the opaque element will have their mouseOver flag set to true. + * + * @param mouseX Mouse X position + * @param mouseY Mouse Y position + * @param consumed True if mouseover event has been consumed. + * @return true if this event has been consumed. + */ + public boolean updateMouseOver(double mouseX, double mouseY, boolean consumed) { + for (GuiElement child : Lists.reverse(getChildren())) { + if (child.isEnabled()) { + consumed |= child.updateMouseOver(mouseX, mouseY, consumed); + } + } + + isMouseOver = !consumed && GuiRender.isInRect(xMin(), yMin(), xSize(), ySize(), mouseX, mouseY) && !blockMouseOver(this, mouseX, mouseY); + return consumed || (isMouseOver && isOpaque()); + } + + //=== Hover Text ===// + + @Override + public Supplier> getTooltip() { + return toolTip; + } + + @Override + public T setTooltipDelay(int tooltipDelay) { + this.hoverTextDelay = tooltipDelay; + return SneakyUtils.unsafeCast(this); + } + + @Override + public int getTooltipDelay() { + return hoverTextDelay; + } + + @Override + public T setTooltip(@Nullable Supplier> tooltip) { + this.toolTip = tooltip; + return SneakyUtils.unsafeCast(this); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiEnergyBar.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiEnergyBar.java new file mode 100644 index 00000000..478e86a8 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiEnergyBar.java @@ -0,0 +1,143 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.BackgroundRender; +import codechicken.lib.gui.modular.lib.Constraints; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.gui.modular.sprite.CCGuiTextures; +import codechicken.lib.gui.modular.sprite.Material; +import codechicken.lib.util.FormatUtil; +import net.minecraft.client.gui.screens.Screen; +import net.minecraft.network.chat.Component; +import org.jetbrains.annotations.NotNull; + +import java.text.DecimalFormat; +import java.text.DecimalFormatSymbols; +import java.util.ArrayList; +import java.util.List; +import java.util.Locale; +import java.util.function.BiFunction; +import java.util.function.Supplier; + +import static net.minecraft.ChatFormatting.*; + +/** + * Created by brandon3055 on 10/09/2023 + */ +public class GuiEnergyBar extends GuiElement implements BackgroundRender { + public static final DecimalFormat COMMA_FORMAT = new DecimalFormat("###,###,###,###,###", DecimalFormatSymbols.getInstance(Locale.ROOT)); + public static final Material EMPTY = CCGuiTextures.getUncached("widgets/energy_empty"); + public static final Material FULL = CCGuiTextures.getUncached("widgets/energy_full"); + + private Supplier energy = () -> 0L; + private Supplier capacity = () -> 0L; + private Material emptyTexture = EMPTY; + private Material fullTexture = FULL; + private BiFunction> toolTipFormatter; + + public GuiEnergyBar(@NotNull GuiParent parent) { + super(parent); + setTooltipDelay(0); + setToolTipFormatter(defaultFormatter()); + } + + /** + * Creates a simple energy bar using a simple slot as a background to make it look nice. + */ + public static EnergyBar simpleBar(@NotNull GuiParent parent) { + GuiRectangle container = GuiRectangle.vanillaSlot(parent); + GuiEnergyBar energyBar = new GuiEnergyBar(container); + Constraints.bind(energyBar, container, 1); + return new EnergyBar(container, energyBar); + } + + public static BiFunction> defaultFormatter() { + return (energy, capacity) -> { + List tooltip = new ArrayList<>(); + tooltip.add(Component.translatable("energy_bar.polylib.energy_storage").withStyle(DARK_AQUA)); + boolean shift = Screen.hasShiftDown(); + tooltip.add(Component.translatable("energy_bar.polylib.capacity") + .withStyle(GOLD) + .append(" ") + .append(Component.literal(shift ? FormatUtil.addCommas(capacity) : FormatUtil.formatNumber(capacity)) + .withStyle(GRAY) + .append(" ") + .append(Component.translatable("energy_bar.polylib.rf") + .withStyle(GRAY) + ) + ) + ); + tooltip.add(Component.translatable("energy_bar.polylib.stored") + .withStyle(GOLD) + .append(" ") + .append(Component.literal(shift ? FormatUtil.addCommas(energy) : FormatUtil.formatNumber(energy)) + .withStyle(GRAY) + ) + .append(" ") + .append(Component.translatable("energy_bar.polylib.rf") + .withStyle(GRAY) + ) + .append(Component.literal(String.format(" (%.2f%%)", ((double) energy / (double) capacity) * 100D)) + .withStyle(GRAY) + ) + ); + return tooltip; + }; + } + + public GuiEnergyBar setEmptyTexture(Material emptyTexture) { + this.emptyTexture = emptyTexture; + return this; + } + + public GuiEnergyBar setFullTexture(Material fullTexture) { + this.fullTexture = fullTexture; + return this; + } + + public GuiEnergyBar setCapacity(long capacity) { + return setCapacity(() -> capacity); + } + + public GuiEnergyBar setCapacity(Supplier capacity) { + this.capacity = capacity; + return this; + } + + public GuiEnergyBar setEnergy(long energy) { + return setEnergy(() -> energy); + } + + public GuiEnergyBar setEnergy(Supplier energy) { + this.energy = energy; + return this; + } + + public long getEnergy() { + return energy.get(); + } + + public long getCapacity() { + return capacity.get(); + } + + /** + * Install a custom formatter to control how the energy tool tip renders. + */ + public GuiEnergyBar setToolTipFormatter(BiFunction> toolTipFormatter) { + this.toolTipFormatter = toolTipFormatter; + setTooltip(() -> this.toolTipFormatter.apply(getEnergy(), getCapacity())); + return this; + } + + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + float p = 1 / 128F; + float height = getCapacity() <= 0 ? 0 : (float) ySize() * (getEnergy() / (float) getCapacity()); + float texHeight = height * p; + render.partialSprite(EMPTY.renderType(GuiRender::texColType), xMin(), yMin(), xMax(), yMax(), EMPTY.sprite(), 0F, 1F - (p * (float) ySize()), p * (float) xSize(), 1F, 0xFFFFFFFF); + render.partialSprite(FULL.renderType(GuiRender::texColType), xMin(), yMin() + (ySize() - height), xMax(), yMax(), FULL.sprite(), 0F, 1F - texHeight, p * (float) xSize(), 1F, 0xFFFFFFFF); + } + + public record EnergyBar(GuiRectangle container, GuiEnergyBar bar) {} +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiEntityRenderer.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiEntityRenderer.java new file mode 100644 index 00000000..ac09f9b3 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiEntityRenderer.java @@ -0,0 +1,264 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.BackgroundRender; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.gui.modular.lib.geometry.Rectangle; +import codechicken.lib.render.CCRenderEventHandler; +import com.mojang.blaze3d.platform.Lighting; +import com.mojang.blaze3d.systems.RenderSystem; +import com.mojang.blaze3d.vertex.PoseStack; +import com.mojang.math.Axis; +import net.minecraft.client.Minecraft; +import net.minecraft.client.gui.GuiGraphics; +import net.minecraft.client.gui.screens.inventory.InventoryScreen; +import net.minecraft.client.player.RemotePlayer; +import net.minecraft.client.renderer.MultiBufferSource; +import net.minecraft.client.renderer.entity.EntityRenderDispatcher; +import net.minecraft.resources.ResourceLocation; +import net.minecraft.world.entity.Entity; +import net.minecraft.world.entity.EntityType; +import net.minecraft.world.entity.LivingEntity; +import net.minecraftforge.registries.ForgeRegistries; +import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; +import org.joml.Matrix4f; +import org.joml.Quaternionf; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.function.Supplier; + +/** + * Created by brandon3055 on 15/11/2023 + */ +public class GuiEntityRenderer extends GuiElement implements BackgroundRender { + public static final Logger LOGGER = LogManager.getLogger(); + private static final Map entityCache = new HashMap<>(); + private static final List invalidEntities = new ArrayList<>(); + + private Supplier rotationSpeed = () -> 1F; + private Supplier lockedRotation = () -> 0F; + private Entity entity; + private ResourceLocation entityName; + private boolean invalidEntity = false; + private Supplier rotationLocked = () -> false; + private Supplier trackMouse = () -> false; + private Supplier drawName = () -> false; + public boolean force2dSize = false; + + public GuiEntityRenderer(@NotNull GuiParent parent) { + super(parent); + } + + public GuiEntityRenderer setEntity(Entity entity) { + this.entity = entity; + if (this.entity == null) { + invalidEntity = true; + return this; + } + + this.entityName = ForgeRegistries.ENTITY_TYPES.getKey(entity.getType()); + invalidEntity = invalidEntities.contains(entityName); + return this; + } + + public GuiEntityRenderer setEntity(ResourceLocation entity) { + this.entityName = entity; + this.entity = entityCache.computeIfAbsent(entity, resourceLocation -> { + EntityType type = ForgeRegistries.ENTITY_TYPES.getValue(entity); + return type == null ? null : type.create(mc().level); + }); + + invalidEntity = this.entity == null; + if (invalidEntities.contains(entityName)) { + invalidEntity = true; + } + + return this; + } + + public GuiEntityRenderer setRotationSpeed(float rotationSpeed) { + this.rotationSpeed = () -> rotationSpeed; + return this; + } + + public GuiEntityRenderer setRotationSpeed(Supplier rotationSpeed) { + this.rotationSpeed = rotationSpeed; + return this; + } + + public float getRotationSpeed() { + return rotationSpeed.get(); + } + + public GuiEntityRenderer setLockedRotation(float lockedRotation) { + this.lockedRotation = () -> lockedRotation; + return this; + } + + public GuiEntityRenderer setLockedRotation(Supplier lockedRotation) { + this.lockedRotation = lockedRotation; + return this; + } + + public float getLockedRotation() { + return lockedRotation.get(); + } + + public GuiEntityRenderer setRotationLocked(boolean rotationLocked) { + this.rotationLocked = () -> rotationLocked; + return this; + } + + public GuiEntityRenderer setRotationLocked(Supplier rotationLocked) { + this.rotationLocked = rotationLocked; + return this; + } + + public boolean isRotationLocked() { + return rotationLocked.get(); + } + + public GuiEntityRenderer setTrackMouse(boolean trackMouse) { + this.trackMouse = () -> trackMouse; + return this; + } + + public GuiEntityRenderer setTrackMouse(Supplier trackMouse) { + this.trackMouse = trackMouse; + return this; + } + + public boolean isTrackMouse() { + return trackMouse.get(); + } + + public GuiEntityRenderer setDrawName(boolean drawName) { + this.drawName = () -> drawName; + return this; + } + + public GuiEntityRenderer setDrawName(Supplier drawName) { + this.drawName = drawName; + return this; + } + + public boolean isDrawName() { + return drawName.get(); + } + + public GuiEntityRenderer setForce2dSize(boolean force2dSize) { + this.force2dSize = force2dSize; + return this; + } + + @Override + public double getBackgroundDepth() { + Rectangle rect = getRectangle(); + float scale = (float) (force2dSize ? (Math.min(rect.height() / entity.getBbHeight(), rect.width() / entity.getBbWidth())) : rect.height() / entity.getBbHeight()); + return scale * 2; + } + + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + if (invalidEntity) return; + + try { + if (entity != null) { + Rectangle rect = getRectangle(); + float scale = (float) (force2dSize ? (Math.min(rect.height() / entity.getBbHeight(), rect.width() / entity.getBbWidth())) : rect.height() / entity.getBbHeight()); + float xPos = (float) (rect.x() + (rect.width() / 2D)); + float yPos = (float) ((yMin() + (ySize() / 2)) + (rect.height() / 2)); + float rotation = rotationLocked.get() ? lockedRotation.get() : (CCRenderEventHandler.renderTime + partialTicks) * rotationSpeed.get(); + if (entity instanceof LivingEntity living) { + int eyeOffset = (int) ((entity.getEyeHeight()) * scale); + if (trackMouse.get()) { + renderEntityInInventoryFollowsMouse(render, xPos, yPos, scale, xPos - (float) mouseX, yPos - (float) mouseY - eyeOffset, living); + } else { + renderEntityInInventoryWithRotation(render, xPos, yPos, scale, rotation, living); + } + } + } + } catch (Throwable e) { + invalidEntity = true; + invalidEntities.add(entityName); + LOGGER.error("Failed to render entity in GUI. This is not a bug there are just some entities that can not be rendered like this."); + LOGGER.error("Entity: " + entity, e); + } + } + + public static void renderEntityInInventoryFollowsMouse(GuiRender render, double pX, double pY, double pScale, float offsetX, float offsetY, LivingEntity pEntity) { + float xAngle = (float)Math.atan(offsetX / 40.0F); + float yAngle = (float)Math.atan(offsetY / 40.0F); + renderEntityInInventoryFollowsAngle(render, pX, pY, pScale, xAngle, yAngle, pEntity); + } + + public static void renderEntityInInventoryFollowsAngle(GuiRender render, double pX, double pY, double pScale, float angleX, float angleY, LivingEntity pEntity) { + Quaternionf quaternionf = (new Quaternionf()).rotateZ((float)Math.PI); + Quaternionf quaternionf1 = (new Quaternionf()).rotateX(angleY * 20.0F * ((float)Math.PI / 180F)); + quaternionf.mul(quaternionf1); + float f2 = pEntity.yBodyRot; + float f3 = pEntity.getYRot(); + float f4 = pEntity.getXRot(); + float f5 = pEntity.yHeadRotO; + float f6 = pEntity.yHeadRot; + pEntity.yBodyRot = 180.0F + angleX * 20.0F; + pEntity.setYRot(180.0F + angleX * 40.0F); + pEntity.setXRot(-angleY * 20.0F); + pEntity.yHeadRot = pEntity.getYRot(); + pEntity.yHeadRotO = pEntity.getYRot(); + renderEntityInInventory(render, pX, pY, pScale, quaternionf, quaternionf1, pEntity); + pEntity.yBodyRot = f2; + pEntity.setYRot(f3); + pEntity.setXRot(f4); + pEntity.yHeadRotO = f5; + pEntity.yHeadRot = f6; + } + + public static void renderEntityInInventoryWithRotation(GuiRender render, double xPos, double yPos, double scale, double rotation, LivingEntity living) { + Quaternionf quaternionf = new Quaternionf().rotateZ((float)Math.PI); + Quaternionf quaternionf1 = Axis.YP.rotationDegrees((float) rotation); + quaternionf.mul(quaternionf1); + float f2 = living.yBodyRot; + float f3 = living.getYRot(); + float f4 = living.getXRot(); + float f5 = living.yHeadRotO; + float f6 = living.yHeadRot; + living.yBodyRot = 180.0F; + living.setYRot(180.0F); + living.setXRot(0); + living.yHeadRot = living.getYRot(); + living.yHeadRotO = living.getYRot(); + renderEntityInInventory(render, xPos, yPos, scale, quaternionf, quaternionf1, living); + living.yBodyRot = f2; + living.setYRot(f3); + living.setXRot(f4); + living.yHeadRotO = f5; + living.yHeadRot = f6; + } + + public static void renderEntityInInventory(GuiRender render, double pX, double pY, double pScale, Quaternionf quat, @Nullable Quaternionf pCameraOrientation, LivingEntity pEntity) { + render.pose().pushPose(); + render.pose().translate(pX, pY, 50.0D); + render.pose().mulPoseMatrix((new Matrix4f()).scaling((float)pScale, (float)pScale, (float)(-pScale))); + render.pose().mulPose(quat); + Lighting.setupForEntityInInventory(); + EntityRenderDispatcher entityrenderdispatcher = Minecraft.getInstance().getEntityRenderDispatcher(); + if (pCameraOrientation != null) { + pCameraOrientation.conjugate(); + entityrenderdispatcher.overrideCameraOrientation(pCameraOrientation); + } + + entityrenderdispatcher.setRenderShadow(false); + RenderSystem.runAsFancy(() -> entityrenderdispatcher.render(pEntity, 0.0D, 0.0D, 0.0D, 0.0F, 1.0F, render.pose(), render.buffers(), 15728880)); + render.flush(); + entityrenderdispatcher.setRenderShadow(true); + render.pose().popPose(); + Lighting.setupFor3DItems(); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiEventProvider.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiEventProvider.java new file mode 100644 index 00000000..393ba6c3 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiEventProvider.java @@ -0,0 +1,122 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import org.apache.logging.log4j.util.TriConsumer; +import org.jetbrains.annotations.NotNull; + +import java.util.ArrayList; +import java.util.List; +import java.util.function.BiConsumer; + +/** + * Created by brandon3055 on 15/11/2023 + */ +public class GuiEventProvider extends GuiElement { + + private boolean ignoreConsumed = false; + private final List> clickListeners = new ArrayList<>(); + private final List> releaseListeners = new ArrayList<>(); + private final List> movedListeners = new ArrayList<>(); + private final List> scrollListeners = new ArrayList<>(); + private final List> keyPressListeners = new ArrayList<>(); + private final List> keyReleaseListeners = new ArrayList<>(); + private final List> charTypedListeners = new ArrayList<>(); + + public GuiEventProvider(@NotNull GuiParent parent) { + super(parent); + } + + public GuiEventProvider setIgnoreConsumed(boolean ignoreConsumed) { + this.ignoreConsumed = ignoreConsumed; + return this; + } + + public GuiEventProvider onMouseClick(TriConsumer listener) { + clickListeners.add(listener); + return this; + } + + public GuiEventProvider onMouseRelease(TriConsumer listener) { + releaseListeners.add(listener); + return this; + } + + public GuiEventProvider onMouseMove(BiConsumer listener) { + movedListeners.add(listener); + return this; + } + + public GuiEventProvider onScroll(TriConsumer listener) { + scrollListeners.add(listener); + return this; + } + + public GuiEventProvider onKeyPress(TriConsumer listener) { + keyPressListeners.add(listener); + return this; + } + + public GuiEventProvider onKeyRelease(TriConsumer listener) { + keyReleaseListeners.add(listener); + return this; + } + + public GuiEventProvider onCharTyped(BiConsumer listener) { + charTypedListeners.add(listener); + return this; + } + + @Override + public boolean mouseClicked(double mouseX, double mouseY, int button, boolean consumed) { + if (ignoreConsumed || !consumed) { + clickListeners.forEach(e -> e.accept(mouseX, mouseY, button)); + } + return super.mouseClicked(mouseX, mouseY, button, consumed); + } + + @Override + public boolean mouseReleased(double mouseX, double mouseY, int button, boolean consumed) { + if (ignoreConsumed || !consumed) { + releaseListeners.forEach(e -> e.accept(mouseX, mouseY, button)); + } + return super.mouseReleased(mouseX, mouseY, button, consumed); + } + + @Override + public void mouseMoved(double mouseX, double mouseY) { + movedListeners.forEach(e -> e.accept(mouseX, mouseY)); + super.mouseMoved(mouseX, mouseY); + } + + @Override + public boolean mouseScrolled(double mouseX, double mouseY, double scroll, boolean consumed) { + if (ignoreConsumed || !consumed) { + scrollListeners.forEach(e -> e.accept(mouseX, mouseY, scroll)); + } + return super.mouseScrolled(mouseX, mouseY, scroll, consumed); + } + + @Override + public boolean keyPressed(int key, int scancode, int modifiers, boolean consumed) { + if (ignoreConsumed || !consumed) { + keyPressListeners.forEach(e -> e.accept(key, scancode, modifiers)); + } + return super.keyPressed(key, scancode, modifiers, consumed); + } + + @Override + public boolean keyReleased(int key, int scancode, int modifiers, boolean consumed) { + if (ignoreConsumed || !consumed) { + keyReleaseListeners.forEach(e -> e.accept(key, scancode, modifiers)); + } + return super.keyReleased(key, scancode, modifiers, consumed); + } + + @Override + public boolean charTyped(char character, int modifiers, boolean consumed) { + if (ignoreConsumed || !consumed) { + charTypedListeners.forEach(e -> e.accept(character, modifiers)); + } + return super.charTyped(character, modifiers, consumed); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiFluidTank.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiFluidTank.java new file mode 100644 index 00000000..a2d4eb9a --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiFluidTank.java @@ -0,0 +1,252 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.BackgroundRender; +import codechicken.lib.gui.modular.lib.Constraints; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.gui.modular.sprite.CCGuiTextures; +import codechicken.lib.gui.modular.sprite.Material; +import codechicken.lib.util.FormatUtil; +import net.minecraft.client.Minecraft; +import net.minecraft.client.renderer.texture.TextureAtlas; +import net.minecraft.client.renderer.texture.TextureAtlasSprite; +import net.minecraft.network.chat.Component; +import net.minecraft.network.chat.Style; +import net.minecraft.resources.ResourceLocation; +import net.minecraft.world.level.material.Fluid; +import net.minecraft.world.level.material.Fluids; +import net.minecraftforge.client.extensions.common.IClientFluidTypeExtensions; +import net.minecraftforge.fluids.FluidStack; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.ArrayList; +import java.util.List; +import java.util.function.BiFunction; +import java.util.function.Supplier; + +import static net.minecraft.ChatFormatting.*; + +/** + * When implementing this tank, you must specify the tank capacity in mb, + * And then you have two options for specifying the tank contents. + * You can set the fluid and the amount stored, + * Or you can provide a {@link FluidStack} + *

+ * Created by brandon3055 on 11/09/2023 + */ +public class GuiFluidTank extends GuiElement implements BackgroundRender { + //TODO make a better texture, This feels a little too.. cluttered. + public static final Material DEFAULT_WINDOW = CCGuiTextures.getUncached("widgets/tank_window"); + + private int gaugeColour = 0xFF909090; + private boolean drawGauge = true; + private Material window = null; + private Supplier capacity = () -> 10000L; + private Supplier fluidStack = () -> FluidStack.EMPTY; + + private BiFunction> toolTipFormatter; + + public GuiFluidTank(@NotNull GuiParent parent) { + super(parent); + setTooltipDelay(0); + setToolTipFormatter(defaultFormatter()); + } + + /** + * Creates a simple tank using a simple slot as a background to make it look nice. + */ + public static FluidTank simpleTank(@NotNull GuiParent parent) { + GuiRectangle container = GuiRectangle.vanillaSlot(parent); + GuiFluidTank energyBar = new GuiFluidTank(container); + Constraints.bind(energyBar, container, 1); + return new FluidTank(container, energyBar); + } + + /** + * Sets the capacity of this tank in milli-buckets. + */ + public GuiFluidTank setCapacity(long capacity) { + return setCapacity(() -> capacity); + } + + /** + * Supply the capacity of this tank in milli-buckets. + */ + public GuiFluidTank setCapacity(Supplier capacity) { + this.capacity = capacity; + return this; + } + + /** + * Allows you to set the current stored fluid stack. + */ + public GuiFluidTank setFluidStack(FluidStack fluidStack) { + return setFluidStack(() -> fluidStack); + } + + /** + * Allows you to supply the current stored fluid stack. + */ + public GuiFluidTank setFluidStack(Supplier fluidStack) { + this.fluidStack = fluidStack; + return this; + } + + /** + * Install a custom formatter to control how the fluid tool tip renders. + */ + public GuiFluidTank setToolTipFormatter(BiFunction> toolTipFormatter) { + this.toolTipFormatter = toolTipFormatter; + setTooltip(() -> this.toolTipFormatter.apply(getFluidStack(), getCapacity())); + return this; + } + + /** + * Sets the tank window texture, Will be tiled to fit the tank size. + * + * @param window New window texture or null for no window texture. + */ + public GuiFluidTank setWindow(@Nullable Material window) { + this.window = window; + return this; + } + + /** + * Enable the built-in fluid gauge lines. + */ + public GuiFluidTank setDrawGauge(boolean drawGauge) { + this.drawGauge = drawGauge; + return this; + } + + /** + * Sets the colour of the built-in fluid gauge lines + */ + public GuiFluidTank setGaugeColour(int gaugeColour) { + this.gaugeColour = gaugeColour; + return this; + } + + public Long getCapacity() { + return capacity.get(); + } + + public FluidStack getFluidStack() { + return fluidStack.get(); + } + + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + FluidStack stack = getFluidStack(); + Material fluidMat = Material.fromSprite(getStillTexture(stack)); + + if (!stack.isEmpty() && fluidMat != null) { + int fluidColor = getColour(stack); + float height = getCapacity() <= 0 ? 0 : (float) ySize() * (stack.getAmount() / (float) getCapacity()); + render.tileSprite(fluidMat.renderType(GuiRender::texColType), xMin(), yMax() - height, xMax(), yMax(), fluidMat.sprite(), fluidColor); + } + + if (window != null) { + render.tileSprite(window.renderType(GuiRender::texColType), xMin(), yMin(), xMax(), yMax(), window.sprite(), 0xFFFFFFFF); + } + + gaugeColour = 0xFF000000; + if (drawGauge) { + double spacing = computeGaugeSpacing(); + if (spacing == 0) return; + + double pos = spacing; + while (pos + 1 < ySize()) { + double width = xSize() / 4; + double yPos = yMax() - 1 - pos; + render.fill(xMax() - width, yPos, xMax(), yPos + 1, gaugeColour); + pos += spacing; + } + } + } + + private double computeGaugeSpacing() { + double ySize = ySize(); + double capacity = getCapacity(); + if (ySize / (capacity / 100D) > 3) return ySize / (capacity / 100D); + else if (ySize / (capacity / 500D) > 3) return ySize / (capacity / 500D); + else if (ySize / (capacity / 1000D) > 3) return ySize / (capacity / 1000D); + else if (ySize / (capacity / 5000D) > 3) return ySize / (capacity / 5000D); + else if (ySize / (capacity / 10000D) > 3) return ySize / (capacity / 10000D); + else if (ySize / (capacity / 50000D) > 3) return ySize / (capacity / 50000D); + else if (ySize / (capacity / 100000D) > 3) return ySize / (capacity / 100000D); + return 0; + } + + public static BiFunction> defaultFormatter() { + return (fluidStack, capacity) -> { + List tooltip = new ArrayList<>(); + tooltip.add(Component.translatable("fluid_tank.polylib.fluid_storage").withStyle(DARK_AQUA)); + if (!fluidStack.isEmpty()) { + tooltip.add(Component.translatable("fluid_tank.polylib.contains") + .withStyle(GOLD) + .append(" ") + .append(fluidStack.getDisplayName().copy() + .setStyle(Style.EMPTY + .withColor(getColour(fluidStack)) + ) + ) + ); + } + + tooltip.add(Component.translatable("fluid_tank.polylib.capacity") + .withStyle(GOLD) + .append(" ") + .append(Component.literal(FormatUtil.addCommas(capacity)) + .withStyle(GRAY) + .append(" ") + .append(Component.translatable("fluid_tank.polylib.mb") + .withStyle(GRAY) + ) + ) + ); + tooltip.add(Component.translatable("fluid_tank.polylib.stored") + .withStyle(GOLD) + .append(" ") + .append(Component.literal(FormatUtil.addCommas(fluidStack.getAmount())) + .withStyle(GRAY) + ) + .append(" ") + .append(Component.translatable("fluid_tank.polylib.mb") + .withStyle(GRAY) + ) + .append(Component.literal(String.format(" (%.2f%%)", ((double) fluidStack.getAmount() / (double) capacity) * 100D)) + .withStyle(GRAY) + ) + ); + return tooltip; + }; + } + + //TODO These could maybe go in FluidUtils? but they are client side only so... + + public static int getColour(FluidStack fluidStack) { + return fluidStack.getFluid() == Fluids.EMPTY ? -1 : IClientFluidTypeExtensions.of(fluidStack.getFluid()).getTintColor(fluidStack); + } + + public static int getColour(Fluid fluid) { + return fluid == Fluids.EMPTY ? -1 : IClientFluidTypeExtensions.of(fluid).getTintColor(); + } + + @Nullable + public static TextureAtlasSprite getStillTexture(FluidStack stack) { + if (stack.getFluid() == Fluids.EMPTY) return null; + ResourceLocation texture = IClientFluidTypeExtensions.of(stack.getFluid()).getStillTexture(stack); + return Minecraft.getInstance().getTextureAtlas(TextureAtlas.LOCATION_BLOCKS).apply(texture); + } + + @Nullable + public static TextureAtlasSprite getStillTexture(Fluid fluid) { + if (fluid == Fluids.EMPTY) return null; + ResourceLocation texture = IClientFluidTypeExtensions.of(fluid).getStillTexture(); + return Minecraft.getInstance().getTextureAtlas(TextureAtlas.LOCATION_BLOCKS).apply(texture); + } + + public record FluidTank(GuiRectangle container, GuiFluidTank tank) {} +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiItemStack.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiItemStack.java new file mode 100644 index 00000000..2ff85905 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiItemStack.java @@ -0,0 +1,118 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.BackgroundRender; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import net.minecraft.world.item.ItemStack; +import org.jetbrains.annotations.NotNull; + +import java.util.function.Supplier; + +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.HEIGHT; +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.WIDTH; + +/** + * A simple gui element that renders an item stack. + * This width and height of this element should be constrained to the same value, + * The stack size is based on the element size. + * constrain size to 16x16 for the standard gui stack size. + *

+ * Created by brandon3055 on 03/09/2023 + */ +public class GuiItemStack extends GuiElement implements BackgroundRender { + private Supplier stack; + private Supplier decorate = () -> true; + private Supplier toolTip = () -> true; + + public GuiItemStack(@NotNull GuiParent parent) { + this(parent, () -> ItemStack.EMPTY); + } + + public GuiItemStack(@NotNull GuiParent parent, ItemStack itemStack) { + super(parent); + setStack(itemStack); + } + + public GuiItemStack(@NotNull GuiParent parent, Supplier provider) { + super(parent); + setStack(provider); + } + + public GuiItemStack setStack(Supplier stackProvider) { + this.stack = stackProvider; + return this; + } + + public GuiItemStack setStack(ItemStack stack) { + this.stack = () -> stack; + return this; + } + + /** + * Enable item stack decorations. + * Meaning, Damage bar, Stack size, Item cool down, etc. (Default Enabled) + */ + public GuiItemStack enableStackDecoration(boolean enableDecoration) { + return enableStackDecoration(() -> enableDecoration); + } + + /** + * Enable item stack decorations. + * Meaning, Damage bar, Stack size, Item cool down, etc. (Default Enabled) + */ + public GuiItemStack enableStackDecoration(Supplier enableDecoration) { + this.decorate = enableDecoration; + return this; + } + + /** + * Enable the default item stack tooltip. (Default Enabled) + * Note: If the {@link GuiItemStack} element has a tooltip applied via one of the element #setTooltip methods, + * That will override the item stack tool tip. + */ + public GuiItemStack enableStackToolTip(boolean enableToolTip) { + return enableStackToolTip(() -> enableToolTip); + } + + /** + * Enable the default item stack tooltip. (Default Enabled) + * Note: If the {@link GuiItemStack} element has a tooltip applied via one of the element #setTooltip methods, + * That will override the item stack tool tip. + */ + public GuiItemStack enableStackToolTip(Supplier enableToolTip) { + this.toolTip = enableToolTip; + return this; + } + + //=== Internal methods ===// + + public double getStackSize() { + return Math.max(getValue(WIDTH), getValue(HEIGHT)); + } + + @Override + public double getBackgroundDepth() { + return getStackSize(); + } + + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + ItemStack stack = this.stack.get(); + if (stack.isEmpty()) return; + + render.renderItem(stack, xMin(), yMin(), getStackSize(), (int) (xMin() + (xSize() * yMin()))); + if (decorate.get()) { + render.renderItemDecorations(stack, xMin(), yMin(), getStackSize()); + } + } + + @Override + public boolean renderOverlay(GuiRender render, double mouseX, double mouseY, float partialTicks, boolean consumed) { + if (super.renderOverlay(render, mouseX, mouseY, partialTicks, consumed)) return true; + if (isMouseOver() && !stack.get().isEmpty()) { + render.renderTooltip(stack.get(), mouseX, mouseY); + return true; + } + return false; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiList.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiList.java new file mode 100644 index 00000000..909d5d53 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiList.java @@ -0,0 +1,221 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.SliderState; +import codechicken.lib.gui.modular.lib.geometry.*; +import codechicken.lib.math.MathHelper; +import net.minecraft.network.chat.Component; +import org.jetbrains.annotations.NotNull; + +import java.util.*; +import java.util.function.BiFunction; + +import static codechicken.lib.gui.modular.lib.geometry.Constraint.*; +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * GuiList, as the name suggests allows you to display a list of objects. + * The list type can be whatever you want, and you can install a converter to + * map list objects to elements for display. + *

+ * The default converter simply displays the toString() value of the object. + *

+ * Element width will be fixed to the width of the GuiList, element height can be whatever you want. + *

+ * Note on adding child elements to this: + * If any child elements extend beyond the bounds of this element, that part will be culled. + * Also, child elements will always end up bellow the list items when the list is updated. + *

+ * Created by brandon3055 on 21/09/2023 + */ +public class GuiList extends GuiElement> { + + /** + * This is made available primarily for debugging purposes where it can be useful to see what's going on behind the scenes. + */ + public boolean enableScissor = true; + private double yScrollPos = 0; + private double contentHeight = 0; + private double itemSpacing = 1; + private boolean rebuild = true; + private GuiSlider hiddenBar = null; + + private final List listContent = new ArrayList<>(); + private final Map> elementMap = new HashMap<>(); + private final LinkedList> visible = new LinkedList<>(); + + private BiFunction, E, ? extends GuiElement> displayBuilder = (parent, e) -> { + GuiText text = new GuiText(parent, () -> Component.literal(String.valueOf(e))).setWrap(true); + text.constrain(GeoParam.HEIGHT, Constraint.dynamic(() -> (double) font().wordWrapHeight(text.getText(), (int) text.xSize()))); + return text; + }; + + public GuiList(@NotNull GuiParent parent) { + super(parent); + this.setZStacking(false); + this.setRenderCull(getRectangle()); + } + + public boolean add(E e) { + rebuild = true; + return listContent.add(e); + } + + public boolean remove(E e) { + rebuild = true; + return listContent.remove(e); + } + + /** + * You are allowed to modify this list directly, but if you do you must call + * {@link #markDirty()} otherwise the display elements will not get updated. + */ + public List getList() { + return listContent; + } + + public void markDirty() { + this.rebuild = true; + } + + public GuiList setDisplayBuilder(BiFunction, E, ? extends GuiElement> displayBuilder) { + this.displayBuilder = displayBuilder; + return this; + } + + public GuiList setItemSpacing(double itemSpacing) { + this.itemSpacing = itemSpacing; + return this; + } + + public SliderState scrollState() { + return SliderState.forScrollBar(() -> yScrollPos, e -> { + yScrollPos = e; + updateVisible(); + }, () -> MathHelper.clip(ySize() / contentHeight, 0, 1)); + } + + /** + * You can choose to attach a scroll bar to this element the same way you would a {@link GuiScrolling} + * But sometimes you just want to be able to mouse-wheel scroll without an actual scroll bar. + *

+ * This method will add a hidden scroll bar to enable mouse wheel scrolling and middle-click dragging without the need for an actual scroll bar. + * The scroll bar will be an invisible zero width element on the right side of this list. + */ + public GuiList addHiddenScrollBar() { + if (hiddenBar != null) removeChild(hiddenBar); + hiddenBar = new GuiSlider(this, Axis.Y) + .setSliderState(scrollState()) + .setScrollableElement(this) + .constrain(TOP, match(get(TOP))) + .constrain(LEFT, relative(get(RIGHT), -5)) + .constrain(BOTTOM, match(get(BOTTOM))) + .constrain(RIGHT, match(get(RIGHT))); + return this; + } + + public GuiList removeHiddenScrollBar() { + if (hiddenBar != null) removeChild(hiddenBar); + return this; + } + + //=== Internal Logic ===// + + public double hiddenSize() { + return Math.max(contentHeight - ySize(), 0); + } + + @Override + public void tick(double mouseX, double mouseY) { + if (rebuild) { + rebuildElements(); + } + super.tick(mouseX, mouseY); + } + + public void rebuildElements() { + elementMap.values().forEach(this::removeChild); + elementMap.clear(); + + for (E item : listContent) { + GuiElement next = displayBuilder.apply(this, item); + next.constrain(LEFT, match(get(LEFT))); + next.constrain(RIGHT, match(get(RIGHT))); + removeChild(next); + elementMap.put(item, next); + } + rebuild = false; + updateVisible(); + } + + public Map> getElementMap() { + return elementMap; + } + + public void scrollTo(E scrollTo) { + if (rebuild) { + rebuild = false; + rebuildElements(); + } + + if (elementMap.containsKey(scrollTo)) { + scrollState().setPos(0); + double yMax = yMin(); + + for (E item : getList()) { + GuiElement e = elementMap.get(item); + if (e != null) { + yMax += e.ySize() + 1; + if (item.equals(scrollTo)) break; + } + } + + if (yMax > yMax()) { + double move = yMax - yMax(); + scrollState().setPos(move / hiddenSize()); + } + } + } + + private void updateVisible() { + visible.forEach(this::removeChild); + visible.clear(); + contentHeight = 0; + if (listContent.isEmpty()) return; + + for (GuiElement item : elementMap.values()) { + contentHeight += item.ySize() + itemSpacing; + } + contentHeight -= itemSpacing; + + double winTop = yMin(); + double winBottom = yMax(); + + double yPos = winTop + (yScrollPos * -hiddenSize()); + for (E item : listContent) { + GuiElement element = elementMap.get(item); + if (element == null) continue; + double top = yPos; + double bottom = yPos + element.ySize(); + + if ((top >= winTop && top <= winBottom) || (bottom >= winTop && bottom <= winBottom)) { + addChild(element); + visible.add(element); + element.constrain(TOP, literal(top)); + } + yPos = bottom + itemSpacing; + } + } + + @Override + public boolean blockMouseOver(GuiElement element, double mouseX, double mouseY) { + return super.blockMouseOver(element, mouseX, mouseY) || (element.isDescendantOf(this) && !isMouseOver()); + } + + @Override + public void render(GuiRender render, double mouseX, double mouseY, float partialTicks) { + if (enableScissor) render.pushScissorRect(getRectangle()); + super.render(render, mouseX, mouseY, partialTicks); + if (enableScissor) render.popScissor(); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiManipulable.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiManipulable.java new file mode 100644 index 00000000..d95835e4 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiManipulable.java @@ -0,0 +1,401 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.ContentElement; +import codechicken.lib.gui.modular.lib.CursorHelper; +import codechicken.lib.gui.modular.lib.geometry.Constraint; +import codechicken.lib.gui.modular.lib.geometry.GeoParam; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.gui.modular.lib.geometry.Rectangle; +import net.minecraft.client.Minecraft; +import net.minecraft.client.gui.Font; +import net.minecraft.resources.ResourceLocation; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import static codechicken.lib.gui.modular.lib.geometry.Constraint.literal; +import static codechicken.lib.gui.modular.lib.geometry.Constraint.match; +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * This element can be used to create movable/resizable guis of gui elements. + * This is achieved via a "contentElement" to which all child elements should eb attached. + * Initially the bounds of the content element will match the parent {@link GuiManipulable} element. + * However, depending on which features are enabled, it is possible for the user to resize the + * content element by clicking and dragging the edges, or move the element by clicking and dragging + * a specified "dragArea". + *

+ * It should be noted that the constraints on the underlying {@link GuiManipulable} should be fairly rigid. + * Things like dynamic constraint changes will not translate through to the contentElement, + *

+ * If a UI resize occurs the content element's bounds will be reset to default. + * You can also trigger a manual reset by calling {@link #resetBounds()} + *

+ * Created by brandon3055 on 13/11/2023 + */ +public class GuiManipulable extends GuiElement implements ContentElement> { + private final GuiElement contentElement; + + private int dragXOffset = 0; + private int dragYOffset = 0; + private boolean isDragging = false; + private boolean dragPos = false; + private boolean dragTop = false; + private boolean dragLeft = false; + private boolean dragBottom = false; + private boolean dragRight = false; + private boolean enableCursors = false; + + //Made available for external position restraints + public int xMin = 0; + public int xMax = 0; + public int yMin = 0; + public int yMax = 0; + + protected Rectangle minSize = Rectangle.create(0, 0, 50, 50); + protected Rectangle maxSize = Rectangle.create(0, 0, 256, 256); + protected Runnable onMovedCallback = null; + protected Runnable onResizedCallback = null; + protected PositionRestraint positionRestraint = draggable -> { + if (xMin < 0) { + int move = -xMin; + xMin += move; + xMax += move; + } else if (xMax > scaledScreenWidth()) { + int move = xMax - scaledScreenWidth(); + xMin -= move; + xMax -= move; + } + if (yMin < 0) { + int move = -yMin; + yMin += move; + yMax += move; + } else if (yMax > scaledScreenHeight()) { + int move = yMax - scaledScreenHeight(); + yMin -= move; + yMax -= move; + } + }; + + private GuiElement moveHandle; + private GuiElement leftHandle; + private GuiElement rightHandle; + private GuiElement topHandle; + private GuiElement bottomHandle; + + public GuiManipulable(@NotNull GuiParent parent) { + super(parent); + this.contentElement = new GuiElement<>(this) + .constrain(LEFT, Constraint.dynamic(() -> (double) xMin)) + .constrain(RIGHT, Constraint.dynamic(() -> (double) xMax)) + .constrain(TOP, Constraint.dynamic(() -> (double) yMin)) + .constrain(BOTTOM, Constraint.dynamic(() -> (double) yMax)); + moveHandle = new GuiRectangle(contentElement); + leftHandle = new GuiRectangle(contentElement); + rightHandle = new GuiRectangle(contentElement); + topHandle = new GuiRectangle(contentElement); + bottomHandle = new GuiRectangle(contentElement); + } + + public GuiManipulable resetBounds() { + xMin = (int)xMin(); + xMax = (int)xMax(); + yMin = (int)yMin(); + yMax = (int)yMax(); + return this; + } + + @Override + public GuiManipulable constrain(GeoParam param, @Nullable Constraint constraint) { + return super.constrain(param, constraint).resetBounds(); //TODO, This will break if strict constraints are enabled... + } + + @Override + public void onScreenInit(Minecraft mc, Font font, int screenWidth, int screenHeight) { + super.onScreenInit(mc, font, screenWidth, screenHeight); + resetBounds(); + } + + @Override + public GuiElement getContentElement() { + return contentElement; + } + + public GuiManipulable addResizeHandles(int handleSize, boolean includeTopHandle) { + if (includeTopHandle) addTopHandle(handleSize); + addLeftHandle(handleSize); + addRightHandle(handleSize); + addBottomHandle(handleSize); + return this; + } + + public GuiManipulable addTopHandle(int handleSize) { + this.topHandle + .constrain(TOP, match(contentElement.get(TOP))) + .constrain(LEFT, match(contentElement.get(LEFT))) + .constrain(RIGHT, match(contentElement.get(RIGHT))) + .constrain(HEIGHT, literal(handleSize)); + return this; + } + + public GuiManipulable addBottomHandle(int handleSize) { + this.bottomHandle + .constrain(BOTTOM, match(contentElement.get(BOTTOM))) + .constrain(LEFT, match(contentElement.get(LEFT))) + .constrain(RIGHT, match(contentElement.get(RIGHT))) + .constrain(HEIGHT, literal(handleSize)); + return this; + } + + public GuiManipulable addLeftHandle(int handleSize) { + this.leftHandle + .constrain(LEFT, match(contentElement.get(LEFT))) + .constrain(TOP, match(contentElement.get(TOP))) + .constrain(BOTTOM, match(contentElement.get(BOTTOM))) + .constrain(WIDTH, literal(handleSize)); + return this; + } + + public GuiManipulable addRightHandle(int handleSize) { + this.rightHandle + .constrain(RIGHT, match(contentElement.get(RIGHT))) + .constrain(TOP, match(contentElement.get(TOP))) + .constrain(BOTTOM, match(contentElement.get(BOTTOM))) + .constrain(WIDTH, literal(handleSize)); + return this; + } + + public GuiManipulable addMoveHandle(int handleSize) { + this.moveHandle + .constrain(TOP, match(contentElement.get(TOP))) + .constrain(LEFT, match(contentElement.get(LEFT))) + .constrain(RIGHT, match(contentElement.get(RIGHT))) + .constrain(HEIGHT, literal(handleSize)); + return this; + } + + /** + * You can use this to retrieve the current move handle. + * You are free to update the constraints on this handle, but it must be constrained relative to the content element. + */ + public GuiElement getMoveHandle() { + return moveHandle; + } + + /** + * You can use this to retrieve the current left resize handle. + * You are free to update the constraints on this handle, but it must be constrained relative to the content element. + */ + public GuiElement getLeftHandle() { + return leftHandle; + } + + /** + * You can use this to retrieve the current right resize handle. + * You are free to update the constraints on this handle, but it must be constrained relative to the content element. + */ + public GuiElement getRightHandle() { + return rightHandle; + } + + /** + * You can use this to retrieve the current top resize handle. + * You are free to update the constraints on this handle, but it must be constrained relative to the content element. + */ + public GuiElement getTopHandle() { + return topHandle; + } + + /** + * You can use this to retrieve the current bottom resize handle. + * You are free to update the constraints on this handle, but it must be constrained relative to the content element. + */ + public GuiElement getBottomHandle() { + return bottomHandle; + } + + /** + * Enables rendering of custom mouse cursors when hovering over a draggable handle. + */ + public GuiManipulable enableCursors(boolean enableCursors) { + this.enableCursors = enableCursors; + return this; + } + + public GuiManipulable setOnMovedCallback(Runnable onMovedCallback) { + this.onMovedCallback = onMovedCallback; + return this; + } + + public GuiManipulable setOnResizedCallback(Runnable onResizedCallback) { + this.onResizedCallback = onResizedCallback; + return this; + } + + public GuiManipulable setPositionRestraint(PositionRestraint positionRestraint) { + this.positionRestraint = positionRestraint; + return this; + } + + public void setMinSize(Rectangle minSize) { + this.minSize = minSize; + } + + public void setMaxSize(Rectangle maxSize) { + this.maxSize = maxSize; + } + + public Rectangle getMinSize() { + return minSize; + } + + public Rectangle getMaxSize() { + return maxSize; + } + + + @Override + public void tick(double mouseX, double mouseY) { + if (enableCursors) { + boolean posFlag = moveHandle != null && moveHandle.isMouseOver(); + boolean topFlag = topHandle != null && topHandle.isMouseOver(); + boolean leftFlag = leftHandle != null && leftHandle.isMouseOver(); + boolean bottomFlag = bottomHandle != null && bottomHandle.isMouseOver(); + boolean rightFlag = rightHandle != null && rightHandle.isMouseOver(); + boolean any = posFlag || topFlag || leftFlag || bottomFlag || rightFlag; + + if (any) { + if (posFlag) { + getModularGui().setCursor(CursorHelper.DRAG); + } else if ((topFlag && leftFlag) || (bottomFlag && rightFlag)) { + getModularGui().setCursor(CursorHelper.RESIZE_TLBR); + } else if ((topFlag && rightFlag) || (bottomFlag && leftFlag)) { + getModularGui().setCursor(CursorHelper.RESIZE_TRBL); + } else if (topFlag || bottomFlag) { + getModularGui().setCursor(CursorHelper.RESIZE_V); + } else { + getModularGui().setCursor(CursorHelper.RESIZE_H); + } + } + } + + super.tick(mouseX, mouseY); + } + + public void startDragging() { + double mouseX = getModularGui().computeMouseX(); + double mouseY = getModularGui().computeMouseY(); + dragXOffset = (int) (mouseX - xMin); + dragYOffset = (int) (mouseY - yMin); + isDragging = true; + dragPos = true; + } + + @Override + public boolean mouseClicked(double mouseX, double mouseY, int button) { + if (super.mouseClicked(mouseX, mouseY, button)) return true; + + boolean posFlag = moveHandle != null && moveHandle.isMouseOver(); + boolean topFlag = topHandle != null && topHandle.isMouseOver(); + boolean leftFlag = leftHandle != null && leftHandle.isMouseOver(); + boolean bottomFlag = bottomHandle != null && bottomHandle.isMouseOver(); + boolean rightFlag = rightHandle != null && rightHandle.isMouseOver(); + + if (posFlag || topFlag || leftFlag || bottomFlag || rightFlag) { + dragXOffset = (int) (mouseX - xMin); + dragYOffset = (int) (mouseY - yMin); + isDragging = true; + if (posFlag) { + dragPos = true; + } else { + dragTop = topFlag; + dragLeft = leftFlag; + dragBottom = bottomFlag; + dragRight = rightFlag; + } + return true; + } + + return false; + } + + @Override + public void mouseMoved(double mouseX, double mouseY) { + if (isDragging) { + int xMove = (int) (mouseX - dragXOffset) - xMin; + int yMove = (int) (mouseY - dragYOffset) - yMin; + if (dragPos) { + Rectangle previous = Rectangle.create(xMin, yMin, xMax - xMin, yMax - yMin); + xMin += xMove; + xMax += xMove; + yMin += yMove; + yMax += yMove; + validatePosition(); + onMoved(); + } else { + Rectangle min = getMinSize(); + Rectangle max = getMaxSize(); + if (dragTop) { + yMin += yMove; + if (yMax - yMin < min.height()) yMin = yMax - (int) min.height(); + if (yMax - yMin > max.height()) yMin = yMax - (int) max.height(); + if (yMin < 0) yMin = 0; + } + if (dragLeft) { + xMin += xMove; + if (xMax - xMin < min.width()) xMin = xMax - (int) min.width(); + if (xMax - xMin > max.width()) xMin = xMax - (int) max.width(); + if (xMin < 0) xMin = 0; + } + if (dragBottom) { + yMax = yMin + (dragYOffset + yMove); + if (yMax - yMin < min.height()) yMax = yMin + (int) min.height(); + if (yMax - yMin > max.height()) yMax = yMin + (int) max.height(); + if (yMax > scaledScreenHeight()) yMax = scaledScreenHeight(); + } + if (dragRight) { + xMax = xMin + (dragXOffset + xMove); + if (xMax - xMin < min.width()) xMax = xMin + (int) min.width(); + if (xMax - xMin > max.width()) xMax = xMin + (int) max.width(); + if (xMax > scaledScreenWidth()) xMax = scaledScreenWidth(); + } + validatePosition(); + onResized(); + } + } + super.mouseMoved(mouseX, mouseY); + } + + @Override + public boolean mouseReleased(double mouseX, double mouseY, int button, boolean consumed) { + if (isDragging) { + validatePosition(); + } + isDragging = dragPos = dragTop = dragLeft = dragBottom = dragRight = false; + return super.mouseReleased(mouseX, mouseY, button, consumed); + } + + protected void validatePosition() { + double x = xMin; + double y = yMin; + positionRestraint.restrainPosition(this); + if ((x != xMin || y != yMin) && onMovedCallback != null) { + onMovedCallback.run(); + } + } + + protected void onMoved() { + if (onMovedCallback != null) { + onMovedCallback.run(); + } + } + + protected void onResized() { + if (onResizedCallback != null) { + onResizedCallback.run(); + } + } + + public interface PositionRestraint { + void restrainPosition(GuiManipulable draggable); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiProgressIcon.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiProgressIcon.java new file mode 100644 index 00000000..3e56ce61 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiProgressIcon.java @@ -0,0 +1,115 @@ +package codechicken.lib.gui.modular.elements; + + +import codechicken.lib.gui.modular.lib.BackgroundRender; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.geometry.Axis; +import codechicken.lib.gui.modular.lib.geometry.Direction; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.gui.modular.sprite.Material; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.function.Supplier; + +/** + * This can be used to create a simple progress indicator like those used in machines like furnaces. + *

+ * The background texture (if one is used) and the animated texture must be the same shape and size, + * They must be designed so that the animated texture can be rendered directly on top of the background texture with no offset. + * The animated texture should not have any empty space on ether end as the entire width of the texture is used in the animation. + *

+ * Texture must be designed for left to right animation, + *

+ * Created by brandon3055 on 04/09/2023 + */ +public class GuiProgressIcon extends GuiElement implements BackgroundRender { + + private Material background = null; + private Material animated; + private Supplier progress = () -> 0D; + private Direction direction = Direction.RIGHT; + + public GuiProgressIcon(@NotNull GuiParent parent, Material animated) { + super(parent); + this.animated = animated; + } + + public GuiProgressIcon(@NotNull GuiParent parent, Material background, Material animated) { + super(parent); + this.background = background; + this.animated = animated; + } + + public GuiProgressIcon(@NotNull GuiParent parent) { + super(parent); + } + + /** + * Set the direction this progress icon is pointing, Default is RIGHT + */ + public GuiProgressIcon setDirection(Direction direction) { + this.direction = direction; + return this; + } + + /** + * Sets the background texture, aka the "empty" texture. + */ + public GuiProgressIcon setBackground(@Nullable Material background) { + this.background = background; + return this; + } + + /** + * Sets the texture that will be animated. + */ + public GuiProgressIcon setAnimated(Material animated) { + this.animated = animated; + return this; + } + + /** + * Set the current progress to a fixed value. + * + * @see #setProgress(Supplier) + */ + public GuiProgressIcon setProgress(double progress) { + return setProgress(() -> progress); + } + + /** + * Attach a supplier that returns the current progress value for this progress icon (0 to 1) + */ + public GuiProgressIcon setProgress(Supplier progress) { + this.progress = progress; + return this; + } + + public double getProgress() { + return progress.get(); + } + + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + render.pose().pushPose(); + + double width = direction.getAxis() == Axis.X ? xSize() : ySize(); + double height = direction.getAxis() == Axis.X ? ySize() : xSize(); + + render.pose().translate(xMin() + (xSize() / 2), yMin() + (ySize() / 2), 0); + render.pose().mulPose(com.mojang.math.Axis.ZP.rotationDegrees((float) Direction.RIGHT.rotationTo(direction))); + + double halfWidth = width / 2; + double halfHeight = height / 2; + if (background != null) { + render.tex(background, -halfWidth, -halfHeight, halfWidth, halfHeight, 0xFFFFFFFF); + } + + if (animated == null) return; + float progress = (float) getProgress(); + render.partialSprite(animated.renderType(GuiRender::texColType), -halfWidth, -halfHeight, -halfWidth + (width* progress), -halfHeight + height, animated.sprite(), 0F, 0F, progress, 1F, 0xFFFFFFFF); + + render.pose().popPose(); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiRectangle.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiRectangle.java new file mode 100644 index 00000000..0d034702 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiRectangle.java @@ -0,0 +1,165 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.BackgroundRender; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import org.jetbrains.annotations.NotNull; + +import java.util.function.Supplier; + +/** + * Used to draw a simple rectangle on the screen. + * Can specify separate (or no) border colours and fill colours. + * Can also render using the "shadedRectangle" render type. + *

+ * Created by brandon3055 on 28/08/2023 + */ +public class GuiRectangle extends GuiElement implements BackgroundRender { + private Supplier fill = null; + private Supplier border = null; + + private Supplier borderWidth = () -> 1D; + + private Supplier shadeTopLeft; + private Supplier shadeBottomRight; + private Supplier shadeCorners; + + /** + * @param parent parent {@link GuiParent}. + */ + public GuiRectangle(@NotNull GuiParent parent) { + super(parent); + } + + /** + * Creates a rectangle that mimics the appearance of a vanilla inventory slot. + * Uses shadedRect to create the 3D "inset" look. + */ + public static GuiRectangle vanillaSlot(@NotNull GuiParent parent) { + return new GuiRectangle(parent).shadedRect(0xFF373737, 0xFFffffff, 0xFF8b8b8b, 0xFF8b8b8b); + } + + /** + * Creates a rectangle that mimics the appearance of a vanilla inventory slot, except inverted + * Uses shadedRect to create the 3D "popped out" appearance + */ + public static GuiRectangle invertedSlot(@NotNull GuiParent parent) { + return new GuiRectangle(parent).shadedRect(0xFFffffff, 0xFF373737, 0xFF8b8b8b, 0xFF8b8b8b); + } + + /** + * Creates a rectangle similar in appearance to a vanilla button, but with no texture and no black border. + */ + public static GuiRectangle planeButton(@NotNull GuiParent parent) { + return new GuiRectangle(parent).shadedRect(0xFFaaaaaa, 0xFF545454, 0xFF6f6f6f); + } + + public static GuiRectangle toolTipBackground(@NotNull GuiParent parent) { + return toolTipBackground(parent, 0xF0100010, 0x505000FF, 0x5028007f); + } + + public static GuiRectangle toolTipBackground(@NotNull GuiParent parent, int backgroundColour, int borderColourTop, int borderColourBottom) { + return toolTipBackground(parent, backgroundColour, backgroundColour, borderColourTop, borderColourBottom); + } + + public static GuiRectangle toolTipBackground(@NotNull GuiParent parent, int backgroundColourTop, int backgroundColourBottom, int borderColourTop, int borderColourBottom) { + return new GuiRectangle(parent) { + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + render.toolTipBackground(xMin(), yMin(), xSize(), ySize(), backgroundColourTop, backgroundColourBottom, borderColourTop, borderColourBottom, false); + } + }; + } + + public GuiRectangle border(int border) { + return border(() -> border); + } + + public GuiRectangle border(Supplier border) { + this.border = border; + return this; + } + + public GuiRectangle fill(int fill) { + return fill(() -> fill); + } + + public GuiRectangle fill(Supplier fill) { + this.fill = fill; + return this; + } + + public GuiRectangle rectangle(int fill, int border) { + return rectangle(() -> fill, () -> border); + } + + public GuiRectangle rectangle(Supplier fill, Supplier border) { + this.fill = fill; + this.border = border; + return this; + } + + public GuiRectangle shadedRect(int topLeft, int bottomRight, int fill) { + return shadedRect(() -> topLeft, () -> bottomRight, () -> fill); + } + + public GuiRectangle shadedRect(Supplier topLeft, Supplier bottomRight, Supplier fill) { + return shadedRect(topLeft, bottomRight, () -> GuiRender.midColour(topLeft.get(), bottomRight.get()), fill); + } + + public GuiRectangle shadedRect(int topLeft, int bottomRight, int cornerMix, int fill) { + return shadedRect(() -> topLeft, () -> bottomRight, () -> cornerMix, () -> fill); + } + + public GuiRectangle shadedRect(Supplier topLeft, Supplier bottomRight, Supplier cornerMix, Supplier fill) { + this.fill = fill; + this.shadeTopLeft = topLeft; + this.shadeBottomRight = bottomRight; + this.shadeCorners = cornerMix; + return this; + } + + public GuiRectangle setShadeTopLeft(Supplier shadeTopLeft) { + this.shadeTopLeft = shadeTopLeft; + return this; + } + + public GuiRectangle setShadeBottomRight(Supplier shadeBottomRight) { + this.shadeBottomRight = shadeBottomRight; + return this; + } + + public GuiRectangle setShadeCorners(Supplier shadeCorners) { + this.shadeCorners = shadeCorners; + return this; + } + + public GuiRectangle setShadeCornersAuto() { + this.shadeCorners = () -> GuiRender.midColour(shadeTopLeft.get(), shadeBottomRight.get()); + return this; + } + + public GuiRectangle borderWidth(double borderWidth) { + return borderWidth(() -> borderWidth); + } + + public GuiRectangle borderWidth(Supplier borderWidth) { + this.borderWidth = borderWidth; + return this; + } + + public double getBorderWidth() { + return borderWidth.get(); + } + + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + if (shadeTopLeft != null && shadeBottomRight != null && shadeCorners != null) { + render.shadedRect(getRectangle(), getBorderWidth(), shadeTopLeft.get(), shadeBottomRight.get(), shadeCorners.get(), fill == null ? 0 : fill.get()); + } else if (border != null) { + render.borderRect(getRectangle(), getBorderWidth(), fill == null ? 0 : fill.get(), border.get()); + } else if (fill != null) { + render.rect(getRectangle(), fill.get()); + } + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiScrolling.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiScrolling.java new file mode 100644 index 00000000..c0c5f9e0 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiScrolling.java @@ -0,0 +1,219 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.Constraints; +import codechicken.lib.gui.modular.lib.ContentElement; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.SliderState; +import codechicken.lib.gui.modular.lib.geometry.Axis; +import codechicken.lib.gui.modular.lib.geometry.Constraint; +import codechicken.lib.gui.modular.lib.geometry.GeoParam; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.math.MathHelper; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import static codechicken.lib.gui.modular.lib.geometry.Constraint.match; +import static codechicken.lib.gui.modular.lib.geometry.Constraint.relative; +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * So the logic behind this element is as follows. + * This element contains a base "Content Element" that holds all the scrollable content. + * The content element's position is controlled by the {@link GuiScrolling} + * But its {@link GeoParam#WIDTH} and {@link GeoParam#HEIGHT} constraints can be set by the user, + * Or they can be set to dynamically adjust to the child elements added to it. + *

+ * The bounds of the {@link GuiScrolling} represent the "view window" + * When scrolling up/down, left/right the Content Element is effectively just moving around behind the view window + * and everything outside the view window is scissored off. + * Any events that occur outside the view window are not propagated to scroll element. + * Calls to {@link #isMouseOver()} from an area of an element that is outside the view window will return false. + *

+ * Elements that are completely outside the view window will not be rendered at all for efficiency. + *

+ * Created by brandon3055 on 01/09/2023 + */ +public class GuiScrolling extends GuiElement implements ContentElement> { + + /** + * This is made available primarily for debugging purposes where it can be useful to see what's going on behind the scenes. + */ + public boolean enableScissor = true; + private GuiElement contentElement; + private double xScrollPos = 0; + private double yScrollPos = 0; + private double contentWidth = 0; + private double contentHeight = 0; + private boolean setup = false; + + /** + * @param parent parent {@link GuiParent}. + */ + public GuiScrolling(@NotNull GuiParent parent) { + super(parent); + installContainerElement(new ContentElement(this)); + } + + //=== Scroll element setup ===// + + /** + * Retrieves the content element that holds all the scrolling elements. + * You must add all of your scrolling content to this element. + * Scrolling content must also be constrained relative to this element. + *

+ * The {@link GeoParam#TOP} and {@link GeoParam#LEFT} constraints for this element are set by the {@link GuiScrolling} and must not be overridden. + * These are used to control the 'scrolling' of the element. + *

+ * By default, the {@link GeoParam#WIDTH} and {@link GeoParam#HEIGHT} (and therefor also BOTTOM, RIGHT) are dynamically constrained to match the outer bounds of the scrolling elements. + * So attempting to constrain the content to any of these dynamic parameters would result in a stack overflow. + * You can however override the WIDTH and HEIGHT constraints if you wish. + * This can be useful if you wish to create something like a fixed width scrolling list where the width of each scrolling element is bound to the width of the list. + *

+ * The most important thing to note, Especially when manually constraining the WIDTH and HEIGHT of the content element, + * All scrolling elements must be withing the bounds of the content element. Anything outside the content element's bounds will not be visible. + * + * @return The content element. + */ + @Override + public GuiElement getContentElement() { + return contentElement; + } + + /** + * This allows you to install a custom container element. + * The elements constraints will automatically be set by this method. + *

+ * After calling this method you may override the container element WIDTH and HEIGHT constraints as described in the documentation for {@link #getContentElement()} + * But you must not touch the TOP or LEFT constraints. + *

+ * Important thing to note, By default the container element is preinstalled before any children can be added, meaning any children added to the {@link GuiScrolling} + * will render on top of the scrolling content. + * As this method allows you to set a new child as the container element, any children added before the new content element, will render under the content element. + * + * @param element The new container element. + */ + public void installContainerElement(GuiElement element) { + if (element.getParent() != this) throw new IllegalStateException("Content element must be a child of the GuiScrollingBase it is being installed in"); + if (contentElement != null) removeChild(contentElement); + setup = true; + contentElement = element; + contentElement.setRenderCull(getRectangle()); + contentElement.constrain(TOP, Constraint.relative(get(TOP), () -> yScrollPos * -hiddenSize(Axis.Y))); + contentElement.constrain(LEFT, Constraint.relative(get(LEFT), () -> xScrollPos * -hiddenSize(Axis.X))); + contentElement.constrain(WIDTH, Constraint.dynamic(() -> contentElement.getChildBounds().xMax() - contentElement.xMin())); + contentElement.constrain(HEIGHT, Constraint.dynamic(() -> contentElement.getChildBounds().yMax() - contentElement.yMin())); + setup = false; + } + + /** + * @return a {@link SliderState} that can be used to get or control the scroll position of the specified axis. + */ + public SliderState scrollState(Axis axis) { + return switch (axis) { + case X -> SliderState.forScrollBar(() -> xScrollPos, e -> xScrollPos = e, () -> MathHelper.clip(xSize() / contentElement.xSize(), 0, 1)); + case Y -> SliderState.forScrollBar(() -> yScrollPos, e -> yScrollPos = e, () -> MathHelper.clip(ySize() / contentElement.ySize(), 0, 1)); + }; + } + + //=== Internal logic ===// + + /** + * @return the total content size / length for the given axis + */ + public double totalSize(Axis axis) { + return switch (axis) { + case X -> contentWidth; + case Y -> contentHeight; + }; + } + + /** + * @return the hidden content size / length for the given axis (How much of the content is outside the view area) + */ + public double hiddenSize(Axis axis) { + return switch (axis) { + case X -> Math.max(contentWidth - xSize(), 0); + case Y -> Math.max(contentHeight - ySize(), 0); + }; + } + + @Override + public void tick(double mouseX, double mouseY) { + super.tick(mouseX, mouseY); + //These can not be generated dynamically, Doing so would result in a calculation loop, aka a stack overflow. + contentWidth = contentElement.xSize(); + contentHeight = contentElement.ySize(); + } + + @Override + public boolean blockMouseOver(GuiElement element, double mouseX, double mouseY) { + return super.blockMouseOver(element, mouseX, mouseY) || (element.isDescendantOf(contentElement) && !this.isMouseOver()); + } + + //=== Rendering ===// + + @Override + protected boolean renderChild(GuiElement child, GuiRender render, double mouseX, double mouseY, float partialTicks) { + boolean scissor = child == contentElement && enableScissor; + if (scissor) render.pushScissorRect(getRectangle()); + boolean ret = super.renderChild(child, render, mouseX, mouseY, partialTicks); + if (scissor) render.popScissor(); + return ret; + } + + private class ContentElement extends GuiElement { + /** + * @param parent parent {@link GuiParent}. + */ + public ContentElement(@NotNull GuiParent parent) { + super(parent); + } + + @Override + public ContentElement constrain(GeoParam param, @Nullable Constraint constraint) { + if (!setup && (param == TOP || param == LEFT)) throw new IllegalStateException("Can not override TOP or LEFT constraints on content element, These are used to control the scrolling behavior!"); + return super.constrain(param, constraint); + } + } + + public static ScrollWindow simpleScrollWindow(@NotNull GuiParent parent, boolean verticalScrollBar, boolean horizontalScrollBar) { + GuiElement container = new GuiElement<>(parent); + GuiRectangle background = GuiRectangle.vanillaSlot(container) + .constrain(TOP, match(container.get(TOP))) + .constrain(LEFT, match(container.get(LEFT))) + .constrain(BOTTOM, relative(container.get(BOTTOM), horizontalScrollBar ? -10 : 0)) + .constrain(RIGHT, relative(container.get(RIGHT), verticalScrollBar ? -10 : 0)); + + GuiScrolling scroll = new GuiScrolling(background); + Constraints.bind(scroll, background, 1); + + GuiSlider.ScrollBar verticalBar = null; + if (verticalScrollBar) { + verticalBar = GuiSlider.vanillaScrollBar(container, Axis.Y); + verticalBar.container() + .constrain(TOP, match(container.get(TOP))) + .constrain(BOTTOM, relative(container.get(BOTTOM), horizontalScrollBar ? -10 : 0)) + .constrain(RIGHT, match(container.get(RIGHT))) + .constrain(WIDTH, Constraint.literal(9)); + verticalBar.slider() + .setSliderState(scroll.scrollState(Axis.Y)) + .setScrollableElement(scroll); + } + GuiSlider.ScrollBar horizontalBar = null; + if (horizontalScrollBar) { + horizontalBar = GuiSlider.vanillaScrollBar(container, Axis.X); + horizontalBar.container() + .constrain(BOTTOM, match(container.get(BOTTOM))) + .constrain(LEFT, match(container.get(LEFT))) + .constrain(RIGHT, relative(container.get(RIGHT), verticalScrollBar ? -10 : 0)) + .constrain(HEIGHT, Constraint.literal(9)); + horizontalBar.slider() + .setSliderState(scroll.scrollState(Axis.X)) + .setScrollableElement(scroll); + } + return new ScrollWindow(container, scroll, verticalBar, horizontalBar); + } + + public record ScrollWindow(GuiElement container, GuiScrolling scrolling, @Nullable GuiSlider.ScrollBar verticalBar, @Nullable GuiSlider.ScrollBar horizontalBar) { + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiSlider.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiSlider.java new file mode 100644 index 00000000..2157ccf8 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiSlider.java @@ -0,0 +1,266 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.*; +import codechicken.lib.gui.modular.lib.geometry.*; +import codechicken.lib.math.MathHelper; +import org.jetbrains.annotations.NotNull; + +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * This can be used as the base for anything that requires the linear movement of an element between two position. + * e.g. Scroll bars, Slide controls and slide indicators. + *

+ * Implementation is simple, Simply install a "Slide Element", this will be the moving element, + * The movement of this element is confined to the bounds og the {@link GuiSlider} + *

+ * The position of the slider is managed via the installed {@link SliderState} + *

+ * Created by brandon3055 on 02/09/2023 + */ +public class GuiSlider extends GuiElement { + private final Axis axis; + private SliderState state = SliderState.create(0.1); + private GuiElement slider; + private double outOfBoundsDist = 50; + private GuiElement scrollableElement; + + private int dragButton = GuiButton.LEFT_CLICK; + private int scrollDragButton = GuiButton.MIDDLE_CLICK; + private boolean middleClickScroll = false; + /** + * This should theoretically never be needed, But just in case... + */ + public boolean invertDragScroll = false; + + private boolean dragging = false; + private double slideStartPos = 0; + private Position clickPos = Position.create(0, 0); + private boolean scrollableDragging = false; + + /** + * Creates a basic gui slider that moves along the specified axis. + * This includes a default slider element the width of which is bound to the GuiSlider, + * And the length of which is controlled by {@link SliderState#sliderRatio()} + */ + public GuiSlider(@NotNull GuiParent parent, Axis axis) { + super(parent); + this.axis = axis; + installSlider(new GuiElement<>(this)); + bindSliderLength(); + bindSliderWidth(); + } + + public GuiSlider(@NotNull GuiParent parent, Axis axis, GuiElement slider) { + super(parent); + this.axis = axis; + installSlider(slider); + } + + /** + * Vanilla does not really seem to have a standard for its scroll bars, + * But this is something that should at least fit in to a typical vanilla gui. + */ + public static ScrollBar vanillaScrollBar(GuiElement parent, Axis axis) { + GuiRectangle background = GuiRectangle.vanillaSlot(parent); + + GuiSlider slider = new GuiSlider(background, axis); + Constraints.bind(slider, background, 1); + + slider.installSlider(GuiRectangle.planeButton(slider)) + .bindSliderLength() + .bindSliderWidth(); + + GuiRectangle sliderHighlight = new GuiRectangle(slider.getSlider()) + .fill(0x5000b6FF) + .setEnabled(() -> slider.getSlider().isMouseOver()); + + Constraints.bind(sliderHighlight, slider.getSlider()); + + return new ScrollBar(background, slider, sliderHighlight); + } + + /** + * Set the slider state used by this slider element. + * The slider state is used to get and set the slider position. + * It also controls scroll speed. + */ + public GuiSlider setSliderState(SliderState state) { + this.state = state; + return this; + } + + /** + * For use cases where this slider is controlling something like a scroll element. + * This enables scrolling when the cursor is over the scrollable element. + * It can also enable scrolling via middle-click + drag. + */ + public GuiSlider setScrollableElement(GuiElement scrollableElement) { + return setScrollableElement(scrollableElement, true); + } + + /** + * For use cases where this slider is controlling something like a scroll element. + * This enables scrolling when the cursor is over the scrollable element. + * It can also enable scrolling via middle-click + drag. + */ + public GuiSlider setScrollableElement(GuiElement scrollableElement, boolean middleClickScroll) { + this.scrollableElement = scrollableElement; + this.middleClickScroll = middleClickScroll; + return this; + } + + /** + * Install an element to be used as the sliding element. + * The sliders minimum position (meaning either LEFT or TOP) on the moving axis will be constrained the {@link GuiSlider} + * Attempting to override this constraint after installing the slider element will break the slider. + *

+ * The size constraints, and position constraint for the non-moving axis need to be set by the implementor. + * + * @see #bindSliderLength() + * @see #bindSliderWidth() + */ + public GuiSlider installSlider(GuiElement slider) { + if (slider.getParent() != this) throw new IllegalStateException("slider element must be a child of the GuiSlider it is being installed in"); + if (this.slider != null) removeChild(this.slider); + this.slider = slider; + switch (axis) { + case X -> slider.constrain(LEFT, Constraint.relative(get(LEFT), () -> (getValue(WIDTH) - slider.getValue(WIDTH)) * state.getPos())); + case Y -> slider.constrain(TOP, Constraint.relative(get(TOP), () -> (getValue(HEIGHT) - slider.getValue(HEIGHT)) * state.getPos())); + } + return this; + } + + /** + * Sets up constraints to automatically control the slider element length. + * The slider element length will be controlled by {@link SliderState#sliderRatio()} + * This is used for things like gui scroll bars where the bar length changes based on the ratio of content in view. + */ + public GuiSlider bindSliderLength() { + switch (axis) {//Ensure we don't accidentally over-constrain + case X -> slider.constrain(RIGHT, null).constrain(WIDTH, Constraint.dynamic(() -> getValue(WIDTH) * state.sliderRatio())); + case Y -> slider.constrain(BOTTOM, null).constrain(HEIGHT, Constraint.dynamic(() -> getValue(HEIGHT) * state.sliderRatio())); + } + return this; + } + + /** + * Binds the sliders position and size on the non-moving axis to the width and pos of the {@link GuiSlider} + */ + public GuiSlider bindSliderWidth() { + switch (axis) {//Ensure we don't accidentally over-constrain + case X -> slider.constrain(HEIGHT, null).constrain(TOP, Constraint.match(get(TOP))).constrain(BOTTOM, Constraint.match(get(BOTTOM))); + case Y -> slider.constrain(WIDTH, null).constrain(LEFT, Constraint.match(get(LEFT))).constrain(RIGHT, Constraint.match(get(RIGHT))); + } + return this; + } + + /** + * @return the installed slider element. + */ + public GuiElement getSlider() { + return slider; + } + + /** + * @return True if the slider is currently being dragged by the user. + */ + public boolean isDragging() { + return dragging; + } + + /** + * Set the out-of-bounds distance, + * If the cursor is dragged more than this distance from the slider bounds on the no-moving axis, + * the slider will snap back to its original position until the cursor moves back into bounds. + * Default is 50, -1 will disable the snap-back functionality. + */ + public GuiSlider setOutOfBoundsDist(double outOfBoundsDist) { + this.outOfBoundsDist = outOfBoundsDist; + return this; + } + + /** + * @param dragButton The mouse button used to drag this slider (Default {@link GuiButton#LEFT_CLICK}) + */ + public GuiSlider setDragButton(int dragButton) { + this.dragButton = dragButton; + return this; + } + + /** + * @param scrollDragButton The button used to scroll by clicking and dragging the defined scrollableElement (Default {@link GuiButton#MIDDLE_CLICK}) + * @see #setScrollableElement(GuiElement, boolean) + */ + public GuiSlider setScrollDragButton(int scrollDragButton) { + this.scrollDragButton = scrollDragButton; + return this; + } + + @Override + public boolean mouseClicked(double mouseX, double mouseY, int button) { + dragging = false; + clickPos = Position.create(mouseX, mouseY); + slideStartPos = state.getPos(); + if (button == dragButton && isMouseOver()) { + if (!slider.isMouseOver()) { + clickPos = Position.create(slider.xCenter(), slider.yCenter()); + handleDrag(mouseX, mouseY); + } + dragging = true; + return true; + } + if (button == scrollDragButton && scrollableElement != null && scrollableElement.isMouseOver()) { + scrollableDragging = true; + } + return false; + } + + @Override + public boolean mouseReleased(double mouseX, double mouseY, int button, boolean consumed) { + dragging = scrollableDragging = false; + return super.mouseReleased(mouseX, mouseY, button, consumed); + } + + @Override + public void mouseMoved(double mouseX, double mouseY) { + if (dragging || (scrollableDragging && scrollableElement != null)) { + handleDrag(mouseX, mouseY); + } + super.mouseMoved(mouseX, mouseY); + } + + private void handleDrag(double mouseX, double mouseY) { + Position mousePos = Position.create(mouseX, mouseY); + Rectangle rect = dragging || scrollableElement == null ? getRectangle() : scrollableElement.getRectangle(); + + if (dragging && outOfBoundsDist >= -1 && rect.distance(axis.opposite(), mousePos) > outOfBoundsDist) { + state.setPos(slideStartPos); + return; + } + + double travel = rect.size(axis) - slider.getRectangle().size(axis); + if (travel <= 0) return; + double clickPos = this.clickPos.get(axis); + double currentPos = mousePos.get(axis); + double movement = (currentPos - clickPos) / travel; + + if (scrollableDragging) { + movement *= invertDragScroll ? state.sliderRatio() : -state.sliderRatio(); + } + + state.setPos(MathHelper.clip(slideStartPos + movement, 0, 1)); + } + + @Override + public boolean mouseScrolled(double mouseX, double mouseY, double scroll) { + if (isMouseOver() || (scrollableElement != null && scrollableElement.isMouseOver())) { + if (!state.canScroll(axis)) return false; + state.setPos(MathHelper.clip(state.getPos() + (state.scrollSpeed() * -scroll), 0, 1)); + return true; + } + return false; + } + + public record ScrollBar(GuiRectangle container, GuiSlider slider, GuiRectangle highlight) {} +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiSlots.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiSlots.java new file mode 100644 index 00000000..9ce861eb --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiSlots.java @@ -0,0 +1,312 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.BackgroundRender; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.container.ContainerScreenAccess; +import codechicken.lib.gui.modular.lib.container.SlotGroup; +import codechicken.lib.gui.modular.lib.geometry.Constraint; +import codechicken.lib.gui.modular.lib.geometry.GeoParam; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.gui.modular.sprite.CCGuiTextures; +import codechicken.lib.gui.modular.sprite.Material; +import net.minecraft.world.inventory.Slot; +import org.jetbrains.annotations.NotNull; + +import java.util.function.Function; + +import static codechicken.lib.gui.modular.lib.geometry.Constraint.match; +import static codechicken.lib.gui.modular.lib.geometry.Constraint.relative; +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; +import static net.minecraft.world.inventory.InventoryMenu.BLOCK_ATLAS; + +/** + * This element is used to manage and render a grid of inventory slots in a GUI. + * The width and height of this element are automatically constrained based on the slot configuration. + * However, you can override those constraints, The slot grid will always render in the center of the element nomater the element size. + *

+ * This can be used to render all slots in a {@link SlotGroup} or a sub-set of slots within a group. + *

+ * Created by brandon3055 on 08/09/2023 + */ +public class GuiSlots extends GuiElement implements BackgroundRender { + public static final Material[] ARMOR_SLOTS = new Material[]{Material.fromAtlas(BLOCK_ATLAS, "item/empty_armor_slot_helmet"), Material.fromAtlas(BLOCK_ATLAS, "item/empty_armor_slot_chestplate"), Material.fromAtlas(BLOCK_ATLAS, "item/empty_armor_slot_leggings"), Material.fromAtlas(BLOCK_ATLAS, "item/empty_armor_slot_boots")}; + public static final Material OFF_HAND_SLOT = Material.fromAtlas(BLOCK_ATLAS, "item/empty_armor_slot_shield"); + + private final int firstSlot; + private final int slotCount; + private final int columns; + private final SlotGroup slots; + private final ContainerScreenAccess screenAccess; + + private Material slotTexture = CCGuiTextures.getUncached("widgets/slot"); + private Function slotIcons = slot -> null; + private Function highlightColour = slot -> 0x80ffffff; + private int xSlotSpacing = 0; + private int ySlotSpacing = 0; + + /** + * @param slots The slot group containing the slots that this element will manage. + * @param gridColumns The width of the inventory grid (Typically 9 for standard player or chest inventories) + */ + public GuiSlots(@NotNull GuiParent parent, ContainerScreenAccess screenAccess, SlotGroup slots, int gridColumns) { + this(parent, screenAccess, slots, 0, slots.size(), gridColumns); + } + + /** + * @param slots The slot group containing the slots that this element will manage. + * @param firstSlot Index of the fist slot within the slot group. + * @param slotCount The number of slots that this element will manage. + * @param gridColumns The width of the inventory grid (Typically 9 for standard player or chest inventories) + */ + public GuiSlots(@NotNull GuiParent parent, ContainerScreenAccess screenAccess, SlotGroup slots, int firstSlot, int slotCount, int gridColumns) { + super(parent); + this.screenAccess = screenAccess; + this.slots = slots; + this.firstSlot = firstSlot; + this.slotCount = slotCount; + this.columns = gridColumns; + if (firstSlot + slotCount > slots.size()) { + throw new IllegalStateException("Specified slot range is out of bounds, Last slot in group is at index " + (slots.size() - 1) + " Specified range is from index " + firstSlot + " to " + (firstSlot + slotCount - 1)); + } + int columns = Math.min(gridColumns, slots.size()); + this.constrain(WIDTH, Constraint.dynamic(() -> (double) (columns * 18) + ((columns - 1) * xSlotSpacing))); + int rows = Math.max(1, slots.size() / gridColumns); + this.constrain(GeoParam.HEIGHT, Constraint.dynamic(() -> (double) (rows * 18) + ((rows - 1) * ySlotSpacing))); + for (int index = 0; index < slotCount; index++) { + Slot slot = slots.getSlot(index + firstSlot); + getModularGui().setSlotHandler(slot, this); + } + + updateSlots(parent.getModularGui().getRoot()); + } + + //=== Construction Helpers ===// + + public static GuiSlots singleSlot(@NotNull GuiParent parent, ContainerScreenAccess screenAccess, SlotGroup slots) { + return singleSlot(parent, screenAccess, slots, 0); + } + + public static GuiSlots singleSlot(@NotNull GuiParent parent, ContainerScreenAccess screenAccess, SlotGroup slots, int index) { + return new GuiSlots(parent, screenAccess, slots, index, 1, 1); + } + + public static Player player(@NotNull GuiParent parent, ContainerScreenAccess screenAccess, SlotGroup mainSlots, SlotGroup hotBarSlots) { + return player(parent, screenAccess, mainSlots, hotBarSlots, 3); + } + + public static Player player(@NotNull GuiParent parent, ContainerScreenAccess screenAccess, SlotGroup mainSlots, SlotGroup hotBarSlots, int hotBarSpacing) { + int width = 18 * 9; + int height = 18 * 4 + hotBarSpacing; + GuiElement container = new GuiElement<>(parent) + .setZStacking(false) + .constrain(WIDTH, Constraint.literal(width)) + .constrain(HEIGHT, Constraint.literal(height)); + + GuiSlots main = new GuiSlots(container, screenAccess, mainSlots, 9) + .constrain(TOP, Constraint.midPoint(container.get(TOP), container.get(BOTTOM), height / -2D)) + .constrain(LEFT, Constraint.midPoint(container.get(LEFT), container.get(RIGHT), width / -2D)); + + GuiSlots bar = new GuiSlots(container, screenAccess, hotBarSlots, 9) + .constrain(TOP, relative(main.get(BOTTOM), hotBarSpacing)) + .constrain(LEFT, match(main.get(LEFT))); + return new Player(container, main, bar); + } + + public static PlayerWithArmor playerWithArmor(@NotNull GuiParent parent, ContainerScreenAccess screenAccess, SlotGroup mainSlots, SlotGroup hotBarSlots, SlotGroup armorSlots) { + return playerWithArmor(parent, screenAccess, mainSlots, hotBarSlots, armorSlots, 3, true); + } + + public static PlayerWithArmor playerWithArmor(@NotNull GuiParent parent, ContainerScreenAccess screenAccess, SlotGroup mainSlots, SlotGroup hotBarSlots, SlotGroup armorSlots, int groupSpacing, boolean slotIcons) { + int width = 18 * 10 + groupSpacing; + int height = 18 * 4 + groupSpacing; + GuiElement container = new GuiElement<>(parent) + .setZStacking(false) + .constrain(WIDTH, Constraint.literal(width)) + .constrain(HEIGHT, Constraint.literal(height)); + + GuiSlots armor = new GuiSlots(container, screenAccess, armorSlots, 1) + .setYSlotSpacing(groupSpacing / 3) + .setEmptyIcon(index -> slotIcons ? ARMOR_SLOTS[index] : null) + .constrain(TOP, Constraint.midPoint(container.get(TOP), container.get(BOTTOM), height / -2D)) + .constrain(LEFT, Constraint.midPoint(container.get(LEFT), container.get(RIGHT), width / -2D)); + + GuiSlots main = new GuiSlots(container, screenAccess, mainSlots, 9) + .constrain(TOP, match(armor.get(TOP))) + .constrain(LEFT, relative(armor.get(RIGHT), groupSpacing)); + + GuiSlots bar = new GuiSlots(container, screenAccess, hotBarSlots, 9) + .constrain(TOP, relative(main.get(BOTTOM), groupSpacing)) + .constrain(LEFT, match(main.get(LEFT))); + + return new PlayerWithArmor(container, main, bar, armor); + } + + public static PlayerAll playerAllSlots(@NotNull GuiParent parent, ContainerScreenAccess screenAccess, SlotGroup mainSlots, SlotGroup hotBarSlots, SlotGroup armorSlots, SlotGroup offhandSlots) { + return playerAllSlots(parent, screenAccess, mainSlots, hotBarSlots, armorSlots, offhandSlots, 3, true); + } + + public static PlayerAll playerAllSlots(@NotNull GuiParent parent, ContainerScreenAccess screenAccess, SlotGroup mainSlots, SlotGroup hotBarSlots, SlotGroup armorSlots, SlotGroup offhandSlots, int groupSpacing, boolean slotIcons) { + int width = 18 * 11 + groupSpacing * 2; + int height = 18 * 4 + groupSpacing; + GuiElement container = new GuiElement<>(parent) + .setZStacking(false) + .constrain(WIDTH, Constraint.literal(width)) + .constrain(HEIGHT, Constraint.literal(height)); + + GuiSlots armor = new GuiSlots(container, screenAccess, armorSlots, 1) + .setYSlotSpacing(groupSpacing / 3) + .setEmptyIcon(index -> slotIcons ? ARMOR_SLOTS[index] : null) + .constrain(TOP, Constraint.midPoint(container.get(TOP), container.get(BOTTOM), height / -2D)) + .constrain(LEFT, Constraint.midPoint(container.get(LEFT), container.get(RIGHT), width / -2D)); + + GuiSlots main = new GuiSlots(container, screenAccess, mainSlots, 9) + .constrain(TOP, match(armor.get(TOP))) + .constrain(LEFT, relative(armor.get(RIGHT), groupSpacing)); + + GuiSlots bar = new GuiSlots(container, screenAccess, hotBarSlots, 9) + .constrain(TOP, relative(main.get(BOTTOM), groupSpacing)) + .constrain(LEFT, match(main.get(LEFT))); + + GuiSlots offHand = new GuiSlots(container, screenAccess, offhandSlots, 1) + .setEmptyIcon(index -> slotIcons ? OFF_HAND_SLOT : null) + .constrain(TOP, match(bar.get(TOP))) + .constrain(LEFT, relative(bar.get(RIGHT), groupSpacing)); + + return new PlayerAll(container, main, bar, armor, offHand); + } + + //=== Slots Setup ===// + + /** + * Allows you to use a custom slot texture, The default is the standard vanilla slot. + */ + public GuiSlots setSlotTexture(Material slotTexture) { + this.slotTexture = slotTexture; + return this; + } + + /** + * Sets a custom slot highlight colour (The highlight you get when your cursor is over a slot.) + */ + public GuiSlots setHighlightColour(int highlightColour) { + return setHighlightColour(slot -> highlightColour); + } + + /** + * Allows you to set per-slot highlight colours, The integer passed to the function is the + * index of the slot within the {@link SlotGroup} + */ + public GuiSlots setHighlightColour(Function highlightColour) { + this.highlightColour = highlightColour; + return this; + } + + /** + * Applies a single empty slot icon to all slots. + * Recommended texture size is 16x16 + */ + public GuiSlots setEmptyIcon(Material texture) { + return setEmptyIcon(index -> texture); + } + + /** + * Allows you to provide a texture to be rendered in each slot when the slot is empty. + * Recommended texture size is 16x16 + * + * @param slotIcons A function that is given the slot index within the {@link SlotGroup}, and should return a material or null. + */ + public GuiSlots setEmptyIcon(Function slotIcons) { + this.slotIcons = slotIcons; + return this; + } + + public GuiSlots setXSlotSpacing(int xSlotSpacing) { + this.xSlotSpacing = xSlotSpacing; + return this; + } + + public GuiSlots setYSlotSpacing(int ySlotSpacing) { + this.ySlotSpacing = ySlotSpacing; + return this; + } + + public GuiSlots setSlotSpacing(int xSlotSpacing, int ySlotSpacing) { + this.xSlotSpacing = xSlotSpacing; + this.ySlotSpacing = ySlotSpacing; + return this; + } + + //=== Internal Methods ===// + + @Override + public double getBackgroundDepth() { + return 33; + } + + private void updateSlots(GuiElement root) { + int columns = Math.min(this.columns, slots.size()); + int rows = Math.max(1, slots.size() / columns); + double width = (columns * 18) + (columns - 1) * xSlotSpacing; + double height = (rows * 18) + (rows - 1) * ySlotSpacing; + int top = (int) (yCenter() - (height / 2) - root.yMin()); + int left = (int) (xCenter() - (width / 2) - root.xMin()); + + for (int index = 0; index < slotCount; index++) { + Slot slot = slots.getSlot(index + firstSlot); + int x = index % columns; + int y = index / columns; + slot.x = left + (x * 18) + 1 + (x * xSlotSpacing); + slot.y = top + (y * 18) + 1 + (y * ySlotSpacing); + } + } + + @Override + public void tick(double mouseX, double mouseY) { + super.tick(mouseX, mouseY); + } + + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + GuiElement root = getModularGui().getRoot(); + updateSlots(root); + + Slot highlightSlot = null; + render.pose().pushPose(); + + for (int index = 0; index < slotCount; index++) { + Slot slot = slots.getSlot(index + firstSlot); + render.texRect(slotTexture, slot.x + root.xMin() - 1, slot.y + root.yMin() - 1, 18, 18); + } + + render.pose().translate(0, 0, 0.4); + + for (int index = 0; index < slotCount; index++) { + Slot slot = slots.getSlot(index + firstSlot); + if (!slot.isActive()) continue; + if (!slot.hasItem()) { + Material icon = slotIcons.apply(index + firstSlot); + if (icon != null) { + render.texRect(icon, slot.x + root.xMin(), slot.y + root.yMin(), 16, 16); + } + } + + screenAccess.renderSlot(render, slot); + if (GuiRender.isInRect(slot.x + root.xMin(), slot.y + root.yMin(), 16, 16, mouseX, mouseY) && !blockMouseOver(this, mouseX, mouseY)) { + highlightSlot = slot; + } + } + + if (highlightSlot != null) { + render.pose().translate(0, 0, getBackgroundDepth() - 0.8); + render.rect(highlightSlot.x + root.xMin(), highlightSlot.y + root.yMin(), 16, 16, highlightColour.apply(slots.indexOf(highlightSlot))); + } + + render.pose().popPose(); + } + + public record Player(GuiElement container, GuiSlots main, GuiSlots hotBar) {} + + public record PlayerWithArmor(GuiElement container, GuiSlots main, GuiSlots hotBar, GuiSlots armor) {} + + public record PlayerAll(GuiElement container, GuiSlots main, GuiSlots hotBar, GuiSlots armor, GuiSlots offHand) {} +} diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiText.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiText.java new file mode 100644 index 00000000..2776c2d5 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiText.java @@ -0,0 +1,268 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.ForegroundRender; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.geometry.Align; +import codechicken.lib.gui.modular.lib.geometry.GeoParam; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.gui.modular.lib.geometry.Position; +import com.mojang.blaze3d.vertex.PoseStack; +import com.mojang.math.Axis; +import net.minecraft.client.gui.Font; +import net.minecraft.locale.Language; +import net.minecraft.network.chat.Component; +import net.minecraft.network.chat.FormattedText; +import net.minecraft.util.FormattedCharSequence; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.List; +import java.util.function.Supplier; + +import static codechicken.lib.gui.modular.lib.geometry.Align.MAX; +import static codechicken.lib.gui.modular.lib.geometry.Align.MIN; +import static codechicken.lib.gui.modular.lib.geometry.Constraint.dynamic; + +/** + * Created by brandon3055 on 31/08/2023 + */ +public class GuiText extends GuiElement implements ForegroundRender { + private Supplier text; + private Supplier shadow = () -> true; + private Supplier textColour = () -> 0xFFFFFFFF; + private Supplier rotation = null; + private Position rotatePoint = Position.create(() -> xSize() / 2, () -> ySize() / 2); + private boolean trim = false; + private boolean wrap = false; + private boolean scroll = true; + private Align alignment = Align.CENTER; + //TODO, Arbitrary rotation is fun, But may want to switch to Axis, with option for "reverse" + + /** + * @param parent parent {@link GuiParent}. + */ + public GuiText(@NotNull GuiParent parent) { + this(parent, () -> null); + } + + /** + * @param parent parent {@link GuiParent}. + */ + public GuiText(@NotNull GuiParent parent, @Nullable Component text) { + this(parent, () -> text); + } + + /** + * @param parent parent {@link GuiParent}. + */ + public GuiText(@NotNull GuiParent parent, @NotNull Supplier<@Nullable Component> text) { + super(parent); + this.text = text; + } + + /** + * Apply a dynamic height constraint that sets the height based on text height (accounting for wrapping) + */ + public GuiText autoHeight() { + constrain(GeoParam.HEIGHT, dynamic(() -> wrap ? (double) font().wordWrapHeight(getText(), (int) xSize()) : font().lineHeight)); + return this; + } + + public GuiText setTextSupplier(@NotNull Supplier<@Nullable Component> textSupplier) { + this.text = textSupplier; + return this; + } + + public GuiText setText(@Nullable Component text) { + this.text = () -> text; + return this; + } + + public GuiText setText(@NotNull String text) { + this.text = () -> Component.literal(text); + return this; + } + + public GuiText setTranslatable(@NotNull String translationKey) { + this.text = () -> Component.translatable(translationKey); + return this; + } + + @Nullable + public Component getText() { + return text.get(); + } + + public GuiText setAlignment(Align alignment) { + this.alignment = alignment; + return this; + } + + public Align getAlignment() { + return alignment; + } + + /** + * If set to true the text will be trimmed if it is too long to fit within the bounds on the element. + * Default disabled. + * Setting this to true will automatically disable wrap and scroll ether are enabled. + */ + public GuiText setTrim(boolean trim) { + this.trim = trim; + if (trim) wrap = scroll = false; + return this; + } + + public boolean getTrim() { + return trim; + } + + /** + * Set to true the text will be wrapped (rendered as multiple lines of text) if it is too long to fit within the size of the element. + * Default disabled. + * Setting this to true will automatically disable trim and scroll ether are enabled. + */ + public GuiText setWrap(boolean wrap) { + this.wrap = wrap; + if (wrap) trim = scroll = false; + return this; + } + + public boolean getWrap() { + return wrap; + } + + /** + * Set to true the text scroll using the same logic as vanillas widgets if the text is too long to fit within the elements bounds + * Default enabled. + * Setting this to true will automatically disable trim and wrap ether are enabled. + */ + public GuiText setScroll(boolean scroll) { + this.scroll = scroll; + if (scroll) trim = wrap = false; + return this; + } + + public boolean getScroll() { + return scroll; + } + + public GuiText setShadow(@NotNull Supplier shadow) { + this.shadow = shadow; + return this; + } + + public GuiText setShadow(boolean shadow) { + this.shadow = () -> shadow; + return this; + } + + public boolean getShadow() { + return shadow.get(); + } + + public GuiText setTextColour(Supplier textColour) { + this.textColour = textColour; + return this; + } + + public GuiText setTextColour(int textColour) { + this.textColour = () -> textColour; + return this; + } + + public int getTextColour() { + return textColour.get(); + } + + /** + * Set rotation angle in degrees + * //TODO, Does not work when text scroll is used + */ + public GuiText setRotation(double rotation) { + return setRotation(() -> rotation); + } + + /** + * Dynamic rotation control, Set to null to disable rotation. + * Rotation angle is in degrees + * //TODO, Does not work when text scroll is used + */ + public GuiText setRotation(@Nullable Supplier rotation) { + this.rotation = rotation; + return this; + } + + /** + * Sets the point around which the text rotates relative to the top left of the element. + * Default rotation point is a dynamic point set to the dead center of the element. + */ + public GuiText setRotatePoint(Position rotatePoint) { + this.rotatePoint = rotatePoint; + return this; + } + + @Override + public double getForegroundDepth() { + return 0.035; + } + + @Override + public void renderForeground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + Component component = getText(); + if (component == null) return; + Font font = render.font(); + + int textHeight = font.lineHeight; + int textWidth = font.width(component); + boolean tooLong = textWidth > xSize(); + double yPos = (yMin() + ySize() / 2 - textHeight / 2D) + 1; //Adding 1 here makes the text look 'visually' centered, Text height includes the height of the optional underline. + + PoseStack stack = render.pose(); + if (rotation != null) { + stack.pushPose(); + stack.translate(xMin() + rotatePoint.x(), yMin() + rotatePoint.y(), 0); + stack.mulPose(Axis.ZP.rotationDegrees(rotation.get().floatValue())); + stack.translate(-xMin() - rotatePoint.x(), -yMin() - rotatePoint.y(), 0); + } + + //Draw Trimmed + if (tooLong && getTrim()) { + Component tail = Component.literal("...").setStyle(getText().getStyle()); + FormattedText head = font.getSplitter().headByWidth(component, (int) xSize() - font.width(tail), getText().getStyle()); + FormattedCharSequence formatted = Language.getInstance().getVisualOrder(FormattedText.composite(head, tail)); + textWidth = font.width(formatted); + + double xPos = alignment == MIN ? xMin() : alignment == MAX ? xMax() - textWidth : xMin() + xSize() / 2 - textWidth / 2D; + render.drawString(formatted, xPos, yPos, getTextColour(), getShadow()); + } + //Draw Wrapped + else if (tooLong && wrap) { + textHeight = font.wordWrapHeight(component, (int) xSize()); + List list = font.split(component, (int) xSize()); + + yPos = yMin() + ySize() / 2 - textHeight / 2D; + for (FormattedCharSequence line : list) { + int lineWidth = font.width(line); + double xPos = alignment == MIN ? xMin() : alignment == MAX ? xMax() - lineWidth : xMin() + xSize() / 2 - lineWidth / 2D; + render.drawString(line, xPos, yPos, getTextColour(), getShadow()); + yPos += font.lineHeight; + } + } + //Draw Scrolling + else if (tooLong && scroll) { + render.pushScissorRect(getRectangle()); + render.drawScrollingString(component, xMin(), yPos, xMax(), getTextColour(), getShadow(), false); + render.popScissor(); + } + //Draw + else { + double xPos = alignment == MIN ? xMin() : alignment == MAX ? xMax() - textWidth : xMin() + xSize() / 2 - textWidth / 2D; + render.drawString(component, xPos, yPos, getTextColour(), getShadow()); + } + + if (rotation != null) { + stack.popPose(); + } + } +} \ No newline at end of file diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiTextField.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiTextField.java new file mode 100644 index 00000000..680b918c --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiTextField.java @@ -0,0 +1,660 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.BackgroundRender; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.TextState; +import codechicken.lib.gui.modular.lib.geometry.Constraint; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import com.mojang.blaze3d.platform.InputConstants; +import com.mojang.blaze3d.vertex.DefaultVertexFormat; +import com.mojang.blaze3d.vertex.VertexFormat; +import net.minecraft.SharedConstants; +import net.minecraft.Util; +import net.minecraft.client.Minecraft; +import net.minecraft.client.gui.screens.Screen; +import net.minecraft.client.renderer.GameRenderer; +import net.minecraft.client.renderer.RenderStateShard; +import net.minecraft.client.renderer.RenderType; +import net.minecraft.network.chat.Component; +import net.minecraft.network.chat.Style; +import net.minecraft.util.FormattedCharSequence; +import net.minecraft.util.Mth; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; +import org.lwjgl.glfw.GLFW; + +import java.util.Objects; +import java.util.function.BiFunction; +import java.util.function.Predicate; +import java.util.function.Supplier; + +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * TODO, Re write this, Its currently mostly pulled from the TextField in Gui v2 + *

+ * Created by brandon3055 on 03/09/2023 + */ +public class GuiTextField extends GuiElement implements BackgroundRender { + private static final RenderType HIGHLIGHT_TYPE = RenderType.create("text_field_highlight", DefaultVertexFormat.POSITION_COLOR, VertexFormat.Mode.QUADS, 256, RenderType.CompositeState.builder() + .setShaderState(new RenderStateShard.ShaderStateShard(GameRenderer::getPositionColorShader)) + .setColorLogicState(RenderStateShard.OR_REVERSE_COLOR_LOGIC) + .createCompositeState(false)); + + private int tick; + private int cursorPos; + private int maxLength = 32; + private int displayPos; + private int highlightPos; + private boolean focused; + private boolean shiftPressed; + + private Supplier isEditable = () -> true; + private Supplier isFocusable = () -> true; + private Supplier canLoseFocus = () -> true; + + private Runnable onEditComplete = null; + private Runnable onEnterPressed = null; + + private TextState textState = TextState.simpleState(""); + private Supplier shadow = () -> true; + private Supplier textColor = () -> 0xe0e0e0; + + private Supplier suggestion = null; + private Supplier suggestionColour = () -> 0x7f7f80; + + private Predicate filter = Objects::nonNull; + private BiFunction formatter = (string, pos) -> FormattedCharSequence.forward(string, Style.EMPTY); + + public GuiTextField(@NotNull GuiParent parent) { + super(parent); + } + + /** + * Creates a simple text box with a simple bordered background. + * Using colours 0xFF000000, 0xFFFFFFFF, 0xE0E0E0 will get you a text box identical to the one in a command block + */ + public static TextField create(GuiElement parent, int backgroundColour, int borderColour, int textColour) { + GuiRectangle background = new GuiRectangle(parent) + .rectangle(backgroundColour, borderColour); + + GuiTextField textField = new GuiTextField(background) + .setTextColor(textColour) + .constrain(TOP, Constraint.relative(background.get(TOP), 1)) + .constrain(BOTTOM, Constraint.relative(background.get(BOTTOM), -1)) + .constrain(LEFT, Constraint.relative(background.get(LEFT), 4)) + .constrain(RIGHT, Constraint.relative(background.get(RIGHT), -4)); + + return new TextField(background, textField); + } + + //=== Text field setup ===// + + /** + * Called when the user clicks outside the text box, or when they press enter + */ + public GuiTextField setOnEditComplete(Runnable onEditComplete) { + this.onEditComplete = onEditComplete; + return this; + } + + /** + * Called when the user presses enter key (Including numpad enter) with the text box ion focus + */ + public GuiTextField setEnterPressed(Runnable onEnterPressed) { + this.onEnterPressed = onEnterPressed; + return this; + } + + /** + * The {@link TextState} is an accessor for the current text value. + * It simply contains string getter and setter methods. + * You can use this to link the text field to some external value. + */ + public GuiTextField setTextState(TextState textState) { + this.textState = textState; + return this; + } + + public TextState getTextState() { + return textState; + } + + public GuiTextField setTextColor(Supplier textColor) { + this.textColor = textColor; + return this; + } + + public GuiTextField setTextColor(int textColor) { + return setTextColor(() -> textColor); + } + + /** + * Should the text be rendered with a shadow? + */ + public GuiTextField setShadow(Supplier shadow) { + this.shadow = shadow; + return this; + } + + /** + * Should the text be rendered with a shadow? + */ + public GuiTextField setShadow(boolean shadow) { + return setShadow(() -> shadow); + } + + /** + * Set the "suggestion" text that is displayed when the text field is empty. + */ + public GuiTextField setSuggestion(Component suggestion) { + return setSuggestion(suggestion == null ? null : () -> suggestion); + } + + /** + * Set the "suggestion" text that is displayed when the text field is empty. + */ + public GuiTextField setSuggestion(@Nullable Supplier suggestion) { + this.suggestion = suggestion; + return this; + } + + /** + * Set the colour of the suggestion text. + */ + public void setSuggestionColour(Supplier suggestionColour) { + this.suggestionColour = suggestionColour; + } + + /** + * Allows you to apply a fielder to this text field. + * Whenever this field's value is updated, If the new value does not pass this filter + * It will not be applied. + */ + public GuiTextField setFilter(Predicate filter) { + this.filter = filter; + return this; + } + + /** + * Formats the current text value for display to the user. + * The integer is the current display start position within the current field value. + */ + public GuiTextField setFormatter(BiFunction formatter) { + this.formatter = formatter; + return this; + } + + /** + * If set to false, it will not be possible for the text field to lose focus via normal means. + * Focus can still be set via {@link #setFocus(boolean)} + */ + public GuiTextField setCanLoseFocus(boolean canLoseFocus) { + return setCanLoseFocus(() -> canLoseFocus); + } + + /** + * If set to false, it will not be possible for the text field to lose focus via normal means. + * Focus can still be set via {@link #setFocus(boolean)} + */ + public GuiTextField setCanLoseFocus(Supplier canLoseFocus) { + this.canLoseFocus = canLoseFocus; + return this; + } + + /** + * If false, It will not be possible to focus this element by clicking on it. + */ + public GuiTextField setFocusable(boolean focusable) { + return setFocusable(() -> focusable); + } + + /** + * If false, It will not be possible to focus this element by clicking on it. + */ + public GuiTextField setFocusable(Supplier focusable) { + this.isFocusable = focusable; + return this; + } + + /** + * If false, It will not be possible for the user to edit the value of this text field. + */ + public GuiTextField setEditable(boolean editable) { + return setEditable(() -> editable); + } + + /** + * If false, It will not be possible for the user to edit the value of this text field. + */ + public GuiTextField setEditable(Supplier editable) { + this.isEditable = editable; + return this; + } + + /** + * Sets the maximum allowed text length + */ + public GuiTextField setMaxLength(int newWidth) { + String value = getValue(); + maxLength = newWidth; + if (value.length() > newWidth) { + textState.setText(value.substring(0, newWidth)); + } + return this; + } + + private int getMaxLength() { + return maxLength; + } + + //=== Text field logic ===// + + /** + * Note, Initial value should be set after element is constrained, + * If element width is zero when set, nothing will render until the field is updated. + * TODO, I need to fix this. Element size should be able to change dynamically without things breaking. + */ + public GuiTextField setValue(String newValue) { + if (this.filter.test(newValue)) { + if (newValue.length() > maxLength) { + textState.setText(newValue.substring(0, maxLength)); + } else { + textState.setText(newValue); + } + moveCursorToEnd(); + setHighlightPos(cursorPos); + } + return this; + } + + public String getValue() { + return textState.getText(); + } + + public String getHighlighted() { + int i = Math.min(cursorPos, highlightPos); + int j = Math.max(cursorPos, highlightPos); + return getValue().substring(i, j); + } + + public void insertText(String text) { + String value = getValue(); + int selectStart = Math.min(cursorPos, highlightPos); + int selectEnd = Math.max(cursorPos, highlightPos); + int freeSpace = maxLength - value.length() - (selectStart - selectEnd); + String toInsert = SharedConstants.filterText(text); + int insertLen = toInsert.length(); + if (freeSpace < insertLen) { + toInsert = toInsert.substring(0, freeSpace); + insertLen = freeSpace; + } + + String newValue = (new StringBuilder(value)).replace(selectStart, selectEnd, toInsert).toString(); + if (filter.test(newValue)) { + textState.setText(newValue); + setCursorPosition(selectStart + insertLen); + setHighlightPos(cursorPos); + } + } + + private void deleteText(int i) { + if (Screen.hasControlDown()) { + deleteWords(i); + } else { + deleteChars(i); + } + } + + public void deleteWords(int i) { + if (!getValue().isEmpty()) { + if (highlightPos != cursorPos) { + insertText(""); + } else { + deleteChars(getWordPosition(i) - cursorPos); + } + } + } + + public void deleteChars(int i1) { + String value = getValue(); + if (!value.isEmpty()) { + if (highlightPos != cursorPos) { + insertText(""); + } else { + int i = getCursorPos(i1); + int j = Math.min(i, cursorPos); + int k = Math.max(i, cursorPos); + if (j != k) { + String s = (new StringBuilder(value)).delete(j, k).toString(); + if (filter.test(s)) { + textState.setText(s); + moveCursorTo(j); + } + } + } + } + } + + public int getWordPosition(int i) { + return getWordPosition(i, getCursorPosition()); + } + + private int getWordPosition(int i, int i1) { + return getWordPosition(i, i1, true); + } + + private int getWordPosition(int i1, int i2, boolean b) { + String value = getValue(); + int i = i2; + boolean flag = i1 < 0; + int j = Math.abs(i1); + + for (int k = 0; k < j; ++k) { + if (!flag) { + int l = value.length(); + i = value.indexOf(32, i); + if (i == -1) { + i = l; + } else { + while (b && i < l && value.charAt(i) == ' ') ++i; + } + } else { + while (b && i > 0 && value.charAt(i - 1) == ' ') --i; + while (i > 0 && value.charAt(i - 1) != ' ') --i; + } + } + + return i; + } + + public void moveCursor(int pos) { + moveCursorTo(getCursorPos(pos)); + } + + private int getCursorPos(int i) { + return Util.offsetByCodepoints(getValue(), cursorPos, i); + } + + public void moveCursorTo(int pos, boolean notify) { + setCursorPosition(pos); + if (!shiftPressed) { + setHighlightPos(cursorPos); + } + } + + public void moveCursorTo(int pos) { + moveCursorTo(pos, true); + } + + public void setCursorPosition(int pos) { + cursorPos = Mth.clamp(pos, 0, getValue().length()); + } + + public void moveCursorToStart() { + moveCursorTo(0); + } + + public void moveCursorToEnd(boolean notify) { + moveCursorTo(getValue().length(), notify); + } + + public void moveCursorToEnd() { + moveCursorToEnd(true); + } + + public boolean isEditable() { + return isEditable.get(); + } + + public void setFocus(boolean focused) { + if (this.focused && !focused && onEditComplete != null) { + onEditComplete.run(); + } + this.focused = focused; + } + + public boolean isFocused() { + return focused; + } + + public int getCursorPosition() { + return cursorPos; + } + + public void setHighlightPos(int newPos) { + String value = getValue(); + int length = value.length(); + highlightPos = Mth.clamp(newPos, 0, length); + + if (displayPos > length) { + displayPos = length; + } + + int width = (int) xSize(); + String visibleText = font().plainSubstrByWidth(value.substring(displayPos), width); + int endPos = displayPos + visibleText.length(); + if (highlightPos == displayPos) { + displayPos -= font().plainSubstrByWidth(value, width, true).length(); + } + + if (highlightPos > endPos) { + displayPos += highlightPos - endPos; + } else if (highlightPos <= displayPos) { + displayPos -= displayPos - highlightPos; + } + + displayPos = Mth.clamp(displayPos, 0, length); + } + + //=== Input Handling ===// + + @Override + public boolean keyPressed(int key, int scancode, int modifiers) { + if (!canConsumeInput()) { + return false; + } else { + shiftPressed = Screen.hasShiftDown(); + if (Screen.isSelectAll(key)) { + moveCursorToEnd(); + setHighlightPos(0); + return true; + } else if (Screen.isCopy(key)) { + Minecraft.getInstance().keyboardHandler.setClipboard(getHighlighted()); + return true; + } else if (Screen.isPaste(key)) { + if (isEditable()) { + insertText(Minecraft.getInstance().keyboardHandler.getClipboard()); + } + + return true; + } else if (Screen.isCut(key)) { + Minecraft.getInstance().keyboardHandler.setClipboard(getHighlighted()); + if (isEditable()) { + insertText(""); + } + + return true; + } else { + switch (key) { + case InputConstants.KEY_BACKSPACE: + if (isEditable()) { + shiftPressed = false; + deleteText(-1); + shiftPressed = Screen.hasShiftDown(); + } + + return true; + case InputConstants.KEY_NUMPADENTER: + case InputConstants.KEY_RETURN: { + if (onEditComplete != null) { + onEditComplete.run(); + } + if (onEnterPressed != null) { + onEnterPressed.run(); + } + } + case InputConstants.KEY_INSERT: + case InputConstants.KEY_DOWN: + case InputConstants.KEY_UP: + case InputConstants.KEY_PAGEUP: + case InputConstants.KEY_PAGEDOWN: + default: + //Consume key presses when we are typing so we dont do something dumb like close the screen when you type e + return key != GLFW.GLFW_KEY_ESCAPE; + case InputConstants.KEY_DELETE: + if (isEditable()) { + shiftPressed = false; + deleteText(1); + shiftPressed = Screen.hasShiftDown(); + } + return true; + case InputConstants.KEY_RIGHT: + if (Screen.hasControlDown()) { + moveCursorTo(getWordPosition(1)); + } else { + moveCursor(1); + } + return true; + case InputConstants.KEY_LEFT: + if (Screen.hasControlDown()) { + moveCursorTo(getWordPosition(-1)); + } else { + moveCursor(-1); + } + return true; + case InputConstants.KEY_HOME: + moveCursorToStart(); + return true; + case InputConstants.KEY_END: + moveCursorToEnd(); + return true; + } + } + } + } + + public boolean canConsumeInput() { + return isFocused() && isEditable() && isEnabled(); + } + + @Override + public boolean keyReleased(int key, int scancode, int modifiers, boolean consumed) { + this.shiftPressed = Screen.hasShiftDown(); + return super.keyReleased(key, scancode, modifiers, consumed); + } + + @Override + public boolean charTyped(char charTyped, int charCode) { + if (!canConsumeInput()) { + return false; + } else if (SharedConstants.isAllowedChatCharacter(charTyped)) { + if (isEditable()) { + insertText(Character.toString(charTyped)); + } + return true; + } else { + return false; + } + } + + @Override + public boolean mouseClicked(double mouseX, double mouseY, int button, boolean consumed) { + consumed = super.mouseClicked(mouseX, mouseY, button, consumed); + + boolean mouseOver = isMouseOver(); + if (isFocused() && !mouseOver) { + setFocus(isFocusable.get() && !canLoseFocus.get()); + } + + if (consumed) return true; + + if (canLoseFocus.get()) { + setFocus(mouseOver && isFocusable.get()); + } else { + setFocus(isFocusable.get()); + } + + if (isFocused() && mouseOver && button == 0) { + int i = (int) (Mth.floor(mouseX) - xMin()); + String s = font().plainSubstrByWidth(getValue().substring(displayPos), (int) xSize()); + moveCursorTo(font().plainSubstrByWidth(s, i).length() + displayPos); + return true; + } else { + return false; + } + } + + //=== Rendering ===// + + @Override + public double getBackgroundDepth() { + return 0.04; + } + + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + String value = getValue(); + int colour = textColor.get(); + + int textStart = cursorPos - displayPos; + int highlightStart = highlightPos - displayPos; + String displayText = font().plainSubstrByWidth(value.substring(displayPos), (int) xSize()); + boolean flag = textStart >= 0 && textStart <= displayText.length(); + boolean cursorBlink = isFocused() && tick / 6 % 2 == 0 && flag; + double drawX = xMin(); + double drawY = yMin() + ((ySize() - 8) / 2); + int drawEnd = (int) drawX; + + if (highlightStart > displayText.length()) { + highlightStart = displayText.length(); + } + + if (!displayText.isEmpty()) { + String drawString = flag ? displayText.substring(0, textStart) : displayText; + drawEnd = render.drawString(formatter.apply(drawString, displayPos), drawX, drawY, colour, shadow.get()); + } + + boolean flag2 = cursorPos < value.length() || value.length() >= getMaxLength(); + int k1 = drawEnd; + + if (!flag) { + k1 = (int) (textStart > 0 ? drawX + xSize() : drawX); + } else if (flag2) { + k1 = drawEnd - 1; + --drawEnd; + } + + if (!displayText.isEmpty() && flag && textStart < displayText.length()) { + render.drawString(formatter.apply(displayText.substring(textStart), cursorPos), drawEnd, drawY, colour, shadow.get()); + } + + if (suggestion != null && value.isEmpty()) { + render.drawString(suggestion.get(), (float) (k1 - 1), (float) drawY, suggestionColour.get(), shadow.get()); + } + + if (cursorBlink) { + if (flag2) { + render.fill(k1, drawY - 1, k1 + 1, drawY + 1 + 9, -3092272); + } else { + render.drawString("_", (float) k1, (float) drawY, colour, shadow.get()); + } + } + + if (highlightStart != textStart) { + int l1 = (int) (drawX + font().width(displayText.substring(0, highlightStart))); + render.pose().translate(0, 0, 0.035); + render.fill(HIGHLIGHT_TYPE, k1, drawY - 1, l1 - 1, drawY + 1 + 9, 0xFF0000FF); + render.pose().translate(0, 0, -0.035); + } + } + + @Override + public void tick(double mouseX, double mouseY) { + super.tick(mouseX, mouseY); + tick++; + } + + public record TextField(GuiRectangle container, GuiTextField field) {} +} + diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiTextList.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiTextList.java new file mode 100644 index 00000000..865b935c --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiTextList.java @@ -0,0 +1,173 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.ForegroundRender; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.geometry.Align; +import codechicken.lib.gui.modular.lib.geometry.GeoParam; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import net.minecraft.client.gui.Font; +import net.minecraft.network.chat.Component; +import org.jetbrains.annotations.NotNull; + +import java.util.List; +import java.util.function.Supplier; + +import static codechicken.lib.gui.modular.lib.geometry.Align.MAX; +import static codechicken.lib.gui.modular.lib.geometry.Align.MIN; +import static codechicken.lib.gui.modular.lib.geometry.Constraint.dynamic; + +/** + * Created by brandon3055 on 10/10/2023 + */ +public class GuiTextList extends GuiElement implements ForegroundRender { + private Supplier> text; + private Supplier shadow = () -> true; + private Supplier textColour = () -> 0xFFFFFFFF; + private boolean scroll = true; + private Align horizontalAlign = Align.CENTER; + private Align verticalAlign = Align.TOP; + private int lineSpacing = 0; + + /** + * @param parent parent {@link GuiParent}. + */ + public GuiTextList(@NotNull GuiParent parent) { + this(parent, () -> null); + } + + /** + * @param parent parent {@link GuiParent}. + */ + public GuiTextList(@NotNull GuiParent parent, List text) { + this(parent, () -> text); + } + + /** + * @param parent parent {@link GuiParent}. + */ + public GuiTextList(@NotNull GuiParent parent, @NotNull Supplier> text) { + super(parent); + this.text = text; + } + + /** + * Apply a dynamic height constraint that sets the height based on text height (accounting for wrapping) + */ + public GuiTextList autoHeight() { + constrain(GeoParam.HEIGHT, dynamic(() -> (double) ((font().lineHeight + lineSpacing) * getText().size()) - 1)); + return this; + } + + public GuiTextList setTextSupplier(@NotNull Supplier> textSupplier) { + this.text = textSupplier; + return this; + } + + public GuiTextList setText(List text) { + this.text = () -> text; + return this; + } + + public List getText() { + return text.get(); + } + + public GuiTextList setHorizontalAlign(Align horizontalAlign) { + this.horizontalAlign = horizontalAlign; + return this; + } + + public GuiTextList setVerticalAlign(Align verticalAlign) { + this.verticalAlign = verticalAlign; + return this; + } + + public Align getHorizontalAlign() { + return horizontalAlign; + } + + public Align getVerticalAlign() { + return verticalAlign; + } + + public GuiTextList setLineSpacing(int lineSpacing) { + this.lineSpacing = lineSpacing; + return this; + } + + public int getLineSpacing() { + return lineSpacing; + } + + /** + * Set to true the text scroll using the same logic as vanillas widgets if the text is too long to fit within the elements bounds + * Default enabled. + * Setting this to true will automatically disable trim and wrap ether are enabled. + */ + public GuiTextList setScroll(boolean scroll) { + this.scroll = scroll; + return this; + } + + public boolean getScroll() { + return scroll; + } + + public GuiTextList setShadow(@NotNull Supplier shadow) { + this.shadow = shadow; + return this; + } + + public GuiTextList setShadow(boolean shadow) { + this.shadow = () -> shadow; + return this; + } + + public boolean getShadow() { + return shadow.get(); + } + + public GuiTextList setTextColour(Supplier textColour) { + this.textColour = textColour; + return this; + } + + public GuiTextList setTextColour(int textColour) { + this.textColour = () -> textColour; + return this; + } + + public int getTextColour() { + return textColour.get(); + } + + @Override + public double getForegroundDepth() { + return 0.035; + } + + @Override + public void renderForeground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + List list = getText(); + if (list.isEmpty()) return; + Font font = render.font(); + + double height = (list.size() * (font.lineHeight + lineSpacing)) - lineSpacing; + double yPos = verticalAlign == MIN ? yMin() : verticalAlign == MAX ? yMax() - height : (yCenter() - (height / 2)) + 1; + for (Component line : list) { + int textWidth = font.width(line); + boolean tooLong = textWidth > xSize(); + + if (tooLong && scroll) { + render.pushScissorRect(getRectangle()); + render.drawScrollingString(line, xMin(), yPos, xMax(), getTextColour(), getShadow(), false); + render.popScissor(); + } else { + double xPos = horizontalAlign == MIN ? xMin() : horizontalAlign == MAX ? xMax() - textWidth : xMin() + xSize() / 2 - textWidth / 2D; + render.drawString(line, xPos, yPos, getTextColour(), getShadow()); + } + + yPos += font.lineHeight + lineSpacing; + } + } +} \ No newline at end of file diff --git a/src/main/java/codechicken/lib/gui/modular/elements/GuiTexture.java b/src/main/java/codechicken/lib/gui/modular/elements/GuiTexture.java new file mode 100644 index 00000000..da5c0577 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/elements/GuiTexture.java @@ -0,0 +1,110 @@ +package codechicken.lib.gui.modular.elements; + +import codechicken.lib.gui.modular.lib.BackgroundRender; +import codechicken.lib.gui.modular.lib.GuiRender; +import codechicken.lib.gui.modular.lib.geometry.Borders; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.gui.modular.sprite.Material; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.function.Supplier; + +/** + * Created by brandon3055 on 28/08/2023 + */ +public class GuiTexture extends GuiElement implements BackgroundRender { + private Supplier getMaterial; + private Supplier colour = () -> 0xFFFFFFFF; + private Borders dynamicBorders = null; + + /** + * @param parent parent {@link GuiParent}. + */ + public GuiTexture(@NotNull GuiParent parent) { + super(parent); + } + + public GuiTexture(@NotNull GuiParent parent, Supplier supplier) { + super(parent); + setMaterial(supplier); + } + + public GuiTexture(@NotNull GuiParent parent, Material material) { + super(parent); + setMaterial(material); + } + + public GuiTexture setMaterial(Supplier supplier) { + this.getMaterial = supplier; + return this; + } + + public GuiTexture setMaterial(Material material) { + this.getMaterial = () -> material; + return this; + } + + @Nullable + public Material getMaterial() { + return getMaterial == null ? null : getMaterial.get(); + } + + /** + * Enables dynamic texture resizing though the use of cutting and tiling. + * Only works with textures that can be cut up and tiled without issues, e.g. background textures or button textures. + * This method uses the standard border with of 5 pixels on all sides. + */ + public GuiTexture dynamicTexture() { + return dynamicTexture(5); + } + + /** + * Enables dynamic texture resizing though the use of cutting and tiling. + * Only works with textures that can be cut up and tiled without issues, e.g. background textures or button textures. + * The border parameters indicate the width of border around the texture that must be maintained during the cutting and tiling process. + * For standardisation purposes the border width should be >= 5 + */ + public GuiTexture dynamicTexture(int textureBorders) { + return dynamicTexture(Borders.create(textureBorders)); + } + + /** + * Enables dynamic texture resizing though the use of cutting and tiling. + * Only works with textures that can be cut up and tiled without issues, e.g. background textures or button textures. + * The border parameters indicate the width of border around the texture that must be maintained during the cutting and tiling process. + * For standardisation purposes the border width should be >= 5 + */ + public GuiTexture dynamicTexture(Borders textureBorders) { + dynamicBorders = textureBorders; + return this; + } + + /** + * Allows you to set an argb colour. + * This colour will be applied when rendering the texture. + */ + public GuiTexture setColour(int colourARGB) { + return setColour(() -> colourARGB); + } + + /** + * Allows you to set an argb colour provider. + * This colour will be applied when rendering the texture. + */ + public GuiTexture setColour(Supplier colour) { + this.colour = colour; + return this; + } + + @Override + public void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks) { + Material material = getMaterial(); + if (material == null) return; + if (dynamicBorders != null) { + render.dynamicTex(material, getRectangle(), dynamicBorders, colour.get()); + } else { + render.texRect(material, getRectangle(), colour.get()); + } + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/BackgroundRender.java b/src/main/java/codechicken/lib/gui/modular/lib/BackgroundRender.java new file mode 100644 index 00000000..0291e8af --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/BackgroundRender.java @@ -0,0 +1,34 @@ +package codechicken.lib.gui.modular.lib; + +import com.mojang.blaze3d.vertex.PoseStack; + +/** + * Allows a Gui Elements to render content behind child elements. + * This is the default render mode for the majority of elements. + *

+ * Created by brandon3055 on 07/08/2023 + */ +public interface BackgroundRender { + + /** + * Specifies the z depth of the background content. + * After {@link #renderBackground(GuiRender, double, double, float)} is called, the PoseStack will be translated by this amount in the z direction + * before any assigned child elements are rendered. + * Recommended minimum depth is 0.01 or 0.035 if this element renders text. (text shadows are rendered with a 0.03 offset) + * + * @return the z height of the background content. + */ + default double getBackgroundDepth() { + return 0.01; + } + + /** + * Used to render content behind this elements child elements. + * When rendering element content, always use the {@link PoseStack} available via the provided {@link GuiRender} + * Where applicable, always use push/pop to ensure the stack is returned to its original state after your rendering is complete. + * + * @param render Contains gui context information as well as essential render methods/utils including the PoseStack. + */ + void renderBackground(GuiRender render, double mouseX, double mouseY, float partialTicks); + +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/ColourState.java b/src/main/java/codechicken/lib/gui/modular/lib/ColourState.java new file mode 100644 index 00000000..850e0f14 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/ColourState.java @@ -0,0 +1,72 @@ +package codechicken.lib.gui.modular.lib; + +import codechicken.lib.colour.Colour; +import codechicken.lib.colour.ColourARGB; + +import java.util.Locale; +import java.util.function.Consumer; +import java.util.function.Supplier; + +/** + * Created by brandon3055 on 19/11/2023 + */ +public interface ColourState { + + int get(); + + default ColourARGB getColour() { + return new ColourARGB(get()); + } + + void set(int colour); + + default void set(Colour colour) { + set(colour.argb()); + } + + default String getHexColour() { + return Integer.toHexString(get()).toUpperCase(Locale.ROOT); + } + + default void setHexColour(String hexColour) { + try { + set(Integer.parseUnsignedInt(hexColour, 16)); + } catch (Throwable e) { + set(0); + } + } + + static ColourState create() { + return create(null); + } + + static ColourState create(Consumer listener) { + return new ColourState() { + int colour = 0; + @Override + public int get() { + return colour; + } + + @Override + public void set(int colour) { + this.colour = colour; + if (listener != null) listener.accept(colour); + } + }; + } + + static ColourState create(Supplier getter, Consumer setter) { + return new ColourState() { + @Override + public int get() { + return getter.get(); + } + + @Override + public void set(int colour) { + setter.accept(colour); + } + }; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/Constraints.java b/src/main/java/codechicken/lib/gui/modular/lib/Constraints.java new file mode 100644 index 00000000..1cb4de5b --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/Constraints.java @@ -0,0 +1,209 @@ +package codechicken.lib.gui.modular.lib; + +import codechicken.lib.gui.modular.lib.geometry.Borders; +import codechicken.lib.gui.modular.lib.geometry.ConstrainedGeometry; + +import java.util.function.Supplier; + +import static codechicken.lib.gui.modular.lib.geometry.Constraint.*; +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * This class contains a bunch of static helper methods that can be used to quickly apply common constraints. + * The plan is to keep adding common constraints to this class as they pop up. + *

+ * Created by brandon3055 on 28/08/2023 + */ +public class Constraints { + + /** + * Bind an elements geometry to a reference element. + * + * @param element The element to be bound. + * @param reference The element to be bound to. + */ + public static void bind(ConstrainedGeometry element, ConstrainedGeometry reference) { + bind(element, reference, 0.0); + } + + /** + * Bind an elements geometry to a reference element with border offsets. (Border offsets may be positive or negative) + * + * @param element The element to be bound. + * @param reference The element to be bound to. + */ + public static void bind(ConstrainedGeometry element, ConstrainedGeometry reference, double borders) { + bind(element, reference, borders, borders, borders, borders); + } + + /** + * Bind an elements geometry to a reference element with border offsets. (Border offsets may be positive or negative) + * + * @param element The element to be bound. + * @param reference The element to be bound to. + */ + public static void bind(ConstrainedGeometry element, ConstrainedGeometry reference, double top, double left, double bottom, double right) { + element.constrain(TOP, relative(reference.get(TOP), top)); + element.constrain(LEFT, relative(reference.get(LEFT), left)); + element.constrain(BOTTOM, relative(reference.get(BOTTOM), -bottom)); + element.constrain(RIGHT, relative(reference.get(RIGHT), -right)); + } + + /** + * Bind an elements geometry to a reference element with border offsets. (Border offsets may be positive or negative) + * The border offsets are dynamic, meaning if the values stored in the {@link Borders} object are updated, this binding will reflect those changes automatically. + * + * @param element The element to be bound. + * @param reference The element to be bound to. + * @param borders Border offsets. + */ + public static void bind(ConstrainedGeometry element, ConstrainedGeometry reference, Borders borders) { + element.constrain(TOP, relative(reference.get(TOP), borders::top)); + element.constrain(LEFT, relative(reference.get(LEFT), borders::left)); + element.constrain(BOTTOM, relative(reference.get(BOTTOM), () -> -borders.bottom())); + element.constrain(RIGHT, relative(reference.get(RIGHT), () -> -borders.right())); + } + + + public static void size(ConstrainedGeometry element, double width, double height) { + element.constrain(WIDTH, literal(width)); + element.constrain(HEIGHT, literal(height)); + } + + public static void size(ConstrainedGeometry element, Supplier width, Supplier height) { + element.constrain(WIDTH, dynamic(width)); + element.constrain(HEIGHT, dynamic(height)); + } + + public static void pos(ConstrainedGeometry element, double left, double top) { + element.constrain(LEFT, literal(left)); + element.constrain(TOP, literal(top)); + } + + public static void pos(ConstrainedGeometry element, Supplier left, Supplier top) { + element.constrain(LEFT, dynamic(left)); + element.constrain(TOP, dynamic(top)); + } + + public static void center(ConstrainedGeometry element, ConstrainedGeometry centerOn) { + element.constrain(TOP, midPoint(centerOn.get(TOP), centerOn.get(BOTTOM), () -> element.ySize() / -2)); + element.constrain(LEFT, midPoint(centerOn.get(LEFT), centerOn.get(RIGHT), () -> element.xSize() / -2)); + } + + /** + * Constrain the specified element to a position inside the specified targetElement. + * See the following image for an example of what each LayoutPos does: + * https://ss.brandon3055.com/e89a6 + * + * @param target The element whose position we are setting. + * @param reference The reference element, the element that target will be placed inside. + * @param position The layout position. + */ + public static void placeInside(ConstrainedGeometry target, ConstrainedGeometry reference, LayoutPos position) { + placeInside(target, reference, position, 0, 0); + } + + /** + * Constrain the specified element to a position inside the specified targetElement. + * See the following image for an example of what each LayoutPos does: + * https://ss.brandon3055.com/e89a6 + * + * @param target The element whose position we are setting. + * @param reference The reference element, the element that target will be placed inside. + * @param position The layout position. + * @param xOffset Optional X offset to be applied to the final position. + * @param yOffset Optional Y offset to be applied to the final position. + */ + public static void placeInside(ConstrainedGeometry target, ConstrainedGeometry reference, LayoutPos position, double xOffset, double yOffset) { + switch (position) { + case TOP_LEFT -> target + .constrain(TOP, relative(reference.get(TOP), yOffset)) + .constrain(LEFT, relative(reference.get(LEFT), xOffset)); + case TOP_CENTER -> target + .constrain(TOP, relative(reference.get(TOP), yOffset)) + .constrain(LEFT, midPoint(reference.get(LEFT), reference.get(RIGHT), () -> (target.xSize() / -2) + xOffset)); + case TOP_RIGHT -> target + .constrain(TOP, relative(reference.get(TOP), yOffset)) + .constrain(RIGHT, relative(reference.get(RIGHT), xOffset)); + case MIDDLE_RIGHT -> target + .constrain(TOP, midPoint(reference.get(TOP), reference.get(BOTTOM), () -> (target.ySize() / -2) + yOffset)) + .constrain(RIGHT, relative(reference.get(RIGHT), xOffset)); + case BOTTOM_RIGHT -> target + .constrain(BOTTOM, relative(reference.get(BOTTOM), yOffset)) + .constrain(RIGHT, relative(reference.get(RIGHT), xOffset)); + case BOTTOM_CENTER -> target + .constrain(BOTTOM, relative(reference.get(BOTTOM), yOffset)) + .constrain(LEFT, midPoint(reference.get(LEFT), reference.get(RIGHT), () -> (target.xSize() / -2) + xOffset)); + case BOTTOM_LEFT -> target + .constrain(BOTTOM, relative(reference.get(BOTTOM), yOffset)) + .constrain(LEFT, relative(reference.get(LEFT), xOffset)); + case MIDDLE_LEFT -> target + .constrain(TOP, midPoint(reference.get(TOP), reference.get(BOTTOM), () -> (target.ySize() / -2) + yOffset)) + .constrain(LEFT, relative(reference.get(LEFT), xOffset)); + } + } + + /** + * Constrain the specified element to a position outside the specified targetElement. + * See the following image for an example of what each LayoutPos does: + * https://ss.brandon3055.com/baa7c + * + * @param target The element whose position we are setting. + * @param reference The reference element, the element that target will be placed outside of. + * @param position The layout position. + */ + public static void placeOutside(ConstrainedGeometry target, ConstrainedGeometry reference, LayoutPos position) { + placeOutside(target, reference, position, 0, 0); + } + + /** + * Constrain the specified element to a position outside the specified targetElement. + * See the following image for an example of what each LayoutPos does: + * https://ss.brandon3055.com/baa7c + * + * @param target The element whose position we are setting. + * @param reference The reference element, the element that target will be placed outside of. + * @param position The layout position. + * @param xOffset Optional X offset to be applied to the final position. + * @param yOffset Optional Y offset to be applied to the final position. + */ + public static void placeOutside(ConstrainedGeometry target, ConstrainedGeometry reference, LayoutPos position, double xOffset, double yOffset) { + switch (position) { + case TOP_LEFT -> target + .constrain(BOTTOM, relative(reference.get(TOP), yOffset)) + .constrain(RIGHT, relative(reference.get(LEFT), xOffset)); + case TOP_CENTER -> target + .constrain(BOTTOM, relative(reference.get(TOP), yOffset)) + .constrain(LEFT, midPoint(reference.get(LEFT), reference.get(RIGHT), () -> (target.xSize() / -2) + xOffset)); + case TOP_RIGHT -> target + .constrain(BOTTOM, relative(reference.get(TOP), yOffset)) + .constrain(LEFT, relative(reference.get(RIGHT), xOffset)); + case MIDDLE_RIGHT -> target + .constrain(TOP, midPoint(reference.get(TOP), reference.get(BOTTOM), () -> (target.ySize() / -2) + yOffset)) + .constrain(LEFT, relative(reference.get(RIGHT), xOffset)); + case BOTTOM_RIGHT -> target + .constrain(TOP, relative(reference.get(BOTTOM), yOffset)) + .constrain(LEFT, relative(reference.get(RIGHT), xOffset)); + case BOTTOM_CENTER -> target + .constrain(TOP, relative(reference.get(BOTTOM), yOffset)) + .constrain(LEFT, midPoint(reference.get(LEFT), reference.get(RIGHT), () -> (target.xSize() / -2) + xOffset)); + case BOTTOM_LEFT -> target + .constrain(TOP, relative(reference.get(BOTTOM), yOffset)) + .constrain(RIGHT, relative(reference.get(LEFT), xOffset)); + case MIDDLE_LEFT -> target + .constrain(TOP, midPoint(reference.get(TOP), reference.get(BOTTOM), () -> (target.ySize() / -2) + yOffset)) + .constrain(RIGHT, relative(reference.get(LEFT), xOffset)); + } + } + + public enum LayoutPos { + TOP_LEFT, + TOP_CENTER, + TOP_RIGHT, + MIDDLE_RIGHT, + MIDDLE_LEFT, + BOTTOM_RIGHT, + BOTTOM_CENTER, + BOTTOM_LEFT + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/ContentElement.java b/src/main/java/codechicken/lib/gui/modular/lib/ContentElement.java new file mode 100644 index 00000000..6db976a5 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/ContentElement.java @@ -0,0 +1,14 @@ +package codechicken.lib.gui.modular.lib; + + +import codechicken.lib.gui.modular.elements.GuiElement; + +/** + * Implemented by elements that have a separate child element to which content should be added. + * e.g. scroll element, manipulable element etc... + * Created by brandon3055 on 13/11/2023 + */ +public interface ContentElement> { + + T getContentElement(); +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/CursorHelper.java b/src/main/java/codechicken/lib/gui/modular/lib/CursorHelper.java new file mode 100644 index 00000000..4ef4cd96 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/CursorHelper.java @@ -0,0 +1,100 @@ +package codechicken.lib.gui.modular.lib; + +import codechicken.lib.CodeChickenLib; +import net.minecraft.client.Minecraft; +import net.minecraft.resources.ResourceLocation; +import net.minecraft.server.packs.resources.Resource; +import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; +import org.jetbrains.annotations.Nullable; +import org.lwjgl.BufferUtils; +import org.lwjgl.glfw.GLFW; +import org.lwjgl.glfw.GLFWImage; +import org.lwjgl.system.MemoryUtil; + +import javax.imageio.ImageIO; +import java.awt.*; +import java.awt.image.BufferedImage; +import java.nio.ByteBuffer; +import java.util.HashMap; +import java.util.Map; +import java.util.Optional; + +/** + * Created by brandon3055 on 11/5/20. + */ +public class CursorHelper { + public static final Logger LOGGER = LogManager.getLogger(); + + public static final ResourceLocation DRAG = new ResourceLocation(CodeChickenLib.MOD_ID, "textures/gui/cursors/drag.png"); + public static final ResourceLocation RESIZE_H = new ResourceLocation(CodeChickenLib.MOD_ID, "textures/gui/cursors/resize_h.png"); + public static final ResourceLocation RESIZE_V = new ResourceLocation(CodeChickenLib.MOD_ID, "textures/gui/cursors/resize_v.png"); + public static final ResourceLocation RESIZE_TRBL = new ResourceLocation(CodeChickenLib.MOD_ID, "textures/gui/cursors/resize_diag_trbl.png"); + public static final ResourceLocation RESIZE_TLBR = new ResourceLocation(CodeChickenLib.MOD_ID, "textures/gui/cursors/resize_diag_tlbr.png"); + + private static final Map cursors = new HashMap<>(); + private static ResourceLocation active = null; + + private static long createCursor(ResourceLocation cursorTexture) { + try { + Resource resource = Minecraft.getInstance().getResourceManager().getResource(cursorTexture).orElse(null); + if (resource == null) return MemoryUtil.NULL; + BufferedImage bufferedimage = ImageIO.read(resource.open()); + GLFWImage glfwImage = imageToGLFWImage(bufferedimage); + return GLFW.glfwCreateCursor(glfwImage, 16, 16); + } catch (Exception e) { + LOGGER.warn("An error occurred while creating cursor", e); + } + return 0; + } + + private static GLFWImage imageToGLFWImage(BufferedImage image) { + if (image.getType() != BufferedImage.TYPE_INT_ARGB_PRE) { + final BufferedImage convertedImage = new BufferedImage(image.getWidth(), image.getHeight(), BufferedImage.TYPE_INT_ARGB_PRE); + final Graphics2D graphics = convertedImage.createGraphics(); + final int targetWidth = image.getWidth(); + final int targetHeight = image.getHeight(); + graphics.drawImage(image, 0, 0, targetWidth, targetHeight, null); + graphics.dispose(); + image = convertedImage; + } + final ByteBuffer buffer = BufferUtils.createByteBuffer(image.getWidth() * image.getHeight() * 4); + for (int i = 0; i < image.getHeight(); i++) { + for (int j = 0; j < image.getWidth(); j++) { + int colorSpace = image.getRGB(j, i); + buffer.put((byte) ((colorSpace << 8) >> 24)); + buffer.put((byte) ((colorSpace << 16) >> 24)); + buffer.put((byte) ((colorSpace << 24) >> 24)); + buffer.put((byte) (colorSpace >> 24)); + } + } + buffer.flip(); + final GLFWImage result = GLFWImage.create(); + result.set(image.getWidth(), image.getHeight(), buffer); + return result; + } + + public static void setCursor(@Nullable ResourceLocation cursor) { + if (cursor != active) { + active = cursor; + long window = Minecraft.getInstance().getWindow().getWindow(); + long newCursor = active == null ? 0 : cursors.computeIfAbsent(cursor, CursorHelper::createCursor); + GLFW.glfwSetCursor(window, newCursor); + } + } + + public static void resetCursor() { + if (active != null) { + setCursor(null); + } + } + + public static void onResourceReload() { + cursors.values().forEach(cursor -> { + if (cursor != MemoryUtil.NULL) { + GLFW.glfwDestroyCursor(cursor); + } + }); + cursors.clear(); + } +} \ No newline at end of file diff --git a/src/main/java/codechicken/lib/gui/modular/lib/DynamicTextures.java b/src/main/java/codechicken/lib/gui/modular/lib/DynamicTextures.java new file mode 100644 index 00000000..aa15ddfa --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/DynamicTextures.java @@ -0,0 +1,30 @@ +package codechicken.lib.gui.modular.lib; + +import net.minecraft.resources.ResourceLocation; + +import java.util.function.Function; + +/** + * Designed for use with DynamicTextureProvider + *

+ * Created by brandon3055 on 07/09/2023 + */ +public interface DynamicTextures { + + void makeTextures(Function textures); + + default String dynamicTexture(Function textures, ResourceLocation dynamicInput, ResourceLocation outputLocation, int width, int height, int border) { + return textures.apply(new DynamicTexture(dynamicInput, outputLocation, width, height, border, border, border, border)); + } + + default String dynamicTexture(Function textures, ResourceLocation dynamicInput, ResourceLocation outputLocation, int width, int height, int topBorder, int leftBorder, int bottomBorder, int rightBorder) { + return textures.apply(new DynamicTexture(dynamicInput, outputLocation, width, height, topBorder, leftBorder, bottomBorder, rightBorder)); + } + + record DynamicTexture(ResourceLocation dynamicInput, ResourceLocation outputLocation, int width, int height, int topBorder, int leftBorder, int bottomBorder, int rightBorder) { + public String guiTexturePath() { + return outputLocation.getPath().replace("textures/gui/", "").replace(".gui", ""); + } + } + +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/ElementEvents.java b/src/main/java/codechicken/lib/gui/modular/lib/ElementEvents.java new file mode 100644 index 00000000..d4edcbe1 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/ElementEvents.java @@ -0,0 +1,272 @@ +package codechicken.lib.gui.modular.lib; + +import com.google.common.collect.Lists; +import codechicken.lib.gui.modular.elements.GuiElement; + +import java.util.List; + +/** + * This class defines the default implementation for all Screen events. + * Input events in Modular GUI v2 work similar to v2, events are passed to all elements recursively in a top-down order, + * and if any element 'consumes' the event, it will not be passed any further down the chain. + * However, this approach had issues in v2, because there are certain situations where an element needs to receive an event even if it has been consumed. + * To deal with that we now have two methods for each event, the main handler method that will always get called, and uses a 'consumed' flag to track whether the event has been consumed, + * as well as a simpler convenience method that will only get called if the event has not already been canceled, and does not require you to call super. + *

+ * Created by brandon3055 on 09/08/2023 + */ +public interface ElementEvents { + + /** + * @return An unmodifiable list of all assigned child elements assigned to this parent. The list should be sorted in the order they were added. + */ + List> getChildren(); + + //=== Mouse Events ==// + + /** + * Called whenever the cursor position changes. + * Vanillas mouseDragged is not passed through because it is redundant. + * All mouse drag functionality can be archived using available events. + * + * @param mouseX new mouse X position + * @param mouseY new mouse Y position + */ + default void mouseMoved(double mouseX, double mouseY) { + for (GuiElement child : Lists.reverse(getChildren())) { + if (child.isEnabled()) { + child.mouseMoved(mouseX, mouseY); + } + } + } + + /** + * Override this method to implement handling for the mouseClicked event. + * This event propagates through the entire gui element stack from top to bottom, If eny element consumes the event it will not propagate any further. + * For rare cases where you need to receive this even if it has been consumed, you can override {@link #mouseClicked(double, double, int, boolean)} + *

+ * Note: You do not need to call super when overriding this interface method. + * + * @param mouseX Mouse X position + * @param mouseY Mouse Y position + * @param button Mouse Button + * @return true to consume event. + */ + default boolean mouseClicked(double mouseX, double mouseY, int button) { + return false; + } + + /** + * Root handler for mouseClick event. This method will always be called for all elements even if the event has already been consumed. + * There are a few uses for this method, but the fast majority of mouseClick handling should be implemented via {@link #mouseClicked(double, double, int)} + *

+ * Note: If overriding this method, do so with caution, You must either return true (if you wish to consume the event) or you must return the result of the super call. + * + * @param mouseX Mouse X position + * @param mouseY Mouse Y position + * @param button Mouse Button + * @param consumed Will be true if this action has already been consumed. + * @return true if this event has been consumed. + */ + default boolean mouseClicked(double mouseX, double mouseY, int button, boolean consumed) { + for (GuiElement child : Lists.reverse(getChildren())) { + if (child.isEnabled()) { + consumed |= child.mouseClicked(mouseX, mouseY, button, consumed); + } + } + return consumed || mouseClicked(mouseX, mouseY, button) || blockMouseEvents(); + } + + /** + * Override this method to implement handling for the mouseReleased event. + * This event propagates through the entire gui element stack from top to bottom, If eny element consumes the event it will not propagate any further. + * For rare cases where you need to receive this even if it has been consumed, you can override {@link #mouseReleased(double, double, int, boolean)} + *

+ * Note: You do not need to call super when overriding this interface method. + * + * @param mouseX Mouse X position + * @param mouseY Mouse Y position + * @param button Mouse Button + * @return true to consume event. + */ + default boolean mouseReleased(double mouseX, double mouseY, int button) { + return false; + } + + /** + * Root handler for mouseReleased event. This method will always be called for all elements even if the event has already been consumed. + * There are a few uses for this method, but the fast majority of mouseReleased handling should be implemented via {@link #mouseReleased(double, double, int)} + *

+ * Note: If overriding this method, do so with caution, You must either return true (if you wish to consume the event) or you must return the result of the super call. + * + * @param mouseX Mouse X position + * @param mouseY Mouse Y position + * @param button Mouse Button + * @param consumed Will be true if this action has already been consumed. + * @return true if this event has been consumed. + */ + default boolean mouseReleased(double mouseX, double mouseY, int button, boolean consumed) { + for (GuiElement child : Lists.reverse(getChildren())) { + if (child.isEnabled()) { + consumed |= child.mouseReleased(mouseX, mouseY, button, consumed); + } + } + return consumed || mouseReleased(mouseX, mouseY, button) || blockMouseEvents(); + } + + /** + * Override this method to implement handling for the mouseScrolled event. + * This event propagates through the entire gui element stack from top to bottom, If eny element consumes the event it will not propagate any further. + * For rare cases where you need to receive this even if it has been consumed, you can override {@link #mouseScrolled(double, double, double, boolean)} + *

+ * Note: You do not need to call super when overriding this interface method. + * + * @param mouseX Mouse X position + * @param mouseY Mouse Y position + * @param scroll Scroll direction and amount + * @return true to consume event. + */ + default boolean mouseScrolled(double mouseX, double mouseY, double scroll) { + return false; + } + + /** + * Root handler for mouseScrolled event. This method will always be called for all elements even if the event has already been consumed. + * There are a few uses for this method, but the fast majority of mouseScrolled handling should be implemented via {@link #mouseScrolled(double, double, double)} + *

+ * Note: If overriding this method, do so with caution, You must either return true (if you wish to consume the event) or you must return the result of the super call. + * + * @param mouseX Mouse X position + * @param mouseY Mouse Y position + * @param scroll Scroll direction and amount + * @param consumed Will be true if this action has already been consumed. + * @return true if this event has been consumed. + */ + default boolean mouseScrolled(double mouseX, double mouseY, double scroll, boolean consumed) { + for (GuiElement child : Lists.reverse(getChildren())) { + if (child.isEnabled()) { + consumed |= child.mouseScrolled(mouseX, mouseY, scroll, consumed); + } + } + return consumed || mouseScrolled(mouseX, mouseY, scroll) || blockMouseEvents(); + } + + /** + * @return True to prevent mouse events from being passed to elements bellow this element. + */ + default boolean blockMouseEvents() { + return false; + } + + //=== Keyboard Events ==// + + /** + * Override this method to implement handling for the keyPressed event. + * This event propagates through the entire gui element stack from top to bottom, If eny element consumes the event it will not propagate any further. + * For rare cases where you need to receive this even if it has been consumed, you can override {@link #keyPressed(int, int, int, boolean)} + *

+ * Note: You do not need to call super when overriding this interface method. + * + * @param key the keyboard key that was pressed. + * @param scancode the system-specific scancode of the key + * @param modifiers bitfield describing which modifier keys were held down. + * @return true to consume event. + */ + default boolean keyPressed(int key, int scancode, int modifiers) { + return false; + } + + /** + * Root handler for keyPressed event. This method will always be called for all elements even if the event has already been consumed. + * There are a few uses for this method, but the fast majority of keyPressed handling should be implemented via {@link #keyPressed(int, int, int)} + *

+ * Note: If overriding this method, do so with caution, You must either return true (if you wish to consume the event) or you must return the result of the super call. + * + * @param key the keyboard key that was pressed. + * @param scancode the system-specific scancode of the key + * @param modifiers bitfield describing which modifier keys were held down. + * @param consumed Will be true if this action has already been consumed. + * @return true if this event has been consumed. + */ + default boolean keyPressed(int key, int scancode, int modifiers, boolean consumed) { + for (GuiElement child : Lists.reverse(getChildren())) { + if (child.isEnabled()) { + consumed |= child.keyPressed(key, scancode, modifiers, consumed); + } + } + return consumed || keyPressed(key, scancode, modifiers); + } + + /** + * Override this method to implement handling for the keyReleased event. + * This event propagates through the entire gui element stack from top to bottom, If eny element consumes the event it will not propagate any further. + * For rare cases where you need to receive this even if it has been consumed, you can override {@link #keyReleased(int, int, int, boolean)} + *

+ * Note: You do not need to call super when overriding this interface method. + * + * @param key the keyboard key that was released. + * @param scancode the system-specific scancode of the key + * @param modifiers bitfield describing which modifier keys were held down. + * @return true to consume event. + */ + default boolean keyReleased(int key, int scancode, int modifiers) { + return false; + } + + /** + * Root handler for keyReleased event. This method will always be called for all elements even if the event has already been consumed. + * There are a few uses for this method, but the fast majority of keyReleased handling should be implemented via {@link #keyReleased(int, int, int)} + *

+ * Note: If overriding this method, do so with caution, You must either return true (if you wish to consume the event) or you must return the result of the super call. + * + * @param key the keyboard key that was released. + * @param scancode the system-specific scancode of the key + * @param modifiers bitfield describing which modifier keys were held down. + * @param consumed Will be true if this action has already been consumed. + * @return true if this event has been consumed. + */ + default boolean keyReleased(int key, int scancode, int modifiers, boolean consumed) { + for (GuiElement child : Lists.reverse(getChildren())) { + if (child.isEnabled()) { + consumed |= child.keyReleased(key, scancode, modifiers, consumed); + } + } + return consumed || keyReleased(key, scancode, modifiers); + } + + /** + * Override this method to implement handling for the charTyped event. + * This event propagates through the entire gui element stack from top to bottom, If eny element consumes the event it will not propagate any further. + * For rare cases where you need to receive this even if it has been consumed, you can override {@link #charTyped(char, int, boolean)} + *

+ * Note: You do not need to call super when overriding this interface method. + * + * @param character The character typed. + * @param modifiers bitfield describing which modifier keys were held down. + * @return true to consume event. + */ + default boolean charTyped(char character, int modifiers) { + return false; + } + + /** + * Root handler for charTyped event. This method will always be called for all elements even if the event has already been consumed. + * There are a few uses for this method, but the fast majority of charTyped handling should be implemented via {@link #charTyped(char, int)} + *

+ * Note: If overriding this method, do so with caution, You must either return true (if you wish to consume the event) or you must return the result of the super call. + * + * @param character The character typed. + * @param modifiers bitfield describing which modifier keys were held down. + * @param consumed Will be true if this action has already been consumed. + * @return true if this event has been consumed. + */ + default boolean charTyped(char character, int modifiers, boolean consumed) { + for (GuiElement child : Lists.reverse(getChildren())) { + if (child.isEnabled()) { + consumed |= child.charTyped(character, modifiers, consumed); + } + } + return consumed || charTyped(character, modifiers); + } + +} \ No newline at end of file diff --git a/src/main/java/codechicken/lib/gui/modular/lib/ForegroundRender.java b/src/main/java/codechicken/lib/gui/modular/lib/ForegroundRender.java new file mode 100644 index 00000000..76dfcfb3 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/ForegroundRender.java @@ -0,0 +1,32 @@ +package codechicken.lib.gui.modular.lib; + +import com.mojang.blaze3d.vertex.PoseStack; + +/** + * Allows a Gui Elements to render content in front of child elements. + * Note: Most elements should use {@link BackgroundRender} to render their content. + *

+ * Created by brandon3055 on 07/08/2023 + */ +public interface ForegroundRender { + + /** + * Specifies the z depth of the foreground content. + * Used when calculating the total depth of this gui element. + * Recommended minimum depth is 0.01 or 0.035 if this element renders text. (text shadows are rendered with a 0.03 offset) + * + * @return the z height of the background content. + */ + default double getForegroundDepth() { + return 0.01; + } + + /** + * Used to render content in front of this elements child elements. + * When rendering element content, always use the {@link PoseStack} available via the provided {@link GuiRender} + * Where applicable, always use push/pop to ensure the stack is returned to its original state after your rendering is complete. + * + * @param render Contains gui context information as well as essential render methods/utils including the PoseStack. + */ + void renderForeground(GuiRender render, double mouseX, double mouseY, float partialTicks); +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/GuiProvider.java b/src/main/java/codechicken/lib/gui/modular/lib/GuiProvider.java new file mode 100644 index 00000000..32dbcbb5 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/GuiProvider.java @@ -0,0 +1,42 @@ +package codechicken.lib.gui.modular.lib; + +import codechicken.lib.gui.modular.ModularGui; +import codechicken.lib.gui.modular.elements.GuiElement; +import codechicken.lib.gui.modular.lib.container.ContainerGuiProvider; + +/** + * This interface is used to build modular gui Screens. + * For modular gui container screens use {@link ContainerGuiProvider} + * + * Created by brandon3055 on 19/08/2023 + */ +public interface GuiProvider { + + /** + * Override this to defile a custom root gui element. + * Useful if you want to use something like a background texture or a manipulable element as the root element. + * + * @param gui The modular GUI. + * @return the root gui element. + */ + default GuiElement createRootElement(ModularGui gui) { + return new GuiElement<>(gui); + } + + /** + * Use this method to build the modular gui. + *

+ * Initialize the gui root element with {@link ModularGui#initStandardGui(int, int)}, {@link ModularGui#initFullscreenGui()} + * This applies bindings to fix the size and position of the root element, you can also do this manually for custom configurations. + *

+ * Build your gui by adding and configuring your desired gui elements. + * Elements must be added to the root gui element which is obtainable via gui.getRoot() + *

+ * Note: gui elements are added on construction, meaning you do not need to use element.addChild. + * Instead, just construct the elements, and pass in the root element (or any other initialized element) as the parent. + * + * @param gui The modular gui instance. + */ + void buildGui(ModularGui gui); + +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/GuiRender.java b/src/main/java/codechicken/lib/gui/modular/lib/GuiRender.java new file mode 100644 index 00000000..a91b7bd8 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/GuiRender.java @@ -0,0 +1,1728 @@ +package codechicken.lib.gui.modular.lib; + +import codechicken.lib.gui.modular.lib.geometry.Borders; +import codechicken.lib.gui.modular.lib.geometry.Rectangle; +import codechicken.lib.gui.modular.sprite.Material; +import com.mojang.blaze3d.platform.Lighting; +import com.mojang.blaze3d.systems.RenderSystem; +import com.mojang.blaze3d.vertex.DefaultVertexFormat; +import com.mojang.blaze3d.vertex.PoseStack; +import com.mojang.blaze3d.vertex.VertexConsumer; +import com.mojang.blaze3d.vertex.VertexFormat; +import net.minecraft.CrashReport; +import net.minecraft.CrashReportCategory; +import net.minecraft.ReportedException; +import net.minecraft.Util; +import net.minecraft.client.Minecraft; +import net.minecraft.client.gui.Font; +import net.minecraft.client.gui.GuiGraphics; +import net.minecraft.client.gui.components.AbstractWidget; +import net.minecraft.client.gui.screens.Screen; +import net.minecraft.client.gui.screens.inventory.tooltip.ClientTooltipComponent; +import net.minecraft.client.gui.screens.inventory.tooltip.ClientTooltipPositioner; +import net.minecraft.client.gui.screens.inventory.tooltip.DefaultTooltipPositioner; +import net.minecraft.client.player.LocalPlayer; +import net.minecraft.client.renderer.GameRenderer; +import net.minecraft.client.renderer.MultiBufferSource; +import net.minecraft.client.renderer.RenderStateShard; +import net.minecraft.client.renderer.RenderType; +import net.minecraft.client.renderer.texture.OverlayTexture; +import net.minecraft.client.renderer.texture.SpriteContents; +import net.minecraft.client.renderer.texture.TextureAtlasSprite; +import net.minecraft.client.resources.model.BakedModel; +import net.minecraft.network.chat.Component; +import net.minecraft.network.chat.FormattedText; +import net.minecraft.network.chat.HoverEvent; +import net.minecraft.network.chat.Style; +import net.minecraft.resources.ResourceLocation; +import net.minecraft.util.FormattedCharSequence; +import net.minecraft.util.Mth; +import net.minecraft.world.entity.LivingEntity; +import net.minecraft.world.inventory.tooltip.TooltipComponent; +import net.minecraft.world.item.ItemDisplayContext; +import net.minecraft.world.item.ItemStack; +import net.minecraft.world.level.Level; +import net.minecraftforge.client.ForgeHooksClient; +import net.minecraftforge.client.event.RenderTooltipEvent; +import net.minecraftforge.common.MinecraftForge; +import org.jetbrains.annotations.Nullable; +import org.joml.Matrix4f; +import org.joml.Vector2ic; + +import java.util.List; +import java.util.Optional; +import java.util.stream.Collectors; + +/** + * This class primarily based on GuiHelper from BrandonsCore + * But its implementation is heavily inspired by the new GuiGraphics system in 1.20+ + *

+ * The purpose of this class is to provide most of the basic rendering functions required to render various GUI geometry. + * This includes things like simple rectangles, textures, strings, etc. + *

+ * Created by brandon3055 on 29/06/2023 + */ +public class GuiRender { + public static final RenderType SOLID = RenderType.gui(); + + //Used for things like events that require the vanilla GuiGraphics + private final RenderWrapper renderWrapper; + + private final Minecraft mc; + private final PoseStack pose; + private final ScissorHandler scissorHandler = new ScissorHandler(); + private final MultiBufferSource.BufferSource buffers; + private boolean batchDraw; + private Font fontOverride; + + public GuiRender(Minecraft mc, PoseStack poseStack, MultiBufferSource.BufferSource buffers) { + this.mc = mc; + this.pose = poseStack; + this.buffers = buffers; + this.renderWrapper = new RenderWrapper(this); + } + + public GuiRender(Minecraft mc, MultiBufferSource.BufferSource buffers) { + this(mc, new PoseStack(), buffers); + } + + public static GuiRender convert(GuiGraphics graphics) { + return new GuiRender(Minecraft.getInstance(), graphics.pose(), graphics.bufferSource()); + } + + public PoseStack pose() { + return pose; + } + + public MultiBufferSource.BufferSource buffers() { + return buffers; + } + + public Minecraft mc() { + return mc; + } + + public Font font() { + return fontOverride == null ? mc().font : fontOverride; + } + + public int guiWidth() { + return mc().getWindow().getGuiScaledWidth(); + } + + public int guiHeight() { + return mc().getWindow().getGuiScaledHeight(); + } + + /** + * Allows you to override the font renderer used for all text rendering. + * Be sure to set the override back to null when you are finished using your custom font! + * + * @param font The font to use, or null to disable override. + */ + public void overrideFont(@Nullable Font font) { + this.fontOverride = font; + } + + /** + * Allow similar render calls to be batched together into a single draw for better render efficiency. + * All render calls in batch must use the same render type. + * + * @param batch callback in which the rendering should be implemented. + */ + public void batchDraw(Runnable batch) { + flush(); + batchDraw = true; + batch.run(); + batchDraw = false; + flush(); + } + + private void flushIfUnBatched() { + if (!batchDraw) flush(); + } + + private void flushIfBatched() { + if (batchDraw) flush(); + } + + public void flush() { + RenderSystem.disableDepthTest(); + buffers.endBatch(); + RenderSystem.enableDepthTest(); + } + + /** + * Only use this as a last resort! It may explode... Have fun! + * + * @return A Vanilla GuiGraphics instance that wraps this {@link GuiRender} + */ + @Deprecated + public RenderWrapper guiGraphicsWrapper() { + return renderWrapper; + } + + //=== Un-Textured geometry ===// + + /** + * Fill rectangle with solid colour + */ + public void rect(Rectangle rectangle, int colour) { + this.rect(SOLID, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), colour); + } + + /** + * Fill rectangle with solid colour + */ + public void rect(RenderType type, Rectangle rectangle, int colour) { + this.rect(type, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), colour); + } + + /** + * Fill rectangle with solid colour + */ + public void rect(double x, double y, double width, double height, int colour) { + this.fill(SOLID, x, y, x + width, y + height, colour); + } + + /** + * Fill rectangle with solid colour + */ + public void rect(RenderType type, double x, double y, double width, double height, int colour) { + this.fill(type, x, y, x + width, y + height, colour); + } + + /** + * Fill area with solid colour + */ + public void fill(double xMin, double yMin, double xMax, double yMax, int colour) { + this.fill(SOLID, xMin, yMin, xMax, yMax, colour); + } + + /** + * Fill area with solid colour + */ + public void fill(RenderType type, double xMin, double yMin, double xMax, double yMax, int colour) { + if (xMax < xMin) { + double min = xMax; + xMax = xMin; + xMin = min; + } + if (yMax < yMin) { + double min = yMax; + yMax = yMin; + yMin = min; + } + + Matrix4f mat = pose.last().pose(); + VertexConsumer buffer = buffers.getBuffer(type); + buffer.vertex(mat, (float) xMax, (float) yMax, 0).color(colour).endVertex(); //R-B + buffer.vertex(mat, (float) xMax, (float) yMin, 0).color(colour).endVertex(); //R-T + buffer.vertex(mat, (float) xMin, (float) yMin, 0).color(colour).endVertex(); //L-T + buffer.vertex(mat, (float) xMin, (float) yMax, 0).color(colour).endVertex(); //L-B + flushIfUnBatched(); + } + + /** + * Fill area with colour gradient from top to bottom + */ + public void gradientFillV(double xMin, double yMin, double xMax, double yMax, int topColour, int bottomColour) { + this.gradientFillV(SOLID, xMin, yMin, xMax, yMax, topColour, bottomColour); + } + + /** + * Fill area with colour gradient from top to bottom + */ + public void gradientFillV(RenderType type, double xMin, double yMin, double xMax, double yMax, int topColour, int bottomColour) { + VertexConsumer buffer = buffers().getBuffer(type); + float sA = a(topColour) / 255.0F; + float sR = r(topColour) / 255.0F; + float sG = g(topColour) / 255.0F; + float sB = b(topColour) / 255.0F; + float eA = a(bottomColour) / 255.0F; + float eR = r(bottomColour) / 255.0F; + float eG = g(bottomColour) / 255.0F; + float eB = b(bottomColour) / 255.0F; + Matrix4f mat = pose.last().pose(); + buffer.vertex(mat, (float) xMax, (float) yMax, 0).color(eR, eG, eB, eA).endVertex(); //R-B + buffer.vertex(mat, (float) xMax, (float) yMin, 0).color(sR, sG, sB, sA).endVertex(); //R-T + buffer.vertex(mat, (float) xMin, (float) yMin, 0).color(sR, sG, sB, sA).endVertex(); //L-T + buffer.vertex(mat, (float) xMin, (float) yMax, 0).color(eR, eG, eB, eA).endVertex(); //L-B + this.flushIfUnBatched(); + } + + /** + * Fill area with colour gradient from left to right + */ + public void gradientFillH(double xMin, double yMin, double xMax, double yMax, int leftColour, int rightColour) { + this.gradientFillH(SOLID, xMin, yMin, xMax, yMax, leftColour, rightColour); + } + + /** + * Fill area with colour gradient from left to right + */ + public void gradientFillH(RenderType type, double xMin, double yMin, double xMax, double yMax, int leftColour, int rightColour) { + VertexConsumer buffer = buffers().getBuffer(type); + float sA = a(leftColour) / 255.0F; + float sR = r(leftColour) / 255.0F; + float sG = g(leftColour) / 255.0F; + float sB = b(leftColour) / 255.0F; + float eA = a(rightColour) / 255.0F; + float eR = r(rightColour) / 255.0F; + float eG = g(rightColour) / 255.0F; + float eB = b(rightColour) / 255.0F; + Matrix4f mat = pose.last().pose(); + buffer.vertex(mat, (float) xMax, (float) yMax, 0).color(eR, eG, eB, eA).endVertex(); //R-B + buffer.vertex(mat, (float) xMax, (float) yMin, 0).color(eR, eG, eB, eA).endVertex(); //R-T + buffer.vertex(mat, (float) xMin, (float) yMin, 0).color(sR, sG, sB, sA).endVertex(); //L-T + buffer.vertex(mat, (float) xMin, (float) yMax, 0).color(sR, sG, sB, sA).endVertex(); //L-B + this.flushIfUnBatched(); + } + + /** + * Draw a bordered rectangle of specified with specified border width, border colour and fill colour. + */ + public void borderRect(Rectangle rectangle, double borderWidth, int fillColour, int borderColour) { + borderFill(rectangle.x(), rectangle.y(), rectangle.xMax(), rectangle.yMax(), borderWidth, fillColour, borderColour); + } + + /** + * Draw a bordered rectangle of specified with specified border width, border colour and fill colour. + */ + public void borderRect(double x, double y, double width, double height, double borderWidth, int fillColour, int borderColour) { + borderFill(x, y, x + width, y + height, borderWidth, fillColour, borderColour); + } + + /** + * Draw a bordered rectangle of specified with specified border width, border colour and fill colour. + */ + public void borderRect(RenderType type, Rectangle rectangle, double borderWidth, int fillColour, int borderColour) { + borderFill(type, rectangle.x(), rectangle.y(), rectangle.xMax(), rectangle.yMax(), borderWidth, fillColour, borderColour); + } + + /** + * Draw a bordered rectangle of specified with specified border width, border colour and fill colour. + */ + public void borderRect(RenderType type, double x, double y, double width, double height, double borderWidth, int fillColour, int borderColour) { + borderFill(type, x, y, x + width, y + height, borderWidth, fillColour, borderColour); + } + + /** + * Draw a border of specified with, fill internal area with solid colour. + */ + public void borderFill(double xMin, double yMin, double xMax, double yMax, double borderWidth, int fillColour, int borderColour) { + borderFill(SOLID, xMin, yMin, xMax, yMax, borderWidth, fillColour, borderColour); + } + + /** + * Draw a border of specified with, fill internal area with solid colour. + */ + public void borderFill(RenderType type, double xMin, double yMin, double xMax, double yMax, double borderWidth, int fillColour, int borderColour) { + if (batchDraw) { //Draw batched for efficiency, unless already doing a batch draw. + borderFillInternal(type, xMin, yMin, xMax, yMax, borderWidth, fillColour, borderColour); + } else { + batchDraw(() -> borderFillInternal(type, xMin, yMin, xMax, yMax, borderWidth, fillColour, borderColour)); + } + } + + private void borderFillInternal(RenderType type, double xMin, double yMin, double xMax, double yMax, double borderWidth, int fillColour, int borderColour) { + fill(type, xMin, yMin, xMax, yMin + borderWidth, borderColour); //Top + fill(type, xMin, yMin + borderWidth, xMin + borderWidth, yMax - borderWidth, borderColour); //Left + fill(type, xMin, yMax - borderWidth, xMax, yMax, borderColour); //Bottom + fill(type, xMax - borderWidth, yMin + borderWidth, xMax, yMax - borderWidth, borderColour); //Right + if (fillColour != 0) //No point rendering fill if there is no fill colour + fill(type, xMin + borderWidth, yMin + borderWidth, xMax - borderWidth, yMax - borderWidth, fillColour); //Fill + } + + /** + * Can be used to create the illusion of an inset / outset rectangle. This is identical to the way inventory slots are rendered except in code rather than via a texture. + * Example Usage: render.shadedFill(0, 0, 18, 18, 1, 0xFF373737, 0xFFffffff, 0xFF8b8b8b, 0xFF8b8b8b); //Renders a vanilla style inventory slot + * This can also be used to render things like buttons that appear to actually "push in" when you press them. + */ + public void shadedRect(Rectangle rectangle, double borderWidth, int topLeftColour, int bottomRightColour, int fillColour) { + shadedFill(SOLID, rectangle.x(), rectangle.y(), rectangle.xMax(), rectangle.yMax(), borderWidth, topLeftColour, bottomRightColour, midColour(topLeftColour, bottomRightColour), fillColour); + } + + /** + * Can be used to create the illusion of an inset / outset rectangle. This is identical to the way inventory slots are rendered except in code rather than via a texture. + * Example Usage: render.shadedFill(0, 0, 18, 18, 1, 0xFF373737, 0xFFffffff, 0xFF8b8b8b, 0xFF8b8b8b); //Renders a vanilla style inventory slot + * This can also be used to render things like buttons that appear to actually "push in" when you press them. + */ + public void shadedRect(double x, double y, double width, double height, double borderWidth, int topLeftColour, int bottomRightColour, int fillColour) { + shadedFill(SOLID, x, y, x + width, y + height, borderWidth, topLeftColour, bottomRightColour, midColour(topLeftColour, bottomRightColour), fillColour); + } + + /** + * Can be used to create the illusion of an inset / outset rectangle. This is identical to the way inventory slots are rendered except in code rather than via a texture. + * Example Usage: render.shadedFill(0, 0, 18, 18, 1, 0xFF373737, 0xFFffffff, 0xFF8b8b8b, 0xFF8b8b8b); //Renders a vanilla style inventory slot + * This can also be used to render things like buttons that appear to actually "push in" when you press them. + */ + public void shadedRect(Rectangle rectangle, double borderWidth, int topLeftColour, int bottomRightColour, int cornerMixColour, int fillColour) { + shadedFill(SOLID, rectangle.x(), rectangle.y(), rectangle.xMax(), rectangle.yMax(), borderWidth, topLeftColour, bottomRightColour, cornerMixColour, fillColour); + } + + /** + * Can be used to create the illusion of an inset / outset rectangle. This is identical to the way inventory slots are rendered except in code rather than via a texture. + * Example Usage: render.shadedFill(0, 0, 18, 18, 1, 0xFF373737, 0xFFffffff, 0xFF8b8b8b, 0xFF8b8b8b); //Renders a vanilla style inventory slot + * This can also be used to render things like buttons that appear to actually "push in" when you press them. + */ + public void shadedRect(double x, double y, double width, double height, double borderWidth, int topLeftColour, int bottomRightColour, int cornerMixColour, int fillColour) { + shadedFill(SOLID, x, y, x + width, y + height, borderWidth, topLeftColour, bottomRightColour, cornerMixColour, fillColour); + } + + /** + * Can be used to create the illusion of an inset / outset rectangle. This is identical to the way inventory slots are rendered except in code rather than via a texture. + * Example Usage: render.shadedFill(0, 0, 18, 18, 1, 0xFF373737, 0xFFffffff, 0xFF8b8b8b, 0xFF8b8b8b); //Renders a vanilla style inventory slot + * This can also be used to render things like buttons that appear to actually "push in" when you press them. + */ + public void shadedRect(RenderType type, double x, double y, double width, double height, double borderWidth, int topLeftColour, int bottomRightColour, int cornerMixColour, int fillColour) { + shadedFill(type, x, y, x + width, y + height, borderWidth, topLeftColour, bottomRightColour, cornerMixColour, fillColour); + } + + /** + * Can be used to create the illusion of an inset / outset area. This is identical to the way inventory slots are rendered except in code rather than via a texture. + * Example Usage: render.shadedFill(0, 0, 18, 18, 1, 0xFF373737, 0xFFffffff, 0xFF8b8b8b, 0xFF8b8b8b); //Renders a vanilla style inventory slot + * This can also be used to render things like buttons that appear to actually "push in" when you press them. + */ + public void shadedFill(double xMin, double yMin, double xMax, double yMax, double borderWidth, int topLeftColour, int bottomRightColour, int fillColour) { + shadedFill(SOLID, xMin, yMin, xMax, yMax, borderWidth, topLeftColour, bottomRightColour, midColour(topLeftColour, bottomRightColour), fillColour); + } + + /** + * Can be used to create the illusion of an inset / outset area. This is identical to the way inventory slots are rendered except in code rather than via a texture. + * Example Usage: render.shadedFill(0, 0, 18, 18, 1, 0xFF373737, 0xFFffffff, 0xFF8b8b8b, 0xFF8b8b8b); //Renders a vanilla style inventory slot + * This can also be used to render things like buttons that appear to actually "push in" when you press them. + */ + public void shadedFill(double xMin, double yMin, double xMax, double yMax, double borderWidth, int topLeftColour, int bottomRightColour, int cornerMixColour, int fillColour) { + shadedFill(SOLID, xMin, yMin, xMax, yMax, borderWidth, topLeftColour, bottomRightColour, cornerMixColour, fillColour); + } + + /** + * Can be used to create the illusion of an inset / outset area. This is identical to the way inventory slots are rendered except in code rather than via a texture. + * Example Usage: render.shadedFill(0, 0, 18, 18, 1, 0xFF373737, 0xFFffffff, 0xFF8b8b8b, 0xFF8b8b8b); //Renders a vanilla style inventory slot + * This can also be used to render things like buttons that appear to actually "push in" when you press them. + */ + public void shadedFill(RenderType type, double xMin, double yMin, double xMax, double yMax, double borderWidth, int topLeftColour, int bottomRightColour, int cornerMixColour, int fillColour) { + if (batchDraw) { //Draw batched for efficiency, unless already doing a batch draw. + shadedFillInternal(type, xMin, yMin, xMax, yMax, borderWidth, topLeftColour, bottomRightColour, cornerMixColour, fillColour); + } else { + batchDraw(() -> shadedFillInternal(type, xMin, yMin, xMax, yMax, borderWidth, topLeftColour, bottomRightColour, cornerMixColour, fillColour)); + } + } + + public void shadedFillInternal(RenderType type, double xMin, double yMin, double xMax, double yMax, double borderWidth, int topLeftColour, int bottomRightColour, int cornerMixColour, int fillColour) { + fill(type, xMin, yMin, xMax - borderWidth, yMin + borderWidth, topLeftColour); //Top + fill(type, xMin, yMin + borderWidth, xMin + borderWidth, yMax - borderWidth, topLeftColour); //Left + fill(type, xMin + borderWidth, yMax - borderWidth, xMax, yMax, bottomRightColour); //Bottom + fill(type, xMax - borderWidth, yMin + borderWidth, xMax, yMax - borderWidth, bottomRightColour); //Right + fill(type, xMax - borderWidth, yMin, xMax, yMin + borderWidth, cornerMixColour); //Top Right Corner + fill(type, xMin, yMax - borderWidth, xMin + borderWidth, yMax, cornerMixColour); //Bottom Left Corner + + if (fillColour != 0) //No point rendering fill if there is no fill colour + fill(type, xMin + borderWidth, yMin + borderWidth, xMax - borderWidth, yMax - borderWidth, fillColour); //Fill + } + + //=== Generic Backgrounds ===// + + /** + * Draws a rectangle / background with a style matching vanilla tool tips. + */ + public void toolTipBackground(double x, double y, double width, double height) { + toolTipBackground(x, y, width, height, 0xF0100010, 0x505000FF, 0x5028007f); + } + + /** + * Draws a rectangle / background with a style matching vanilla tool tips. + * Vanilla Default Colours: 0xF0100010, 0x505000FF, 0x5028007f + */ + public void toolTipBackground(double x, double y, double width, double height, int backgroundColour, int borderColourTop, int borderColourBottom) { + toolTipBackground(x, y, width, height, backgroundColour, backgroundColour, borderColourTop, borderColourBottom, false); + } + + /** + * Draws a rectangle / background with a style matching vanilla tool tips. + * Vanilla Default Colours: 0xF0100010, 0xF0100010, 0x505000FF, 0x5028007f + */ + public void toolTipBackground(double x, double y, double width, double height, int backgroundColourTop, int backgroundColourBottom, int borderColourTop, int borderColourBottom, boolean empty) { + if (batchDraw) { //Draw batched for efficiency, unless already doing a batch draw. + toolTipBackgroundInternal(x, y, x + width, y + height, backgroundColourTop, backgroundColourBottom, borderColourTop, borderColourBottom, false); + } else { + batchDraw(() -> toolTipBackgroundInternal(x, y, x + width, y + height, backgroundColourTop, backgroundColourBottom, borderColourTop, borderColourBottom, false)); + } + } + + private void toolTipBackgroundInternal(double xMin, double yMin, double xMax, double yMax, int backgroundColourTop, int backgroundColourBottom, int borderColourTop, int borderColourBottom, boolean empty) { + fill(xMin + 1, yMin, xMax - 1, yMin + 1, backgroundColourTop); // Top + fill(xMin + 1, yMax - 1, xMax - 1, yMax, backgroundColourBottom); // Bottom + gradientFillV(xMin, yMin + 1, xMin + 1, yMax - 1, backgroundColourTop, backgroundColourBottom); // Left + gradientFillV(xMax - 1, yMin + 1, xMax, yMax - 1, backgroundColourTop, backgroundColourBottom); // Right + if (!empty) { + gradientFillV(xMin + 1, yMin + 1, xMax - 1, yMax - 1, backgroundColourTop, backgroundColourBottom); // Fill + } + gradientFillV(xMin + 1, yMin + 1, xMin + 2, yMax - 1, borderColourTop, borderColourBottom); // Left Accent + gradientFillV(xMax - 2, yMin + 1, xMax - 1, yMax - 1, borderColourTop, borderColourBottom); // Right Accent + fill(xMin + 2, yMin + 1, xMax - 2, yMin + 2, borderColourTop); // Top Accent + fill(xMin + 2, yMax - 2, xMax - 2, yMax - 1, borderColourBottom); // Bottom Accent + } + + //=== Textured geometry ===// + + //Sprite plus RenderType + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void spriteRect(RenderType type, Rectangle rectangle, TextureAtlasSprite sprite) { + spriteRect(type, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), sprite, 1F, 1F, 1F, 1F); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void spriteRect(RenderType type, Rectangle rectangle, TextureAtlasSprite sprite, int argb) { + spriteRect(type, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), sprite, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void spriteRect(RenderType type, Rectangle rectangle, TextureAtlasSprite sprite, float red, float green, float blue, float alpha) { + spriteRect(type, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), sprite, red, green, blue, alpha); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void spriteRect(RenderType type, double x, double y, double width, double height, TextureAtlasSprite sprite) { + spriteRect(type, x, y, width, height, sprite, 1F, 1F, 1F, 1F); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void spriteRect(RenderType type, double x, double y, double width, double height, TextureAtlasSprite sprite, int argb) { + spriteRect(type, x, y, width, height, sprite, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void spriteRect(RenderType type, double x, double y, double width, double height, TextureAtlasSprite sprite, float red, float green, float blue, float alpha) { + sprite(type, x, y, x + width, y + height, sprite, red, green, blue, alpha); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void sprite(RenderType type, double xMin, double yMin, double xMax, double yMax, TextureAtlasSprite sprite) { + sprite(type, xMin, yMin, xMax, yMax, sprite, 1F, 1F, 1F, 1F); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void sprite(RenderType type, double xMin, double yMin, double xMax, double yMax, TextureAtlasSprite sprite, int argb) { + sprite(type, xMin, yMin, xMax, yMax, sprite, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void sprite(RenderType type, double xMin, double yMin, double xMax, double yMax, TextureAtlasSprite sprite, float red, float green, float blue, float alpha) { + VertexConsumer buffer = buffers().getBuffer(type); + Matrix4f mat = pose.last().pose(); + buffer.vertex(mat, (float) xMax, (float) yMax, 0).color(red, green, blue, alpha).uv(sprite.getU1(), sprite.getV1()).endVertex(); //R-B + buffer.vertex(mat, (float) xMax, (float) yMin, 0).color(red, green, blue, alpha).uv(sprite.getU1(), sprite.getV0()).endVertex(); //R-T + buffer.vertex(mat, (float) xMin, (float) yMin, 0).color(red, green, blue, alpha).uv(sprite.getU0(), sprite.getV0()).endVertex(); //L-T + buffer.vertex(mat, (float) xMin, (float) yMax, 0).color(red, green, blue, alpha).uv(sprite.getU0(), sprite.getV1()).endVertex(); //L-B + flushIfUnBatched(); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void spriteRect(RenderType type, Rectangle rectangle, int rotation, TextureAtlasSprite sprite) { + spriteRect(type, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), rotation, sprite, 1F, 1F, 1F, 1F); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void spriteRect(RenderType type, Rectangle rectangle, int rotation, TextureAtlasSprite sprite, int argb) { + spriteRect(type, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), rotation, sprite, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void spriteRect(RenderType type, Rectangle rectangle, int rotation, TextureAtlasSprite sprite, float red, float green, float blue, float alpha) { + spriteRect(type, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), rotation, sprite, red, green, blue, alpha); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void spriteRect(RenderType type, double x, double y, double width, double height, int rotation, TextureAtlasSprite sprite) { + spriteRect(type, x, y, width, height, rotation, sprite, 1F, 1F, 1F, 1F); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void spriteRect(RenderType type, double x, double y, double width, double height, int rotation, TextureAtlasSprite sprite, int argb) { + spriteRect(type, x, y, width, height, rotation, sprite, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void spriteRect(RenderType type, double x, double y, double width, double height, int rotation, TextureAtlasSprite sprite, float red, float green, float blue, float alpha) { + sprite(type, x, y, x + width, y + height, rotation, sprite, red, green, blue, alpha); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void sprite(RenderType type, double xMin, double yMin, double xMax, double yMax, int rotation, TextureAtlasSprite sprite) { + sprite(type, xMin, yMin, xMax, yMax, rotation, sprite, 1F, 1F, 1F, 1F); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void sprite(RenderType type, double xMin, double yMin, double xMax, double yMax, int rotation, TextureAtlasSprite sprite, int argb) { + sprite(type, xMin, yMin, xMax, yMax, rotation, sprite, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void sprite(RenderType type, double xMin, double yMin, double xMax, double yMax, int rotation, TextureAtlasSprite sprite, float red, float green, float blue, float alpha) { + float[] u = {sprite.getU0(), sprite.getU1(), sprite.getU1(), sprite.getU0()}; + float[] v = {sprite.getV1(), sprite.getV1(), sprite.getV0(), sprite.getV0()}; + VertexConsumer buffer = buffers().getBuffer(type); + Matrix4f mat = pose.last().pose(); + buffer.vertex(mat, (float) xMax, (float) yMax, 0).color(red, green, blue, alpha).uv(u[(1 + rotation) % 4], v[(1 + rotation) % 4]).endVertex(); //R-B + buffer.vertex(mat, (float) xMax, (float) yMin, 0).color(red, green, blue, alpha).uv(u[(2 + rotation) % 4], v[(2 + rotation) % 4]).endVertex(); //R-T + buffer.vertex(mat, (float) xMin, (float) yMin, 0).color(red, green, blue, alpha).uv(u[(3 + rotation) % 4], v[(3 + rotation) % 4]).endVertex(); //L-T + buffer.vertex(mat, (float) xMin, (float) yMax, 0).color(red, green, blue, alpha).uv(u[(0 + rotation) % 4], v[(0 + rotation) % 4]).endVertex(); //L-B + flushIfUnBatched(); + } + + //Partial Sprite + + //TODO figure out if there is a way to make this work. +// /** +// * Draws a subsection of a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX +// * Texture will be resized / reshaped as appropriate to fit the defined area. +// *

+// * This is similar to {@link #partialSprite(RenderType, double, double, double, double, TextureAtlasSprite, float, float, float, float, int)} +// * Except the input uv values are in texture coordinates. So to draw a full 16x16 sprite with this you would supply 0, 0, 16, 16 +// * +// * @param rotation Rotates sprite clockwise in 90 degree steps. +// */ +// public void partialSprite(RenderType type, double xMin, double yMin, double xMax, double yMax, int rotation, TextureAtlasSprite sprite, int texXMin, int texYMin, int texXMax, int texYMax, int argb) { +// float width = sprite.contents().width(); +// float height = sprite.contents().height(); +// partialSprite(type, xMin, yMin, xMax, yMax, rotation, sprite, texXMin / width, texYMin / height, texXMax / width, texYMax / height, argb); +// } +// + +// /** +// * Draws a subsection of a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX +// * Texture will be resized / reshaped as appropriate to fit the defined area. +// * Valid input u/v value range is 0 to 1 [0, 0, 1, 1 would render the full sprite] +// * +// * @param rotation Rotates sprite clockwise in 90 degree steps. +// */ +// public void partialSprite(RenderType type, double xMin, double yMin, double xMax, double yMax, int rotation, TextureAtlasSprite sprite, float uMin, float vMin, float uMax, float vMax, int argb) { +// partialSprite(type, xMin, yMin, xMax, yMax, rotation, sprite, uMin, vMin, uMax, vMax, r(argb), g(argb), b(argb), a(argb)); +// } +// +// /** +// * Draws a subsection of a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX +// * Texture will be resized / reshaped as appropriate to fit the defined area. +// * Valid input u/v value range is 0 to 1 [0, 0, 1, 1 would render the full sprite] +// * +// * @param rotation Rotates sprite clockwise in 90 degree steps. +// */ +// public void partialSprite(RenderType type, double xMin, double yMin, double xMax, double yMax, int rotation, TextureAtlasSprite sprite, float left, float top, float right, float bottom, float red, float green, float blue, float alpha) { +// VertexConsumer buffer = buffers().getBuffer(type); +// Matrix4f mat = pose.last().pose(); +// rotation = Math.floorMod(rotation, 4); +// +// float[] sub = {left, top, right, bottom}; +// left = sub[rotation % 4]; +// top = sub[(rotation + 1) % 4]; +// right = sub[(rotation + 2) % 4]; +// bottom = sub[(rotation + 3) % 4]; +// +// float ul = sprite.getU1() - sprite.getU0(); +// float vl = sprite.getV1() - sprite.getV0(); +// float u0 = sprite.getU0() + (left * ul); +// float v0 = sprite.getV0() + (top * vl); +// float u1 = sprite.getU0() + (right * ul); +// float v1 = sprite.getV0() + (bottom * vl); +// float[] u = {u0, u1, u1, u0}; +// float[] v = {v1, v1, v0, v0}; +// buffer.vertex(mat, (float) xMax, (float) yMax, 0).color(red, green, blue, alpha).uv(u[(1 + rotation) % 4], v[(1 + rotation) % 4]).endVertex(); //R-B +// buffer.vertex(mat, (float) xMax, (float) yMin, 0).color(red, green, blue, alpha).uv(u[(2 + rotation) % 4], v[(2 + rotation) % 4]).endVertex(); //R-T +// buffer.vertex(mat, (float) xMin, (float) yMin, 0).color(red, green, blue, alpha).uv(u[(3 + rotation) % 4], v[(3 + rotation) % 4]).endVertex(); //L-T +// buffer.vertex(mat, (float) xMin, (float) yMax, 0).color(red, green, blue, alpha).uv(u[(0 + rotation) % 4], v[(0 + rotation) % 4]).endVertex(); //L-B +// flushIfUnBatched(); +// } + + /** + * Draws a subsection of a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + *

+ * This is similar to {@link #partialSprite(RenderType, double, double, double, double, TextureAtlasSprite, float, float, float, float, int)} + * Except the input uv values are in texture coordinates. So to draw a full 16x16 sprite with this you would supply 0, 0, 16, 16 + */ + public void partialSpriteTex(RenderType type, double xMin, double yMin, double xMax, double yMax, TextureAtlasSprite sprite, double texXMin, double texYMin, double texXMax, double texYMax, int argb) { + partialSpriteTex(type, xMin, yMin, xMax, yMax, sprite, texXMin, texYMin, texXMax, texYMax, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a subsection of a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + *

+ * This is similar to {@link #partialSprite(RenderType, double, double, double, double, TextureAtlasSprite, float, float, float, float, int)} + * Except the input uv values are in texture coordinates. So to draw a full 16x16 sprite with this you would supply 0, 0, 16, 16 + */ + public void partialSpriteTex(RenderType type, double xMin, double yMin, double xMax, double yMax, TextureAtlasSprite sprite, double texXMin, double texYMin, double texXMax, double texYMax, float red, float green, float blue, float alpha) { + int width = sprite.contents().width(); + int height = sprite.contents().height(); + partialSprite(type, xMin, yMin, xMax, yMax, sprite, (float) texXMin / width, (float) texYMin / height, (float) texXMax / width, (float) texYMax / height, red, green, blue, alpha); + } + + /** + * Draws a subsection of a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + * Valid input u/v value range is 0 to 1 [0, 0, 1, 1 would render the full sprite] + */ + public void partialSprite(RenderType type, double xMin, double yMin, double xMax, double yMax, TextureAtlasSprite sprite, float uMin, float vMin, float uMax, float vMax, int argb) { + partialSprite(type, xMin, yMin, xMax, yMax, sprite, uMin, vMin, uMax, vMax, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a subsection of a TextureAtlasSprite using the given render type, Vertex format should be POSITION_COLOR_TEX + * Texture will be resized / reshaped as appropriate to fit the defined area. + * Valid input u/v value range is 0 to 1 [0, 0, 1, 1 would render the full sprite] + */ + public void partialSprite(RenderType type, double xMin, double yMin, double xMax, double yMax, TextureAtlasSprite sprite, float uMin, float vMin, float uMax, float vMax, float red, float green, float blue, float alpha) { + VertexConsumer buffer = buffers().getBuffer(type); + Matrix4f mat = pose.last().pose(); + float u0 = sprite.getU0(); + float v0 = sprite.getV0(); + float u1 = sprite.getU1(); + float v1 = sprite.getV1(); + float ul = u1 - u0; + float vl = v1 - v0; + buffer.vertex(mat, (float) xMax, (float) yMax, 0).color(red, green, blue, alpha).uv(u0 + (uMax * ul), v0 + (vMax * vl)).endVertex(); //R-B + buffer.vertex(mat, (float) xMax, (float) yMin, 0).color(red, green, blue, alpha).uv(u0 + (uMax * ul), v0 + (vMin * vl)).endVertex(); //R-T + buffer.vertex(mat, (float) xMin, (float) yMin, 0).color(red, green, blue, alpha).uv(u0 + (uMin * ul), v0 + (vMin * vl)).endVertex(); //L-T + buffer.vertex(mat, (float) xMin, (float) yMax, 0).color(red, green, blue, alpha).uv(u0 + (uMin * ul), v0 + (vMax * vl)).endVertex(); //L-B + flushIfUnBatched(); + } + + /** + * Draw a sprite tiled to fit the specified area. + * Sprite is drawn from the top-left so sprite will be tiled right and down. + */ + public void tileSprite(RenderType type, double xMin, double yMin, double xMax, double yMax, TextureAtlasSprite sprite, int argb) { + tileSprite(type, xMin, yMin, xMax, yMax, sprite, sprite.contents().width(), sprite.contents().height(), argb); + } + + /** + * Draw a sprite tiled to fit the specified area. + * Sprite is drawn from the top-left so sprite will be tiled right and down. + * + * @param textureWidth Set base width of the sprite texture in pixels + * @param textureHeight Set base height of the sprite texture in pixels + */ + public void tileSprite(RenderType type, double xMin, double yMin, double xMax, double yMax, TextureAtlasSprite sprite, int textureWidth, int textureHeight, int argb) { + tileSprite(type, xMin, yMin, xMax, yMax, sprite, textureWidth, textureHeight, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draw a sprite tiled to fit the specified area. + * Sprite is drawn from the top-left so sprite will be tiled right and down. + */ + public void tileSprite(RenderType type, double xMin, double yMin, double xMax, double yMax, TextureAtlasSprite sprite, float red, float green, float blue, float alpha) { + tileSprite(type, xMin, yMin, xMax, yMax, sprite, sprite.contents().width(), sprite.contents().height(), red, green, blue, alpha); + } + + /** + * Draw a sprite tiled to fit the specified area. + * Sprite is drawn from the top-left so sprite will be tiled right and down. + * + * @param textureWidth Set base width of the sprite texture in pixels + * @param textureHeight Set base height of the sprite texture in pixels + */ + public void tileSprite(RenderType type, double xMin, double yMin, double xMax, double yMax, TextureAtlasSprite sprite, int textureWidth, int textureHeight, float red, float green, float blue, float alpha) { + double width = xMax - xMin; + double height = yMax - yMin; + if (width <= textureWidth && height <= textureHeight) { + partialSprite(type, xMin, yMin, xMax, yMax, sprite, 0F, 0F, (float) width / textureWidth, (float) height / textureHeight, red, green, blue, alpha); + } else { + Runnable draw = () -> { + double xPos = xMin; + do { + double sectionWidth = Math.min(textureWidth, xMax - xPos); + double uWidth = sectionWidth / textureWidth; + double yPos = yMin; + do { + double sectionHeight = Math.min(textureHeight, yMax - yPos); + double vWidth = sectionHeight / textureHeight; + partialSprite(type, xPos, yPos, xPos + sectionWidth, yPos + sectionHeight, sprite, 0, 0, (float) uWidth, (float) vWidth, red, green, blue, alpha); + yPos += textureHeight; + } while (yPos < yMax); + xPos += textureWidth; + } while (xPos < xMax); + }; + if (batchDraw) { + draw.run(); + } else { + batchDraw(draw); + } + } + } + + //Material + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void texRect(Material material, Rectangle rectangle) { + texRect(material, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), 1F, 1F, 1F, 1F); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void texRect(Material material, Rectangle rectangle, int argb) { + texRect(material, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void texRect(Material material, Rectangle rectangle, float red, float green, float blue, float alpha) { + texRect(material, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), red, green, blue, alpha); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void texRect(Material material, double x, double y, double width, double height) { + texRect(material, x, y, width, height, 1F, 1F, 1F, 1F); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void texRect(Material material, double x, double y, double width, double height, int argb) { + texRect(material, x, y, width, height, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void texRect(Material material, double x, double y, double width, double height, float red, float green, float blue, float alpha) { + tex(material, x, y, x + width, y + height, red, green, blue, alpha); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void tex(Material material, double xMin, double yMin, double xMax, double yMax) { + tex(material, xMin, yMin, xMax, yMax, 1F, 1F, 1F, 1F); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void tex(Material material, double xMin, double yMin, double xMax, double yMax, int argb) { + tex(material, xMin, yMin, xMax, yMax, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + */ + public void tex(Material material, double xMin, double yMin, double xMax, double yMax, float red, float green, float blue, float alpha) { + TextureAtlasSprite sprite = material.sprite(); + VertexConsumer buffer = material.buffer(buffers, GuiRender::texColType); + Matrix4f mat = pose.last().pose(); + buffer.vertex(mat, (float) xMax, (float) yMax, 0).color(red, green, blue, alpha).uv(sprite.getU1(), sprite.getV1()).endVertex(); //R-B + buffer.vertex(mat, (float) xMax, (float) yMin, 0).color(red, green, blue, alpha).uv(sprite.getU1(), sprite.getV0()).endVertex(); //R-T + buffer.vertex(mat, (float) xMin, (float) yMin, 0).color(red, green, blue, alpha).uv(sprite.getU0(), sprite.getV0()).endVertex(); //L-T + buffer.vertex(mat, (float) xMin, (float) yMax, 0).color(red, green, blue, alpha).uv(sprite.getU0(), sprite.getV1()).endVertex(); //L-B + flushIfUnBatched(); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void texRect(Material material, int rotation, Rectangle rectangle) { + texRect(material, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), rotation, 1F, 1F, 1F, 1F); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void texRect(Material material, int rotation, Rectangle rectangle, int argb) { + texRect(material, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), rotation, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void texRect(Material material, int rotation, Rectangle rectangle, float red, float green, float blue, float alpha) { + texRect(material, rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height(), rotation, red, green, blue, alpha); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void texRect(Material material, int rotation, double x, double y, double width, double height) { + texRect(material, x, y, width, height, rotation, 1F, 1F, 1F, 1F); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void texRect(Material material, int rotation, double x, double y, double width, double height, int argb) { + texRect(material, x, y, width, height, rotation, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void texRect(Material material, double x, double y, double width, double height, int rotation, float red, float green, float blue, float alpha) { + tex(material, x, y, x + width, y + height, rotation, red, green, blue, alpha); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void tex(Material material, int rotation, double xMin, double yMin, double xMax, double yMax) { + tex(material, xMin, yMin, xMax, yMax, rotation, 1F, 1F, 1F, 1F); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void tex(Material material, double xMin, double yMin, double xMax, double yMax, int rotation, int argb) { + tex(material, xMin, yMin, xMax, yMax, rotation, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * Draws a texture sprite derived from the provided material. + * Texture will be resized / reshaped as appropriate to fit the defined area. + * + * @param rotation Rotates sprite clockwise in 90 degree steps. + */ + public void tex(Material material, double xMin, double yMin, double xMax, double yMax, int rotation, float red, float green, float blue, float alpha) { + TextureAtlasSprite sprite = material.sprite(); + VertexConsumer buffer = material.buffer(buffers, GuiRender::texColType); + float[] u = {sprite.getU0(), sprite.getU1(), sprite.getU1(), sprite.getU0()}; + float[] v = {sprite.getV1(), sprite.getV1(), sprite.getV0(), sprite.getV0()}; + Matrix4f mat = pose.last().pose(); + buffer.vertex(mat, (float) xMax, (float) yMax, 0).color(red, green, blue, alpha).uv(u[(1 + rotation) % 4], v[(1 + rotation) % 4]).endVertex(); //R-B + buffer.vertex(mat, (float) xMax, (float) yMin, 0).color(red, green, blue, alpha).uv(u[(2 + rotation) % 4], v[(2 + rotation) % 4]).endVertex(); //R-T + buffer.vertex(mat, (float) xMin, (float) yMin, 0).color(red, green, blue, alpha).uv(u[(3 + rotation) % 4], v[(3 + rotation) % 4]).endVertex(); //L-T + buffer.vertex(mat, (float) xMin, (float) yMax, 0).color(red, green, blue, alpha).uv(u[(0 + rotation) % 4], v[(0 + rotation) % 4]).endVertex(); //L-B + flushIfUnBatched(); + } + + //Slice and stitch + + /** + * This can be used to take something like a generic bordered background texture and dynamically resize it to draw at any size and shape you want. + * This is done by cutting up the texture and stitching it back to together using, cutting and tiling as required. + * The border parameters indicate the width of the borders around the texture, e.g. a vanilla gui texture has 4 pixel borders. + */ + public void dynamicTex(Material material, Rectangle rectangle, Borders borders, int argb) { + dynamicTex(material, (int) rectangle.x(), (int) rectangle.y(), (int) rectangle.width(), (int) rectangle.height(), (int) borders.top(), (int) borders.left(), (int) borders.bottom(), (int) borders.right(), argb); + } + + /** + * This can be used to take something like a generic bordered background texture and dynamically resize it to draw at any size and shape you want. + * This is done by cutting up the texture and stitching it back to together using, cutting and tiling as required. + * The border parameters indicate the width of the borders around the texture, e.g. a vanilla gui texture has 4 pixel borders. + */ + public void dynamicTex(Material material, Rectangle rectangle, int topBorder, int leftBorder, int bottomBorder, int rightBorder, int argb) { + dynamicTex(material, (int) rectangle.x(), (int) rectangle.y(), (int) rectangle.width(), (int) rectangle.height(), topBorder, leftBorder, bottomBorder, rightBorder, argb); + } + + /** + * This can be used to take something like a generic bordered background texture and dynamically resize it to draw at any size and shape you want. + * This is done by cutting up the texture and stitching it back to together using, cutting and tiling as required. + * The border parameters indicate the width of the borders around the texture, e.g. a vanilla gui texture has 4 pixel borders. + */ + public void dynamicTex(Material material, int x, int y, int width, int height, int topBorder, int leftBorder, int bottomBorder, int rightBorder, int argb) { + dynamicTex(material, x, y, width, height, topBorder, leftBorder, bottomBorder, rightBorder, r(argb), g(argb), b(argb), a(argb)); + } + + /** + * This can be used to take something like a generic bordered background texture and dynamically resize it to draw at any size and shape you want. + * This is done by cutting up the texture and stitching it back to together using, cutting and tiling as required. + * The border parameters indicate the width of the borders around the texture, e.g. a vanilla gui texture has 4 pixel borders. + */ + public void dynamicTex(Material material, Rectangle rectangle, Borders borders) { + dynamicTex(material, (int) rectangle.x(), (int) rectangle.y(), (int) rectangle.width(), (int) rectangle.height(), (int) borders.top(), (int) borders.left(), (int) borders.bottom(), (int) borders.right()); + } + + /** + * This can be used to take something like a generic bordered background texture and dynamically resize it to draw at any size and shape you want. + * This is done by cutting up the texture and stitching it back to together using, cutting and tiling as required. + * The border parameters indicate the width of the borders around the texture, e.g. a vanilla gui texture has 4 pixel borders. + */ + public void dynamicTex(Material material, Rectangle rectangle, int topBorder, int leftBorder, int bottomBorder, int rightBorder) { + dynamicTex(material, (int) rectangle.x(), (int) rectangle.y(), (int) rectangle.width(), (int) rectangle.height(), topBorder, leftBorder, bottomBorder, rightBorder); + } + + /** + * This can be used to take something like a generic bordered background texture and dynamically resize it to draw at any size and shape you want. + * This is done by cutting up the texture and stitching it back to together using, cutting and tiling as required. + * The border parameters indicate the width of the borders around the texture, e.g. a vanilla gui texture has 4 pixel borders. + */ + public void dynamicTex(Material material, int x, int y, int width, int height, int topBorder, int leftBorder, int bottomBorder, int rightBorder) { + dynamicTex(material, x, y, width, height, topBorder, leftBorder, bottomBorder, rightBorder, 1F, 1F, 1F, 1F); + } + + /** + * This can be used to take something like a generic bordered background texture and dynamically resize it to draw at any size and shape you want. + * This is done by cutting up the texture and stitching it back to together using, cutting and tiling as required. + * The border parameters indicate the width of the borders around the texture, e.g. a vanilla gui texture has 4 pixel borders. + */ + public void dynamicTex(Material material, int x, int y, int width, int height, int topBorder, int leftBorder, int bottomBorder, int rightBorder, float red, float green, float blue, float alpha) { + if (batchDraw) {//Draw batched for efficiency, unless already doing a batch draw. + dynamicTexInternal(material, x, y, width, height, topBorder, leftBorder, bottomBorder, rightBorder, red, green, blue, alpha); + } else { + batchDraw(() -> dynamicTexInternal(material, x, y, width, height, topBorder, leftBorder, bottomBorder, rightBorder, red, green, blue, alpha)); + } + } + + //Todo, This method can probably be made a lot more efficient. + private void dynamicTexInternal(Material material, int xPos, int yPos, int xSize, int ySize, int topBorder, int leftBorder, int bottomBorder, int rightBorder, float red, float green, float blue, float alpha) { + TextureAtlasSprite sprite = material.sprite(); + VertexConsumer buffer = material.buffer(buffers, GuiRender::texColType); + Matrix4f mat = pose.last().pose(); + SpriteContents contents = sprite.contents(); + int texWidth = contents.width(); + int texHeight = contents.height(); + int trimWidth = texWidth - leftBorder - rightBorder; + int trimHeight = texHeight - topBorder - bottomBorder; + if (xSize <= texWidth) trimWidth = Math.min(trimWidth, xSize - rightBorder); + if (xSize <= 0 || ySize <= 0 || trimWidth <= 0 || trimHeight <= 0) return; + + for (int x = 0; x < xSize; ) { + int rWidth = Math.min(xSize - x, trimWidth); + int trimU = 0; + if (x != 0) { + if (x + leftBorder + trimWidth <= xSize) { + trimU = leftBorder; + } else { + trimU = (texWidth - (xSize - x)); + } + } + + //Top & Bottom trim + bufferDynamic(buffer, mat, sprite, xPos + x, yPos, trimU, 0, rWidth, topBorder, red, green, blue, alpha); + bufferDynamic(buffer, mat, sprite, xPos + x, yPos + ySize - bottomBorder, trimU, texHeight - bottomBorder, rWidth, bottomBorder, red, green, blue, alpha); + + rWidth = Math.min(xSize - x - leftBorder - rightBorder, trimWidth); + for (int y = 0; y < ySize; ) { + int rHeight = Math.min(ySize - y - topBorder - bottomBorder, trimHeight); + int trimV; + if (y + (texHeight - topBorder - bottomBorder) <= ySize) { + trimV = topBorder; + } else { + trimV = texHeight - (ySize - y); + } + + //Left & Right trim + if (x == 0 && y + topBorder < ySize - bottomBorder) { + bufferDynamic(buffer, mat, sprite, xPos, yPos + y + topBorder, 0, trimV, leftBorder, rHeight, red, green, blue, alpha); + bufferDynamic(buffer, mat, sprite, xPos + xSize - rightBorder, yPos + y + topBorder, trimU + texWidth - rightBorder, trimV, rightBorder, rHeight, red, green, blue, alpha); + } + + //Core + if (y + topBorder < ySize - bottomBorder && x + leftBorder < xSize - rightBorder) { + bufferDynamic(buffer, mat, sprite, xPos + x + leftBorder, yPos + y + topBorder, leftBorder, topBorder, rWidth, rHeight, red, green, blue, alpha); + } + y += trimHeight; + } + x += trimWidth; + } + } + + private void bufferDynamic(VertexConsumer builder, Matrix4f mat, TextureAtlasSprite tex, int x, int y, double textureX, double textureY, int width, int height, float red, float green, float blue, float alpha) { + int w = tex.contents().width(); + int h = tex.contents().height(); + //@formatter:off + builder.vertex(mat, x, y + height, 0).color(red, green, blue, alpha).uv(tex.getU((textureX / w) * 16D), tex.getV(((textureY + height) / h) * 16)).endVertex(); + builder.vertex(mat, x + width, y + height, 0).color(red, green, blue, alpha).uv(tex.getU(((textureX + width) / w) * 16), tex.getV(((textureY + height) / h) * 16)).endVertex(); + builder.vertex(mat, x + width, y, 0).color(red, green, blue, alpha).uv(tex.getU(((textureX + width) / w) * 16), tex.getV(((textureY) / h) * 16)).endVertex(); + builder.vertex(mat, x, y, 0).color(red, green, blue, alpha).uv(tex.getU((textureX / w) * 16), tex.getV(((textureY) / h) * 16)).endVertex(); + //@formatter:on + } + + //=== Strings ===// + + /** + * Draw string with shadow. + */ + public int drawString(@Nullable String message, double x, double y, int colour) { + return drawString(message, x, y, colour, true); + } + + public int drawString(@Nullable String message, double x, double y, int colour, boolean shadow) { + if (message == null) return 0; + int i = font().drawInBatch(message, (float) x, (float) y, colour, shadow, pose.last().pose(), buffers, Font.DisplayMode.NORMAL, 0, 15728880, font().isBidirectional()); + this.flushIfUnBatched(); + return i; + } + + /** + * Draw string with shadow. + */ + public int drawString(FormattedCharSequence message, double x, double y, int colour) { + return drawString(message, x, y, colour, true); + } + + public int drawString(FormattedCharSequence message, double x, double y, int colour, boolean shadow) { + int i = font().drawInBatch(message, (float) x, (float) y, colour, shadow, pose.last().pose(), buffers, Font.DisplayMode.NORMAL, 0, 15728880); + this.flushIfUnBatched(); + return i; + } + + /** + * Draw string with shadow. + */ + public int drawString(Component message, double x, double y, int colour) { + return drawString(message, x, y, colour, true); + } + + public int drawString(Component message, double x, double y, int colour, boolean shadow) { + return drawString(message.getVisualOrderText(), x, y, colour, shadow); + } + + /** + * Draw wrapped string with shadow. + */ + public void drawWordWrap(FormattedText message, double x, double y, int width, int colour) { + drawWordWrap(message, x, y, width, colour, false); + } + + public void drawWordWrap(FormattedText message, double x, double y, int width, int colour, boolean shadow) { + drawWordWrap(message, x, y, width, colour, shadow, font().lineHeight); + } + + public void drawWordWrap(FormattedText message, double x, double y, int width, int colour, boolean shadow, double spacing) { + for (FormattedCharSequence formattedcharsequence : font().split(message, width)) { + drawString(formattedcharsequence, x, y, colour, shadow); + y += spacing; + } + } + + /** + * Draw centered string with shadow. (centered on x position) + */ + public void drawCenteredString(String message, double x, double y, int colour) { + drawCenteredString(message, x, y, colour, true); + } + + public void drawCenteredString(String message, double x, double y, int colour, boolean shadow) { + drawString(message, x - font().width(message) / 2D, y, colour, shadow); + } + + /** + * Draw centered string with shadow. (centered on x position) + */ + public void drawCenteredString(Component message, double x, double y, int colour) { + drawCenteredString(message, x, y, colour, true); + } + + public void drawCenteredString(Component message, double x, double y, int colour, boolean shadow) { + FormattedCharSequence formattedcharsequence = message.getVisualOrderText(); + drawString(formattedcharsequence, x - font().width(formattedcharsequence) / 2D, y, colour, shadow); + } + + /** + * Draw centered string with shadow. (centered on x position) + */ + public void drawCenteredString(FormattedCharSequence message, double x, double y, int colour) { + drawCenteredString(message, x, y, colour, true); + } + + public void drawCenteredString(FormattedCharSequence message, double x, double y, int colour, boolean shadow) { + drawString(message, x - font().width(message) / 2D, y, colour, shadow); + } + + /** + * If text is to long ti fit between x and xMaz, the text will scroll from left to right. + * Otherwise, will render centered. + * This is mostly copied from {@link AbstractWidget#renderScrollingString(GuiGraphics, Font, Component, int, int, int, int, int)} + */ + @SuppressWarnings ("JavadocReference") + public void drawScrollingString(Component component, double x, double y, double xMax, int colour, boolean shadow) { + drawScrollingString(component, x, y, xMax, colour, shadow, true); + } + + /** + * If text is to long ti fit between x and xMaz, the text will scroll from left to right. + * Otherwise, will render centered. + * This is mostly copied from {@link net.minecraft.client.gui.components.AbstractWidget#renderScrollingString(GuiGraphics, Font, Component, int, int, int, int, int)} + */ + @SuppressWarnings ("JavadocReference") + public void drawScrollingString(Component component, double x, double y, double xMax, int colour, boolean shadow, boolean doScissor) { + int textWidth = font().width(component); + double width = xMax - x; + if (textWidth > width) { + double outside = textWidth - width; + double anim = (double) Util.getMillis() / 1000.0; + double e = Math.max(outside * 0.5, 3.0); + double f = Math.sin(1.5707963267948966 * Math.cos(6.283185307179586 * anim / e)) / 2.0 + 0.5; + double offset = Mth.lerp(f, 0.0, outside); + if (doScissor) pushScissor(x, y - 1, xMax, y + font().lineHeight + 1); + drawString(component, x - offset, y, colour, shadow); + if (doScissor) popScissor(); + } else { + drawCenteredString(component, (x + xMax) / 2, y, colour, shadow); + } + } + + //=== Tool Tips ===// + + private ItemStack tooltipStack = ItemStack.EMPTY; + + public void renderTooltip(ItemStack stack, double mouseX, double mouseY) { + renderTooltip(stack, mouseX, mouseY, 0xf0100010, 0xf0100010, 0x505000ff, 0x5028007f); + } + + public void renderTooltip(ItemStack stack, double mouseX, double mouseY, int backgroundTop, int backgroundBottom, int borderTop, int borderBottom) { + this.tooltipStack = stack; + this.toolTipWithImage(Screen.getTooltipFromItem(this.mc(), stack), stack.getTooltipImage(), mouseX, mouseY, backgroundTop, backgroundBottom, borderTop, borderBottom); + this.tooltipStack = ItemStack.EMPTY; + } + + public void toolTipWithImage(List tooltips, Optional tooltipImage, ItemStack stack, double mouseX, double mouseY) { + toolTipWithImage(tooltips, tooltipImage, stack, mouseX, mouseY, 0xf0100010, 0xf0100010, 0x505000ff, 0x5028007f); + } + + public void toolTipWithImage(List tooltips, Optional tooltipImage, ItemStack stack, double mouseX, double mouseY, int backgroundTop, int backgroundBottom, int borderTop, int borderBottom) { + this.tooltipStack = stack; + this.toolTipWithImage(tooltips, tooltipImage, mouseX, mouseY, backgroundTop, backgroundBottom, borderTop, borderBottom); + this.tooltipStack = ItemStack.EMPTY; + } + + public void toolTipWithImage(List tooltip, Optional tooltipImage, double mouseX, double mouseY) { + toolTipWithImage(tooltip, tooltipImage, mouseX, mouseY, 0xf0100010, 0xf0100010, 0x505000ff, 0x5028007f); + } + + public void toolTipWithImage(List tooltip, Optional tooltipImage, double mouseX, double mouseY, int backgroundTop, int backgroundBottom, int borderTop, int borderBottom) { + List list = ForgeHooksClient.gatherTooltipComponents(tooltipStack, tooltip, tooltipImage, (int) mouseX, guiWidth(), guiHeight(), font()); + this.renderTooltipInternal(list, mouseX, mouseY, backgroundTop, backgroundBottom, borderTop, borderBottom, DefaultTooltipPositioner.INSTANCE); + } + + public void renderTooltip(Component message, double mouseX, double mouseY) { + renderTooltip(message, mouseX, mouseY, 0xf0100010, 0xf0100010, 0x505000ff, 0x5028007f); + } + + public void renderTooltip(Component message, double mouseX, double mouseY, int backgroundTop, int backgroundBottom, int borderTop, int borderBottom) { + this.renderTooltip(List.of(message.getVisualOrderText()), mouseX, mouseY, backgroundTop, backgroundBottom, borderTop, borderBottom); + } + + public void componentTooltip(List tooltips, double mouseX, double mouseY) { + componentTooltip(tooltips, mouseX, mouseY, 0xf0100010, 0xf0100010, 0x505000ff, 0x5028007f); + } + + public void componentTooltip(List tooltips, double mouseX, double mouseY, int backgroundTop, int backgroundBottom, int borderTop, int borderBottom) { + List list = ForgeHooksClient.gatherTooltipComponents(tooltipStack, tooltips, Optional.empty(), (int) mouseX, guiWidth(), guiHeight(), font()); + this.renderTooltipInternal(list, mouseX, mouseY, backgroundTop, backgroundBottom, borderTop, borderBottom, DefaultTooltipPositioner.INSTANCE); + } + + public void componentTooltip(List tooltips, double mouseX, double mouseY, ItemStack stack) { + componentTooltip(tooltips, mouseX, mouseY, 0xf0100010, 0xf0100010, 0x505000ff, 0x5028007f, stack); + } + + public void componentTooltip(List tooltips, double mouseX, double mouseY, int backgroundTop, int backgroundBottom, int borderTop, int borderBottom, ItemStack stack) { + this.tooltipStack = stack; + List list = ForgeHooksClient.gatherTooltipComponents(tooltipStack, tooltips, Optional.empty(), (int) mouseX, guiWidth(), guiHeight(), font()); + this.renderTooltipInternal(list, mouseX, mouseY, backgroundTop, backgroundBottom, borderTop, borderBottom, DefaultTooltipPositioner.INSTANCE); + this.tooltipStack = ItemStack.EMPTY; + } + + public void renderTooltip(List tooltips, double mouseX, double mouseY) { + renderTooltip(tooltips, mouseX, mouseY, 0xf0100010, 0xf0100010, 0x505000ff, 0x5028007f); + } + + public void renderTooltip(List tooltips, double mouseX, double mouseY, int backgroundTop, int backgroundBottom, int borderTop, int borderBottom) { + this.renderTooltipInternal(tooltips.stream().map(ClientTooltipComponent::create).collect(Collectors.toList()), mouseX, mouseY, backgroundTop, backgroundBottom, borderTop, borderBottom, DefaultTooltipPositioner.INSTANCE); + } + + public void renderTooltip(List tooltips, ClientTooltipPositioner positioner, double mouseX, double mouseY, int backgroundTop, int backgroundBottom, int borderTop, int borderBottom) { + this.renderTooltipInternal(tooltips.stream().map(ClientTooltipComponent::create).collect(Collectors.toList()), mouseX, mouseY, backgroundTop, backgroundBottom, borderTop, borderBottom, positioner); + } + + private void renderTooltipInternal(List tooltips, double mouseX, double mouseY, int backgroundTop, int backgroundBottom, int borderTop, int borderBottom, ClientTooltipPositioner positioner) { + if (!tooltips.isEmpty()) { + RenderTooltipEvent.Pre event = ForgeHooksClient.onRenderTooltipPre(tooltipStack, guiGraphicsWrapper(), (int) mouseX, (int) mouseY, guiWidth(), guiHeight(), tooltips, font(), positioner); + if (event.isCanceled()) return; + + + int width = 0; + int height = tooltips.size() == 1 ? -2 : 0; + for (ClientTooltipComponent line : tooltips) { + width = Math.max(width, line.getWidth(event.getFont())); + height += line.getHeight(); + } + + Vector2ic position = positioner.positionTooltip(guiWidth(), guiHeight(), event.getX(), event.getY(), width, height); + int xPos = position.x(); + int yPos = Math.max(position.y(), 3); //Default positioner allows negative y-pos for some reason... + + RenderTooltipEvent.Color colorEvent = new RenderTooltipEvent.Color(tooltipStack, guiGraphicsWrapper(), (int) mouseX, (int) mouseY, font(), backgroundTop, borderTop, borderBottom, tooltips); + colorEvent.setBackgroundEnd(backgroundBottom); + MinecraftForge.EVENT_BUS.post(colorEvent); + + toolTipBackground(xPos - 3, yPos - 3, width + 6, height + 6, colorEvent.getBackgroundStart(), colorEvent.getBackgroundEnd(), colorEvent.getBorderStart(), colorEvent.getBorderEnd(), false); + int linePos = yPos; + + for (int i = 0; i < tooltips.size(); ++i) { + ClientTooltipComponent component = tooltips.get(i); + component.renderText(event.getFont(), xPos, linePos, pose.last().pose(), buffers); + linePos += component.getHeight() + (i == 0 ? 2 : 0); + } + + linePos = yPos; + + for (int i = 0; i < tooltips.size(); ++i) { + ClientTooltipComponent component = tooltips.get(i); + component.renderImage(event.getFont(), xPos, linePos, renderWrapper); + linePos += component.getHeight() + (i == 0 ? 2 : 0); + } + } + } + + public void renderComponentHoverEffect(@Nullable Style style, int mouseX, int mouseY) { + if (style != null && style.getHoverEvent() != null) { + HoverEvent event = style.getHoverEvent(); + HoverEvent.ItemStackInfo stackInfo = event.getValue(HoverEvent.Action.SHOW_ITEM); + if (stackInfo != null) { + renderTooltip(stackInfo.getItemStack(), mouseX, mouseY); + } else { + HoverEvent.EntityTooltipInfo tooltipInfo = event.getValue(HoverEvent.Action.SHOW_ENTITY); + if (tooltipInfo != null) { + if (mc().options.advancedItemTooltips) { + componentTooltip(tooltipInfo.getTooltipLines(), mouseX, mouseY); + } + } else { + Component component = event.getValue(HoverEvent.Action.SHOW_TEXT); + if (component != null) { + renderTooltip(font().split(component, Math.max(this.guiWidth() / 2, 200)), mouseX, mouseY); + } + } + } + } + } + + //=== ItemStacks ===// + + /** + * Renders an item stack on the screen. + * Important Note: Required z clearance is 32, This must be accounted for when setting the z depth in an element using this render method. + */ + public void renderItem(ItemStack stack, double x, double y) { + renderItem(stack, x, y, 16); + } + + /** + * Renders an item stack on the screen. + * Important Note: Required z clearance is size*2, This must be accounted for when setting the z depth in an element using this render method. + */ + public void renderItem(ItemStack stack, double x, double y, double size) { + this.renderItem(mc().player, mc().level, stack, x, y, size, 0); + } + + /** + * Renders an item stack on the screen. + * Important Note: Required z clearance is size*2, This must be accounted for when setting the z depth in an element using this render method. + */ + public void renderItem(ItemStack stack, double x, double y, double size, int modelRand) { + this.renderItem(mc().player, mc().level, stack, x, y, size, modelRand); + } + + /** + * Renders an item stack on the screen. + * Important Note: Required z clearance is 32, This must be accounted for when setting the z depth in an element using this render method. + */ + public void renderFakeItem(ItemStack stack, double x, double y) { + renderFakeItem(stack, x, y, 16); + } + + /** + * Renders an item stack on the screen. + * Important Note: Required z clearance is size*2, This must be accounted for when setting the z depth in an element using this render method. + */ + public void renderFakeItem(ItemStack stack, double x, double y, double size) { + this.renderItem(null, mc().level, stack, x, y, size, 0); + } + + /** + * Renders an item stack on the screen. + * Important Note: Required z clearance is 32, This must be accounted for when setting the z depth in an element using this render method. + */ + public void renderItem(LivingEntity entity, ItemStack stack, double x, double y, int modelRand) { + renderItem(entity, stack, x, y, 16, modelRand); + } + + /** + * Renders an item stack on the screen. + * Important Note: Required z clearance is size*2, This must be accounted for when setting the z depth in an element using this render method. + */ + public void renderItem(LivingEntity entity, ItemStack stack, double x, double y, double size, int modelRand) { + this.renderItem(entity, entity.level(), stack, x, y, size, modelRand); + } + + /** + * Renders an item stack on the screen. + * Important Note: Required z clearance is size*2, This must be accounted for when setting the z depth in an element using this render method. + * + * @param size Width and height of the stack in pixels (Standard default is 16) + * @param modelRand A somewhat random value used in model gathering, Not very important, Can just use 0 or x/y position. + */ + public void renderItem(@Nullable LivingEntity entity, @Nullable Level level, ItemStack stack, double x, double y, double size, int modelRand) { + if (!stack.isEmpty()) { + BakedModel bakedmodel = mc().getItemRenderer().getModel(stack, level, entity, modelRand); + pose.pushPose(); + pose.translate(x + (size / 2D), y + (size / 2D), size); + try { + pose.mulPoseMatrix((new Matrix4f()).scaling(1.0F, -1.0F, 1.0F)); + pose.scale((float) size, (float) size, (float) size); + boolean flag = !bakedmodel.usesBlockLight(); + if (flag) Lighting.setupForFlatItems(); + mc().getItemRenderer().render(stack, ItemDisplayContext.GUI, false, pose, buffers, 0xf000f0, OverlayTexture.NO_OVERLAY, bakedmodel); + this.flush(); + if (flag) Lighting.setupFor3DItems(); + } catch (Throwable throwable) { + CrashReport crashreport = CrashReport.forThrowable(throwable, "Rendering item"); + CrashReportCategory crashreportcategory = crashreport.addCategory("Item being rendered"); + crashreportcategory.setDetail("Item Type", () -> String.valueOf(stack.getItem())); + crashreportcategory.setDetail("Item Stack", () -> String.valueOf(stack.getItem())); + crashreportcategory.setDetail("Item Damage", () -> String.valueOf(stack.getDamageValue())); + crashreportcategory.setDetail("Item NBT", () -> String.valueOf(stack.getTag())); + crashreportcategory.setDetail("Item Foil", () -> String.valueOf(stack.hasFoil())); + throw new ReportedException(crashreport); + } + pose.popPose(); + } + } + + /** + * Draw item decorations (Count, Damage, Cool-down) + * This should be rendered at the same position and size as the item. + * There is no need to fiddle with z offsets or anything, just call renderItemDecorations after renderItem and it will work. + * Z depth requirements are the same as the renderItem method. + */ + public void renderItemDecorations(ItemStack stack, double x, double y) { + renderItemDecorations(stack, x, y, 16); + } + + /** + * Draw item decorations (Count, Damage, Cool-down) + * This should be rendered at the same position and size as the item. + * There is no need to fiddle with z offsets or anything, just call renderItemDecorations after renderItem and it will work. + * Z depth requirements are the same as the renderItem method. + */ + public void renderItemDecorations(ItemStack stack, double x, double y, double size) { + this.renderItemDecorations(stack, x, y, size, null); + } + + /** + * Draw item decorations (Count, Damage, Cool-down) + * This should be rendered at the same position and size as the item. + * There is no need to fiddle with z offsets or anything, just call renderItemDecorations after renderItem and it will work. + * Z depth requirements are the same as the renderItem method. + */ + public void renderItemDecorations(ItemStack stack, double x, double y, @Nullable String text) { + renderItemDecorations(stack, x, y, 16, text); + } + + /** + * Draw item decorations (Count, Damage, Cool-down) + * This should be rendered at the same position and size as the item. + * There is no need to fiddle with z offsets or anything, just call renderItemDecorations after renderItem and it will work. + * Z depth requirements are the same as the renderItem method. + */ + public void renderItemDecorations(ItemStack stack, double x, double y, double size, @Nullable String text) { + if (!stack.isEmpty()) { + pose.pushPose(); + float scale = (float) size / 16F; + pose.translate(x, y, (size * 2) - 0.1); + pose.scale(scale, scale, 1F); + pose.translate(-x, -y, 0); + + if (stack.getCount() != 1 || text != null) { + String s = text == null ? String.valueOf(stack.getCount()) : text; + drawString(s, x + 19 - 2 - font().width(s), y + 6 + 3, 0xffffff, true); + } + + if (stack.isBarVisible()) { + int l = stack.getBarWidth(); + int i = stack.getBarColor(); + double j = x + 2; + double k = y + 13; + pose.translate(0.0F, 0.0F, 0.04); + fill(j, k, j + 13, k + 2, 0xff000000); + pose.translate(0.0F, 0.0F, 0.02); + fill(j, k, j + l, k + 1, i | 0xff000000); + } + + LocalPlayer localplayer = mc().player; + float f = localplayer == null ? 0.0F : localplayer.getCooldowns().getCooldownPercent(stack.getItem(), mc().getFrameTime()); + if (f > 0.0F) { + double i1 = y + Mth.floor(16.0F * (1.0F - f)); + double j1 = i1 + Mth.ceil(16.0F * f); + pose.translate(0.0F, 0.0F, 0.02); + fill(x, i1, x + 16, j1, Integer.MAX_VALUE); + } + + pose.popPose(); + if (size == 16) { + net.minecraftforge.client.ItemDecoratorHandler.of(stack).render(guiGraphicsWrapper(), font(), stack, (int) x, (int) y); + } + } + } + + + //=== Entity? ===// + + + //=== Render Utils ===// + + public void pushScissorRect(Rectangle rectangle) { + pushScissorRect(rectangle.x(), rectangle.y(), rectangle.width(), rectangle.height()); + } + + public void pushScissorRect(double x, double y, double width, double height) { + flushIfBatched(); + scissorHandler.pushGuiScissor(x, y, width, height); + } + + public void pushScissor(double xMin, double yMin, double xMax, double yMax) { + flushIfBatched(); + scissorHandler.pushGuiScissor(xMin, yMin, xMax - xMin, yMax - yMin); + } + + public void popScissor() { + scissorHandler.popScissor(); + } + + /** + * Sets the render system shader colour, Effect will vary depending on what is being rendered. + * Ideally this should be avoided in favor of render calls that accept colour. + */ + public void setColor(float red, float green, float blue, float alpha) { + this.flushIfBatched(); + RenderSystem.setShaderColor(red, green, blue, alpha); + } + + //=== Static Utils ===// + + public static boolean isInRect(double minX, double minY, double width, double height, double testX, double testY) { + return ((testX >= minX && testX < minX + width) && (testY >= minY && testY < minY + height)); + } + + public static boolean isInRect(int minX, int minY, int width, int height, double testX, double testY) { + return ((testX >= minX && testX < minX + width) && (testY >= minY && testY < minY + height)); + } + + /** + * Mixes the two input colours by adding up the R, G, B and A values of each input. + */ + public static int mixColours(int colour1, int colour2) { + return mixColours(colour1, colour2, false); + } + + /** + * Mixes the two input colours by adding up the R, G, B and A values of each input. + * + * @param subtract If true, subtract colour2 from colour1, otherwise add colour2 to colour1. + */ + public static int mixColours(int colour1, int colour2, boolean subtract) { + int alpha1 = colour1 >> 24 & 255; + int alpha2 = colour2 >> 24 & 255; + int red1 = colour1 >> 16 & 255; + int red2 = colour2 >> 16 & 255; + int green1 = colour1 >> 8 & 255; + int green2 = colour2 >> 8 & 255; + int blue1 = colour1 & 255; + int blue2 = colour2 & 255; + + int alpha = Mth.clamp(alpha1 + (subtract ? -alpha2 : alpha2), 0, 255); + int red = Mth.clamp(red1 + (subtract ? -red2 : red2), 0, 255); + int green = Mth.clamp(green1 + (subtract ? -green2 : green2), 0, 255); + int blue = Mth.clamp(blue1 + (subtract ? -blue2 : blue2), 0, 255); + + return (alpha & 0xFF) << 24 | (red & 0xFF) << 16 | (green & 0xFF) << 8 | blue & 0xFF; + } + + /** + * Returns a colour half-way between the two input colours. + * The R, G, B and A channels are extracted from each input, + * Then for each chanel, a midpoint is determined, + * And a new colour is constructed based on the midpoint of each channel. + */ + public static int midColour(int colour1, int colour2) { + int alpha1 = colour1 >> 24 & 255; + int alpha2 = colour2 >> 24 & 255; + int red1 = colour1 >> 16 & 255; + int red2 = colour2 >> 16 & 255; + int green1 = colour1 >> 8 & 255; + int green2 = colour2 >> 8 & 255; + int blue1 = colour1 & 255; + int blue2 = colour2 & 255; + return (alpha2 + (alpha1 - alpha2) / 2 & 0xFF) << 24 | (red2 + (red1 - red2) / 2 & 0xFF) << 16 | (green2 + (green1 - green2) / 2 & 0xFF) << 8 | blue2 + (blue1 - blue2) / 2 & 0xFF; + } + + private static float r(int argb) { + return (argb >> 16 & 255) / 255F; + } + + private static float g(int argb) { + return (argb >> 8 & 255) / 255F; + } + + private static float b(int argb) { + return (argb & 255) / 255F; + } + + private static float a(int argb) { + return (argb >>> 24) / 255F; + } + + //Render Type Builders + + public static RenderType texType(ResourceLocation location) { + return RenderType.create("tex_type", DefaultVertexFormat.POSITION_TEX, VertexFormat.Mode.QUADS, 256, RenderType.CompositeState.builder() + .setShaderState(new RenderStateShard.ShaderStateShard(GameRenderer::getPositionTexShader)) + .setTextureState(new RenderStateShard.TextureStateShard(location, false, false)) + .setTransparencyState(RenderStateShard.TRANSLUCENT_TRANSPARENCY) + .setCullState(RenderStateShard.NO_CULL) + .createCompositeState(false)); + } + + public static RenderType texColType(ResourceLocation location) { + return RenderType.create("tex_col_type", DefaultVertexFormat.POSITION_COLOR_TEX, VertexFormat.Mode.QUADS, 256, RenderType.CompositeState.builder() + .setShaderState(new RenderStateShard.ShaderStateShard(GameRenderer::getPositionColorTexShader)) + .setTextureState(new RenderStateShard.TextureStateShard(location, false, false)) + .setTransparencyState(RenderStateShard.TRANSLUCENT_TRANSPARENCY) + .setCullState(RenderStateShard.NO_CULL) + .createCompositeState(false)); + } + + /** + * This exists to allow thing like the Tooltip events to still function correctly, hopefully without exploding... + */ + public static class RenderWrapper extends GuiGraphics { + private final GuiRender wrapped; + + private RenderWrapper(GuiRender wrapped) { + super(wrapped.mc(), wrapped.pose(), wrapped.buffers()); + this.wrapped = wrapped; + } + + @Override + public void drawManaged(Runnable runnable) { + wrapped.batchDraw(runnable); + } + + @Override + public void flush() { + wrapped.flush(); + } + + @Override + public void flushIfManaged() { + wrapped.flushIfBatched(); + } + + @Override + public void flushIfUnmanaged() { + wrapped.flushIfUnBatched(); + } + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/ScissorHandler.java b/src/main/java/codechicken/lib/gui/modular/lib/ScissorHandler.java new file mode 100644 index 00000000..06952dec --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/ScissorHandler.java @@ -0,0 +1,75 @@ +package codechicken.lib.gui.modular.lib; + +import com.google.common.collect.Queues; +import com.mojang.blaze3d.platform.Window; +import com.mojang.blaze3d.systems.RenderSystem; +import net.minecraft.client.Minecraft; +import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; + +import java.util.Deque; + +/** + * Created by brandon3055 on 29/06/2023 + */ +public class ScissorHandler { + public static final Logger LOGGER = LogManager.getLogger(); + private final Deque stack = Queues.newArrayDeque(); + + public void pushGuiScissor(double x, double y, double width, double height) { + Window window = Minecraft.getInstance().getWindow(); + int windowHeight = window.getHeight(); + double scale = window.getGuiScale(); + double scX = x * scale; + double scY = (double) windowHeight - (y + height) * scale; + double scW = Math.max(width * scale, 0); + double scH = Math.max(height * scale, 0); + pushScissor((int) scX, (int) scY, (int) scW, (int) scH); + } + + public void pushScissor(int x, int y, int width, int height) { + int xMax = x + width; + int yMax = y + height; + stack.addLast(ScissorState.createState(x, y, xMax, yMax, stack.peekLast()).apply()); + } + + public void popScissor() { + if (stack.isEmpty()) { + LOGGER.error("Scissor stack underflow"); + } + stack.removeLast(); + ScissorState active = stack.peekLast(); + if (active != null) { + active.apply(); + } else { + RenderSystem.disableScissor(); + } + } + + private record ScissorState(int x, int y, int xMax, int yMax) { + + private ScissorState apply() { + RenderSystem.enableScissor(x, y, xMax - x, yMax - y); + return this; + } + + private static ScissorState createState(int newX, int newY, int newXMax, int newYMax, ScissorState prevState) { + if (prevState != null) { + int x = Math.max(prevState.x, newX); + int y = Math.max(prevState.y, newY); + int xMax = Math.min(prevState.xMax, newXMax); + int yMax = Math.min(prevState.yMax, newYMax); + Minecraft mc = Minecraft.getInstance(); + if (x < 0) x = 0; + if (y < 0) y = 0; + if (xMax > mc.getWindow().getScreenWidth()) xMax = mc.getWindow().getScreenWidth(); + if (yMax > mc.getWindow().getScreenHeight()) yMax = mc.getWindow().getScreenHeight(); + if (xMax < x) xMax = x; + if (yMax < y) yMax = y; + return new ScissorState(x, y, xMax, yMax); + } else { + return new ScissorState(newX, newY, newXMax, newYMax); + } + } + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/SliderState.java b/src/main/java/codechicken/lib/gui/modular/lib/SliderState.java new file mode 100644 index 00000000..a9724798 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/SliderState.java @@ -0,0 +1,139 @@ +package codechicken.lib.gui.modular.lib; + +import codechicken.lib.gui.modular.lib.geometry.Axis; +import net.minecraft.client.gui.screens.Screen; +import org.jetbrains.annotations.Nullable; + +import java.util.function.Consumer; +import java.util.function.Supplier; + +/** + * The primary interface for managing getting and setting the position of slider elements. + *

+ * Created by brandon3055 on 01/09/2023 + */ +public interface SliderState { + + /** + * @return the current position. (Between 0 and 1) + */ + double getPos(); + + /** + * Set the current slide position, + * When using this method, make sure the provided value can not go outside the valid range of 0 to 1. + * + * @param pos Set the current position (Between 0 and 1) + */ + void setPos(double pos); + + /** + * The slider ratio is computed by dividing the length of the slider element by the total length of the slider track. + * This is primarily for things like setting the size of scroll bar sliders, and calibrating scrolling via middle-click + drag. + * + * @return scroll ratio, viewable content size / total content size (Range 0 to 1) + */ + default double sliderRatio() { + return 0.1; + } + + /** + * For controlling a slider via mouse scroll wheel. + * You can return a negative value to invert the scroll direction. + * + * @return The amount added to the position per scroll increment. + */ + default double scrollSpeed() { + double ratio = sliderRatio(); + return ratio < 0.1 ? ratio * 0.1 : ratio * ratio; + } + + /** + * @param scrollAxis The moving axis of the slider element. + * @return true if the current scroll wheel event should affect this slider. + */ + default boolean canScroll(Axis scrollAxis) { + return true; + } + + /** + * Creates a basic slide state which stores its position internally. + * Useful for things like simple slide control elements. + */ + static SliderState create(double speed) { + return create(speed, null); + } + + /** + * Creates a basic slide state which stores its position internally. + * And allows you to attach a change listener. + * Useful for things like simple slide control elements. + */ + static SliderState create(double speed, @Nullable Consumer changeListener) { + return new SliderState() { + double pos = 0; + + @Override + public double getPos() { + return pos; + } + + @Override + public void setPos(double pos) { + this.pos = pos; + if (changeListener != null) { + changeListener.accept(pos); + } + } + + @Override + public double scrollSpeed() { + return speed; + } + }; + } + + static SliderState forScrollBar(Supplier getPos, Consumer setPos, Supplier getRatio) { + return new SliderState() { + @Override + public double getPos() { + return getPos.get(); + } + + @Override + public void setPos(double pos) { + setPos.accept(pos); + } + + @Override + public double sliderRatio() { + return getRatio.get(); + } + + @Override + public boolean canScroll(Axis scrollAxis) { + //Controls scrolling left and right when shift key is down. + return (scrollAxis == Axis.Y) != Screen.hasShiftDown(); + } + }; + } + + static SliderState forSlider(Supplier getPos, Consumer setPos, Supplier getSpeed) { + return new SliderState() { + @Override + public double getPos() { + return getPos.get(); + } + + @Override + public void setPos(double pos) { + setPos.accept(pos); + } + + @Override + public double scrollSpeed() { + return getSpeed.get(); + } + }; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/TextState.java b/src/main/java/codechicken/lib/gui/modular/lib/TextState.java new file mode 100644 index 00000000..f92af296 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/TextState.java @@ -0,0 +1,56 @@ +package codechicken.lib.gui.modular.lib; + +import codechicken.lib.gui.modular.elements.GuiTextField; +import org.jetbrains.annotations.Nullable; + +import java.util.function.Consumer; +import java.util.function.Supplier; + +/** + * The primary interface for managing getting and setting the text in a {@link GuiTextField} + *

+ * Created by brandon3055 on 03/09/2023 + */ +public interface TextState { + + String getText(); + + void setText(String text); + + static TextState simpleState(String defaultValue) { + return simpleState(defaultValue, null); + } + + static TextState simpleState(String defaultValue, @Nullable Consumer changeListener) { + return new TextState() { + private String value = defaultValue; + + @Override + public String getText() { + return value; + } + + @Override + public void setText(String text) { + this.value = text; + if (changeListener != null) { + changeListener.accept(value); + } + } + }; + } + + static TextState create(Supplier getValue, Consumer setValue) { + return new TextState() { + @Override + public String getText() { + return getValue.get(); + } + + @Override + public void setText(String text) { + setValue.accept(text); + } + }; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/TooltipHandler.java b/src/main/java/codechicken/lib/gui/modular/lib/TooltipHandler.java new file mode 100644 index 00000000..7bc1bb20 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/TooltipHandler.java @@ -0,0 +1,105 @@ +package codechicken.lib.gui.modular.lib; + +import codechicken.lib.gui.modular.elements.GuiElement; +import net.covers1624.quack.util.SneakyUtils; +import net.minecraft.network.chat.Component; +import org.jetbrains.annotations.Nullable; + +import java.util.Collections; +import java.util.List; +import java.util.function.Supplier; + +/** + * Created by brandon3055 on 01/09/2023 + */ +public interface TooltipHandler> { + + Supplier> getTooltip(); + + /** + * Set a delay before element tooltip is displayed. + * Default delay is 10 ticks. + */ + T setTooltipDelay(int tooltipDelay); + + int getTooltipDelay(); + + /** + * Add hover text that is to be displayed when the user hovers their cursor over this element. (with a delay of 10 ticks) + * If you have multiple stacked elements with tooltips, only the top most element under the cursor will display its hover text. + * + * @param Tooltip A single tooltip text component supplier. + * @see #setTooltipDelay(int) + */ + default T setTooltipSingle(@Nullable Component Tooltip) { + return setTooltip(Tooltip == null ? null : () -> Collections.singletonList(Tooltip)); + } + + /** + * Add hover text that is to be displayed when the user hovers their cursor over this element. (with a delay of 10 ticks) + * If you have multiple stacked elements with tooltips, only the top most element under the cursor will display its hover text. + * + * @param Tooltip A single tooltip text component supplier. + * @see #setTooltipDelay(int) + */ + default T setTooltip(Component... Tooltip) { + return setTooltip(Tooltip == null ? null : () -> List.of(Tooltip)); + } + + /** + * Add hover text that is to be displayed when the user hovers their cursor over this element. (with a delay of 10 ticks) + * If you have multiple stacked elements with tooltips, only the top most element under the cursor will display its hover text. + * + * @param tooltip A single line tooltip component supplier. + * @see #setTooltipDelay(int) + */ + default T setTooltipSingle(@Nullable Supplier tooltip) { + return setTooltip(tooltip == null ? null : () -> Collections.singletonList(tooltip.get())); + } + + /** + * Add hover text that is to be displayed when the user hovers their cursor over this element. (with a delay of 10 ticks) + * If you have multiple stacked elements with tooltips, only the top most element under the cursor will display its hover text. + * + * @param tooltip The tooltip lines. If null or empty, hover text will be disabled + * @see #setTooltipDelay(int) + */ + default T setTooltip(@Nullable List tooltip) { + return setTooltip(tooltip == null ? null : () -> tooltip); + } + + /** + * Add hover text that is to be displayed when the user hovers their cursor over this element. (with a delay of 10 ticks) + * If you have multiple stacked elements with tooltips, only the top most element under the cursor will display its hover text. + * + * @param tooltip The tooltip lines. If null or the returned list is empty, hover text will be disabled + * @see #setTooltipDelay(int) + */ + T setTooltip(@Nullable Supplier> tooltip); + + /** + * Add hover text that is to be displayed when the user hovers their cursor over this element. + * If you have multiple stacked elements with tooltips, only the top most element under the cursor will display its hover text. + * + * @param tooltip The tooltip lines. If null or the returned list is empty, hover text will be disabled + * @param tooltipDelay Delay before hover text is shown. + */ + default T setTooltip(@Nullable Supplier> tooltip, int tooltipDelay) { + setTooltip(tooltip); + setTooltipDelay(tooltipDelay); + return SneakyUtils.unsafeCast(this); + } + + /** + * The method responsible for rendering element tool tips. + * Called from {@link GuiElement#renderOverlay(GuiRender, double, double, float, boolean)} + */ + default boolean renderTooltip(GuiRender render, double mouseX, double mouseY) { + Supplier> supplier = getTooltip(); + if (supplier == null) return false; + List list = supplier.get(); + if (list.isEmpty()) return false; + render.componentTooltip(list, mouseX, mouseY); + return true; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/container/ContainerGuiProvider.java b/src/main/java/codechicken/lib/gui/modular/lib/container/ContainerGuiProvider.java new file mode 100644 index 00000000..92282ee8 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/container/ContainerGuiProvider.java @@ -0,0 +1,33 @@ +package codechicken.lib.gui.modular.lib.container; + +import codechicken.lib.gui.modular.ModularGui; +import codechicken.lib.gui.modular.ModularGuiContainer; +import codechicken.lib.gui.modular.lib.GuiProvider; +import net.minecraft.client.gui.screens.inventory.MenuAccess; +import net.minecraft.world.inventory.AbstractContainerMenu; + +/** + * Created by brandon3055 on 08/09/2023 + */ +public abstract class ContainerGuiProvider implements GuiProvider { + + private ContainerScreenAccess screenAccess; + + public void setMenuAccess(ContainerScreenAccess screenAccess) { + this.screenAccess = screenAccess; + } + + @Override + public final void buildGui(ModularGui gui) { + buildGui(gui, screenAccess); + } + + /** + * This is the same as {@link GuiProvider#buildGui(ModularGui)} except you have access to the {@link MenuAccess} + * The given menu accessor should always be the parent screen unless your using some custom modular gui implementation. + * + * @param gui The modular gui instance. + * @param screenAccess The screen access (This will be a gui class that extends {@link ModularGuiContainer} + */ + public abstract void buildGui(ModularGui gui, ContainerScreenAccess screenAccess); +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/container/ContainerScreenAccess.java b/src/main/java/codechicken/lib/gui/modular/lib/container/ContainerScreenAccess.java new file mode 100644 index 00000000..d8998dd2 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/container/ContainerScreenAccess.java @@ -0,0 +1,22 @@ +package codechicken.lib.gui.modular.lib.container; + +import codechicken.lib.gui.modular.elements.GuiSlots; +import codechicken.lib.gui.modular.lib.GuiRender; +import net.minecraft.client.gui.screens.inventory.MenuAccess; +import net.minecraft.world.inventory.AbstractContainerMenu; +import net.minecraft.world.inventory.Slot; + +/** + * Used by {@link ContainerGuiProvider} + * Provides access to the ContainerMenu as well as some essential functions. + *

+ * Created by brandon3055 on 08/09/2023 + */ +public interface ContainerScreenAccess extends MenuAccess { + + /** + * This is the modular gui friendly method used by elements such as {@link GuiSlots} to render inventory item stacks. + */ + void renderSlot(GuiRender render, Slot slot); + +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/container/DataSync.java b/src/main/java/codechicken/lib/gui/modular/lib/container/DataSync.java new file mode 100644 index 00000000..4b0cb9ec --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/container/DataSync.java @@ -0,0 +1,48 @@ +package codechicken.lib.gui.modular.lib.container; + +import codechicken.lib.data.MCDataInput; +import codechicken.lib.inventory.container.data.AbstractDataStore; +import codechicken.lib.inventory.container.modular.ModularGuiContainerMenu; + +import java.util.function.Supplier; + +/** + * This class provides a convenient way to synchronize server side data with a client side screen via the ContainerMenu + *

+ * Created by brandon3055 on 09/09/2023 + */ +public class DataSync { + public static final int PKT_SEND_CHANGES = 255; + private final ModularGuiContainerMenu containerMenu; + private final AbstractDataStore dataStore; + private final Supplier valueGetter; + + public DataSync(ModularGuiContainerMenu containerMenu, AbstractDataStore dataStore, Supplier valueGetter) { + this.containerMenu = containerMenu; + this.dataStore = dataStore; + this.valueGetter = valueGetter; + containerMenu.dataSyncs.add(this); + } + + public T get() { + return dataStore.get(); + } + + /** + * This should only ever be called server side! + */ + public void detectAndSend() { + if (dataStore.isSameValue(valueGetter.get())) { + return; + } + dataStore.set(valueGetter.get()); + containerMenu.sendPacketToClient(PKT_SEND_CHANGES, buf -> { + buf.writeByte((byte) containerMenu.dataSyncs.indexOf(this)); + dataStore.toBytes(buf); + }); + } + + public void handleSyncPacket(MCDataInput buf) { + dataStore.fromBytes(buf); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/container/SlotGroup.java b/src/main/java/codechicken/lib/gui/modular/lib/container/SlotGroup.java new file mode 100644 index 00000000..1b6ff101 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/container/SlotGroup.java @@ -0,0 +1,119 @@ +package codechicken.lib.gui.modular.lib.container; + +import codechicken.lib.gui.modular.elements.GuiSlots; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.inventory.container.modular.ModularGuiContainerMenu; +import codechicken.lib.inventory.container.modular.ModularSlot; +import net.minecraft.world.Container; +import net.minecraft.world.entity.EquipmentSlot; +import net.minecraft.world.entity.Mob; +import net.minecraft.world.entity.player.Inventory; +import net.minecraft.world.inventory.Slot; +import net.minecraft.world.item.ItemStack; +import net.minecraft.world.item.enchantment.EnchantmentHelper; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collections; +import java.util.List; +import java.util.function.BiFunction; +import java.util.function.Function; + +/** + * Used to configure slots a set of slots in a ContainerMenu. + * The SlotGroup can then be passed to ModularGui elements such as {@link GuiSlots} in order to give the gui control over slot positioning and rendering. + *

+ * Ideally you should use a separate SlotGroup for each 'group' of slots, Meaning the players main inventory and hot bar should be separate ranges. + * That said, if you have multiple slots spread out across something like a machine gui, You can add them all to one SlotGroup then + * pass each individual slot from that range to a {@link GuiSlots#singleSlot(GuiParent, ContainerScreenAccess, SlotGroup, int)} + *

+ * Created by brandon3055 on 08/09/2023 + */ +public class SlotGroup { + + public final int zone; + public final List quickMoveTo; + + private final ModularGuiContainerMenu containerMenu; + private final List slots = new ArrayList<>(); + + public SlotGroup(ModularGuiContainerMenu containerMenu, int zone, int... quickMoveTo) { + this.zone = zone; + this.containerMenu = containerMenu; + this.quickMoveTo = Arrays.stream(quickMoveTo).boxed().toList(); + } + + public ModularSlot addSlot(ModularSlot slot) { + slots.add(slot); + containerMenu.addSlot(slot); + containerMenu.mapSlot(slot, this); + return slot; + } + + /** + * Convenient method for adding multiple slots. + * + * @param slotCount The number of slots to be added. + * @param startIndex Slot starting index. + * @param makeSlot Builder used to create the slots, Input integer will start at startIndex and increment by one for each slot. + */ + public void addSlots(int slotCount, int startIndex, Function makeSlot) { + for (int index = startIndex; index < startIndex + slotCount; index++) { + addSlot(makeSlot.apply(index)); + } + } + + public void addAllSlots(Container container) { + addAllSlots(container, ModularSlot::new); + } + + public void addAllSlots(Container container, BiFunction makeSlot) { + for (int index = 0; index < container.getContainerSize(); index++) { + addSlot(makeSlot.apply(container, index)); + } + } + + public void addPlayerMain(Inventory inventory) { + addSlots(27, 9, index -> new ModularSlot(inventory, index)); + } + + public void addPlayerBar(Inventory inventory) { + addSlots(9, 0, index -> new ModularSlot(inventory, index)); + } + + public void addPlayerArmor(Inventory inventory) { + for (int i = 0; i < 4; ++i) { + EquipmentSlot slot = EquipmentSlot.byTypeAndIndex(EquipmentSlot.Type.ARMOR, 3 - i); + addSlot(new ModularSlot(inventory, 39 - i) + .onSet((oldStack, newStack) -> onEquipItem(inventory, slot, newStack, oldStack)) + .setStackLimit(stack -> 1) + .setValidator(stack -> slot == Mob.getEquipmentSlotForItem(stack)) + .setCanRemove((player, stack) -> stack.isEmpty() || player.isCreative() || !EnchantmentHelper.hasBindingCurse(stack)) + ); + } + } + + public void addPlayerOffhand(Inventory inventory) { + addSlot(new ModularSlot(inventory, 40).onSet((oldStack, newStack) -> onEquipItem(inventory, EquipmentSlot.OFFHAND, newStack, oldStack))); + } + + static void onEquipItem(Inventory inventory, EquipmentSlot slot, ItemStack newStack, ItemStack oldStack) { + inventory.player.onEquipItem(slot, oldStack, newStack); + } + + public int size() { + return slots.size(); + } + + public ModularSlot getSlot(int index) { + return slots.get(index); + } + + public int indexOf(Slot slot) { + return slots.indexOf(slot); + } + + public List slots() { + return Collections.unmodifiableList(slots); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/Align.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Align.java new file mode 100644 index 00000000..c38b84b3 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Align.java @@ -0,0 +1,14 @@ +package codechicken.lib.gui.modular.lib.geometry; + +/** + * Created by brandon3055 on 31/08/2023 + */ +public enum Align { + MIN, + CENTER, + MAX; + public static Align LEFT = MIN; + public static Align RIGHT = MAX; + public static Align TOP = MIN; + public static Align BOTTOM = MAX; +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/Axis.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Axis.java new file mode 100644 index 00000000..7f4ac8f2 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Axis.java @@ -0,0 +1,13 @@ +package codechicken.lib.gui.modular.lib.geometry; + +/** + * Created by brandon3055 on 02/09/2023 + */ +public enum Axis { + X, + Y; + + public Axis opposite() { + return this == X ? Y : X; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/AxisConfig.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/AxisConfig.java new file mode 100644 index 00000000..5fbd204b --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/AxisConfig.java @@ -0,0 +1,96 @@ +package codechicken.lib.gui.modular.lib.geometry; + +import org.apache.commons.lang3.function.TriFunction; +import org.jetbrains.annotations.Nullable; + +/** + * Denies how each of the three axis parameters are computed based on the available constraints. + * Note: If both min and max are defined, size is ignored. + */ +public enum AxisConfig { + //@formatter:off + // | Compute Axis Min | Compute Axis Max | Compute Axis Size | + /** + * If nothing is constrained. + * min=0, max=0, size=0 + * */ + NONE (0, (min, max, size) -> 0D, (min, max, size) -> 0D, (min, max, size) -> 0D), + /** + * If only Min is constrained. + * min=min, max=min, size=0 + * */ + MIN_ONLY (1, (min, max, size) -> min.get(), (min, max, size) -> min.get(), (min, max, size) -> 0D), + /** + * If only Max is constrained. + * min=max, max=max, size=0 + * */ + MAX_ONLY (1, (min, max, size) -> max.get(), (min, max, size) -> max.get(), (min, max, size) -> 0D), + /** + * If only Size is constrained. + * min=0, max=size, size=size + * */ + SIZE_ONLY (1, (min, max, size) -> 0D, (min, max, size) -> size.get(), (min, max, size) -> size.get()), + /** + * If Min and Size are constrained . + * min=min, max=min+size, size=size + * */ + MIN_SIZE (2, (min, max, size) -> min.get(), (min, max, size) -> min.get() + size.get(), (min, max, size) -> size.get()), + /** + * If Max and Size are constrained. + * min=max-size, max=max, size=size + * */ + MAX_SIZE (2, (min, max, size) -> max.get() - size.get(), (min, max, size) -> max.get(), (min, max, size) -> size.get()), + /** + * If Min and Max are constrained. + * min=min, max=max, size=max-min + * */ + MIN_MAX (2, (min, max, size) -> min.get(), (min, max, size) -> max.get(), (min, max, size) -> max.get() - min.get()), + /** + * If Min, Max and Size are constrained the Size is ignored. + * min=min, max=max, size=max-min + * */ + MIN_MAX_SIZE(3, (min, max, size) -> min.get(), (min, max, size) -> max.get(), (min, max, size) -> max.get() - min.get()); + //@formatter:on + + public final int constraints; + public final TriFunction min; + public final TriFunction max; + public final TriFunction size; + //[min][max][size] TODO, I Clean this up once i confirm everything works correctly. + private static final AxisConfig[][][] LOOKUP = new AxisConfig[][][]{ + { //Min = 0 + { //Max = 0 + NONE, //Size = 0 + SIZE_ONLY //Size = 1 + }, + { //Max = 1 + MAX_ONLY, //Size = 0 + MAX_SIZE //Size = 1 + } + }, + { //Min = 1 + { //Max = 0 + MIN_ONLY, //Size = 0 + MIN_SIZE //Size = 1 + }, + { //Max = 1 + MIN_MAX, //Size = 0 + MIN_MAX_SIZE//Size = 1 + } + } + }; + + AxisConfig(int constraints, + TriFunction min, + TriFunction max, + TriFunction size) { + this.constraints = constraints; + this.min = min; + this.max = max; + this.size = size; + } + + public static AxisConfig getConfigFor(@Nullable Constraint min, @Nullable Constraint max, @Nullable Constraint size) { + return LOOKUP[min != null ? 1 : 0][max != null ? 1 : 0][size != null ? 1 : 0]; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/Borders.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Borders.java new file mode 100644 index 00000000..d9ef9ce7 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Borders.java @@ -0,0 +1,118 @@ +package codechicken.lib.gui.modular.lib.geometry; + +import java.util.Objects; + +/** + * Created by brandon3055 on 28/08/2023 + */ +public final class Borders { + public double top; + public double left; + public double bottom; + public double right; + + public Borders(double top, double left, double bottom, double right) { + this.top = top; + this.left = left; + this.bottom = bottom; + this.right = right; + } + + public static Borders create(double borders) { + return create(borders, borders); + } + + public static Borders create(double leftRight, double topBottom) { + return create(topBottom, leftRight, topBottom, leftRight); + } + + public static Borders create(double top, double left, double bottom, double right) { + return new Borders(top, left, bottom, right); + } + + public double top() { + return top; + } + + public double left() { + return left; + } + + public double bottom() { + return bottom; + } + + public double right() { + return right; + } + + public Borders top(double top) { + this.top = top; + return this; + } + + public Borders left(double left) { + this.left = left; + return this; + } + + public Borders bottom(double bottom) { + this.bottom = bottom; + return this; + } + + public Borders right(double right) { + this.right = right; + return this; + } + + public Borders setTopBottom(double topBottom) { + return top(topBottom).bottom(topBottom); + } + + public Borders setLeftRight(double leftRight) { + return left(leftRight).setLeftRight(leftRight); + } + + public Borders setBorders(double borders) { + return setBorders(borders, borders); + } + + public Borders setBorders(double topBottom, double leftRight) { + return setBorders(topBottom, leftRight, topBottom, leftRight); + } + + public Borders setBorders(double top, double left, double bottom, double right) { + this.top = top; + this.left = left; + this.bottom = bottom; + this.right = right; + return this; + } + + @Override + public boolean equals(Object obj) { + if (obj == this) return true; + if (obj == null || obj.getClass() != this.getClass()) return false; + var that = (Borders) obj; + return Double.doubleToLongBits(this.top) == Double.doubleToLongBits(that.top) && + Double.doubleToLongBits(this.left) == Double.doubleToLongBits(that.left) && + Double.doubleToLongBits(this.bottom) == Double.doubleToLongBits(that.bottom) && + Double.doubleToLongBits(this.right) == Double.doubleToLongBits(that.right); + } + + @Override + public int hashCode() { + return Objects.hash(top, left, bottom, right); + } + + @Override + public String toString() { + return "Borders[" + + "top=" + top + ", " + + "left=" + left + ", " + + "bottom=" + bottom + ", " + + "right=" + right + ']'; + } + +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/ConstrainedGeometry.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/ConstrainedGeometry.java new file mode 100644 index 00000000..b0c62353 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/ConstrainedGeometry.java @@ -0,0 +1,376 @@ +package codechicken.lib.gui.modular.lib.geometry; + +import codechicken.lib.gui.modular.elements.GuiElement; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.function.Supplier; + +import static codechicken.lib.gui.modular.lib.geometry.GeoParam.*; + +/** + * This is the base class used to define the size and position of a GuiElement. + *

+ * The geometry system is designed to be very user-friendly, yet very powerful, with the ability to do all sorts of fun and interesting things. + * But if all you want is the ability to set simple size and position values, then there is no need to read any further. + * Just use the basic setters setXPos, setYPos, setWidth, setHeight and everything should just work. + * If you want to dive deeper, read on. + *

+ * The geometry system is based around 6 core parameters: xMin, xMax, xSize, yMin, yMax and ySize. See: {@link GeoParam}
+ * These parameters are defined using 'Constraints' See: {@link Constraint}
+ * When properly constrained, it will be possible to request values for any of these 6 parameters.
+ * For each axis (x and y) there are 3 constraints: min, max and size.
+ * In order for an axis to be properly defined 2 of these constraints need to be set.
+ * e.g. (xMin and xSize) or (xMin and xMax) or (xMax and xSize)
+ * Using any of these combinations the position and size of an axis can be computed. + *

+ * Look at {@link AxisConfig} for details on the values that get returned when too few constraints are set.
+ * TLDR: We will always try to return "reasonable" default values. e.g. if only size is constrained, xMin will default + * to 0 and xMax will default to 0 + xSize.
+ * If all three constraints are set then the size constraint will be ignored.
+ * You can use {@link #strictMode(boolean)} to change this behavior. + *

+ * There are a number of different types of constraints that can be used, including completely custom constraints.
+ * Take a look at the {@link Constraint} class for a detailed list with explanations. + *

+ * Note: + * All position and size values in ModularGui use doubles. This is because floating point position and size values + * can be very useful at times. But they can also cause a lot of ugly visual artifacts when not used properly. + * So by default, all the builtin constraints will cast their outputs to an integer value. + * If you need floating point precision, you can enable it by calling .precise() on any of the builtin constraints. + *

+ * Created by brandon3055 on 29/06/2023 + */ +public abstract class ConstrainedGeometry> implements GuiParent { + + private Constraint xMin = null; + private Constraint xMax = null; + private Constraint xSize = null; + private AxisConfig xAxis = AxisConfig.NONE; + + private Constraint yMin = null; + private Constraint yMax = null; + private Constraint ySize = null; + private AxisConfig yAxis = AxisConfig.NONE; + + //Permanently bound immutable position and rectangle elements. + private final Position position = Position.create(this); + private final Rectangle rectangle = Rectangle.create(this); + private final Rectangle.Mutable childBounds = getRectangle().mutable(); + + private boolean strictMode = false; + + @NotNull + public abstract GuiParent getParent(); + + public GeoRef getParent(GeoParam param) { + return getParent().get(param); + } + + //=== Simple Setters ===// + + /** + * Simple method for setting the x position of this element. + *

+ * Constrains the left side of this element, to the left side of the parent element, + * with an offset that is calculated by subtracting parent's current x pos from the given x pos. + *

+ * In other words, if the parent element's position changes, this element will move with it. + * + * @param x The X position in screen space. + * @return This Element. + */ + public T setXPos(double x) { + GeoRef parentLeft = getParent(LEFT); + return constrain(LEFT, Constraint.relative(parentLeft, x - parentLeft.get())); + } + + /** + * Simple method for setting the y position of this element. + *

+ * Constrains the top side of this element, to the top side of the parent element, + * with an offset that is calculated by subtracting parent's current y pos from the given y pos. + *

+ * In other words, if the parent element's position changes, this element will move with it. + * + * @param y The Y position in screen space. + * @return This Element. + */ + public T setYPos(double y) { + GeoRef parentTop = getParent(LEFT); + return constrain(LEFT, Constraint.relative(parentTop, y - parentTop.get())); + } + + /** + * Convenience method for setting both x and y positions. + * + * @param x The X position in screen space. + * @param y The Y position in screen space. + * @return This Element. + * @see #setXPos(double) + * @see #setYPos(double) + */ + public T setPos(double x, double y) { + return setXPos(x).setYPos(y); + } + + /** + * Simple method for setting the width of this element. + * + * @param width The width to apply. + * @return This Element. + */ + public T setWidth(double width) { + return constrain(WIDTH, Constraint.literal(width)); + } + + /** + * Simple method for setting the height of this element. + * + * @param height The height to apply. + * @return This Element. + */ + public T setHeight(double height) { + return constrain(HEIGHT, Constraint.literal(height)); + } + + /** + * Convenience method for setting both width and height. + * + * @param width The width to apply. + * @param height The height to apply. + * @return This Element. + * @see #setWidth(double) + * @see #setHeight(double) + */ + public T setSize(double width, double height) { + return setWidth(width).setHeight(height); + } + + //=== Everything related to constraint based geometry ===// + + /** + * @return The position of the Left edge of this element. + */ + @Override + public double xMin() { + return xAxis.min.apply(xMin, xMax, xSize); + } + + /** + * @return The position of the Right edge of this element. + */ + @Override + public double xMax() { + return xAxis.max.apply(xMin, xMax, xSize); + } + + /** + * @return The Width of this element. + */ + @Override + public double xSize() { + return xAxis.size.apply(xMin, xMax, xSize); + } + + /** + * @return The position of the Top edge of this element. + */ + @Override + public double yMin() { + return yAxis.min.apply(yMin, yMax, ySize); + } + + /** + * @return The position of the Bottom edge of this element. + */ + @Override + public double yMax() { + return yAxis.max.apply(yMin, yMax, ySize); + } + + /** + * @return The Height of this element. + */ + @Override + public double ySize() { + return yAxis.size.apply(yMin, yMax, ySize); + } + + /** + * Returns a reference to the specified geometry parameter. + * This is primarily used when defining geometry constraints. + * But it can also be used as a simple {@link Supplier} + * that will return the current parameter value when requested. + *

+ * Note: The returned geometry reference will always be valid + * + * @param param The geometry parameter. + * @return A Geometry Reference + */ + @Override + public GeoRef get(GeoParam param) { + return new GeoRef(this, param); + } + + /** + * @param param The geometry parameter to be constrained. + * @param constraint The constraint to apply + * @return This Element. + */ + @SuppressWarnings ("unchecked") + public T constrain(GeoParam param, @Nullable Constraint constraint) { + if (constraint != null && constraint.axis() != null && constraint.axis() != param.axis) { + throw new IllegalStateException("Attempted to apply constraint for axis: " + constraint.axis() + ", to Parameter: " + param); + } + if (param.axis == Axis.X) { + constrainX(param, constraint); + } else if (param.axis == Axis.Y) { + constrainY(param, constraint); + } + return (T) this; + } + + /** + * Clear any configured constraints and reset this element to default unconstrained state. + * Convenient when reconfiguring an elements constraints or applying constraints to an element + * with an existing, unknown constraint configuration. + */ + @SuppressWarnings ("unchecked") + public T clearConstraints() { + xMin = xMax = xSize = yMin = yMax = ySize = null; + xAxis = yAxis = AxisConfig.NONE; + return (T) this; + } + + private void constrainX(GeoParam param, @Nullable Constraint constraint) { + if (param == GeoParam.LEFT) { + xMin = constraint; + } else if (param == GeoParam.RIGHT) { + xMax = constraint; + } else if (param == WIDTH) { + xSize = constraint; + } + xAxis = AxisConfig.getConfigFor(xMin, xMax, xSize); + validate(); + } + + private void constrainY(GeoParam param, @Nullable Constraint constraint) { + if (param == GeoParam.TOP) { + yMin = constraint; + } else if (param == GeoParam.BOTTOM) { + yMax = constraint; + } else if (param == GeoParam.HEIGHT) { + ySize = constraint; + } + yAxis = AxisConfig.getConfigFor(yMin, yMax, ySize); + validate(); + } + + /** + * Strict mode is intended to help catch potential mistakes when writing modular GUIs + *

+ * Enforces a strict requirement for each exist to have two and only two constraints. + * Any attempt to over-constrain an axis will throw an immediate fatal exception. + * If an axis is under-constrained then a fatal exception will be thrown when a value from the axis is queried. + *

+ * Strict mode applies to this element, and recursively to all children of this element. + * + * @param strictMode Enable strict mode. + * @return the geometry object. + */ + @SuppressWarnings ("unchecked") + public T strictMode(boolean strictMode) { + this.strictMode = strictMode; + //TODO Propagate to children (Will be handled in the base GuiElement) + return (T) this; + } + + //TODO This needs to be called from the parent element somewhere. Possibly on tick or render + //Ideally i would like to find a way to only call it once. we need to account for constraints being modified after initial element construction. + public void validate() { + if (strictMode) { + if (xAxis.constraints != 2) { + throw new IllegalStateException(String.format("X axis of element: %s is %s constrained!", getParent(), xAxis.constraints < 2 ? "under" : "over")); + } else if (yAxis.constraints != 2) { + throw new IllegalStateException(String.format("Y axis of element: %s is %s constrained!", getParent(), yAxis.constraints < 2 ? "under" : "over")); + } + } + } + + public void clearGeometryCache() { + if (xMin != null) xMin.markDirty(); + if (xMax != null) xMax.markDirty(); + if (xSize != null) xSize.markDirty(); + if (yMin != null) yMin.markDirty(); + if (yMax != null) yMax.markDirty(); + if (ySize != null) ySize.markDirty(); + getChildren().forEach(ConstrainedGeometry::clearGeometryCache); + } + + //=== Geometry Utilities ===// + + /** + * Returns a {@link Position} that is permanently bound to this element. + */ + public Position getPosition() { + return position; + } + + /** + * Returns a {@link Rectangle} that is permanently bound to this element. + */ + public Rectangle getRectangle() { + return rectangle; + } + + public double xCenter() { + return xMin() + (xSize() / 2); + } + + public double yCenter() { + return yMin() + (ySize() / 2); + } + + /** + * Returns a new {@link Rectangle} the bounds of which will enclose this element and all of its child + * elements recursively. + */ + public Rectangle.Mutable getEnclosingRect() { + return addBoundsToRect(getRectangle().mutable()); + } + + /** + * Expands the bounds of the given rectangle (if needed) so that they enclose this element. + * And all of its child elements recursively. + */ + public Rectangle.Mutable addBoundsToRect(Rectangle.Mutable enclosingRect) { + enclosingRect.combine(getRectangle()); + for (GuiElement element : getChildren()) { + if (element.isEnabled()) { + element.addBoundsToRect(enclosingRect); + } + } + return enclosingRect; + } + + /** + * @return a rectangle, the bounds of which enclose all enabled child elements. + * If there are no enabled child elements the returned rect will have the position of this element, with zero size. + */ + public Rectangle.Mutable getChildBounds() { + boolean set = false; + for (GuiElement element : getChildren()) { + if (element.isEnabled()) { + if (!set) { + childBounds.set(element.getRectangle()); + set = true; + } else { + element.addBoundsToRect(childBounds); + } + } + } + if (!set) childBounds.setPos(xMin(), yMin()).setSize(0, 0); + return childBounds; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/Constraint.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Constraint.java new file mode 100644 index 00000000..63d80344 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Constraint.java @@ -0,0 +1,154 @@ +package codechicken.lib.gui.modular.lib.geometry; + +import org.jetbrains.annotations.Nullable; + +import java.util.function.Supplier; + +/** + * Constraints are used to define an elements position and shape by constraining the elements geometry parameters. {@link GeoParam} + * Constraints can be as simple as literal values representing an elements exact position and / or size in screen space, + * They can be relative coordinates, (Relative to another element's Geometry) + * Or they can be used to supply completely custom dynamic values. + *

+ * All the built-in constraints are implemented and documented in this class. + *

+ * Created by brandon3055 on 30/06/2023 + * + * @see ConstrainedGeometry + */ +public sealed interface Constraint permits ConstraintImpl { + + /** + * This method will return the current value of this constraint. + * This could be a fixed stored value, or a dynamically computed value + * depending on the type of constraint. + *

+ * All position and size values in ModularGui are doubles. + * However, by default, all the builtin constraints will cast their outputs to integer values. + * This avoids a lot of random visual artifacts that can occur when using floating point values in MC Screens. + *

+ * If you need floating-point precision, it can be enabled calling .precise() on any of the builtin constraints. + * + * @return The computed or stored value of this constraint. + */ + double get(); + + /** + * @return the axis this constraint applies to, Ether X, Y or null for undefined. + */ + @Nullable + Axis axis(); + + /** + * This is part of a late addition to improve performance. + * Rather than computing a constraint value every single time it is queried, which can be many, many times per render frame, + * We now cache the constraint value, and that cache is cleared at the start of each render frame. + */ + void markDirty(); + + /** + * This is the most basic constraint. It constrains a parameter to a single fixed value. + * + * @param value The fixed value that will be returned by this constraint. + */ + static ConstraintImpl.Literal literal(double value) { + return new ConstraintImpl.Literal(value); + } + + /** + * Constrains a parameter to the value provided by the given supplier. + * + * @param valueSupplier The dynamic value supplier. + */ + static ConstraintImpl.Dynamic dynamic(Supplier valueSupplier) { + return new ConstraintImpl.Dynamic(valueSupplier); + } + + /** + * Contains a parameter to the exact value of the given reference. + * This is effectively a relative constraint with no offset. + * + * @param relativeTo The relative geometry. + */ + static ConstraintImpl.Relative match(GeoRef relativeTo) { + return new ConstraintImpl.Relative(relativeTo, 0); + } + + /** + * Contains a parameter to the given reference plus the provided fixed offset. + * + * @param relativeTo The relative geometry. + * @param offset The offset to apply. + */ + static ConstraintImpl.Relative relative(GeoRef relativeTo, double offset) { + return new ConstraintImpl.Relative(relativeTo, offset); + } + + /** + * Contains a parameter to the given reference plus the provided dynamic offset. + * + * @param relativeTo The relative geometry. + * @param offset The dynamic offset to apply. + */ + static ConstraintImpl.RelativeDynamic relative(GeoRef relativeTo, Supplier offset) { + return new ConstraintImpl.RelativeDynamic(relativeTo, offset); + } + + /** + * Contains a parameter to a fixed position between the two provided references. + * Note: it is possible to go outside the given range if the given position is greater than 1 or less than 0. + * To prevent this call .clamp() on the returned constraint. + * + * @param start The Start position. + * @param end The End position. + * @param position The position between start and end. (0=start to 1=end) + */ + static ConstraintImpl.Between between(GeoRef start, GeoRef end, double position) { + return new ConstraintImpl.Between(start, end, position); + } + + /** + * Contains a parameter to a dynamic position between the two provided references. + * Note: it is possible to go outside the given range if the given position is greater than 1 or less than 0. + * To prevent this call .clamp() on the returned constraint. + * + * @param start The Start position. + * @param end The End position. + * @param position The dynamic position between start and end. (0=start to 1=end) + */ + static ConstraintImpl.BetweenDynamic between(GeoRef start, GeoRef end, Supplier position) { + return new ConstraintImpl.BetweenDynamic(start, end, position); + } + + /** + * Contains a parameter to the mid-point between the two provided references. + * + * @param start The Start position. + * @param end The End position. + */ + static ConstraintImpl.MidPoint midPoint(GeoRef start, GeoRef end) { + return new ConstraintImpl.MidPoint(start, end, 0); + } + + /** + * Contains a parameter to the mid-point between the two provided references with a fixed offset. + * + * @param start The Start position. + * @param end The End position. + * @param offset offset distance. + */ + static ConstraintImpl.MidPoint midPoint(GeoRef start, GeoRef end, double offset) { + return new ConstraintImpl.MidPoint(start, end, offset); + } + + /** + * Contains a parameter to the mid-point between the two provided references with a dynamic offset. + * + * @param start The Start position. + * @param end The End position. + * @param offset offset distance suppler. + */ + static ConstraintImpl.MidPointDynamic midPoint(GeoRef start, GeoRef end, Supplier offset) { + return new ConstraintImpl.MidPointDynamic(start, end, offset); + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/ConstraintImpl.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/ConstraintImpl.java new file mode 100644 index 00000000..646abe93 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/ConstraintImpl.java @@ -0,0 +1,294 @@ +package codechicken.lib.gui.modular.lib.geometry; + +import net.minecraft.util.Mth; +import org.jetbrains.annotations.Nullable; + +import java.util.function.Supplier; + +/** + * Created by brandon3055 on 04/07/2023 + */ +public abstract non-sealed class ConstraintImpl> implements Constraint { + + protected boolean precise = false; + protected Axis axis = null; + private double value; + private boolean isDirty = true; + + /** + * @return True if precise mode is enabled. + * @see #precise() + */ + public boolean isPrecise() { + return precise; + } + + /** + * By default, all constraint values are cast to (int). + * This helps avoid a lot of visual artifacts that can occur when using floating point positions in MC Screens. + *

+ * Calling this enables precise mode, which allows floating point precision to be used. + * + * @return The Constraint. + */ + @SuppressWarnings("unchecked") + public T precise() { + this.precise = true; + return (T) this; + } + + @Override + public double get() { + if (isDirty) { + value = isPrecise() ? getImpl() : (int) getImpl(); + isDirty = false; + } + return value; + } + + @Override + public @Nullable Axis axis() { + return axis; + } + + @SuppressWarnings("unchecked") + public T setAxis(@Nullable Axis axis) { + this.axis = axis; + return (T) this; + } + + @Override + public void markDirty() { + isDirty = true; + } + + protected abstract double getImpl(); + + public static class Literal extends ConstraintImpl { + protected final double value; + + /** + * Returns the literal value supplied. + */ + public Literal(double value) { + this.value = value; + } + + @Override + protected double getImpl() { + return value; + } + } + + public static class Dynamic extends ConstraintImpl { + protected final Supplier value; + + /** + * Returns a dynamic value that will be retrieved from the provided supplier. + */ + public Dynamic(Supplier value) { + this.value = value; + } + + @Override + protected double getImpl() { + return value.get(); + } + } + + public static class Relative extends ConstraintImpl { + protected final GeoRef relTo; + protected final double offset; + + /** + * Returns the value of the provided reference plus the given offset. + */ + public Relative(GeoRef relTo, double offset) { + this.setAxis(relTo.parameter.axis); + this.relTo = relTo; + this.offset = offset; + } + + @Override + protected double getImpl() { + return relTo.get() + getOffset(); + } + + public double getOffset() { + return offset; + } + } + + public static class RelativeDynamic extends ConstraintImpl { + protected final GeoRef relTo; + protected final Supplier offset; + + /** + * Returns the value of the provided reference plus the given dynamic offset. + */ + public RelativeDynamic(GeoRef relTo, Supplier offset) { + this.setAxis(relTo.parameter.axis); + this.relTo = relTo; + this.offset = offset; + } + + @Override + protected double getImpl() { + return relTo.get() + getOffset(); + } + + public double getOffset() { + return offset.get(); + } + } + + public static class Between extends ConstraintImpl { + protected final GeoRef start; + protected final GeoRef end; + protected final double pos; + protected boolean clamp = false; + + public Between(GeoRef start, GeoRef end, double pos) { + this.setAxis(start.parameter.axis); + if (start.parameter.axis != end.parameter.axis) { + throw new IllegalStateException("Attempted to define a 'Between' Constraint with parameters on different axes."); + } + this.start = start; + this.end = end; + this.pos = pos; + } + + @Override + protected double getImpl() { + return start.get() + (end.get() - start.get()) * getPos(); + } + + public double getPos() { + return clamp ? Mth.clamp(pos, 0, 1) : pos; + } + + public double getStart() { + return start.get(); + } + + public double getEnd() { + return end.get(); + } + + /** + * Ensure the output can not go bellow the min reference or above the max reference. + */ + public Between clamp() { + this.clamp = true; + return this; + } + } + + public static class BetweenDynamic extends ConstraintImpl { + protected final GeoRef start; + protected final GeoRef end; + protected final Supplier pos; + protected boolean clamp = false; + + public BetweenDynamic(GeoRef start, GeoRef end, Supplier pos) { + this.setAxis(start.parameter.axis); + if (start.parameter.axis != end.parameter.axis) { + throw new IllegalStateException("Attempted to define a 'Between' Constraint with parameters on different axes."); + } + this.start = start; + this.end = end; + this.pos = pos; + } + + @Override + protected double getImpl() { + return start.get() + (end.get() - start.get()) * getPos(); + } + + public double getPos() { + return clamp ? Mth.clamp(pos.get(), 0, 1) : pos.get(); + } + + public double getStart() { + return start.get(); + } + + public double getEnd() { + return end.get(); + } + + /** + * Ensure the output can not go bellow the min reference or above the max reference. + */ + public BetweenDynamic clamp() { + this.clamp = true; + return this; + } + } + + public static class MidPoint extends ConstraintImpl { + protected final GeoRef start; + protected final GeoRef end; + protected final double offset; + + public MidPoint(GeoRef start, GeoRef end, double offset) { + this.setAxis(start.parameter.axis); + if (start.parameter.axis != end.parameter.axis) { + throw new IllegalStateException("Attempted to define a 'MidPoint' Constraint with parameters on different axes."); + } + this.start = start; + this.end = end; + this.offset = offset; + } + + @Override + protected double getImpl() { + return start.get() + ((end.get() - start.get()) / 2) + getOffset(); + } + + public double getOffset() { + return offset; + } + + public double getStart() { + return start.get(); + } + + public double getEnd() { + return end.get(); + } + } + + public static class MidPointDynamic extends ConstraintImpl { + protected final GeoRef start; + protected final GeoRef end; + protected final Supplier offset; + + public MidPointDynamic(GeoRef start, GeoRef end, Supplier offset) { + this.setAxis(start.parameter.axis); + if (start.parameter.axis != end.parameter.axis) { + throw new IllegalStateException("Attempted to define a 'MidPoint' Constraint with parameters on different axes."); + } + this.start = start; + this.end = end; + this.offset = offset; + } + + @Override + protected double getImpl() { + return start.get() + ((end.get() - start.get()) / 2) + getOffset(); + } + + public double getOffset() { + return offset.get(); + } + + public double getStart() { + return start.get(); + } + + public double getEnd() { + return end.get(); + } + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/Direction.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Direction.java new file mode 100644 index 00000000..9e2513e8 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Direction.java @@ -0,0 +1,40 @@ +package codechicken.lib.gui.modular.lib.geometry; + +/** + * Created by brandon3055 on 04/09/2023 + */ +public enum Direction { + UP(Axis.Y), + LEFT(Axis.X), + DOWN(Axis.Y), + RIGHT(Axis.X); + + private static Direction[] VALUES = values(); + + private final Axis axis; + + Direction(Axis axis) { + this.axis = axis; + } + + public Axis getAxis() { + return axis; + } + + public Direction opposite() { + if (axis == Axis.X) return this == LEFT ? RIGHT : LEFT; + else return this == UP ? DOWN : UP; + } + + public Direction rotateCW() { + return values()[(ordinal() + VALUES.length - 1) % VALUES.length]; + } + + public Direction rotateCCW() { + return values()[(ordinal() + 1) % VALUES.length]; + } + + public double rotationTo(Direction other) { + return (this.ordinal() - other.ordinal()) * 90; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/GeoParam.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/GeoParam.java new file mode 100644 index 00000000..0f37720f --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/GeoParam.java @@ -0,0 +1,32 @@ +package codechicken.lib.gui.modular.lib.geometry; + +/** + * Used to define the 6 core parameters that make up an elements geometry + * These are named Left, Right, Width, Top, Bottom, Height. + * These names were chosen for ease of use, and to make it clear what they represent. + * Internally they are known as xMin, xMax, xSize, yMin, yMax, ySize. + *

+ * Created by brandon3055 on 30/06/2023 + */ +public enum GeoParam { + /** X_MIN */ + LEFT(Axis.X), + /** X_MAX */ + RIGHT(Axis.X), + /** X_SIZE */ + WIDTH(Axis.X), + + /** Y_MIN */ + TOP(Axis.Y), + /** Y_MAX */ + BOTTOM(Axis.Y), + /** Y_SIZE */ + HEIGHT(Axis.Y); + + public final Axis axis; + + GeoParam(Axis axis) { + this.axis = axis; + } + +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/GeoRef.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/GeoRef.java new file mode 100644 index 00000000..93ebf68e --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/GeoRef.java @@ -0,0 +1,36 @@ +package codechicken.lib.gui.modular.lib.geometry; + +import java.util.function.Supplier; + +/** + * Used to access one of the 6 core parameters that make up an element's geometry. + *

+ * The primary purpose of this class is to provide a convenient way to reference a geometry + * parameter when defining constraints. + * It also helps make the code more debuggable. + * I could just make literally everything a lambda, but that makes debugging kinda painful when things break. + *

+ * Created by brandon3055 on 30/06/2023 + */ +public class GeoRef implements Supplier { + public final GuiParent geometry; + public final GeoParam parameter; + + public GeoRef(GuiParent geometry, GeoParam parameter) { + this.geometry = geometry; + this.parameter = parameter; + } + + @Override + public Double get() { + return geometry.getValue(parameter); + } + + @Override + public String toString() { + return "GeoReference{" + + "geometry=" + geometry + + ", parameter=" + parameter + + '}'; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/GuiParent.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/GuiParent.java new file mode 100644 index 00000000..9a2b8d77 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/GuiParent.java @@ -0,0 +1,190 @@ +package codechicken.lib.gui.modular.lib.geometry; + +import codechicken.lib.gui.modular.ModularGui; +import codechicken.lib.gui.modular.elements.GuiElement; +import net.minecraft.client.Minecraft; +import net.minecraft.client.gui.Font; +import org.jetbrains.annotations.ApiStatus; + +import java.util.List; +import java.util.function.Consumer; +import java.util.function.Supplier; + +/** + * This is the base interface that allows an element or screen to define its basic geometry. + * As well as defining the primary methods for handling child elements. + *

+ * It also provides a way to access some common minecraft fields. + *

+ * Created by brandon3055 on 29/06/2023 + */ +public interface GuiParent> { + + /** + * @return The position of the Left edge of this element. + */ + double xMin(); + + /** + * @return The position of the Right edge of this element. + */ + double xMax(); + + /** + * @return The Width of this element. + */ + double xSize(); + + /** + * @return The position of the Top edge of this element. + */ + double yMin(); + + /** + * @return The position of the Bottom edge of this element. + */ + double yMax(); + + /** + * @return The Height of this element. + */ + double ySize(); + + /** + * Returns a reference to the specified geometry parameter. + * This is primarily used when defining geometry constraints. + * But it can also be used as a simple {@link Supplier} + * that will return the current parameter value when requested. + *

+ * Note: The returned geometry reference will always be valid + * + * @param param The geometry parameter. + * @return A Geometry Reference + */ + default GeoRef get(GeoParam param) { + return new GeoRef(this, param); + } + + /** + * @param param The geometry parameter. + * @return The current value of the specified parameter. + */ + default double getValue(GeoParam param) { + return switch (param) { + case LEFT -> xMin(); + case RIGHT -> xMax(); + case WIDTH -> xSize(); + case TOP -> yMin(); + case BOTTOM -> yMax(); + case HEIGHT -> ySize(); + }; + } + + /** + * @return An unmodifiable list of all assigned child elements assigned to this parent. The list should be sorted in the order they were added. + */ + List> getChildren(); + + /** + * Adds a new child element to this parent. + * You should almost never need to use this because this is handled automatically when an element is created. + *

+ * Note: Due to the way relative coordinates work with the new geometry system, + * Transferring an element to a different parent can have unpredictable results. + * Therefor, to help avoid confusion it is not possible to transfer a child to a new parent using this method. + * + * @param child The child element to be added. + * @throws UnsupportedOperationException - If child has previously been assigned to a different parent. + * @see #adoptChild(GuiElement) + */ + void addChild(GuiElement child); + + /** + * This meant to be a convenience method that allows builder style addition of a child element. + * I'm not sure how useful it will be yet, so it may or may not stay. + * + * @param createChild A consumer that is given this element to be used in the construction of the child element. + * @return The parent element + */ + @ApiStatus.Experimental + @SuppressWarnings ("unchecked") + default T addChild(Consumer createChild) { + createChild.accept((T) this); + return (T) this; + } + + /** + * This method can be used to transfer an already initialized child to this parent element. + * This automatically handles removing the element from its previous parent, adds it to this element. + * Note: This will most likely break any relative constraints on the child's geometry. + * To fix this you will need to re-apply geometry constraints after the transfer. + * + * @param child The child element to be adopted. + */ + void adoptChild(GuiElement child); + + /** + * Allows the removal of a child element. + * Child removal is not instantaneous, Instead all removals occur at the end of the current screen thick. + * This is to avoid any possible concurrency issues. + * + * @param child The child element to be removed. + */ + void removeChild(GuiElement child); + + /** + * Checks if this element is a descendant of the specified. + * @return true if the specified element is a parent or grandparent etc... of this element. + */ + default boolean isDescendantOf(GuiElement ancestor) { + return false; + } + + /** + * @return The minecraft instance. + */ + Minecraft mc(); + + /** + * @return The active font instance. + */ + Font font(); + + /** + * @return The current gui screen width, As returned by mc.getWindow().getGuiScaledWidth() + */ + int scaledScreenWidth(); + + /** + * @return The current gui screen height, As returned by mc.getWindow().getGuiScaledHeight() + */ + int scaledScreenHeight(); + + /** + * @return the parent ModularGui instance. + */ + ModularGui getModularGui(); + + /** + * Called when the minecraft Screen is initialised or resized. + * + * @param mc The Minecraft instance. + * @param font The active font. + * @param screenWidth The current guiScaledWidth. + * @param screenHeight The current guiScaledHeight. + */ + default void onScreenInit(Minecraft mc, Font font, int screenWidth, int screenHeight) { + getChildren().forEach(e -> e.onScreenInit(mc, font, screenWidth, screenHeight)); + } + + /** + * Allows an element to override the {@link GuiElement#isMouseOver()} method of its children. + * This is primarily used for things like scroll elements where mouseover interactions need to be blocked outside the view area. + * + * @param element The element on which isMouseOver is getting called. + * @return true if mouse-over interaction should be blocked for this child element. + */ + default boolean blockMouseOver(GuiElement element, double mouseX, double mouseY) { + return false; + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/Position.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Position.java new file mode 100644 index 00000000..df430507 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Position.java @@ -0,0 +1,103 @@ +package codechicken.lib.gui.modular.lib.geometry; + +import java.util.function.Supplier; + +/** + * Created by brandon3055 on 24/08/2023 + */ +public interface Position { + + double x(); + + double y(); + + default Position offset(double x, double y) { + return create(x() + x, y() + y); + } + + /** + * Returns the position value for the given axis. + */ + default double get(Axis axis) { + return axis == Axis.X ? x() : y(); + } + + static Position create(double x, double y) { + return new Immutable(x, y); + } + + static Position create(Supplier getX, Supplier getY) { + return new Dynamic(getX, getY); + } + + /** + * Creates a new position, bound to the specified parent's position. + * */ + static Position create(GuiParent parent) { + return new Dynamic(parent::xMin, parent::yMin); + } + + record Immutable(@Override double x, @Override double y) implements Position { } + + record Dynamic(Supplier getX, Supplier getY) implements Position { + @Override + public double x() { + return getX.get(); + } + + @Override + public double y() { + return getY.get(); + } + + @Override + public String toString() { + return "Dynamic{" + + "x=" + x() + + ", y=" + y() + + '}'; + } + } + + static class Mutable implements Position { + private double x; + private double y; + + public Mutable(double x, double y) { + this.x = x; + this.y = y; + } + + @Override + public double x() { + return x; + } + + @Override + public double y() { + return y; + } + + @Override + public Position offset(double x, double y) { + this.x += x; + this.y += y; + return this; + } + + public Position set(double x, double y) { + this.x = x; + this.y = y; + return this; + } + + @Override + public String toString() { + return "Mutable{" + + "x=" + x() + + ", y=" + y() + + '}'; + } + } + +} diff --git a/src/main/java/codechicken/lib/gui/modular/lib/geometry/Rectangle.java b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Rectangle.java new file mode 100644 index 00000000..4ad613f7 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/lib/geometry/Rectangle.java @@ -0,0 +1,308 @@ +package codechicken.lib.gui.modular.lib.geometry; + +import net.minecraft.client.renderer.Rect2i; + +import java.util.function.Supplier; + +/** + * Created by brandon3055 on 14/08/2023 + */ +public interface Rectangle { + + Position pos(); + + default double x() { + return pos().x(); + } + + default double y() { + return pos().y(); + } + + double width(); + + double height(); + + default double xMax() { + return x() + width(); + } + + default double yMax() { + return y() + height(); + } + + /** + * Returns a new rectangle with this operation applied + */ + default Rectangle offsetPos(double xAmount, double yAmount) { + return create(x() + xAmount, y() + yAmount, width(), height()); + } + + /** + * Returns a new rectangle with this operation applied + */ + default Rectangle setPos(double newX, double newY) { + return create(newX, newY, width(), height()); + } + + /** + * Returns a new rectangle with this operation applied + */ + default Rectangle setSize(double width, double height) { + return create(x(), y(), width, height); + } + + /** + * Returns a new rectangle with this operation applied + */ + default Rectangle offsetSize(double xAmount, double yAmount) { + return create(x(), y(), width() + xAmount, height() + yAmount); + } + + default Rect2i toRect2i() { + return new Rect2i((int) x(), (int) y(), (int) width(), (int) height()); + } + + default boolean intersects(double x, double y, double w, double h) { + double x0 = x(); + double y0 = y(); + return (x + w > x0 && y + h > y0 && x < x0 + width() && y < y0 + height()); + } + + default boolean intersects(Rectangle other) { + double x0 = x(); + double y0 = y(); + return (other.x() + other.width() > x0 && other.y() + other.height() > y0 && other.x() < x0 + width() && other.y() < y0 + height()); + } + + /** + * Returns a new rectangle that represents the intersection area between the two inputs + */ + default Rectangle intersect(Rectangle other) { + double x = Math.max(x(), other.x()); + double y = Math.max(y(), other.y()); + double width = Math.max(0, Math.min(xMax(), other.xMax()) - x()); + double height = Math.max(0, Math.min(yMax(), other.yMax()) - y()); + return create(x, y, width, height); + } + + /** + * Returns a new rectangle, the bounds of which enclose all the input rectangles. + * + * @param combineWith Rectangles to combine with the start rectangle + */ + default Rectangle combine(Rectangle... combineWith) { + double x = x(); + double y = y(); + double maxX = xMax(); + double maxY = yMax(); + for (Rectangle other : combineWith) { + x = Math.min(x, other.x()); + y = Math.min(y, other.y()); + maxX = Math.max(maxX, other.xMax()); + maxY = Math.max(maxY, other.yMax()); + } + return create(x, y, maxX - x, maxY - y); + } + + default boolean contains(double x, double y) { + return x >= x() && x <= x() + width() && y >= y() && y <= y() + height(); + } + + /** + * @return the size of this rectangle on the given axis. + */ + default double size(Axis axis) { + return axis == Axis.X ? width() : height(); + } + + /** + * @return the min value the specified axis (meaning x() or y()) + */ + default double min(Axis axis) { + return axis == Axis.X ? x() : y(); + } + + /** + * @return the max value the specified axis (meaning x() + width() or y() + height()) + */ + default double max(Axis axis) { + return axis == Axis.X ? xMax() : yMax(); + } + + /** + * Return the distance the given position is from this rectangle on the specified axis. + * Will return 0 if position is inside this rectangle on the given axis. + */ + default double distance(Axis axis, Position position) { + double pos = position.get(axis); + double min = min(axis); + double max = max(axis); + return pos < min ? min - pos : pos > max ? pos - max : 0; + } + + static Rectangle create(Position position, double width, double height) { + return new Immutable(position, width, height); + } + + static Rectangle create(double x, double y, double width, double height) { + return new Immutable(Position.create(x, y), width, height); + } + + /** + * Returns a new rectangle bound to the specified parent's geometry. + */ + static Rectangle create(GuiParent parent) { + return new Dynamic(Position.create(parent), parent::xSize, parent::ySize); + } + + static Rectangle create(Position position, Supplier getWidth, Supplier getHeight) { + return new Dynamic(position, getWidth, getHeight); + } + + default Mutable mutable() { + return new Mutable(new Position.Mutable(x(), y()), width(), height()); + } + + /** + * Should not be created directly + */ + record Immutable(Position position, double xSize, double ySize) implements Rectangle { + @Override + public Position pos() { + return position; + } + + @Override + public double width() { + return xSize; + } + + @Override + public double height() { + return ySize; + } + + @Override + public String toString() { + return "Immutable{" + + "pos=" + pos() + + ", width=" + width() + + ", height=" + height() + + '}'; + } + } + + record Dynamic(Position position, Supplier getWidth, Supplier getHeight) implements Rectangle { + @Override + public Position pos() { + return position; + } + + @Override + public double width() { + return getWidth.get(); + } + + @Override + public double height() { + return getHeight.get(); + } + + @Override + public String toString() { + return "Dynamic{" + + "pos=" + pos() + + ", width=" + width() + + ", height=" + height() + + '}'; + } + } + + class Mutable implements Rectangle { + private Position.Mutable pos; + private double width; + private double height; + + public Mutable(Position.Mutable pos, double width, double height) { + this.pos = pos; + this.width = width; + this.height = height; + } + + @Override + public Position pos() { + return pos; + } + + @Override + public double width() { + return width; + } + + @Override + public double height() { + return height; + } + + @Override + public Rectangle offsetPos(double xAmount, double yAmount) { + pos.offset(xAmount, yAmount); + return this; + } + + @Override + public Rectangle setPos(double newX, double newY) { + pos.set(newX, newY); + return this; + } + + @Override + public Rectangle setSize(double width, double height) { + this.width = width; + this.height = height; + return this; + } + + @Override + public Rectangle offsetSize(double xAmount, double yAmount) { + this.width += xAmount; + this.height += yAmount; + return this; + } + + @Override + public Rectangle intersect(Rectangle other) { + double x = Math.max(x(), other.x()); + double y = Math.max(y(), other.y()); + width = Math.max(0, Math.min(xMax(), other.xMax()) - x()); + height = Math.max(0, Math.min(yMax(), other.yMax()) - y()); + return setPos(x, y); + } + + public void set(Rectangle rectangle) { + this.pos.set(rectangle.x(), rectangle.y()); + this.width = rectangle.width(); + this.height = rectangle.height(); + } + + @Override + public Rectangle combine(Rectangle... combineWith) { + double x = x(); + double y = y(); + double maxX = xMax(); + double maxY = yMax(); + for (Rectangle other : combineWith) { + x = Math.min(x, other.x()); + y = Math.min(y, other.y()); + maxX = Math.max(maxX, other.xMax()); + maxY = Math.max(maxY, other.yMax()); + } + return setPos(x, y).setSize(maxX - x, maxY - y); + } + + public Immutable immutable() { + return new Immutable(Position.create(x(), y()), width(), height()); + } + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/sprite/CCGuiTextures.java b/src/main/java/codechicken/lib/gui/modular/sprite/CCGuiTextures.java new file mode 100644 index 00000000..a3aaf28d --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/sprite/CCGuiTextures.java @@ -0,0 +1,59 @@ +package codechicken.lib.gui.modular.sprite; + +import codechicken.lib.CodeChickenLib; +import net.minecraft.resources.ResourceLocation; + +import java.util.HashMap; +import java.util.Map; +import java.util.function.Supplier; + +/** + * Gui texture handler implementation. + * This sets up a custom atlas that will be populated with all textures in "modid:textures/gui/" + * To use your own textures you can just create a carbon copy of this class with that uses your own modid. + *

+ * Created by brandon3055 on 21/10/2023 + */ +public class CCGuiTextures { + private static final ModAtlasHolder ATLAS_HOLDER = new ModAtlasHolder(CodeChickenLib.MOD_ID, "textures/atlas/gui.png", "gui"); + private static final Map MATERIAL_CACHE = new HashMap<>(); + + /** + * The returned AtlasLoader needs to be registered as a resource reload listener using the appropriate NeoForge / Fabric event. + */ + public static ModAtlasHolder getAtlasHolder() { + return ATLAS_HOLDER; + } + + /** + * Returns a cached Material for the specified gui texture. + * Warning: Do not use this if you intend to use the material with multiple render types. + * The material will cache the first render type it is used with. + * Instead use {@link #getUncached(String)} + * + * @param texture The texture path relative to "modid:gui/" + */ + public static Material get(String texture) { + return MATERIAL_CACHE.computeIfAbsent(CodeChickenLib.MOD_ID + ":" + texture, e -> getUncached(texture)); + } + + public static Material get(Supplier texture) { + return get(texture.get()); + } + + public static Supplier getter(Supplier texture) { + return () -> get(texture.get()); + } + + /** + * Use this to retrieve a new uncached material for the specified gui texture. + * Feel free to hold onto the returned material. + * Storing it somewhere is more efficient than recreating it every render frame. + * + * @param texture The texture path relative to "modid:gui/" + * @return A new Material for the specified gui texture. + */ + public static Material getUncached(String texture) { + return new Material(ATLAS_HOLDER.atlasLocation(), new ResourceLocation(CodeChickenLib.MOD_ID, "gui/" + texture), ATLAS_HOLDER::getSprite); + } +} \ No newline at end of file diff --git a/src/main/java/codechicken/lib/gui/modular/sprite/Material.java b/src/main/java/codechicken/lib/gui/modular/sprite/Material.java new file mode 100644 index 00000000..888b45b3 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/sprite/Material.java @@ -0,0 +1,126 @@ +package codechicken.lib.gui.modular.sprite; + +import com.mojang.blaze3d.platform.NativeImage; +import com.mojang.blaze3d.vertex.VertexConsumer; +import net.minecraft.client.Minecraft; +import net.minecraft.client.renderer.MultiBufferSource; +import net.minecraft.client.renderer.RenderType; +import net.minecraft.client.renderer.texture.SpriteContents; +import net.minecraft.client.renderer.texture.TextureAtlas; +import net.minecraft.client.renderer.texture.TextureAtlasSprite; +import net.minecraft.client.resources.metadata.animation.AnimationMetadataSection; +import net.minecraft.client.resources.metadata.animation.FrameSize; +import net.minecraft.resources.ResourceLocation; +import org.jetbrains.annotations.Nullable; + +import java.util.function.Function; + +/** + * This is similar to Minecraft's {@link net.minecraft.client.resources.model.Material} + * This contains the essential data required to render an atlas sprite. + *

+ * The primary purpose of this class is to make porting between MC versions easier. + * It also allows for loading sprites from a custom texture atlas. Minecraft's material class can only load from vanilla atlases. + *

+ * Created by brandon3055 on 20/08/2023 + */ +public class Material { + private final ResourceLocation atlasLocation; + private final ResourceLocation texture; + private final Function spriteFunction; + + @Nullable + private RenderType renderType; + @Nullable + private net.minecraft.client.resources.model.Material vanillaMat; + + public Material(ResourceLocation atlasLocation, ResourceLocation texture, Function spriteFunction) { + this.atlasLocation = atlasLocation; + this.texture = texture; + this.spriteFunction = spriteFunction; + } + + public ResourceLocation atlasLocation() { + return atlasLocation; + } + + public ResourceLocation texture() { + return texture; + } + + public TextureAtlasSprite sprite() { + return spriteFunction.apply(texture()); + } + + /** + * Returns the cached render type for this material. + * The supplied function will be used to create the render type the first time this method is called. + * + * @param typeBuilder a function that will be used to create the render type if it does not already exist. + * @return The render type for this material. + */ + public RenderType renderType(Function typeBuilder) { + if (this.renderType == null) { + this.renderType = typeBuilder.apply(atlasLocation()); + } + return this.renderType; + } + + /** + * Convenience method to create a vertex consumer using this materials render type. + * + * @param buffers bugger source. + * @param typeBuilder a function that will be used to create the render type if it does not already exist. + */ + public VertexConsumer buffer(MultiBufferSource buffers, Function typeBuilder) { + return buffers.getBuffer(renderType(typeBuilder)); + } + + public net.minecraft.client.resources.model.Material getVanillaMat() { + if (vanillaMat == null) { + vanillaMat = new net.minecraft.client.resources.model.Material(atlasLocation, texture); + } + return vanillaMat; + } + + /** + * Convenient method for getting a material from a vanilla texture atlas. + * + * @return an un-cached material from a vanilla atlas. + */ + public static Material fromAtlas(ResourceLocation atlasLocation, String texture) { + return new Material(atlasLocation, new ResourceLocation(atlasLocation.getNamespace(), texture), e -> Minecraft.getInstance().getTextureAtlas(atlasLocation).apply(e)); + } + + /** + * Create a material from an existing sprite. + * Note: This will only work with sprites from a vanilla atlas. + */ + @Nullable + public static Material fromSprite(@Nullable TextureAtlasSprite sprite) { + if (sprite == null) return null; + return new Material(sprite.atlasLocation(), sprite.contents().name(), e -> Minecraft.getInstance().getTextureAtlas(sprite.atlasLocation()).apply(e)); + } + + public static Material fromRawTexture(ResourceLocation texture) { + return new Material(texture, texture, FullSprite::new); + } + + private static class FullSprite extends TextureAtlasSprite { + private FullSprite(ResourceLocation location) { + super(location, new SpriteContents(location, new FrameSize(1, 1), new NativeImage(1, 1, false), AnimationMetadataSection.EMPTY), 1, 1, 0, 0); + } + + @Override + public float getU(double u) + { + return (float) u / 16; + } + + @Override + public float getV(double v) + { + return (float) v / 16; + } + } +} diff --git a/src/main/java/codechicken/lib/gui/modular/sprite/ModAtlasHolder.java b/src/main/java/codechicken/lib/gui/modular/sprite/ModAtlasHolder.java new file mode 100644 index 00000000..9d361894 --- /dev/null +++ b/src/main/java/codechicken/lib/gui/modular/sprite/ModAtlasHolder.java @@ -0,0 +1,101 @@ +package codechicken.lib.gui.modular.sprite; + +import net.minecraft.client.Minecraft; +import net.minecraft.client.renderer.texture.SpriteLoader; +import net.minecraft.client.renderer.texture.TextureAtlas; +import net.minecraft.client.renderer.texture.TextureAtlasSprite; +import net.minecraft.resources.ResourceLocation; +import net.minecraft.server.packs.PackResources; +import net.minecraft.server.packs.resources.PreparableReloadListener; +import net.minecraft.server.packs.resources.Resource; +import net.minecraft.server.packs.resources.ResourceManager; +import net.minecraft.util.profiling.ProfilerFiller; +import org.jetbrains.annotations.NotNull; + +import java.util.*; +import java.util.concurrent.CompletableFuture; +import java.util.concurrent.Executor; +import java.util.function.Predicate; +import java.util.stream.Stream; + +/** + * Created by brandon3055 on 20/08/2023 + */ +public class ModAtlasHolder implements PreparableReloadListener, AutoCloseable { + private final TextureAtlas textureAtlas; + private final ResourceLocation atlasLocation; + private final ResourceLocation atlasInfoLocation; + private final String modid; + + /** + * Defines a mod texture atlas. + * Must be registered as a resource reload listener via RegisterClientReloadListenersEvent + * This is all that is needed to create a custom texture atlas. + * + * @param modid The mod id of the mod registering this atlas. + * @param atlasLocation The texture atlas location. e.g. "textures/atlas/gui.png" (Will have the modid: prefix added automatically) + * @param atlasInfoLocation The path to the atlas json file relative to modid:atlases/ + * e.g. "gui" will point to modid:atlases/gui.json + */ + public ModAtlasHolder(String modid, String atlasLocation, String atlasInfoLocation) { + this.atlasInfoLocation = new ResourceLocation(modid, atlasInfoLocation); + this.atlasLocation = new ResourceLocation(modid, atlasLocation); + this.textureAtlas = new TextureAtlas(this.atlasLocation); + this.modid = modid; + Minecraft.getInstance().getTextureManager().register(this.textureAtlas.location(), this.textureAtlas); + } + + public ResourceLocation atlasLocation() { + return atlasLocation; + } + + public TextureAtlasSprite getSprite(ResourceLocation resourceLocation) { + return this.textureAtlas.getSprite(resourceLocation); + } + + @Override + public final @NotNull CompletableFuture reload(PreparationBarrier prepBarrier, ResourceManager resourceManager, ProfilerFiller profiler, ProfilerFiller profiler2, Executor executor, Executor executor2) { + Objects.requireNonNull(prepBarrier); + SpriteLoader spriteLoader = SpriteLoader.create(this.textureAtlas); + return spriteLoader.loadAndStitch(new ModResourceManager(resourceManager, modid), this.atlasInfoLocation, 0, executor) + .thenCompose(SpriteLoader.Preparations::waitForUpload) + .thenCompose(prepBarrier::wait) + .thenAcceptAsync((preparations) -> this.apply(preparations, profiler2), executor2); + } + + private void apply(SpriteLoader.Preparations preparations, ProfilerFiller profilerFiller) { + profilerFiller.startTick(); + profilerFiller.push("upload"); + this.textureAtlas.upload(preparations); + profilerFiller.pop(); + profilerFiller.endTick(); + } + + @Override + public void close() { + this.textureAtlas.clearTextureData(); + } + + public static class ModResourceManager implements ResourceManager { + private final ResourceManager wrapped; + private final String modid; + + public ModResourceManager(ResourceManager wrapped, String modid) { + this.wrapped = wrapped; + this.modid = modid; + } + + @Override + public Map listResources(String pPath, Predicate pFilter) { + return wrapped.listResources(pPath, pFilter.and(e -> e.getNamespace().equals(modid))); + } + + //@formatter:off + @Override public Set getNamespaces() { return wrapped.getNamespaces(); } + @Override public List getResourceStack(ResourceLocation pLocation) { return wrapped.getResourceStack(pLocation); } + @Override public Map> listResourceStacks(String pPath, Predicate pFilter) { return wrapped.listResourceStacks(pPath, pFilter); } + @Override public Stream listPacks() { return wrapped.listPacks(); } + @Override public Optional getResource(ResourceLocation pLocation) { return wrapped.getResource(pLocation); } + //@formatter:on + } +} diff --git a/src/main/java/codechicken/lib/internal/ClientInit.java b/src/main/java/codechicken/lib/internal/ClientInit.java index a0186ba8..1e1eb056 100644 --- a/src/main/java/codechicken/lib/internal/ClientInit.java +++ b/src/main/java/codechicken/lib/internal/ClientInit.java @@ -3,13 +3,19 @@ import codechicken.lib.CodeChickenLib; import codechicken.lib.config.ConfigCategory; import codechicken.lib.config.ConfigSyncManager; +import codechicken.lib.gui.modular.lib.CursorHelper; +import codechicken.lib.gui.modular.sprite.CCGuiTextures; import codechicken.lib.model.CompositeItemModel; import codechicken.lib.model.ClassModelLoader; import codechicken.lib.render.CCRenderEventHandler; import codechicken.lib.render.block.BlockRenderingRegistry; import net.covers1624.quack.util.CrashLock; +import net.minecraft.client.Minecraft; +import net.minecraft.server.packs.resources.ResourceManager; +import net.minecraft.server.packs.resources.ResourceManagerReloadListener; import net.minecraftforge.client.event.ClientPlayerNetworkEvent; import net.minecraftforge.client.event.ModelEvent; +import net.minecraftforge.client.event.RegisterClientReloadListenersEvent; import net.minecraftforge.common.MinecraftForge; import net.minecraftforge.eventbus.api.IEventBus; import net.minecraftforge.fml.event.lifecycle.FMLClientSetupEvent; @@ -36,6 +42,7 @@ public static void init() { bus.addListener(ClientInit::onClientSetup); bus.addListener(ClientInit::onRegisterGeometryLoaders); + bus.addListener(ClientInit::onResourceReload); } private static void onClientSetup(FMLClientSetupEvent event) { @@ -74,4 +81,9 @@ private static void onRegisterGeometryLoaders(ModelEvent.RegisterGeometryLoaders event.register("item_composite", new CompositeItemModel()); event.register("class", new ClassModelLoader()); } + + public static void onResourceReload(RegisterClientReloadListenersEvent event) { + event.registerReloadListener(CCGuiTextures.getAtlasHolder()); + event.registerReloadListener((ResourceManagerReloadListener) e -> CursorHelper.onResourceReload()); + } } diff --git a/src/main/java/codechicken/lib/internal/network/CCLNetwork.java b/src/main/java/codechicken/lib/internal/network/CCLNetwork.java index 99b73ea5..5655ef14 100644 --- a/src/main/java/codechicken/lib/internal/network/CCLNetwork.java +++ b/src/main/java/codechicken/lib/internal/network/CCLNetwork.java @@ -15,6 +15,10 @@ public class CCLNetwork { //Client handled. public static final int C_ADD_LANDING_EFFECTS = 1; public static final int C_OPEN_CONTAINER = 10; + public static final int C_GUI_SYNC = 20; + + //Server handled. + public static final int S_GUI_SYNC = 20; //Login handled. public static final int L_CONFIG_SYNC = 1; @@ -22,8 +26,8 @@ public class CCLNetwork { public static void init() { netChannel = PacketCustomChannelBuilder.named(NET_CHANNEL)// .assignClientHandler(() -> ClientPacketHandler::new)// + .assignServerHandler(() -> ServerPacketHandler::new)// .assignLoginHandler(() -> LoginPacketHandler::new)// .build(); } - } diff --git a/src/main/java/codechicken/lib/internal/network/ClientPacketHandler.java b/src/main/java/codechicken/lib/internal/network/ClientPacketHandler.java index 8f7dd554..32bccf89 100644 --- a/src/main/java/codechicken/lib/internal/network/ClientPacketHandler.java +++ b/src/main/java/codechicken/lib/internal/network/ClientPacketHandler.java @@ -1,6 +1,7 @@ package codechicken.lib.internal.network; import codechicken.lib.inventory.container.ICCLContainerType; +import codechicken.lib.inventory.container.modular.ModularGuiContainerMenu; import codechicken.lib.packet.ICustomPacketHandler.IClientPacketHandler; import codechicken.lib.packet.PacketCustom; import codechicken.lib.render.particle.CustomParticleHandler; @@ -17,8 +18,7 @@ import net.minecraft.world.level.block.state.BlockState; import net.minecraftforge.registries.ForgeRegistries; -import static codechicken.lib.internal.network.CCLNetwork.C_ADD_LANDING_EFFECTS; -import static codechicken.lib.internal.network.CCLNetwork.C_OPEN_CONTAINER; +import static codechicken.lib.internal.network.CCLNetwork.*; /** * Created by covers1624 on 14/07/2017. @@ -36,6 +36,7 @@ public void handlePacket(PacketCustom packet, Minecraft mc, ClientPacketListener CustomParticleHandler.addLandingEffects(mc.level, pos, state, vec, numParticles); } case C_OPEN_CONTAINER -> handleOpenContainer(packet, mc); + case C_GUI_SYNC -> ModularGuiContainerMenu.handlePacketFromServer(mc.player, packet); } } diff --git a/src/main/java/codechicken/lib/internal/network/ServerPacketHandler.java b/src/main/java/codechicken/lib/internal/network/ServerPacketHandler.java new file mode 100644 index 00000000..4edab263 --- /dev/null +++ b/src/main/java/codechicken/lib/internal/network/ServerPacketHandler.java @@ -0,0 +1,22 @@ +package codechicken.lib.internal.network; + +import codechicken.lib.inventory.container.modular.ModularGuiContainerMenu; +import codechicken.lib.packet.ICustomPacketHandler.IServerPacketHandler; +import codechicken.lib.packet.PacketCustom; +import net.minecraft.server.level.ServerPlayer; +import net.minecraft.server.network.ServerGamePacketListenerImpl; + +import static codechicken.lib.internal.network.CCLNetwork.S_GUI_SYNC; + +/** + * Created by covers1624 on 14/07/2017. + */ +public class ServerPacketHandler implements IServerPacketHandler { + + @Override + public void handlePacket(PacketCustom packet, ServerPlayer sender, ServerGamePacketListenerImpl handler) { + switch (packet.getType()) { + case S_GUI_SYNC -> ModularGuiContainerMenu.handlePacketFromClient(sender, packet); + } + } +} diff --git a/src/main/java/codechicken/lib/inventory/container/data/AbstractDataStore.java b/src/main/java/codechicken/lib/inventory/container/data/AbstractDataStore.java new file mode 100644 index 00000000..759b2db8 --- /dev/null +++ b/src/main/java/codechicken/lib/inventory/container/data/AbstractDataStore.java @@ -0,0 +1,45 @@ +package codechicken.lib.inventory.container.data; + +import codechicken.lib.data.MCDataInput; +import codechicken.lib.data.MCDataOutput; +import net.minecraft.nbt.Tag; + +import java.util.Objects; + +/** + * The base class of a simple general purpose serializable data system. + * + * Created by brandon3055 on 08/09/2023 + */ +public abstract class AbstractDataStore { + + protected T value; + + public AbstractDataStore(T defaultValue) { + this.value = defaultValue; + } + + public T get() { + return value; + } + + public void set(T value) { + this.value = value; + markDirty(); + } + + public void markDirty(){} + + public abstract void toBytes(MCDataOutput buf); + + public abstract void fromBytes(MCDataInput buf); + + public abstract Tag toTag(); + + public abstract void fromTag(Tag tag); + + public boolean isSameValue(T newValue) { + return Objects.equals(value, newValue); + } +} + diff --git a/src/main/java/codechicken/lib/inventory/container/data/BooleanData.java b/src/main/java/codechicken/lib/inventory/container/data/BooleanData.java new file mode 100644 index 00000000..23bc32c0 --- /dev/null +++ b/src/main/java/codechicken/lib/inventory/container/data/BooleanData.java @@ -0,0 +1,42 @@ +package codechicken.lib.inventory.container.data; + +import codechicken.lib.data.MCDataInput; +import codechicken.lib.data.MCDataOutput; +import net.minecraft.nbt.ByteTag; +import net.minecraft.nbt.NumericTag; +import net.minecraft.nbt.Tag; +import net.minecraft.network.FriendlyByteBuf; + +/** + * Created by brandon3055 on 09/09/2023 + */ +public class BooleanData extends AbstractDataStore { + + public BooleanData() { + super(false); + } + + public BooleanData(boolean defaultValue) { + super(defaultValue); + } + + @Override + public void toBytes(MCDataOutput buf) { + buf.writeBoolean(value); + } + + @Override + public void fromBytes(MCDataInput buf) { + value = buf.readBoolean(); + } + + @Override + public Tag toTag() { + return ByteTag.valueOf(value); + } + + @Override + public void fromTag(Tag tag) { + value = ((NumericTag) tag).getAsByte() != 0; + } +} diff --git a/src/main/java/codechicken/lib/inventory/container/data/ByteData.java b/src/main/java/codechicken/lib/inventory/container/data/ByteData.java new file mode 100644 index 00000000..c438282e --- /dev/null +++ b/src/main/java/codechicken/lib/inventory/container/data/ByteData.java @@ -0,0 +1,42 @@ +package codechicken.lib.inventory.container.data; + +import codechicken.lib.data.MCDataInput; +import codechicken.lib.data.MCDataOutput; +import net.minecraft.nbt.ByteTag; +import net.minecraft.nbt.NumericTag; +import net.minecraft.nbt.Tag; +import net.minecraft.network.FriendlyByteBuf; + +/** + * Created by brandon3055 on 09/09/2023 + */ +public class ByteData extends AbstractDataStore { + + public ByteData() { + super((byte) 0); + } + + public ByteData(byte defaultValue) { + super(defaultValue); + } + + @Override + public void toBytes(MCDataOutput buf) { + buf.writeByte(value); + } + + @Override + public void fromBytes(MCDataInput buf) { + value = buf.readByte(); + } + + @Override + public Tag toTag() { + return ByteTag.valueOf(value); + } + + @Override + public void fromTag(Tag tag) { + value = ((NumericTag) tag).getAsByte(); + } +} diff --git a/src/main/java/codechicken/lib/inventory/container/data/DoubleData.java b/src/main/java/codechicken/lib/inventory/container/data/DoubleData.java new file mode 100644 index 00000000..990f9475 --- /dev/null +++ b/src/main/java/codechicken/lib/inventory/container/data/DoubleData.java @@ -0,0 +1,42 @@ +package codechicken.lib.inventory.container.data; + +import codechicken.lib.data.MCDataInput; +import codechicken.lib.data.MCDataOutput; +import net.minecraft.nbt.DoubleTag; +import net.minecraft.nbt.NumericTag; +import net.minecraft.nbt.Tag; +import net.minecraft.network.FriendlyByteBuf; + +/** + * Created by brandon3055 on 09/09/2023 + */ +public class DoubleData extends AbstractDataStore { + + public DoubleData() { + super(0D); + } + + public DoubleData(double defaultValue) { + super(defaultValue); + } + + @Override + public void toBytes(MCDataOutput buf) { + buf.writeDouble(value); + } + + @Override + public void fromBytes(MCDataInput buf) { + value = buf.readDouble(); + } + + @Override + public Tag toTag() { + return DoubleTag.valueOf(value); + } + + @Override + public void fromTag(Tag tag) { + value = ((NumericTag) tag).getAsDouble(); + } +} diff --git a/src/main/java/codechicken/lib/inventory/container/data/FloatData.java b/src/main/java/codechicken/lib/inventory/container/data/FloatData.java new file mode 100644 index 00000000..6a135bca --- /dev/null +++ b/src/main/java/codechicken/lib/inventory/container/data/FloatData.java @@ -0,0 +1,42 @@ +package codechicken.lib.inventory.container.data; + +import codechicken.lib.data.MCDataInput; +import codechicken.lib.data.MCDataOutput; +import net.minecraft.nbt.FloatTag; +import net.minecraft.nbt.NumericTag; +import net.minecraft.nbt.Tag; +import net.minecraft.network.FriendlyByteBuf; + +/** + * Created by brandon3055 on 09/09/2023 + */ +public class FloatData extends AbstractDataStore { + + public FloatData() { + super(0F); + } + + public FloatData(float defaultValue) { + super(defaultValue); + } + + @Override + public void toBytes(MCDataOutput buf) { + buf.writeFloat(value); + } + + @Override + public void fromBytes(MCDataInput buf) { + value = buf.readFloat(); + } + + @Override + public Tag toTag() { + return FloatTag.valueOf(value); + } + + @Override + public void fromTag(Tag tag) { + value = ((NumericTag) tag).getAsFloat(); + } +} diff --git a/src/main/java/codechicken/lib/inventory/container/data/FluidData.java b/src/main/java/codechicken/lib/inventory/container/data/FluidData.java new file mode 100644 index 00000000..5a5bf64c --- /dev/null +++ b/src/main/java/codechicken/lib/inventory/container/data/FluidData.java @@ -0,0 +1,53 @@ +package codechicken.lib.inventory.container.data; + +import codechicken.lib.data.MCDataInput; +import codechicken.lib.data.MCDataOutput; +import net.minecraft.nbt.CompoundTag; +import net.minecraft.nbt.Tag; +import net.minecraft.network.FriendlyByteBuf; +import net.minecraftforge.fluids.FluidStack; + +/** + * Created by brandon3055 on 09/09/2023 + */ +public class FluidData extends AbstractDataStore { + + public FluidData() { + super(FluidStack.EMPTY); + } + + public FluidData(FluidStack defaultValue) { + super(defaultValue); + } + + @Override + public void set(FluidStack value) { + this.value = value.copy(); + markDirty(); + } + + @Override + public void toBytes(MCDataOutput buf) { + buf.writeFluidStack(value); + } + + @Override + public void fromBytes(MCDataInput buf) { + value = buf.readFluidStack(); + } + + @Override + public Tag toTag() { + return value.writeToNBT(new CompoundTag()); + } + + @Override + public void fromTag(Tag tag) { + value = FluidStack.loadFluidStackFromNBT((CompoundTag) tag); + } + + @Override + public boolean isSameValue(FluidStack newValue) { + return value.equals(newValue); + } +} diff --git a/src/main/java/codechicken/lib/inventory/container/data/IntData.java b/src/main/java/codechicken/lib/inventory/container/data/IntData.java new file mode 100644 index 00000000..fe43551a --- /dev/null +++ b/src/main/java/codechicken/lib/inventory/container/data/IntData.java @@ -0,0 +1,42 @@ +package codechicken.lib.inventory.container.data; + +import codechicken.lib.data.MCDataInput; +import codechicken.lib.data.MCDataOutput; +import net.minecraft.nbt.IntTag; +import net.minecraft.nbt.NumericTag; +import net.minecraft.nbt.Tag; +import net.minecraft.network.FriendlyByteBuf; + +/** + * Created by brandon3055 on 09/09/2023 + */ +public class IntData extends AbstractDataStore { + + public IntData() { + super(0); + } + + public IntData(int defaultValue) { + super(defaultValue); + } + + @Override + public void toBytes(MCDataOutput buf) { + buf.writeVarInt(value); + } + + @Override + public void fromBytes(MCDataInput buf) { + value = buf.readVarInt(); + } + + @Override + public Tag toTag() { + return IntTag.valueOf(value); + } + + @Override + public void fromTag(Tag tag) { + value = ((NumericTag) tag).getAsInt(); + } +} diff --git a/src/main/java/codechicken/lib/inventory/container/data/LongData.java b/src/main/java/codechicken/lib/inventory/container/data/LongData.java new file mode 100644 index 00000000..9b928d9a --- /dev/null +++ b/src/main/java/codechicken/lib/inventory/container/data/LongData.java @@ -0,0 +1,42 @@ +package codechicken.lib.inventory.container.data; + +import codechicken.lib.data.MCDataInput; +import codechicken.lib.data.MCDataOutput; +import net.minecraft.nbt.LongTag; +import net.minecraft.nbt.NumericTag; +import net.minecraft.nbt.Tag; +import net.minecraft.network.FriendlyByteBuf; + +/** + * Created by brandon3055 on 09/09/2023 + */ +public class LongData extends AbstractDataStore { + + public LongData() { + super(0L); + } + + public LongData(long defaultValue) { + super(defaultValue); + } + + @Override + public void toBytes(MCDataOutput buf) { + buf.writeVarLong(value); + } + + @Override + public void fromBytes(MCDataInput buf) { + value = buf.readVarLong(); + } + + @Override + public Tag toTag() { + return LongTag.valueOf(value); + } + + @Override + public void fromTag(Tag tag) { + value = ((NumericTag) tag).getAsLong(); + } +} diff --git a/src/main/java/codechicken/lib/inventory/container/data/ShortData.java b/src/main/java/codechicken/lib/inventory/container/data/ShortData.java new file mode 100644 index 00000000..9a29d0dd --- /dev/null +++ b/src/main/java/codechicken/lib/inventory/container/data/ShortData.java @@ -0,0 +1,42 @@ +package codechicken.lib.inventory.container.data; + +import codechicken.lib.data.MCDataInput; +import codechicken.lib.data.MCDataOutput; +import net.minecraft.nbt.NumericTag; +import net.minecraft.nbt.ShortTag; +import net.minecraft.nbt.Tag; +import net.minecraft.network.FriendlyByteBuf; + +/** + * Created by brandon3055 on 09/09/2023 + */ +public class ShortData extends AbstractDataStore { + + public ShortData() { + super((short) 0); + } + + public ShortData(short defaultValue) { + super(defaultValue); + } + + @Override + public void toBytes(MCDataOutput buf) { + buf.writeShort(value); + } + + @Override + public void fromBytes(MCDataInput buf) { + value = buf.readShort(); + } + + @Override + public Tag toTag() { + return ShortTag.valueOf(value); + } + + @Override + public void fromTag(Tag tag) { + value = ((NumericTag) tag).getAsShort(); + } +} diff --git a/src/main/java/codechicken/lib/inventory/container/modular/ModularGuiContainerMenu.java b/src/main/java/codechicken/lib/inventory/container/modular/ModularGuiContainerMenu.java new file mode 100644 index 00000000..f6f043cc --- /dev/null +++ b/src/main/java/codechicken/lib/inventory/container/modular/ModularGuiContainerMenu.java @@ -0,0 +1,321 @@ +package codechicken.lib.inventory.container.modular; + +import codechicken.lib.data.MCDataInput; +import codechicken.lib.data.MCDataOutput; +import codechicken.lib.gui.modular.elements.GuiSlots; +import codechicken.lib.gui.modular.lib.container.ContainerScreenAccess; +import codechicken.lib.gui.modular.lib.container.DataSync; +import codechicken.lib.gui.modular.lib.container.SlotGroup; +import codechicken.lib.gui.modular.lib.geometry.GuiParent; +import codechicken.lib.internal.network.CCLNetwork; +import codechicken.lib.packet.PacketCustom; +import codechicken.lib.vec.Vector3; +import net.minecraft.network.FriendlyByteBuf; +import net.minecraft.server.level.ServerPlayer; +import net.minecraft.world.entity.player.Inventory; +import net.minecraft.world.entity.player.Player; +import net.minecraft.world.inventory.AbstractContainerMenu; +import net.minecraft.world.inventory.MenuType; +import net.minecraft.world.inventory.Slot; +import net.minecraft.world.item.ItemStack; +import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.function.BiConsumer; +import java.util.function.Consumer; + +import static codechicken.lib.internal.network.CCLNetwork.*; + +/** + * The base abstract ContainerMenu for all modular gui containers. + *

+ * Created by brandon3055 on 08/09/2023 + */ +public abstract class ModularGuiContainerMenu extends AbstractContainerMenu { + private static final Logger LOGGER = LogManager.getLogger(); + + public final Inventory inventory; + public final List slotGroups = new ArrayList<>(); + public final Map slotGroupMap = new HashMap<>(); + public final Map> zonedSlots = new HashMap<>(); + public final List> dataSyncs = new ArrayList<>(); + + protected ModularGuiContainerMenu(@Nullable MenuType menuType, int containerId, Inventory inventory) { + super(menuType, containerId); + this.inventory = inventory; + } + + /** + * Creates and returns a new slot group for this container. + * You can then add your inventory slots to this slot group, similar to how you would normally add slots to the container. + * With one big exception! You do not need to worry about setting slot positions! (Just use 0, 0) + *

+ * Make sure to save your slot groups to accessible fields in your container menu class. + * You will need to pass these to appropriate {@link GuiSlots} / {@link GuiSlots#singleSlot(GuiParent, ContainerScreenAccess, SlotGroup, int)} elements. + * The gui elements will handle positioning and rendering the slots. + *

+ * As far as splitting a containers slots into multiple groups, Typically the players main inventory and hot bar would be added as two separate groups. + * How you handle the containers slots is up to you, For something like a machine with several spread out slots, + * you can still add all the slots to a single group, then pass each individual slot from the group to a single {@link GuiSlots#singleSlot(GuiParent, ContainerScreenAccess, SlotGroup, int)} element. + * + * @param zoneId Used for quick-move (shift click) operations. Each group has a zone id, and you can specify which zones a group can quick-move to. + * Multiple groups can have the same zone id. Quick move work though the groups in a zone in the order the groups here added. + * @param quickMoveTo List of zones this group can quick-move to. + */ + protected SlotGroup createSlotGroup(int zoneId, int... quickMoveTo) { + SlotGroup group = new SlotGroup(this, zoneId, quickMoveTo); + slotGroups.add(group); + return group; + } + + /** + * Convenience method to create a slot group for player slots. + * Configured to quick-move to {@link #remoteSlotGroup()} groups. + * + * @see #createSlotGroup(int, int[]) + */ + protected SlotGroup playerSlotGroup() { + return createSlotGroup(0, 1); + } + + /** + * Convenience method to create a slot group for the 'other side' of the inventory + * So the Block/tile or whatever this inventory is attached to. + * Configured to quick-move to {@link #playerSlotGroup()} groups. + * + * @see #createSlotGroup(int, int[]) + */ + protected SlotGroup remoteSlotGroup() { + return createSlotGroup(1, 0); + } + + //=== Network ===// + /** + * Send a packet to the client side container. + * + * @param packetId message id, Can be any value from 0 to 254, 255 is used by the {@link DataSync} system. + * @param packetWriter Use this callback to write your data to the packet. + */ + public void sendPacketToClient(int packetId, Consumer packetWriter) { + if (inventory.player instanceof ServerPlayer serverPlayer) { + PacketCustom packet = new PacketCustom(CCLNetwork.NET_CHANNEL, C_GUI_SYNC); + packet.writeByte(containerId); + packet.writeByte((byte) packetId); + packetWriter.accept(packet); + packet.sendToPlayer(serverPlayer); + } + } + + /** + * Send a packet to the server side container. + * + * @param packetId message id, Can be any value from 0 to 255 + * @param packetWriter Use this callback to write your data to the packet. + */ + public void sendPacketToServer(int packetId, Consumer packetWriter) { + PacketCustom packet = new PacketCustom(CCLNetwork.NET_CHANNEL, S_GUI_SYNC); + packet.writeByte(containerId); + packet.writeByte((byte) packetId); + packetWriter.accept(packet); + packet.sendToServer(); + } + + public static void handlePacketFromClient(Player player, MCDataInput packet) { + int containerId = packet.readByte(); + int packetId = packet.readByte() & 0xFF; + if (player.containerMenu instanceof ModularGuiContainerMenu menu && menu.containerId == containerId) { + menu.handlePacketFromClient(player, packetId, packet); + } + } + + /** + * Override this in your container menu implementation in order to receive packets sent via {@link #sendPacketToServer(int, Consumer)} + */ + public void handlePacketFromClient(Player player, int packetId, MCDataInput packet) { + + } + + public static void handlePacketFromServer(Player player, MCDataInput packet) { + int containerId = packet.readByte(); + int packetId = packet.readByte() & 0xFF; + if (player.containerMenu instanceof ModularGuiContainerMenu menu && menu.containerId == containerId) { + menu.handlePacketFromServer(player, packetId, packet); + } + } + + /** + * Override this in your container menu implementation in order to receive packets sent via {@link #sendPacketToServer(int, Consumer)} + *

+ * Don't forget to call super if you plan on using the {@link DataSync} system. + */ + public void handlePacketFromServer(Player player, int packetId, MCDataInput packet) { + if (packetId == 255) { + int index = packet.readByte() & 0xFF; + if (dataSyncs.size() > index) { + dataSyncs.get(index).handleSyncPacket(packet); + } + } + } + + //=== Quick Move ===/// + + /** + * Determines if two @link {@link ItemStack} match and can be merged into a single slot + */ + public static boolean canStacksMerge(ItemStack stack1, ItemStack stack2) { + if (stack1.isEmpty() || stack2.isEmpty()) return false; + return ItemStack.matches(stack1, stack2); + } + + /** + * Transfers to the next zone in order, and will loop around to the lowest zone. + * TODO, Would be nice to have better control over quick-move + * Maybe just the ability to specify which zones each group quick-moves to... + */ + @Override + public ItemStack quickMoveStack(@NotNull Player player, int slotIndex) { + Slot slot = getSlot(slotIndex); + if (slot == null || !slot.hasItem()) { + return ItemStack.EMPTY; + } + + SlotGroup group = slotGroupMap.get(slot); + if (group == null) { + return ItemStack.EMPTY; + } + + ItemStack stack = slot.getItem(); + ItemStack result = stack.copy(); + + boolean movedAnything = false; + for (Integer zone : group.quickMoveTo) { + if (!zonedSlots.containsKey(zone)) { + LOGGER.warn("Attempted to quick move to zone id {} but there are no slots assigned to this zone! This is a bug!", zone); + continue; + } + if (moveItemStackTo(stack, zonedSlots.get(zone), false)) { + movedAnything = true; + break; + } + } + + if (!movedAnything) { + return ItemStack.EMPTY; + } + + if (stack.isEmpty()) { + slot.set(ItemStack.EMPTY); + } else { + slot.setChanged(); + } + + slot.onTake(player, stack); + return result; + } + + protected boolean moveItemStackTo(ItemStack stack, List targets, boolean reverse) { + int start = 0; + int end = targets.size(); + boolean moved = false; + int position = start; + if (reverse) { + position = end - 1; + } + + Slot slot; + ItemStack itemStack2; + if (stack.isStackable()) { + while (!stack.isEmpty()) { + if (reverse) { + if (position < start) { + break; + } + } else if (position >= end) { + break; + } + + slot = targets.get(position); + itemStack2 = slot.getItem(); + if (!itemStack2.isEmpty() && ItemStack.isSameItemSameTags(stack, itemStack2)) { + int l = itemStack2.getCount() + stack.getCount(); + if (l <= stack.getMaxStackSize()) { + stack.setCount(0); + itemStack2.setCount(l); + slot.setChanged(); + moved = true; + } else if (itemStack2.getCount() < stack.getMaxStackSize()) { + stack.shrink(stack.getMaxStackSize() - itemStack2.getCount()); + itemStack2.setCount(stack.getMaxStackSize()); + slot.setChanged(); + moved = true; + } + } + + if (reverse) { + --position; + } else { + ++position; + } + } + } + + if (!stack.isEmpty()) { + if (reverse) { + position = end - 1; + } else { + position = start; + } + + while (true) { + if (reverse) { + if (position < start) { + break; + } + } else if (position >= end) { + break; + } + + slot = targets.get(position); + itemStack2 = slot.getItem(); + if (itemStack2.isEmpty() && slot.mayPlace(stack)) { + if (stack.getCount() > slot.getMaxStackSize()) { + slot.set(stack.split(slot.getMaxStackSize())); + } else { + slot.set(stack.split(stack.getCount())); + } + + slot.setChanged(); + moved = true; + break; + } + + if (reverse) { + --position; + } else { + ++position; + } + } + } + + return moved; + } + + //=== Internal Methods ===// + + public void mapSlot(Slot slot, SlotGroup slotGroup) { + slotGroupMap.put(slot, slotGroup); + zonedSlots.computeIfAbsent(slotGroup.zone, e -> new ArrayList<>()).add(slot); + } + + @Override + public void broadcastChanges() { + super.broadcastChanges(); + dataSyncs.forEach(DataSync::detectAndSend); + } +} diff --git a/src/main/java/codechicken/lib/inventory/container/modular/ModularSlot.java b/src/main/java/codechicken/lib/inventory/container/modular/ModularSlot.java new file mode 100644 index 00000000..bfa02872 --- /dev/null +++ b/src/main/java/codechicken/lib/inventory/container/modular/ModularSlot.java @@ -0,0 +1,121 @@ +package codechicken.lib.inventory.container.modular; + +import net.minecraft.world.Container; +import net.minecraft.world.entity.player.Player; +import net.minecraft.world.inventory.Slot; +import net.minecraft.world.item.ItemStack; + +import java.util.function.*; + +/** + * A fully configurable inventory slot. + * If there is anything this slot can not do... Let me know. + *

+ * Created by brandon3055 on 10/09/2023 + */ +public class ModularSlot extends Slot { + private boolean canPlace = true; + private boolean checkContainer = true; + private Supplier enabled = () -> true; + private Predicate validator = stack -> true; + private Function stackLimit = stack -> Integer.MAX_VALUE; + private BiPredicate canRemove = (player, stack) -> true; + private BiConsumer onSet = (oldStack, newStack) -> {}; + + public ModularSlot(Container container, int index) { + this(container, index, 0, 0); + } + + public ModularSlot(Container container, int index, int xPos, int yPos) { + super(container, index, xPos, yPos); + } + + /** + * Configure this slot as an output only slot. + * Items can not be placed in this slot by the player. + */ + public ModularSlot output() { + canPlace = false; + return this; + } + + /** + * Do not use the containers canPlaceItem when checking if an item can be placed. + */ + public ModularSlot noCheck() { + checkContainer = false; + return this; + } + + /** + * Allows you to attach a validator to control what items are allowed in this slot. + * You can also limit a slots allowed contents via the {@link Container#canPlaceItem(int, ItemStack)} method of the container. + * + * @param validator The validator predicate, If the predicate returns false for a stack, the stack will not be placed. + */ + public ModularSlot setValidator(Predicate validator) { + this.validator = validator; + return this; + } + + /** + * Allows you to get a callback when the slot contents are set. + * Parameters given are Old stack then New stack. + */ + public ModularSlot onSet(BiConsumer onSet) { + this.onSet = onSet; + return this; + } + + /** + * Allows you to apply a stack size limit that (if smaller) will override the container and the item stack limits. + */ + public ModularSlot setStackLimit(Function stackLimit) { + this.stackLimit = stackLimit; + return this; + } + + /** + * Allows you to attach a "can remove" predicate that can block removal of a stack from the slot by the player. + */ + public ModularSlot setCanRemove(BiPredicate canRemove) { + this.canRemove = canRemove; + return this; + } + + public ModularSlot setEnabled(Supplier enabled) { + this.enabled = enabled; + return this; + } + + public ModularSlot setEnabled(boolean enabled) { + this.enabled = () -> enabled; + return this; + } + + @Override + public boolean mayPlace(ItemStack itemStack) { + return canPlace && validator.test(itemStack) && (!checkContainer || container.canPlaceItem(getContainerSlot(), itemStack)); + } + + @Override + public boolean mayPickup(Player player) { + return canRemove.test(player, getItem()); + } + + @Override + public void set(ItemStack itemStack) { + onSet.accept(getItem(), itemStack); + super.set(itemStack); + } + + @Override + public boolean isActive() { + return enabled.get(); + } + + @Override + public int getMaxStackSize(ItemStack itemStack) { + return Math.min(super.getMaxStackSize(itemStack), stackLimit.apply(itemStack)); + } +} diff --git a/src/main/java/codechicken/lib/render/CCRenderEventHandler.java b/src/main/java/codechicken/lib/render/CCRenderEventHandler.java index 7bd13335..1f25a147 100644 --- a/src/main/java/codechicken/lib/render/CCRenderEventHandler.java +++ b/src/main/java/codechicken/lib/render/CCRenderEventHandler.java @@ -1,11 +1,13 @@ package codechicken.lib.render; +import codechicken.lib.gui.modular.lib.CursorHelper; import codechicken.lib.raytracer.VoxelShapeBlockHitResult; import codechicken.lib.vec.Matrix4; import net.minecraft.world.phys.BlockHitResult; import net.minecraftforge.api.distmarker.Dist; import net.minecraftforge.api.distmarker.OnlyIn; import net.minecraftforge.client.event.RenderHighlightEvent; +import net.minecraftforge.client.event.ScreenEvent; import net.minecraftforge.common.MinecraftForge; import net.minecraftforge.event.TickEvent; import net.minecraftforge.eventbus.api.EventPriority; diff --git a/src/main/java/codechicken/lib/util/FormatUtil.java b/src/main/java/codechicken/lib/util/FormatUtil.java new file mode 100644 index 00000000..e3395419 --- /dev/null +++ b/src/main/java/codechicken/lib/util/FormatUtil.java @@ -0,0 +1,46 @@ +package codechicken.lib.util; + +import java.text.DecimalFormat; +import java.text.DecimalFormatSymbols; +import java.util.Locale; + +/** + * Created by brandon3055 on 31/10/2023 + */ +public class FormatUtil { + + private static final DecimalFormat DECIMAL_FORMAT = new DecimalFormat("###,###,###,###,###", DecimalFormatSymbols.getInstance(Locale.ROOT)); + + public static String formatNumber(double value) { + if (Math.abs(value) < 1000D) return String.valueOf(value); + else if (Math.abs(value) < 1000000D) return addCommas((int) value); //I mean whats the point of displaying 1.235K instead of 1,235? + else if (Math.abs(value) < 1000000000D) return Math.round(value / 1000D) / 1000D + "M"; + else if (Math.abs(value) < 1000000000000D) return Math.round(value / 1000000D) / 1000D + "G"; + else return Math.round(value / 1000000000D) / 1000D + "T"; + } + + public static String formatNumber(long value) { + if (value == Long.MIN_VALUE) value = Long.MAX_VALUE; + if (Math.abs(value) < 1000L) return String.valueOf(value); + else if (Math.abs(value) < 1000000L) return addCommas(value); + else if (Math.abs(value) < 1000000000L) return Math.round((double)(value / 100000L)) / 10D + "M"; + else if (Math.abs(value) < 1000000000000L) return Math.round((double)(value / 100000000L)) / 10D + "G"; + else if (Math.abs(value) < 1000000000000000L) return Math.round((double)(value / 1000000000L)) / 1000D + "T"; + else if (Math.abs(value) < 1000000000000000000L) return Math.round((double)(value / 1000000000000L)) / 1000D + "P"; + else return Math.round((double) (value / 1000000000000000L)) / 1000D + "E"; + } + + /** + * Add commas to a number e.g. 161253126 > 161,253,126 + */ + public static String addCommas(int value) { + return DECIMAL_FORMAT.format(value); + } + + /** + * Add commas to a number e.g. 161253126 > 161,253,126 + */ + public static String addCommas(long value) { + return DECIMAL_FORMAT.format(value); + } +} diff --git a/src/main/resources/META-INF/accesstransformer.cfg b/src/main/resources/META-INF/accesstransformer.cfg index 03a57caa..0cd041ee 100644 --- a/src/main/resources/META-INF/accesstransformer.cfg +++ b/src/main/resources/META-INF/accesstransformer.cfg @@ -63,3 +63,27 @@ public-f com.mojang.blaze3d.shaders.Uniform m_5679_(Lorg/joml/Matrix4f;)V # set public-f com.mojang.blaze3d.shaders.Uniform m_200759_(Lorg/joml/Matrix3f;)V # set public com.mojang.blaze3d.shaders.Program (Lcom/mojang/blaze3d/shaders/Program$Type;ILjava/lang/String;)V # public com.mojang.blaze3d.shaders.Program$Type m_85571_()I # getGlType + +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen m_280092_(Lnet/minecraft/client/gui/GuiGraphics;Lnet/minecraft/world/inventory/Slot;)V # renderSlot +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen m_280211_(Lnet/minecraft/client/gui/GuiGraphics;Lnet/minecraft/world/item/ItemStack;IILjava/lang/String;)V # renderFloatingItem +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen m_97744_(DD)Lnet/minecraft/world/inventory/Slot; # findSlot +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen m_97818_()V # recalculateQuickCraftRemaining +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen f_97706_ # clickedSlot +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen f_97711_ # draggingItem +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen f_97710_ # isSplittingStack +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen f_97717_ # quickCraftingType +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen f_97720_ # quickCraftingRemainder +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen f_97715_ # snapbackItem +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen f_97714_ # snapbackTime +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen f_97707_ # snapbackEnd +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen f_97712_ # snapbackStartX +public net.minecraft.client.gui.screens.inventory.AbstractContainerScreen f_97713_ # snapbackStartY +public-f net.minecraft.world.inventory.Slot f_40220_ # x +public-f net.minecraft.world.inventory.Slot f_40221_ # y +public net.minecraft.world.inventory.AbstractContainerMenu m_38897_(Lnet/minecraft/world/inventory/Slot;)Lnet/minecraft/world/inventory/Slot; # addSlot + +public net.minecraft.client.renderer.texture.TextureAtlas m_276092_()I # getWidth +public net.minecraft.client.renderer.texture.TextureAtlas m_276095_()I # getHeight +public net.minecraft.client.gui.GuiGraphics (Lnet/minecraft/client/Minecraft;Lcom/mojang/blaze3d/vertex/PoseStack;Lnet/minecraft/client/renderer/MultiBufferSource$BufferSource;)V # +public net.minecraft.client.gui.GuiGraphics m_286081_()V # flushIfUnmanaged +public net.minecraft.client.gui.GuiGraphics m_287246_()V # flushIfManaged \ No newline at end of file diff --git a/src/main/resources/assets/codechickenlib/atlases/gui.json b/src/main/resources/assets/codechickenlib/atlases/gui.json new file mode 100644 index 00000000..5da82735 --- /dev/null +++ b/src/main/resources/assets/codechickenlib/atlases/gui.json @@ -0,0 +1,9 @@ +{ + "sources": [ + { + "type": "directory", + "source": "gui", + "prefix": "gui/" + } + ] +} \ No newline at end of file diff --git a/src/main/resources/assets/codechickenlib/textures/gui/cursors/drag.png b/src/main/resources/assets/codechickenlib/textures/gui/cursors/drag.png new file mode 100644 index 0000000000000000000000000000000000000000..1e8c3266ad6e5f5e3f99ec6652b4ab6e1e6ea8c4 GIT binary patch literal 4183 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 h?L?P) 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 literal 0 HcmV?d00001 diff --git a/src/main/resources/assets/codechickenlib/textures/gui/cursors/resize_diag_trbl.png b/src/main/resources/assets/codechickenlib/textures/gui/cursors/resize_diag_trbl.png new file mode 100644 index 0000000000000000000000000000000000000000..b306c6ba4b86116e5c22d20399474c5dbec375c9 GIT binary patch literal 4339 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 literal 0 HcmV?d00001 diff --git a/src/main/resources/assets/codechickenlib/textures/gui/cursors/resize_h.png b/src/main/resources/assets/codechickenlib/textures/gui/cursors/resize_h.png new file mode 100644 index 0000000000000000000000000000000000000000..07baa61c49b5bb6d4b3c49fe61a15c172599c9f6 GIT binary patch literal 3325 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 Hu0mjfOL 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 literal 0 HcmV?d00001 diff --git a/src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_borderless.png b/src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_borderless.png new file mode 100644 index 0000000000000000000000000000000000000000..f1484440a40e387de4625f5a774eb84fc5c1c9d3 GIT binary patch literal 8152 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 literal 0 HcmV?d00001 diff --git a/src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_borderless_pressed.png b/src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_borderless_pressed.png new file mode 100644 index 0000000000000000000000000000000000000000..7273c079cd5a16f01199ad9a8c3f2d9718a68859 GIT binary patch literal 12424 zcmeHtc|4Ts|Mxw{z9yr^9u-qrhQSObW2tZ&#t@-&2Ez=-Huk+zI+8X~LrmEsl(HnD zj+oP8$rj@n;Y7$5Sz~_pXgQtd`MqAxbDr1p`|mrwZufQH*XMd)pZELvT%Y^8?z`Xc z)`!G|WrP6$5VN!}CxB-x_YmX<{}xdK*#IEbLpo~dX=%9!5CrdmmJnV5!nFYp$WLV+ z2o!*V_s)XHX0Xh);T|#&zKwG-SeE@(P6f+q-)y*T5D*?f3_NFohc;M-f#)Ldpd^0n zs}3x;frsG6KYbox!{Yn-kR=YUq^*I_($GPI?X|RZ475-N+9)N2u7R$uffgF0@i`A*jyjKbK*MsofEhGos%h8hJN#xs|^27W$B-IKzP1lT5#KbD_`2o?clrb zoF2|QV3XBB%Y$G?T!FwpoIbz|5aj0<;O7$*5D*X&5)>ANiHVAch;H93B>|I{QB;tZ zk&{!}siCT*yiY|=PE8-aPYZ!WA{AB97?idV$XlBW0ud4t5)~2MCMLE`dxzW(?SK8_ zJOv~LxjsQ5$^eff1S$#PJO>m(KlvaVzpqrlYaS>sAHRU0kgy17P_c=t6qJV-%E!mc z3)aSf^#HFV-{u_%6MiXMqJVNBO#9^JTY@U4WzVDuZ)SGtxCNaO65g_P+jf~B`mA zYia4%Z_sb&Qtq&CP#aSX^3O;qv10{QmpO?0?}U3G(9M<%RMJaCt#^!nnjGdHHrA_&1x_3J?RO zl(kO^!b~sUDtjiRqC=RGb_;qVyk#eHaMvuC+6J@#J7TB)pP2n1_BSsUAPR+mg9nua zjDZnqNp%0fend*wUP?y&`>J(A4)C^cU-g`^#Dv2uw|KaSeEp{m4j`Yq-pc$^!2y_Y zE1zR8W#ajyEBe2Le}OkOC~nl>JrgtLLI?8q1#$p&<5?EDqYNy>%?_;By5^A@0oyIx zIY3(W(j>fN_Iri1W>tk6njw6G$N}=fDzvY@*JAzBGzV}krS`Kk9%4T5+yXlvySL$1 zzPyg&r#{dZ4lvR6g^;4gljQVyZFJ0{J0oyz{aQ@*?RdDX{J^JaDz?jH}dl$ z5-SuCq>GUe)p3U>gR+WP{d06Jd%ae5A=dc)S5NOIc&35ANUVJCPWFrlC9qY`8>&{;LyF6aiEX6tF);l5gy?$Ti@}?MQvW%wZFVuu`0REVmaX#PB z5)L3y$pOj(ld0?FkQYbgBC-=d-@Scj?9JpHdVakk48`D8@vk)^K= zD@V-O>zb%GXlVPHe)c7~_BQAi_1d{5yFYTFuyUa7(X_>j8o7>Hx&RDltMNXdE4$y2 zAl=W7m#<-E)Uz`0CgFQ0v4RC5=&SN$xkK~^Q0& zYN>HI0xM=~w%^e14hY6_o2An;%e4hscm zn31M+^VQ7&iaBiyUR<^di*&f?hi(RxF0!w=z@EabhT_FCYKAgjRMpXN#3agg1|)@r zUz$7s1I{<2G@{zQkdP{N{4R1FYi*B;0(<$3TESlT3gA!;D-#F9^Q1B(1D9(Hw^z9* zCGTd>1LN$OfiF)uz*FOrU%N-o=6qW3YN0)>0trS*a{woB#2+2pla|e$Hxd($4Z{_H zKI)eBxNCX44i>UK-*>T4jlnCDHUDgj@%i}z9rp^O{ZV*8*59{7-}9!0I2mHu3_ph7 ziWyUbY^Sj=*Et7_WEO(DcA@jUTD={lW!iWqTO;*W+YXY)up(A>YmiTuCvoIO_sm!r z@UD0A;Iua)DFoPo$}5m&MhXwt0y~DTx2ALsRZm-3t3ev8LUZ%awt5qEutFe1F~jY- zjR5+6%l;GHl}`+G7P6NlJ{NDUE}0gWSX$-)@f^Szcpv^R?-F?S-fAbeF)ZJTi ztav$owZ5$R$-Y9Tz$74Hcgu3KRY{WPN888oauKS!0g@O;U~@jEHet%FUNu>{Z5 zWP(%@aNu~w?#ceLBrD;eco}{*N@R3>$)0*wH3r{6O(nv@5%WO>Hp%XUY-x=Q9f*u9 zotke~1H$K0C`T}3eQSlOK$ATD_p~GcF6=jLJ!Vys(g_i6s4~_cBTTh=n>5RlAu>kp z^q^U~X!x{kS|2N3fz;~Xojj5rI@MX!7lnuVgf$#8_6DiNIY0GkLX{>ql+;Wb#f7c3 z3G5&$7`4P0#&|DnD>9&sz8>ya>1rH`SNx?OQ+T79$2bz8sajKjMp+vBdKqfo()36Yu~!*OTdej37w|OqV9Dz54sM<)2#{ zi!Q01%k~DafmVvFv`eCR=pop9?*Lyr)73WUd9MnsFXf*QxJCez+usMv@?>{59t*p5 z0YFRLLrP&`96*N^uc8Vo&wI$a4hqJD?U}TFI9vhWUclEH3HUp+XWq23>fb4L-aV6R)M1|@zdo`w|;&lNzvuq)72+5IOi zvkiKU#=c~BacB*mbiEZh*~gxTJ)I#zc6(QP~gwPs)>eLF)yC^WPsW^DT$ z>vDJ@)OQkO$XmUx*nO!oicp?0+k~*h@_%fH(s6b3psw+o2$$*Djlf%s zw{U4x_uI1>ajZv8dU4Lch_B5`Yr{&{>m!EWhENuB2V&W%&O3DJ=6VkUkZYGDNMx{5 zcN@0xdJNs@&W$h##3F&B>S?QnZ3T#d$)wY=WC)Fprycax8-)R%9!=vWY+&v%0Fo4D zW#z{Lj#p1xhmAEV*So@LSg{t0-fBnT$gwzy39KX^Gcshy$R!QKHlEY;fb9%0X@Vmr zfaCz0?8|BN7t>gAA~^Fy3(hjVe?>C{ZeZ%2O=%W?68sqa+3D=Zn&O#j1?$i?AD*fw z`d}mD84y)2D>Eayg$B;B@q)a%Xg0>U#L1&Sg$#LaSLHW(xP^5>C(Jh%3DJtV`H-HR z*&*UyfjO~#D4zeAJDrR!kfK`wYE9(4F|42)Gp_wA@>B1mnI1U3Ar+3 zEIC?h1*X(w13SlBD@yKA{FcnEuNIrZ+@6|==V=#UdXv+`{JS$1<9yxX6%^%*w0kG5 z;VJVc1KIJi57{@{Gaco}YVH7F1Z9GmrYSm<0rjsnIFo))u>_MlXXvgl5487Ay41;S zJn^`x6X8-PgI;w62eC~+y&!YrVICDTUUG^96pw;zAJDMkq$YH3^By`@2P>uu^Sp^u zg#qsmkJ7`>bED`YnsKP6PL1c_dH>niYw|pp+A%Qz13uR-dex#d4BH>B;(2V%FVx84 z`2gLkm@zQVTqyVNXT?jp+=GoA3(J>E@~-P7;t-8o6zR=e0Ks*cP< zGQ`X|ugfjRg6lWPdPPEmZuQ``-i1=y&1^8OP>te>!N<94(G~%(8w05Iis;NXo-?hH zMA#7g0H9TAXrKIO+645>kHMFp&h+Lp+C4Q7oD2)gPhds@eKulX;>QXdZ0`Rc0Lu$o zY*K^pbW*oOtzUy54$D8Y$l!xgQI%7jEHwzES;aD?bEc-I9S^<6wInek&w(+2H)S4J ziDcFo4c)NXiUiK?;{aR7aHQ7A!0iPe7a4+P!xen9tUS+FZhZ0=XVlm;hOQ$2Y=i#M zsG>L3*%-Z0FcL5Ah?($dz|~A1i=(Y{D>U~{&>9jI!R6t`LvUo@*D1)L*%u#-=1#`@ zeKlavFk$^YlNM6)W6O=ej&v~cBTJ(?ESnLzeSO9k0NjQ3o;yz)d5(pyQOxi>!3?>M z-})`w`v#Ce@$%$h71+2MB&oLN(1`Scq#u#hvh=^ z*m=P2yx*h+7>@`#;Bd@7*a)-VzdiHGNc@%?$@tbPIlE`g0Q^{3!9EubaDLlD{$wBf z6nBmHDrNpAe7w39Y3vP9VexKh@xuKx@4l|e?w|2r3Lf3at>A>DXGF8P6Y{Bk)Dd}b zwU!C4)}GX>gTaRxE56P|+Mn9c6)RZ;Mq(*@z8!8l5--OL5!t$WA`dCrj_0{dC;NK? zjp~5P{wQ#XmPw$3&!cO~EfH(*5(%4(`P}ARz>a0gk;2!#lQ;&TF|||x@1sjsF63+b zO()}_*WIpM5+$@jak6FBDzH=WQehSFV~YjB1yVJAFvzyUVYz#==EnsYyigdFj<9h6 zAMdgU3|soZ=!0j+%?yLn@pqaF`^Wl|ClFkWD$ai9006kQemOQ#^O`#c8IPuq_`3Ix z(8kuhr_^|~*j-B0+QL2rJIAjR@Iv}BBb~7btVlA!^HQau?cyCs>nHa*N=2Gnj=Mz1i>s-?ZdEQY_!IkB=}32Ed-XI{_#lM>uF}yVkbEdXP z??hyzv8x*7WL0SC=>{{!uj`fu73r$M#_tPj7jYwQIKJzP@+^3Rv?-4O_gJ!_lk1`@ ztKCm$76z6hIlv!rYmvZcw_m;Hg!P0=Vnh8rsL3S`uri<1a2?ED^I;`m-HSgko{QAf!NELT)q3{f4#^EFVN~e0OI?vXMOW8g98Y%8jS@2q)@TH&Rkr4M>b*j5=&0x#dv|L{Q6L#06 zK5o5rE-q@spE)zUt+l=uh~xAMrA2vxFKiMhW@dOxGc%=tAb(GaF9`s`Gh;4eEE=$K zjVJ7WId3oj5&wLLS;&L)NvQ5Uuv?~gq@6>ws^>-MCp2AGAA*Od7Vn`D=`?284}$4YR@csHo;`; z&e*>m4}Fpyz0$zbn!K&|wNYTg(S4uqw>K2~lAm`xM;z&ZpZKG?IEXz|d_=aGX0!f# zfXNA`5rbyi-23NtS!}w+I4N9oe^bE`)kgEP3s)7^C7GI+f7#*4mn?KrDvRX zVO5@i-t9PH)ss)pQ{$2$&yF+t%O|JZ`~0$jb+30;(9X|+8E+! zw-EnIUTMWXU99)=7vvc?$#CNw|Ci8)DlCWI=i;SUAc0MLY&8G~Bt8xQ?ZI0NTYh|U zazN+idMuz!4mvBX+J6Ro7aLBoKN@@#ci6z)pQ=Ih@OL9=gi`}H-ccDDhX)YdeMrGd zZX_>?A4Yk)re0Zz;(<|i)WvDx0?bHcibZ4)$u82`-aXRC9qplPykFQT+yDfil7fj! z;Z$Egnn5^5c>~t~EOT#bDl2W61p8o=kAm-0&HRH%N=OZ)hL-w)a7w7Q@_u0@qaY7Y z1A_U%ZxG-WMwuKO9AKcS85S0%5vHTzALONpK%>!`TH2c0+UlT%IxWI4m>90^M^oWK ze8n&)(cFV50l^f1KP4_E(ak?37^AEV)+>GYj~al({ebtQeNzF{Lo=Khpo!4X(xg%~ ze{MkwJ`f6md~@i(Z9%h-2q0+^NHqVDAa~M%P?BG;%FhrU?myZGgar9+(D88BB>9r4 zpeYUP74eTQxo@6-wBSnMMWF_4v;xKchh;Fu^KY^K;TyMRgU-*6fX#p4{=@n^_YGsv z3WqZ=_jeEBx@T#QQRen<;NkC1@i5pZlH5qzI{G9nbre$9Lmla%@2O5CX?v*aXc38e zXc8KY($f72%F>S(O!RXnaiKtP4GM_kfh6hcqKQa#gr1I`I#N#?p-x0->#OT|5IuEt z-E`d%+N7T#Y=S6YR1$rE_KFMT0YYh`J&>L{NOyHzT|F&zBvDUK9i{6ILLo^Wx*lLJ zB>k^Y9_|JQ{e!4Pa5yPcq8CXsz|U)=53b+_CU{GXvbKiS-*@o7#9&Xb0Y>>S#V;iM z?+SYgm1Gx8!;QZq#zm?iCoN$ZwMRBxx-=rk^zM!a^n;P z*tidJF)#}v5rh4M?EU?HG0H!_KK!W0fz!!@7)&%L29rS0AKxSFzkiR|i_kLo<^UHC z=kG!BjQBrUb0?3I5oiw%xdnv=_8+lv>+6iNBL#lF`g-Y0*_cX7N*j~Hfav~}0*x3- z^4Rbb#QIw0PA2+!k-+TnEnL6fr~EgefIxU?f$!sq>N;qYyE+mrIa{Vg>{+026qU%3${VN6jmGOU~>;Icv!hdh4NPgfdC=A>z`O1q^z^xX) z+b@UAzy7y?WwV+%up|^<;Y0%f5h?Bg0qB|Az?%ZWmbe1~qk?jL`VtdP$F;%To4lpD ziG4V$IpdcromLWONM0f~@8J%Mt1tq-pe_Ko_^8Mz!e(o`{wdi}b%XwTgVRzBsy+wS*@Zz zEGqhl65pHGQ%Mb#OY91vPCRBk*r#D1qc7n6wyUVBwv$qpn z1S_x%`gYpWbK-GUcVA^Q$Rr}`_-V$LlMmz8>r3xeP$ypg0(Y`{LC-I%q-OJ-T^V)C!y0;YtMG^Mr*YR2U1l-ke$GqC4< z5E_0s8!9fCW&-ywqZeyi8c6xiio&3QiH74cTH z2O@5{GNbLuWEZ}^$`ZpnAKb&OE5ztCs!tW_8P>AT1Jw=Mjc^?=&ZZ2bwHh&R5I{bKwvaN=4~nH4woocIZygOYANWpDeFYL8FFS z8tLbA!ko6$Ct}JRf7hCbL_*UpT_YrEUMCowfmL{kpKj=!X;|ANUVcHZixOnf57Y9* z=<0MQ27Hd0o;LgA@%|(^<{N2|L+F%Yw6I!kQ=k#$7<}*j$Nhc^u5GVTDXKW1CBj+3 zQv|EJE$r=zu!xhfGxF5Ftk;4Bn2Mm_^y8^u&hYT?oXn!H!*6_-T+Apm*9=c*^`U$2`z2F)l-S#uuju5Ik5dzG@3m+jx%)Wn;QdPK2^1-B*!C$nRlEOE z@pPTVTVx1s+I2xGs3rE|=DYb6Yx|?iXjx_N+|-YQgM+E;p%0G^tx282YnZw5%k&zP zOD9+l^6%La5b_QeQ-IR9cf}gsX(qOnQzs5f;-2p6*|`qcA;%Eec6oroo(Q+-wbrNoNSKslT022@HZn&jpEst3h7 zD|{MQd%7+-oG?Xr`J00j<@SxBwB-B{sPI;$9Kz|nmrgcbI8e!)qn|VPFRc`EE>C)1 zbInZcGoQ&YDl4HYE9_Yom-C;1KUp6E7|nQq9xi?H*s&?RxHDa?$fVUE1MD z2d@ryC{{g3XP2$CN>+%fNVK_-K5DnU-KB+F?oK(BqVRx<(Wfsnh@MA1ts zyd(%ObcsvSFUkdl|77pu@X*qYj z#ME~FIlghVJ0v9W716|R-)*wTo8r`=Jy9>n38_aKBn6l^@FzNoBt(36=kKug5hHKL z?bUz#dNaK^|8#u0;O)F8gqMXyqS~pbL9KMstg~s8M%$GKI&}?3otAraYMAf4f?-0= z0Y^&U1xmDz)kosJCWt=;)GszVOYe(vY`1r}le?5~a5%roM(;w0TZ5KW>D96WsJV}e z(*rk6bH-l1AKHDJ)menhaP(+Ab`V}gmyWWA_iBl@o^sewYq``}ikk`m3=S9BzI;RUUAd@%AQcDeRAwdIJ&$e_@? zeUda&RGC|GsMS#(R@({Qv%j4*#U+;?jjVOG={%aDIZpdS`gYt)IeM-MvE(;l9@7W* zr&5OR_f3WhB&$*FBvMfw3sZI2YSp)pk>HT)gCY0;`#YF6r)T`Kg z*;!dxoDyB*&nZ0?xp(C@@#GylQEQmkF^Z5-e>Nxj-se+vv&G54orYy8l7?MNd*AOB zT$r7mbqrf_mB&>Fzs_@i&v!`B4Vhxc#{(}5ycKQsDsYkT;d&rJv(11vuDk*KI{YB} zE!6G7X`LOLf0d!{e)xuv)bFQ1%`AMEPcfJcmDVQ0b`%XTf~|j->I-F++)f+oG*rSixJ??S0BVPDf8&$8`op|e4lu}V)>bA9e-PWL(Q-eSus=C z;xZv*E429X^Ns5UDLIb8w3gUS{R!4#g{t+7CQ(+o<#N&@@FEPAc5BmuRN^k>Hv}o$ z!78dxgVvMli7z{l)fn5IN7EYfDM4d}e{o7kc{iLtc2N8zSH;OVTSm+uV1j%f`d*lP+nvc9uPh9UU3^OmDu*I&f(BB@i`t7(REzk&DB{T^USq>L1-lQ2Nc3jw_f9`;N zX6Y3r=lRt3jQm-@`wmaAqU+5f%96j+@oMSaS#|kE^s+H?{=>7OHMQ5LiDF{+7Ng_O z+I<1!;>*yPr3MtS>gyE?vYVWn?8awqDVVx=c}I~6cT36ITXwrYlNuHN3tV?aWMP_9 zXwc$BeCoYTBdOX*lc{?wIU+!2s zf5hA$*fw-qh`l#$^H{X`?vy*v7q0Ax!jW41~p>(;?LWOuhLFWIw$^TR?cd>N^vm5j!z>Qp!JpES&J>3Dy1^A8#DYRRc zp50omLbkpNbAFi2vwF`m!_duhbhy>s&CEpAbRK`AJ{D7v8BG2J!iI-C&&Z&MERKEo zu=9>M@z!Cnhk^{MOSbkLTL}|c(p!5i!d32gu`}mFZu5ljUt{ztl^uGAbP~=CuF`XS z_6koMwLZKsOt!JHQMf0WX|S7~fjIF_Yc}lEW!)yr;3G=B$-2Q;Itv;vvt2Sq&z#4e z8Llw4WIgdHRk*t1{8380A>ekZuj? zaB^>EX>4U6ba`-PAZ2)IW&i+q+O3*xlH5Fwg#Ystdj#>DJPs*R5qpC<{yqs-^@rX4 z?o7nE!>)EoMS?&gk$}qWfBo;{{)ay$q|2pTTj`}-{7F6aF!-fi=TEnP{gv!r`;%Um z`2V-}-N%K%Tanjz{#(cE{=xhC^8@ca{QUmDYp*Mzy%u_1crO^WbKsZt)kxkC=Y9XO z#PxlCDZSONrp|C)@3+bA^KY!S^v5Flzx(<|s(5K-?usGo=;LMowqrp&e+Pdv|N2oEDGN*gzoEK$9esIp4FabpVN(0@Mw7<>T+-I=ri0nae1_l6@DszGrw2+>HM^7 zTsOg4*L_WXZr^<%Dv$HYFF*X_x4qu}#>5nMsJw=7hiLD$SVIqYRGgB2{1Yk>Td@(# zJ8VbHQVbuqc-yzzez$MzygauC&tjgB{QBem>EeI#$i)mrPXBTj z0rAf(ZrL3C{&|J}`hp*zN(8gHxpBa8&9lVJ{u5i#;vBgz^LnzvySiQr;3D?U7~>-m zxQdV27k||ds`vP@5ukzPA*L{ifX_KBj0`D~Ev|+Zv^Srr-tG-Kl%TiaCIXRTiIv!p zQv@re41Q{AU})r+Q_i{Mnw!f#j}l8Nxs+0iNN&_vQ_Z#1T3hXPwAfP1t+d)&>uvPV zBOI7}>9x1s`xwCuHX5uoxPS1@3^UF&^DMK@Hv1em@VV2S?{e3>-TfXbthk_wRaRYX z^)ZN4=UsN)ZTCGMu=d0!J^3k5ecIEX@e#FGRR8+<52EH@Q41C+-B&-N#=EZ8 z_Yz)ok`*%|7Lq68RS^KtK{30UPbnyJirLkJh%8`aR&3nziWnhG+lT!4kJx=i?%#@= z%kAHaTloJX=M=jCM&ul!`#WwwL~W_g-4?Nvh4SeY$-XDbQLZ*3vndtNUN9R*vccSU z-I;(*Zi{t4Yw7Zh2cA9i>h*!5xZ%16AJD4{iPu@OFcP0l?bh3~78${3LgzGP?1ywA zm@Sm$@p4)*cCgX%)t@IDja=7GP%*FN9!0P0R$jYBy}mcA=}Fi=R^|9c7>E9RY zx{dhS?fH-8u6keTHk)rP=-w;Zf@PQX^J4wC#qw)0_tiWtMtv?T=GpK3J-)w1@C6q4 zBU*xBSHgVq4($QpaZ|VVy{pnGsW6*^Y=2bY1d#vahuz8aoYKayZ|3b6`o3hF`$UER zCGZ${H4(wR;*fnSyzDF0?6Uy6?*{&pYjeXm6BjgLg>=CAs-1Gq}00k`0);Y!_8JxlHMDK2x~yJGCY*#9p~A&MTEd z+re+mn5^8Xvtb)4b4M9jOV&{*)O&M++bz}d!O;n9T9CX?&CdvP0f^ji*yC{;PTYYq zo2!3aN;iy=3d~UmVMk}fNWQiaHWh}Nh(Vhjv9C~WIrWwuq0fo8)U{Aj6r4qU5v2`i z@=4oWi#N4H(RJ>8N}HbJCx}WJH|{lYaG(yQOU0ck8B|nn*?(fc;_V_Hpr@f=PmlFA zrrPs-v^zU=a*eZ7?WG=)NIG=(xJc$JM0pvY_uCKGeYrs7E)PzcO%-&s1)3uMBxH!c zq6-&mE}xe7@@K+oF+opOetq69qd+bs$b;~*kS}mj@_0bVL-P6O5LcX^=xK%QB)+3j z<=KuhUYrs*{8?S$y^J9FQhe>y{gWjMlxK0 zz{20RuW{`zL}5EVZf?|lA7~|Z2gv+GobEd}E@S}CB#_M`g|)1(ioYXYOjfdFZnW|6FbMr1|++$y)g23fagBAP%=UutAMYA5;& z&#elXwJn1hVwRXZuhY@ga2HgQXRVJi^m*d4&>EN;VCqk6fbJ7%)sV*dJSkE#^*c^? zL8kCwj2FdVPGAL7o;p;c{{>0u`GU8V{i>r@+zO zhOl$&1eo{~ZfR_lkkO^GTbS4G*$+W$qA~V^$S4qckjZgi19}@itL`9d1kWd-dO|oK z%BYculHn1FnRgVMMXk6jO;#|jguN`Tm!o@_+AonGH-zZLIAHHUNo*8QiA%B9B&Ql- zhpAI1iEkEGdS0%`8@9(&`$V@>2Z=j8;rF0W?E9g9A|noi@LraqCKS5^k$w}5`^qp0 zhN*X=x%wsSASa`gkXNx;<>4lbHRWHB-{M*NBBjx*GzFTO4ga-^RZxav1i~bVcy}d3 zDHHl`^yG^=P4YU9EDG^`D<3q$q7A``c%#a_=!saTC_9ywl%cuKXe9CnKi;2Cm3)u$9y6STHl(34j~7}J4dmM6NIh?La1-~#KA=({Qpzj}>l&1dZfTAAyc6y~$ zwh<`JI&>xO6&57VgLgC8F@Vi*^L};2x zB4w;q7(zWIM3%}R<;WK_qU6B7UUx`iCJ8T<7SBg%v?V7-zlJC&N!vb4eYiMvg{Pw- z&2f_uzz#5c62QtMN07XW!&Fh2Rs=VlMxPZkY;d2x9*ISEb zp(e~`x&!)elF(4A3YqbYO1r2K(!1>-?m_T8Feu9lS2za6%_UL{R;sE*HJ(ajb?8;5 zBDzkVQw@fjo*O>0M&a7h^VWlaFuK!*+?io`wf6LP@dc0i6&`y?ui>h#`4Jp<|Dmcg zI^Q&CgxKzwJi@v5GRmTG>AYDLOCulpB}i9P;6j3$2GvCrUMT2fc2jB}CZB+Wc#Sw~ zQ(Lg>lIFZWM6M^oNCxi?d`KhC5+qCJqxv3|p0S7U)|V=sWO-L0t<5l;SrNEZ&Rjog zT}iz#mVg>m?s%KlMXiGc)4*Iyl^z4~Dv!`16Jb5mJPlc+azbzcUD2C+1TkVxgpq`> zN+#x0Sq)2ZrD76OJ|giPXf-m}c3q0u-_}z+D)@P$$SD#_Lo%Zs;a6wjK55`JXh^c! z165k8%oI&Jz(YHYGPvuI7{T2Jg1+Bevx+jrZUWCR-PBEBINsJvtzPDJI~qdp0g1gD zt)Jjf2HoN>*PwG)q*Z4!yYCHTaP`%#QyCHxz!EwJF}?`r_ex1_Kj*4=jN;U&nN+?V zZf8DE3gJS^J*E?ZUO&>$Q>oViiNT-zTDR5R?JX|71!xJ((yt*=Vpr7ms+!ncMh#>< z3CbPMSdfTN8MK(RR#88QVMQ|PdL?vo_tQ+kNQ~5MwDd(E*%KEqL$x}EDEo`DiWL}G z1?&(G@SKnrytS7!4AD-nr)QdFcz>e|AD;_J#90o+#}eIyi^AS$1^^=<%`VI!=`Uvj zz&0$)QVl^WF-tsBa$eo9?E&cLaAq{K`{|5a8rNXR6ZN6=eI^n za|@St{GrjJS&$Nfh^YSchX&nT{ao+nJ}!5w1?sG&Xbn|asXR}|Wc`@LBq@9pW&t56 znP{zp^%999oW=m>P;DYYmPSO2Rn;Y{g1tQfrYb6kjzq2K7=2kdibm9c0x?uoL>=C> zE<>vGl+o*u*$O9wv^6m{ks!VA(sTi+4GP8pJFgV2#L_#frXb&{1*1zXhalnqcC1>^u&MT!O#eB4U&A|GYV>6Ka8S8XW2 zP#4ST7^_n9a#y5TJCh%JEKPiHz_=lFP5KZ(fU_UbG~MmAdB|$G z;!^WItP-yg>2~>8HvZSRUj)YG;+o!L*VlnaZmuy$w!Yvmni=CA)@3 z4*}+W)SeX^LPyBLfkX|&Vml4?Dg(60ZPMjx164q#R8h1>ZkaOa(iu=0eQlK8PjE*> zDXbxD072wCUKrLNsrb-)f03~StwP+7>F-Me=ki#@T{w|FMKa$0K7(4*O7syjU5tVJ zun84c0Bh0gNV+c#8*0Go4uU9)ZkS0Qh!t_t5K#`cvsD)P(&*QLAXDd z=sQTh)~Bx|QUj!GEc2m;-8a2+seAlvwCFC)YddXgwqhn}V}eQ+QxXDd)-&y;Vcu((IC2*?^`(eVCqA zO|q{i46~T6thQb1>sF!R)r`a0qDyrIDnctoH8FYKw{=`6kN_I+lR z5az9i$mJ9wsqLARZJd%l8WKpxHqCjh(zNDSo&xht(}Iz3+4!@)2Y*!Je;WKzkDULk zN8cKANR`bf$FI|)0?gYEq;Hf2>EDWS4EXrKz9lG>pMF|>vRlF#eh{9FO6@iw87N;lfD`Xq=SsHsX47aLK z=z-GGSV?mp>F=P4VkY&PP1^2>xy#0cmm`qu!uAquy_{{whc;DP_NABU;POm_QMUJd zZAi<8h%lN+;g;+@>E~S{-ld_~bL_O1H zf`MaCJ*h!EB4FY+dL0_3-x}yXRxn`{s<6&o#Rf@FjN791w!Gp zg%JA4_iNIXjz*1*aRpRN5blNXtz-h__WM5(m|LqCH_YsYrv9_T6~d;l9T3fTeH{2$o42hfQUMxJ=hA;Ir7zVlA;8bE}hV?lijLv$lqz> zq1+#FmYB-$MbXm(7eKfsBuDB;P%~UBN0M0|4v~F8h$2ny+w4(kDEC=E-sQA2c4+pH z>)xA;bX8++*+d}WmwJnp-vw%Z7AWG`6IThrpI2=p0vV;x}Jir6M0okEGnY8J2FxENKnI@sj=8fGZD61_Mz$gLJi~?UX zB4^T5#qze)4JM#WUwLGBIvrokT7(?5?-PV|dX}c=Srj1En@1L9vSw&{@YeT1f~$`E z^P#3abHLTnoGtB2^Qfn!)|nmhe?1?)lG3_Wdu&&O3i%pr_}rET8VxJ+nx&BNjylw0 zBG(%lbVetBQ!I(*LJnis^U(>>-Mn)7v{Ncw*FF#N&ol?OYpxgFCx|la`u)kmk-DpT z*(!%WK0oy0O{50hJt}t;U_{e)%Of^7Jf#Q3>UTZu-lkKX$-dxZA`3aiNA;C~eV3{4 ze~o^Zt1IH!7i=DG&(r3Fc~1(uMmtc<`O*%gjQIq@>r_(yXTke4i7Z!_#_82pa1`~; zgJeZ8>z=!=&yO4%di~k9_7K^)X`t2j=PV6>8gDR^UsIW#6xD8)trX8Uqt0Sv%!Oju zM)ulNLk~C`+v8S$Ia?*jtK`EtBPrpVPIbOs@p&$m>VItGzV2fuu9cEoL>&D*O0~q< z`HVPB`99~f;WW>DkVebV-#(JQ_3MR47EW1+MLE76#!w`6=1e##Vnx+0=s85!H%zdI zJmjOzzJ8%OvKBsHMd6BnG6j0wu~RoS`Kaf+QD)zhuk&S$W(z&sPvGlo{`e6VYJmcF z0(_1OLI{(p0YHz~RX;X&&)wVmL!wgu1CGAMLXSWG91{^^C*aJtEzeh5^k8hZAFddD zg5aYW9zB)jN$zu?#InynMCM1&9LL~DJ*Izq@`9j7GTmV;ejWmi56B(F{f2S(3Gyei z@0_o`eBR@FY{@9<_``-vg<&0+v|d=!Z863d^nCafL8BIkcur+uwhvuXD(D9*i<9K} z(;KUl4}g#MWnKn|rXGC#fH?QI@@Fu%1ICb8ajPiXnWg3XYkv7s^UI+(+uHhmi=Mpp zGaG(WnCBi%sL&_dK8RW;wmqs^W=R!T9sLgtLbdRb56$XZ^hMvLTPSt+f97{&AvFec z`~Uy}g=s@WP)S2WAaHVTW@&6?001bFeUUv#!$25@-?kQuR2iY`*(cS)f|jCUOO@ZR^n z+JCNm2^FYT&7s5D6wDw~*oj000nXNklDvREV#i-(5?*a67+H~u&P`00-sIRjlMUv?c3f>_3y^8N8+43N^t-&q5sEI+5z z-n;97oaNfQn?EJ__eUSDjn{r}J@v<0Ajb9{d`8N^AO7%r{PsWpLks`_|MlO$|IWiR zQp)Ii3-2SWH6IwFLrNu(Vic=vTf{gzbRCdV#yC2plyC0S_Xa5e+ZJGq?9TgyzBlOh z-R7L(eL{>0jigS7^i%K&3KbX^B)4MIrGeRw@LR&PFph;1u|5)Wp*=g|TD|2R6m z9svIO*DIDKU|9mTEh43ib&asrmK2Z>?J$nT%U##OIa&14C@INCLP%KFfb*$W%Wqr6 zFj(ZA@%A2IO^0a`kMVvvx1m`_+UeB87`aCsO~YVeOowF&O=R%*%Mvh6ZeL%T$h#ePBu%F(!nN^gCl@ap!Z7)2T;{Nf$It%&HD|mqCrj z9P~b->pC@T&cNy9Fikx|NR^SY8itz$0Ow@$F(%z~*L5(aLyQ^56bBVNkO-DY?Dn+l zivhf+MLOMR?VT7ihQaD4`PsGz?~{7`BivwqdLQxrzM=0ehQXlkt!^Y|U|mCF7;6oj z>#(j7DM7u+aJSar{(jW&0g19J{5_`QZgQN@ zJ;uS|I&b=W?kW4PA3QN)1DSaE8V)TZ55z3Pm{y4 zMi?WZJkP$uf~4h4ssvG4Hn=Q7H(r+|BELkJuksd{XW!(eF&*aFH_xln%vuBQ1$&G! zczRxxC>=|nwFcH2tgGM;%d;49EC!Gph}YhC(8#%i_g=%uIiq3q_8u?}R`DW)gkOKX z?qx?jN^dbNLx?+W0O#`nXAR!oHvk642s+1@VQnE*ACWHaCH1DsHPs#a4ooO{fR$2`xi0ctuMg4F$Ka>zMrc`2;UHr}~jeN)3p zZ}~GZWuCn*lIdct!TDUc1Zk}Bi~&ziGn_MsF>6FIHR_GWC>T)xj<}SwtS82Vm)CWt z(_?D@Yh{z`8g#>Tsj4>|LSbr>i)}e14?lWf99@IXyyraoy|fl@#+dNGe}2Vj>h}~e z4Gz@z<)%-kzF7nNhC1}+`+Qply-&JmMnlcH>k_uEVejU;#?sD7K1<`2_7KM&?og`D z4wComd#jsaxpmIs@nO>MyRLw$AiAVn&TYvf6&LC)X~t+YMy{(#ojH1`)K01OFE5LR zr+}Q@tty`X;oAgjjk3P$)x-OwVb7GZw%eN!qh8DzxL!9bOV}Bq^aAS|72vojR+oA9 zO0HgCSN!tpJFeGFLwp?V!EwO43MRyu;C!UBxIlU=M!D{9PfJ5= z7>3+{wZbPtNSJ4@=|TWJpNBoi3kICea*xxg$J5i?;7N@Er&Ev1WzY_<4tE+003IJF z{l4#|dM(RxOo6C1SLm+8G`R+>#LD@m50|hvy6AcKczxZf#z?LuyX0oAmA_-^5&VcJ z^X#>@8&fc3S$rj=WHmH@rtw+#iDj&Ak~X%a0NY`PV_iep#0bhpMR-OG!Vi^cV`K1b zi+Fe#@%FaDm=d9^Or-B6N)PQYLu43iVNLr6Xo%WC)@F@{xAzUx=#&VttP#EIF0C$q z+#nY!*kiG-5z|y6gPU+petv&1J$~sKI!#|jr`$6&st5*RDj`qWqqb<>4jx(KH_rj;hZ>< z&al?t;eN#b{rS)iGbh=#v9z#s)Ok#?OzD!Ol5RNXs)uvXp0H}ebwh% zYw*LjQ>E*dG8RI@Uw(O4!b2^OUZ?6@7T*dT&Xozz<=m?}#WtNG%={L*OR3!aOVulc zaBBeR+IXWfD&s=9;dKp-Ai>nA8=1>_K*|}fudDtY5&3-P5Tt-%Z4hHRJUuVEmU;GA zmY^I30KR>j6pPBhru%gDy6FZB8A;mM)Qt}7yM&n2e4-)9*M$ffJijd8ogu7iP@+WE zCuJ+`T0vCAUe~A`h25#t^rBi3N3NBl{#o^EMq@|r%+XXCB;;lmSM&2SCnz~jRA!FW zavSFlV$6d0OpiJZq%;c7P^KfFuVr$;C=S&CURN(Vky-<3~ zt>@|VPR^=f(n8IG2DIv>RRr^i?#r^(%n&G|reN$j7DsHC( z408Qi;jO`qZ8(`8o1f2Cuf_mx@pJ38s>2W$H|}$iBYDrJBv=9Fqztadus!HaxZJbUqL2Ja(w!JOg7SvY(%qmIzr3 zrLo_er>B(h^0LB*i1Vqxndg!CNZFGz$eEpxpE7>0y+?vV8q3DWIde05FI>JmQ~~Uz za8RSsF~p$jb!&#Z4H04$_($_}YYiSB9o9AO*9qsp&K{@9sUqxs(fL@P*&Ecoz|pHB z$@Nvc`#E*DHcl;wFhW_0_)OwMLxhgQg^%Xxd;q<}Lrr&=K>_M|Eh;Wa(6Kaqbh?Zi zJOb2b)LoIrpq;8HQ~^V8alLw+PQ8M%F;-Lgh6t6x#K-gWqXNqZ^YD{IPG9xguGz>( zliuUVih>G5zgBu!9RRbpzGSWTAcS`NnSrQ(D@~Wx|HqhHQS>M$e>6`oB*^M`sSa~S zN4;rc1Jfg=47c<0c|W>V?$O_X+L56i4JA{g(j;-7Ns3Oq34ztjC^>3~&}Zi9wWL(n zeolON+@%OAraBqg;d=GD_*xRLM_)wjzBiv%cR#`a@&X#ENOfM%C)|jlABAThU>^<9$So_S5R_<=mt1t#+Mkx=7h+6wVOjD9%9@ zuBWx%o2N_O@POiNn6lM;fA>xKs$1Ur6bD&lJuzq{rVE0y=s*v{r+KnhGtWSmu&(DjZ?lBE=7O#*dvtz6C zt){HZ^B8S&YMcup8DY+Urke;0B21iwv z50$q!n0HY<$ZOyBW3Kh__WHyY`2f%N)?s=y+V#5Obn3r!HrX^o)}rCb@vK=<%g{8ui+1GPiW=x02$4R}# zkrc!3I&aM}wHVv;!p@b~ZulKZP^}Am^=z^kja1Tpic!iJ1yUYZC=@p~ zbG)Mlz;LeL0r=|K-&4b-8E$s3(CWgb_+3gzMK>ab0N3{z#}U8&;t{-ubxuPvgn(ff^lzPWSl3k-*Z2L6 zg#NL5m5@@xI88_~Hq#^lKb}k-5JJGZt}2Rl|L(8P{~cWttaZwvUf=)#002ovPDHLk FV1kaFt{ngX literal 0 HcmV?d00001 diff --git a/src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_highlight_borderless.png b/src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_highlight_borderless.png new file mode 100644 index 0000000000000000000000000000000000000000..677d04c1f27e3470d58ec161bcbe8326fe859784 GIT binary patch literal 14008 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 D-GY*S literal 0 HcmV?d00001 diff --git a/src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_vanilla.png b/src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_vanilla.png new file mode 100644 index 0000000000000000000000000000000000000000..5e64e34babe124f2136081e47f342b05ae85af16 GIT binary patch literal 8373 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 LNkvXXu0mjfO5AST literal 0 HcmV?d00001 diff --git a/src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_vanilla_disabled.png b/src/main/resources/assets/codechickenlib/textures/gui/dynamic/button_vanilla_disabled.png new file mode 100644 index 0000000000000000000000000000000000000000..aa29d9977b7b451e1f3f1180ce0538d538eaf9df GIT binary patch literal 6849 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 literal 0 HcmV?d00001 diff --git a/src/main/resources/assets/codechickenlib/textures/gui/dynamic/gui_borderless.png b/src/main/resources/assets/codechickenlib/textures/gui/dynamic/gui_borderless.png new file mode 100644 index 0000000000000000000000000000000000000000..87a64d8b40e597de2f289eb04e33545c063d897e GIT binary patch literal 6281 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 dprjisR`Ic_wPKxI130P>2RmoV8C$=k{{ZiYXL zaB^>EX>4U6ba`-PAZ2)IW&i+q+O1e=a&x&2{LfS52;7(BaIMM>a{M&FRkFXyOjWG1 z_56fKf@q-8km&#U_YnWWkMT|l>q7L&dx;;LY|*ZF7yB8h zam(P{%zr~(_8IrxePG+G@_Apx^@)yaq4mLD(9q5r7wvTt_w~H5mWv|wdErwGud7AR z_qxv#@2?5G^jU=d7w-)l5T)i|Wwh=_8OeX+^uWyDj60hDTG%OLaBWNycymTCPY{rg z-X1z+cN-+aZ=?}hyDA1L>G@HxcK#%5g!-2V7L){W24Adk`F z7=4~DTaSqPl~Gdc=+HATEC{*b$2?vEzYf=dUkR@$v#bE8VZ(w~RY_@}GK8|KsjPL!yIwI zEX!P?ICXQ&FwO>c4)3!nln8AtfDmDC2WON40@5qvlvgrn9h5>GBLK?qT%$G$5KzwP z1}C*O#%mFD)U-FA35spC(_YhUa1(%V#+v|alF0zeT8I1zHX?_zGtN5aybCT0mt6M7 zTkpL0!4t{lV1f-U_z*%2DdcFPjV}5aVvH%~q>}*$y5v(xF{PBV78x65td(J(ai{2F ziY>1A5=tzoc#VFtvsRX{ut0!V00n8~QD)r1^j zCX+@{I1MKqVI!I%!~mh~O1q&Bb{Dz7g`05kZ{eo@7jlB2`+p!O0NqdAenG8eCOyNr zQ58~5r9t%#8Aod+5Mh0*)y1z{-O@TE(sAwD6wHiqt=9$il&cqWV> z%y1c@^hOoBM8(0z8@PbpG)g9OlX6hHjaEnVu@<@)rnSeOojq!%91~evgFA?pq)h?{ z8jm=3H;3+u8bEI#J|1y^@3@l6jNp0_qxOVLoF*7MbVW=GS*R3xSx$P&bu3vsAi`sN zgX=>wBGBIEc;R@#fWWK=J0&+8SY7vH)UU6tb{C_G6b`xn`ICV)u`=7K!fY4fWGIEoq7j8DU~Ki~9)z6J*C zDD2iY_O2yFfl~9$?L{=TV5zTcMU6&*)po)Ul@LVA_n&lvRa^990n_$X<1^AeB22!x z?Ba_LX_xK{8`l`QN86~m9w!4uVv-$djdqTt!H^v)%tV-v5ZQ;;DY9X5H*<zT@7GM!w4}=PHWu0|_cuFmxd=(1KFc3^vF3BZlvq&`0HqGuT^hfkaK{I!{AH zBSGWzv%)VwYIq^);$mdWS@JejZX;o~k%s}K#^*J?R41U0H;F!UDsSWgD|#wy3B1{4 zcNl5$UZ|$t>X%P*KQ(aK+JZmOzGB#7M!W>FC~=YpRkI@3?cLd(MwxaZS|*a&AgS>W z8ri$dpfN^0lf-&TGkj6+ExHAg;L8t;whoCOHD~1Z<&r{t>P>Kt+%8e?3HQ#u`JT#L z&f;uooXrnwl?LgT2*H|nUR#|}c5yw3isw`5N5!-!Id_6*F%?g`A}?QGk`G%1Bu@*y z8bI+mxUEC%Lb3fUi>m-6DfV5*Yt6SS#?du}&E|3%BscWkB)F8816P0nCTQnb%8x*p z*)Hhv^l$+B@D3>jnI@Zz%{!ZHKr3}fi1|7TUSKaptY_5`MUkYGtK@Uopu@OT0^zGV z^yW%oKg7#tKiwJtGI*>Awo{3-@xuzh4xh!-^Ph(I)!HaO+vU|nB8%BOF2zA#jjcIE z!BJY&ESflkF`fKT+R*B8@(5yyPzW{WfM1*Qp;!`Y&-@Xl?pUg0&H>_M5C}vFn-A*U z|FS~wzW(GtRF1ffhV*5)!gvl{haIPezU^Tt@@(q#ORVNPzL{pCc&K)KEJjHQZlW2a zt+_%_b0@>ys(GV$i0MP$uY)du-zk`!@0h-Z__OgKUD7EGJOL5uvd8I?%%c|sk3du! z_Gyv4bR&lO5Dh)Zuw?S-Nt;K+?3AYqg9yLTEH8S7Y|HrOMJ}Q!Kl7mzLSX;S1O(P3BBFT8H-er-|k~% z*nBPBZSJ!~F^Ur1+?m;V)70)8)tFNg|DGO`JHX* zM*4HSxC&u=Ag*kZA8Gf&1fNWGz7e3bWX6S|*p|C?!WSK^S#dVsSv(gKD&N%}1Woc@ zruJxU)?U<_xs2M<8)jzm_prs#!B#wd-2W!t%WdLloDdYySKNK+>xppniFm1?OKRw| ztF}Fk^VbrUdHIVlgXu<<*(xALPkpP(Gi~86&8s1`sRJzaS(nm72EWpM6^ghMJpL}B ziW3`b7JsNX*g?c0Lv^wsD$-G_P=pGhR%q41 z2Y(i;4ld5RI=Bjg;17tCqm!bGl=xjzXc6Nb$349Fy)Sp)0Yam~G^=X@&~)2OClg{e zw<>nOB8*;|afD=M8FP}9g75gcM}Y5lF`ngL_vh$W^A-aFBJnIUOq+OvcxKZ!IPVij zSV>li&xyxPx*+i**A_Bj^56ZkBXg~#>E+L=^>Q;a@^*p8DM#P>Ozvhf*X7tu#Mnq|6OQz{xBO-QLg?Ho~Nz?RJ5P?a${knxeYR#k={ERdjzr zR!}fi2E#dL^ww53uq$ma@u`g9?>cZg3#`Rv#?-EM1yJ9=X)TAX!2xs5bv;{wvxlu~ zxvs6}7$Z)btZop>0$W5pY}S68RquyXT3I)Wh-dHXGc?OX4_;bakBFz^iC+3^&Yv`; zm;TW8UUR-5(kZa&o!Wd|q)Tt7JVrZ#~eFVLpk zGv8*2H(!dX!#f14c(k_MWePu>+Rc;+hb+ogGi5<3Ks016ht|dHaw@Q`Y*iKC4z#kC z$ovWLnePlxwE}AUo->X)0$d8?J8)As> zI~2W}u5AD{y~c<*=8UL%yC>535wRY=Wkzc)l5wmmUSq5{aJxW6Y<~w3>WZYiK_MtD zP~tppjNqL_)IDX2Con;1hf|+Re5cE`4p~XJyEyY;rww!x_`z_vD}(H^oUOGtUhp;a zeOqH2E);;-GcyXNofp7!0$~{xEnuOS<0!wQxH*N(9ZL%FX2=D*zGuMY;wAXvZOw+sC3@%XrL*A8-*?E+KKb_&88qnCbW0Oh>w%FmIPi+0Ie zRS~kTUt_GS6UyN`;4o_iLh%L9Iik;O#0zGPB4X2b z{4=lRya1W5e_H`m8DYTf5|;Qq=fVm83}5bYfs{!YWTedKp~X+3Cj0ug3Z7D=-&%`b zW4ssP8!Fm!aei65OuWmrrhL{9NnqqGVAnDVJOzZNTicP#)H z==hMy&5Vd&-W5k&fr=BsXpx;JF}Z`T0gyosb96RI<^E&Nh4o3SI)gZcYVe{C{bDu6jTX> zw+L46q*0zq^hC-oSj9vYP?+lwSc33UOH8R9D z!MfQjb=}k4o?*3jNmiD^0Gl2%Q3m_x0~uli3lzAoOXdzMHmq)^Xe65m&9=kT?2`>x z{-b_Jn)HLD+{(XJYSs5UI0xVNKuJE`>;R{qVvEO|t4yDAj{^l-Q#x*_C5}1cTeZP* zw>t}{7Fd8q7u=A)$<;p$FYyrN{4FeDQewJl)AgUdugKg0UOMvp$BU?nNAHVR88B~O z|4Z4y7Up%QV6XpV7HVLL+5q8bfA^nm9@u0y6&{1^syGB^id}WQYXa8|9qjC;MLGQr zl(q>p=+bz|1y;1zcOPblK*70|aN72ZTK8eQ?#==c+y8!j#i8;x0bJ_xU~Y}2-nzTS z8ycE{ptia)D4Pwf&EO=eP($2h1;42eiWovachhrk?Oidda&21Yv-cQ+@5A5+d}Xr< zdGW^JuF*WO%@ZG11>O}O>{kT6cO=Lc56e1I@|7Tme#(66lA%t%+W27KzBHhPahCR z1S+XLGeRc)7vY4Dn`JIsUl+a6M37u~U75h-sqx0Tm!kZR%wwp%h+3_ce5U&s&Fwz9 z=Dc*I!zJ{In%Tul$}BsC5N2&2`q@MlU=3l;J})$y{cbRGQ1GPm*ig7QHk?qUOkw^+ zr`MEPUbu)uE&D~?O zeffA(3Z2RbtN$jm3lWhWH??aO7Jn$h4}g_*J;jbdwsN<)LI1b47EfJ&%Z$_ucTd7G z&MtOV-7c{Gb>H8x&|nDFAIsrK<~yiBIacWUCS1T6-9p6;LA3y?yok6|YHppmybVC~ zmv)q`IR7$CpJ1Zg3H~psT>Wo$^|qMQIpbTYAuK=w19Z3`Vn^8N%|(x*_IfDA4I@co z|F|?ZCiHtOkpq%F>w+kl*g(r!g=&%3WZy6%{v||ecb!U+w+sB^`CJuNN12!qAguh( zuIr^D+*^x+h_Xb643U)si*R{7FQlmNO$kk2%3}x9&TNA^gM#Qy_c#HztZ`bM;_orm zB8w5&Iw3_=lc1&4GErMMJaD-bh5%x~eHwP9?tf ztEq9>=>z&c3mvRFZBm*WJsNbmet5YYb|wRT7NGNP7x?q}T$OKbhzvbyua_?~Xh0nkmP z^|)O?BQr(CB|0sPoRpt+z|$);V`}>lO>MJwyKSJ)-=GvNl(!!bWg-?J$3fRb%mnmi zUx2c@&~*wRXjVmIied^1*85V6_yQ%6aYAC3mbbFIi$(8fx09M8!$Tk4;D&zCftH)I zg0E%LbHy^GzD;zU8#C+Pw+qMxPAIet6+5|k{rLiE`cXG1U^$!A)jAN-S3q)m3=EFD zNrx0@dBFzlBI2ed!GoM;pA70u=l>4ewTNgo%LakLCdu2iHTLDsD`=`f&u^f|@IX2| zA#dv_P}n-3_}Q07>bP*Y=fnuk3DK5o3Ag79Q)7X9vE6^WfM z+SOq;do(aRWIbSm%G1E6l=@Dz*mg{tofpB?hlI{G!L1a1JXUA5cwz*C#)j@syK?Rj zQIG}Vn>P3u5LHs{Mip5=1LRp`LhFWZ^b%jXuc6A4Be@=&TC5Y~I>WWVuQ5Jb%re_e zzbD4rJTJtQ)=pNAYaHq>>>yE%6 zk2&A&1Uc@l6y@OLVr-*CYjhqOhLxR2$^b}_RJAapaZq{9@`Io6m>XI$Gks>Yqpsuk(C|#8zyb;S*KK<@ zMa=$S_`#i9Z^Fdc1OFN$&N<^tplpLIZ$~+Pg3oZRQP2uGCfzPz7vO9z7wWfZ{O+=K zkc>b%+|fAoFym5{AuCvpvthRjy#Y^t7g}QjX20CbtdPfH&GnV&o@RBcVDMdL=wQ#yF^1KuqJ5p3}1l>gw^gm5z+t36_6eR{mGd>Sw z+K_$vW$4-*|C%*$lO=u*ck#V;-FLr;IZj^ht_4sFB#1>GK++L`1Ff!)IRgB9;j|{V z!pt{v@M=ZnIpYDzT)WKvS@J>=M{um?H!}dg6LJKCE5+c=3p{b^+AiI_-`dL{!ZSYb z=K9I8E+bYr9?G zkA43sH6dyw!7yP$fs%HeEu>n2x_qj)0{s2Fh@oln>tGX%&GW1hntQ-fcJ>$|K87Sa zPMDw!uQBVf{rt4b5-bku?}3XwR1=)&4o@~&lrdZ)C#7^&fIw>J2Vurou}5NvCBFr` z;{xl^`vvLYP~-2CG+@FTlw&+@%W{;j-dEl?;c#t@xg7rXG?&Gg zE}LQIo#QYU0`#L&1Q%I2m&Ae5#7g#>B< zGqMxuYpFJIhS>l!bbvKldLzdOut`faf!I|MvG@Ky7ZtL-9fQ_4BV-pSOj@P8FJ@A} ztv8IK9E8ZN#S6kk;_?@op036w>Wsi4Dm=-;*67*c0|zvAkg)-X%X3Oj9OuI|^ouaF zxkGCcyN;cuTH+g0;6OUDbk&)|tuZfZQwH!!6iu}Nhb_h>#+V9QD$w{jF3&2KJu^<| znj8yE;ySAxIM5}bO5g6Iv>=l!k(3W`?+2o52cJhZ=?8)0pa*_2F1Xt>hO5j5FsjS= zyY<1YN8%#6xxNfDL{E_-j`SzzjZtMT=zomy_G40D5E^x;yBpGDVz7b7<0WcsgBc@1 zo)|$!=s@}%SjQ%8Pj!e#?KU=9F#bQm2zzjXps8PKjoStOd_F&ABi8+*(>l?w_rdBO zTcJ58tb?Y>@^~QM`|J<5t&B+qPJSTaQkQA6r3=Dv6tRLiN$Q>EMG@BqSQQ?ix)@-b z1#FY`DR0L5Jj~GJYQ>38asDR`OER86F<1Ol)bC)MC+30XLRBJ$m|}i#V?&E7sy(5*}&I(3grg~ZQ#+bN9*#Gb|0TNF>YsGV2A45?J7rkm(5PA zCuR>i(j`y2o=~p%;RCb74QGAn%r>h~N zo6R|09HGxv@pB5i4?A(GyTNrj&Lj5A{OWzBC%cTFTPPu6g^Mr_V&FsmfuXlA^-F!C@%KOKZGcCEnT(AoAO}4gB2}4vtXLTkEr{ zhKUE2qw{x58+!{XS1(5kDhp2=OE@^s<*E!DXR_b!A+KfxMpCB8sX<&l!eazYx$97TRD+)8t3q2wYV48T=o`BuFgWg`U zw7>S=2DE<@%kd)P@OrtrU0Gef6)=kHqP%P0xDhuN*n1|@T@P6gxMMxIHXktFzy08~ zwzlxEO-6I!S*e^p;Co}+BlHeZ^z;I|)O)bP@%!Fs3`}Pn8*FQL8Ts-G*%(;^cfAT2 zIYDo`A1iO4)*fx|XdkXlgMU0ZwNKn!&S$Zb{${$IgA<+ zB)_G9ziRv+q%gz2gw7(t{BKMrT#migYeN`5jeN*90!$H_aUhJM6Msqh~L-VUXoM9$RTD>p$q z*jsn6xEG~qtf_K_RIX#pL$J$o$WZxa^W&)d(@bOgdP%=rFcqe03@cDA{Fp4qt9gvB%6JZa$HO0l+| zryyHeUh`9_AnY1(olVa1noHLomA3oY4}W98@3>m_={|1*V{(zZf=zB#W3^3WZMn1G zsKb^`!tv^#r^on=F7BQP1I*#Ra{#>xqiIWdO{}?>M)>BJEcBxkd1m{+U^fqP?_~mw zYM-mtMbZkLFZk_QU;Kc>COFN`%`V99Gu}hhZ0-hnF56LJgh6D&>cfDK* z6|;LL=mXV@k9!j~=J61eRASfL@ErTOX{i<6;(HQMSM*nIR{1kmgYEaSb|TbrKE%5U zD4_-k+&QW)GA3o;qB<<9OwC%I_T<^FMJ1j;9PFwU)u)h-+*FY6-4F47-qwZ5lAA=P z`!-$^aM_x9;I`Vqu%g6#=AXOMY~+}i&X^s{>y0}ErZ)oXvtVC_d*=eOw6BRr0*!@L zL=7oiP6-^+cpbUDKdK}Ise6M1Pw3-4g?%&$Y*aShoqP_N>A$*~XA<%o_Gz<*szWp- zZ+MqvJiJ(nXXbD%o5zXz#^m}XbfcJzE^=8{ddxkBrUMnTOQ z<3bJHwV>?zMfM`KgTM}FyjvJwv5#rT^L)P=Mo75oftPtVS4}E2{pTO?ol3E6FAje! z;c~87i5ROvRonn#;o-&J*`iPv?=7iO!Kt}8rQcaA~9@5?hcoAlsbVHl}L?&TE8Gj`SSpKR$SmpZ2d zd*j>(U8ec8)Q5WisTZZ1H`qSL&}15YB6dAGsU#C2N1oM5i6UoN0tHhf7ay~spK-61 z5)*uo(Lw^v8s-+E)zI-HMT%zg*E9*d<}V>iVMkk{JZ;K8H5IM& zWwYysvgXB#7bK|1@&hb5g(Yafq>(s z-)K)K8Xdu$%c{9*FsRRWydRDGe>H*^LmbLIQAj9lI-}F%g#yCuZMwevdf2s&I4wHJ zuFJ|Q(d#K5papHPBO;V#EfT%2;$s2w2#T4 zwhxR*Kki$xnDWR8kB}~*b|i7pSy^JGl(Z$o+{?qLDrLqOMCBI`EPmZf!zkwdk^E6! zp%$_4yRsvSK5WXyIDOcE=n{;}@8tA;=#Vj8Jl6J9uj~i!5m@67Jm$BHqkHh~dc`%f1VmZm@o__6tHr7Qg{Ic$0vR0!G&35WA=U5|oT8wh`pnzUxAB+I!X2!WIwU1Nc)TGVvx^qT6kW`?Gd zH4c8;1ns*tdKNyMGJhplyBRyLl(|sqQ-w1bDn*D$CfgYAZw}urwPH|{4%~<=zbB_$ zBo9nrqf+UYwKg$vl)+>kQK@1-mFDn_MHOz;9lBFG|GxX6$aua-mH{`r)n!w-groh6vYCk=JWfEDaTK{D;OLgb zghHKQ0K7yXzSKj*Cv~uIU=pL1y$YUA5k`zAVlHrut$W^CYD+4*r^0%taw4*->^@aD^%Q+4rX1H5guF>eP zCo%~gl|u^zRz+9fn?3+NjFchRF1lw6(9hI8>U4{Y z5{HF;yucH0uE4Ob1Wr%uK>uR_9z>5y2X%lEhrYaWmPx#=fv8XGFC~!~8(?B~NrD?l z2cNFJV#&3s!MSLpHa|rzKH6Cgjl2M34yUwztt>u}q`A{BPt!YKht&&g{2Xxga8A(N znf!&uCxz_nNLdxBpq%lsUV3yQvMV1xC*b;ZZ$|MQT?$pQHgy?z%Xwfg{lM=f=#%)< zZC=-3eU!E*sZ#RuSD=k|>ejKKnLc}lQt@Q&lQn2xR8GC}1OE9>RUC`TuSGge(LYs! z9-^`6TGz1o&8-A|nYD?8^ZZu!5XZ+Kc#IfbndbxveWt^=&^a(6<};hqV);3Bgo^;s zrI{jaM(D4G@{tQHTmWw=wYUOoWWPxJ2~}<1pS8^MxD2~Mmk|zA_1B+PI-4^rNJdOw zLa|d7$3PVKXN^`qYvdqrKVA@FI0}>^qnxQAhUe?40*Dyp2`|DZ`B+^DNjtrr1h{=uqy!0K`LMAM&kzXJGraix7 zAgi7@q?!g^eE=d{X3G$8|0I3C8?psxZAJx4fpG*a{lHX|Hc%Y zMh)#uQ@_5=R_bK*$1Gk+BJXs|TnaZI3Bu|Tc8umT9dmD%#M2Xwdy^7e?060UMzmB7 zDe3G=vHVA}5WXS`c$RJzI~w%9{0xd0g+fFtTg-d-`q<&^gR8aQzQD0y?|9!%f7jcR zs+9(+PmWb2P{nGHIF<%p6U% z`*sDsA{Mvc<^L*;MnxfI=Iun_`^O^EjQ{D3uQ{}J?Pai{#xp`IEJV&Yz*gfqqu|dZ z0rAM`g-%=2I(>?-nae(nYOp{@ebtEv<1qxsn;I=Uq+O!#=MSyWy{(bIl@o%aiZ$N6 zSt{=i)-J=eCw}Nw5n#a4micBhu~DJ)4KMNSm3!PAZZO+7qjorUH3|EM1{>E~YK}O#ogKY?l~#K`&U%~9S*v4Ip$~ao=WzqMV!af=3r|jIonwacdsPS z2fxNw7TcrJ$FlBRwVGaArMbw&OM@U~_{9XFn$1tl;Ha)><$9s@ub;Ih>b2{Mt(B#a z4bJRS&gs0gCq@-g!xA1;}Fbgh`V^#(~K`2Wz;hshH@MYNj_S3n2t*;nuOGE`; z6#tWi-{#s(p@n-rCbMScZkVD$-o@&}&dLSH7tzg_lvkgUHm_!+H^ZK@Df}(i!=Uh-u6OxGzOmL@#E}J+%~K!g*7(E=JbIH7!xs1 zMw;mi2)a1IAf-|55aEchWh?@VpsMhAP#6GgKMO=TX}>EYv>Em|DwEvgZ_1WDIY#N; zn-fcFl*gE72#%=iHKzU-Sa~G!Ay2@G!QTWAE zf3pz}+iz;6)))d~bJ=x^R$GG4W@&7YDE(_X${WQnL~4_3I)Ws8W;?5|y(or0*4HSp zGz(~-^gRVh3d_0>}-)nqD- z6;Jue9m!d_xJwoL$sx)hU?2Qd<$>^aPiWm4g^2aJEmbkY(-;YB^27*A4pi2yD6r#8 zDcRZnnL2?@`TL$o0;0z!Lw5;W&s|GmlkqNau0*RB-F-*tm3TZlm7HiMFU*eM@JzJC zh>M(L6*GQP#)+OS5?}QW{YYTyyVfcgnRF?bKa$KSj#|BQAu$ZjM9j7Yc@XK*5{B1P zR^!jcR;-Q+<#&o0iEB6^l|7~hb|D~5m0)KwPw$nV;#T}1OnjeXQk|flzU;}C^Ck47 zC!?FXaALWH99+n^nQ90BaJ2X!z@D}Dxyx$;KdS@Z1EcSlLYyG4u!JM{`&3tT%xrK zja=UreWd@C(GCCPROw%8a&|PkZ6;kw{c~{@(pY-a2sIAx0>4W>_^=U_nuM85s&J^e zec#0->Gk2zSfngPKfzdw>{lvuyx{Vg*=mpM7Z-`l=t09Lo3NOo!jYq+^u{nFFh0Mw zCf5A;)H$@P#+-XG?Di!5s-cr8iDMeg{`e_%v}0H_lwW5oeEIH?jn46p=6ar}EAtKg zZEJpCL-tY`2>EI?Pix3p9Uwo~hzS-+xRtvcNomi|$}+OvBu+VEssRA{QJjq>e-nz#9(cabsW(ZfXL66L^G*8mi{AR!sh+(c&s zzUCYjhUc1$qlmYRIH#*zAA`=z#GkVQ;lb2^f!{ALtOIqqNRCQ0a}ym7`XKoF4Csx( zjqxM*Da}vU*G}DnSc<=FHg_9t6}dven0um8Y+ctXx;tm({uEIynU_hQzT$yVj_`*) z4zNOp{A?#`seYPsbOw z2%=Uap#%_CV6u<53bmjHZeoQe6(r?HCEpopSmW%F4~Vwkc5rxXz)KbM@K)FXCS6yQ z8SqT`o0;9aXrhfWoPy0QNn8?>17UZqaz9`mT-kfZx6lLKOF{$y%{eQNa;9gHr{xU4 zAPi6Lvw}{AP|W!Ssffh{7;C9aN0Hd>>R7jKf(a@_wd)< z&k_Wr<_lOaC|y2w1jQdZ3LOE!4#DvO#>oye-s9LyzZ1bRJgPf%gF}aaM)+QFg_`EZ z102p;3hArb=?saIgOzxg8}*Z$0lE<@H?P;Lz~? zYw&Ox5aPE^6jucm8I)agOaOymibM0;I0}9RX-O^5<$qavadblo8v)-=aYz!EmXR=wTM7f3{GovT@UZ# zu*}S7ta|rX%%g9_Y5VEPJjZj9e|ZyrzclmHru5*n@DXvxewC>dNGekDG{rBGvVeCqG<@mWpnIG0tcMtCTFC$LPw$2vQK0fWQgB!)}nI;Dnx8LR^Yb*t?N|>pUPi#R$w&>UfPF zuL+lN-}iu@$p?-8d}bLRnB%Y3dDMQesb%@B;bTcZr^nn%LraTl-DAy|Do4X5j8RMU zNv^#jNu07t9 z%-TU0%)JpUfxK(n?tlVzlfxy()3K;I5B)emN&D#~ULO$pqVwH8RPD6J+V>~(yoUx*LDcXHUgQj35HwIl5(k<&gyL|*xS)%Ly|lIOz$M8o=h+~uc~tPYkB(&; zNlx)Ql+}a7=6F(k+q4Ad{rqd6&0WzY(5a#cjI2U~cr@*Y0cy1lckK-Y&yL`P{bM$z zX|EsIzym(5hov%8$IBK;%ZhzkoZJT-0M6YLl3`v5Rc%%g8mX*+g#)f zFHT3-`ultDV$5#gTr_YED){{xQxb^|Y6nS)epzGU70-g(RS_rCY$*X&rUFndNS>o;!^b!^;Th&Dg7}$Y@{V*MDe_flZOwnXo zP^zBa5YGU;vLkNZex$PUhQ2rgpI)mAZl61_OH*)Z{eTPD@`qUdpKiqo0ORPHR zm)X|+F(jX6dfNv96oLpaf|)c4+kPN(!r1Ym_;UF3hqT? zAV9rBz4p;hMh_rnC7Hg+q19#uv$KabIX_l!4V?RJ``WEg>@`dQzT>M*^n|&)qh8rw z5G$Iqadekcw;&SUUsg03)@{H-jd(hckUzo)l8Z@L z`vO#Md4-?zny4bV6xn&qW|HM+Hs&&`A{9Nt6-L98cF=H+or*3s$N}>i5P8HxaEUhA z5wkT-k3|O{OQ908$L9UH_Lu6PZVPWOgx(E-nwZlQfXV_0L#s+GVBetjxo4R+mqWkt=KIm9B8QWEsdw)&<`W#0}(w09F0SDCQ`znMjj zc5a%By%tbLl&@yxewU1o9>5vgZAE^c#zT!b?k$SyVqR4bpwK=>5&gFW^0vr?7d*0p zbwKM~R#Tp5x+LCUAPLi zn}V}QigYlUQ)wJm+#rg1`_Z7VnT;--^>0Xuqv=qjrhxP8FMYo>qlL2qzshoqnk=7a zFVt}$Ia|8MDl|To!@P8sF(s-ASVG=J4+c`k^PuK*CD)9`=eYn}1_!*##mXWJlD)W6 zs}1L%5Z=8neGeaQ zs92ej(c~TTYqBqTz^3(=c(+kW(H~d}abB+5WJK#_cFpqOcgT(zQ@cRhc7(>_!Z2KJ z;KT$^mZEhVxtzwx{kT(P0($%IX!*ZI5#tI8o_5q^!c3QisdPgRJrYwS`-f~vy|y}jC~qs#m>>DH18=mW}ZKF&p&jaS-QxW<;~>n%ibAW;Hl zhYf-D12}diN=O@LIzW|{4T&*R#qrBgm5ob2*{1bNRic?Fk`f!!Mn+}0PPkWOPv=-+ zE~)ox>mIFc4A2%Fvd)EsPY-IakDCRXPFDSWAgSn4eTE}vX$DN_V)}W0V1?h5dRmai zqR>EG(OIsPHEYPHMkA2&pBJ0#!^dow=$4Y}*6iYldJ@&<9X+lV!O!vdJlAgTeNkV(oBKv$rU z_b2|HLhLkC^QtJ8a%}JHVqJI3S)7i8zwYBs`~@TTDgBs289va@eA^%C4yafiDFNa| zy+Km7cx2gxX=VjuD{v?w-CS0Y^Pdmh#lzA0Oag~aB59PUk8QrUql8E-Y@P*eo+;Yw z=ab$;cxE`&py^gNNNG+b-T^w0+wJ3^+HSkm8_~GHzrfFP$<=AJRuqWQ->!uwco?dZ zMh~(I7zy~Fx7OcGx@r4eCCqD>#I^`elXq@d4OP|AlK{}XbH{un2o-EU1fYUnk=!bz ztT|*4)TwI$?3O2pVJmx!xz=ry5Ep!&a&{qC5nXs=StS_wV7a=cx zhlhV7^&Oh3-3!c;?re%~xXxyPE~fS@f!K>`32DuXILllo0{==*FKQub*m4-615 z^<00(Kp^7cBg)NC&uZ&2eC|4Gtte(-{lQ;`M|Ro=6a@rHB#e+y0^|r}<^F8Vh#2!k zk=%|?*&rZyx5hHcTK^-HQ~zGk39V%C)c~f2Pq_ghc6Wg0Xn zgrq;gP?5|{c!RZ8LQ}uq_R2`^GD*s;3Y)R<*Uwq8!$BxyDID{?ba%1WKB)h=IMT~D zdJ>7zaKOIbU$PT$@{nDgw3S6WJG`L(JGsXJsNCpG39w7Zr?_LWXuJN;6pTN1TxKW9 zPo<#dd8~KQbQdqj3Ploy{1Fz-`x{szZhQV^lg4dPW1Hp75ct$WC9|LrhD;!(7c!~H zzzVqSwyJW&IjIIExxG6r6u%m;V-F#6ILO6bOqME5L8whcT|=cNqCGtpoKH%_RXS`; z-GKAzD9%>J=TR+k{N7(rSD`y@#OFPHe?+G~EXk$whDB(&D$86a_yGR|{sPrM!zav^ z2R83(-`u;%@eLHmol>P7wmE(^gemNYZ<00peS&;o4YmXJ$$yuXkMOHnz<;U70vk&{ z_xr?ZgsB{qJJ}CKD^TPm#Xfa9#5zj4#TK$+yYc#zHO2J+WTqE&Ex?p}IEttdNhI%i zVtdG{5Wg{O8r?r8kua{LscOMQ=RiUZXEGE)sP;-*45AK6a;13t_e`=UmSpLT@gA|P zhMQ?T1tL_0Y0G~3Bzhq zn}|VflC9mcj(1V~7FB)5$Bq_Y_15m(7@l0u2}YYy_+souuL)`^Spr#sb>j{c z%MHF7Rio5_7S>OwA z^rZ2}e;orA*sTP5ixFnWV~&=yHrtFq5w;624q0o?owusgWC?TQEs&qejhd*yULNEG z-`bB>=a^)88r{VwW-kY!=1Cd_d*v}|!9+a(QV&|SRp_qpkMuPZ=r4jat%XA}d zr{eBxlZ=ClJEGS>f!sgcD%%Z|?y`!Ce%&2YD`AtE61(4ALGTIWf=h-O_?Lpps`DMV zk-y)$c$~-Q5G!cum}}!my0E%!*;po$|7+7fpLX_vA&}URkuG7h8akM2zk(65n7GFI zbwUj7iEp;?D~%8-3YV13pgp1H40}syR2h)S7e+>80bi5pg=k4WrgwEKytIAV0kz4) zjH{9r2(oF#C1(z0_nsDTBoJ}iYT|s-X@0aKkWzo&%GD<3lwOpz`5W)P&F1Lrr&Fa% zv@RqI*XUdW9QH4DW~+NBS`F2*UyDWIlSwjryrM}CkH`2q&Et|R04;xhyp=Nh`Zlmu zLdn9R-i`?}){?MLQueoF9&~&$`s>I7EYRNN|6#~9Zv5P+wOa+aTTRo z%Wa(-XjhOa&Via|q}o(KQUOtTK!0B}(&1vSNDUfbl}<0hlyk^1<}?;Q1f1>&>JZU# zSmOWLYV2J7>pt@0*rIP?AreVL&Vkk`_Oqw7;GU7mo7SD=9qS`xiMiNn^8=!4Id&$h zfqC!2n5RamL&Lag?5yUvc|wj%^dvs5x>#Ygj(5>tmkk#!U?m}PL)PXU3@Jljs zggBAS$@e@RnBIt7Km|D;4vAzxl2)#(X;8+nGp%4WN&VpPvcGZS96$Yd>z#&T+ASS{ zlz3q;%G&sj6p^6a%tHvYG)Zz>y>kcnq!B&&=4sBc8^cs}$VSOXLR4N8S+(7M8<+I5 ztFIKH`q1>L=p^M{_2vs%OPf(o<`67PL@p_9%5U@k+K1AIK$wa7Z6O=ADE^6QToo3-JplkpJY?yL3VsKC zy>`AIX&f56d=*9XXUkkw<#RspDQB;#Y88CyVOn+tU-yizM%9QPTI(dFY0L4H=^gzX z1_a%vZJx;yNOib^Cw`*+bZ;d@odt?S;jck`Ra{ynDsD%agA-jm4GW1HM@NMef_{zB zJ?~5s=Y8Nl*_})?x62Fz>+wCHM>{)Ws@$n{NpK8#7p!K_@wwyBHcVUkr?6wY6r+JV z3V&trm}D;vc&=&jKwN5cM6k$$k#zK}ij>_>g>Eir(jN#S!>3e$3nk=$Z^TAf&@ud% z4>66b*jhj@@)#>URpd=NyvsLtxP;?#Yw_|x=2PV_Vcg#8K+TKNx6IbQqXN6)e$;9l z-{Gy(shfKm9HYgzNbO`*UgAw-5nPp@%95Y*VbG^^GA>FwpGYGI`B(Y<6u`<`dYZ%`v#gUNSx z{spP;l$o!h-cX2$qF1C!Y3qPYUA3N0qk+0h{qyFC$fcx;VN-&pYei@Vge^a6)zzb3QpzM@rJ?jlSkj{iI-U_#JfywHP+3$rWa@{fz7Z$f17Ptn(>oh(% zJ4cF@Ve5nBj9G&0oeO?-e?WZVY;St7htPJA&=Ge0Eg(yVdRekjKF>c^+-^jFf!%xl zZnuRWE7=elTrjo00d;wYJ@wG;5&*P;oWR}6kGV<_M1N*0`JW5T0%rdO#0&ZAjEw5- z+I1Z$36?r>+XYH*gzjItBq*zq;M?FmHuO0%{s4UUnvm>`vnTY*8YG#U=KyUyD+j?Y@CRRb604 zOxcnhk?0x09TI1m-HCd;F)de?R4x(O`Y>7g+dADCO|(#xk7@rxs@|L9^Ih(2mPG_$_##r8z)V4!|^HQH zy_HGS>=vt_!KkNBsd!(wC=k<%0GCU)^KIdfAwP*+GHy(cN27pbGCfM%$rYS_8;#-& z`(ZJwu_rEH_AotA#eHo)Ape$~r+yik$bU)X5E!hHtLcf3+Q@K?xdB9@!zrLgG^qLv zjNb8No21({z(4v&m$z;+F(w9R<1AN?n6m8lRErDVE0|T|E5{@;>>WrF zSa7)XEsc=beGO>Q!jSD8Qc2br3yJ_)Q~bu%ME;~(?w}i$-N6HM{}Z+NzK*=hld%(RtYs~ z*Am(;jgaOqt_ghguBmztiy8TZhO~?;kUkMUvzwS(U%iG282xSrabD@oedASF{RfNo zo~=ex2S-c>jfPU=P3{ThOJ4_bLb7kXbG1}5ILZzLs}38Nzt69kgB2AQNTis9GxU~~ zh~%SAd1r;6mc=Xm!SzyA5j-CjI|!FZ_HgunN&(*@XfL9kIapzshfdtp3R#{_&vTF~ zcwX@S*|twy6)!WjjZYd$CCQGqLI0y!$q;6cnEgh-b(zF(M#LLj=?RUX+r<92lu!0# zN-~e1{e)K!ri&QHp^?|@u$k`)R_A#~VF7)|E$@meB}`zTuyjf9u1HOElbQv#r)f(yO|+;Id99i6?Rk-fm%UcDS; zC&sy^GEHd#8e1-Egr4(u4E0wGI0V9y zFfS)mN6Zp?m*z994&9wXb0+(u=_CNu=g`QJ7iW0vPqEq@Jil8 zfZf>Hp|qyO8--!tUD;7 zW=vib4mMYCGRJ3mM6Md8uJ4h-)H?ULCN%Jt%)h20(Z#qiW9#pj>)>fH#7$B?>erYB zSg&SeOJNgw>Hr%;`>FVcb`cf_nI260g~#p$OQciugoMKu(338u2<=dvGMV(XfQ@cy z9?M$8*EB{Aqr8T><&B|)fC>W2l0y$Ro_az?E~88w=lwOoL^odHqTa&!>v_(pGHmbp zqIp@HwHwiTtuG9Qo0XAgl148bynj=cp_+>A)>&1q;Qps|rWN~S2i4`|I#@*D-!ITo z(xSQ0(&)zRS+R}E&JLHSg5tUMknWlyG<=?4%DqW%DkB1KmOh9S_A#=3k+%#&O*T~Yge)pay#4TPs#7g3hh!xFX{2%HV14UKzF$%LUN!3$6)8cRoQ>R}-Ob0>bLZ2=>+Lbq z`toLaNK(^|2@@B08M7uOb*-!@3|#7o8DGL`|Hdm4iVTqO)?yWXJ7WEG+jG zkJbiMqh;*=CIk&kV~s);a`%<*aP{(PwI6-ikvK{Y4n%#@dmgLbl6!9qbx_o@cY@Pq zKI@rVcG=Osb7G}^9%cbGb6CK<@X2F#O1m(YX|P(suF?61Wi*BJ{s3{ALo38PEc4yA zD$>uP*afDKtEC)yAdMBo2n`mFqFS`ANV1Q)v9uEyC4zDg=rQYDjvw#|t}&bm6v32k zx;yT3lV9$m;%+_QqyHp~;4lwm?DYSv`~z)$$2?xJ;`urew@Z!s%IFy&e15H6+twjR ztDAvYM{vozH<~wwt^Q|DdD^=8K=Ct`b!GKTPSiuTa6(tjQLWu;XxJ$BCr~{J^0dgM zMfQDZ_PXlyBGp`zSMZ0qgMYD~%VfVJ5EMv#O?4wE1ctzi9NY*vr(mXK{oc6d(2kh~Elz3ex>XB*m* z1BF8CiZO>RBwJ++()^2~ldEhiE-5`~HKLBcETF#S>qD%pwHiqaSyFHWzB$Y0helD-&^v{Y$pOf3N@P{jVErzMOo}z>1a-Ok zg@3CrNTjzDsCg;HLs_o73)HpK{jMOMF&+kWE2Gc3#)~HJw%h^{(R79%C^R!)3b&37 ztCR!(z6ow7c`b%+fWJ#*Ckfp;I_`Jm2!Wv9Ov!CF#Q%FYMs;o(@5v+I{{IL4Z}-y^ dI_@h9pvHN<@5QJ2SkOV{MQUpPYBLY%H1jPbUM?^ZJ zRFx)eq==QKh&Td*DDp1o%;HdsgDCEDS}2q=Y~q zkchF7o(=H31Nevv@B!c3W)kH8TRlFIKCc1I_rOfLl{!<;rUzn!TzD#bd1`H&ONo`ZdDc~lg4 zqyGt~;!6S)^DM}EW}!4Er>k{tZdvx(*NL#F*vi7&*@sVfRfQE-VyixfF3LWy;U2lh zj(k(MLi+MWHv#);LHm)pwN(NE_M5p?t)S4?<)~W0sr18`O6NPZ@UfbbTKR=#%<}$o z`#+oBe>Hnkv_7?JS6%w^$Z-z=?8eBj5OF*Vq7{078Jsve+w2n%7ePqqd)yGCf7tvN z%YDly9zHJQFK>nNPKXsxu@g_80#BAs_kC`?HkqIGD7;U4dM0%>C|sfzmasRs^jS!0 zO=5Ro4@Iv3(?tJh%Ut{$hq-f>yzZ2WXNN^z*r;~r$w%4pUv)P11i2^P3mWJ%cLGKNmkJ>K)=3VH*)W7RbN-_ z+nSMgZuz6`D>Ks~{ZB;H!q4{dx9>dq1UjzT$j~s3-xU!gz9)8Pbi~Pz@#l+G4NC7T z(kp@rAVH_i;)gi#k^%w`_=7U&yZhq zo+ubwa;SGD2sCt9!zDJRV{3>Nuc6xrtGvGLqv?2O_NnV3dqS^HOlerHjMh?rI|EfNYPn=T|L7Pf(I#Q2*Oe+eq@ zynqXAS>DT#N#$t>seFC5?O?#JsX0%Ewb(X#?`DYorYj-OsNI51QYW>eoy@^6cFEV8 ziA!B;Fr8y`y@;C{-r|}fXc3o`8amc~)k>5ZQh-9>Bg6;;N$x?WO_sN2&S16qJmBnTsT#&mMYCVCVo^osqv^MxY5ad)9v z!*6OHy%Z`|xZi<3;InfoHVv#}3lDOf6;IyOZhQ73WDoyvl4DHENh3MEe7593lI+75 zpW`32T%|5N9w0njnh)kp%?M-fTnt`v5E)Y%IC!T$8mas4oRRaf-5fA<)bQi$T~1LF z<-7G2aDkc%u$eK@*@u)PEy{)&ue%nEow5m#ph&r`lC7JIa&ZzK)79nDhS`+O>AoP_ zv(MsUA;ll}6g!#S8gI#&l&1N+82K@UI6I-l97W5{4yRW8;g{&PV^xtWWNx;kHqLd| zVR5@F;LUQ|4jZ|HM{q@ysnz#NkVe?>jq6xaes^FE;lY#v`%cgm9NPMTJk{>(BZaMqHL~Ty^%j=(9 z$30k?vteK3E=)*d%~qesmLTXm$H>|U3kQc8;)PwBS7Y?EQIVHTl_#oi$hu_~i4Y;} zL|3TE`#np(Mv~)+ZK9sR7dKxts`)DCeY<;k$w#z>XD0W=B2xI)Q$1$PZG9<8ag(^a zN{#Rulm40nVS-mEI-h+HtCj0E6|7&|0A=k=7iMOJi5^gq7PQZe;u=wIJog`%%0q4L z#RzMtJ-WhW(3X?WD@TVRgS6(7`3-%dR1fypHk*EmD^|Ppg*l&}H7fY=`uo=+MN+$M z#rLEZq!}NCn!Ml^XPrn*?N>OVh=2a-&eo;ld|w8ut{W%$)|MGZR$wM-A@F%buq-9U zZL$C6Mt8@r&3-MVP1PUHLzs`X+Yb=ry%#!&;f}|vuGF?pS+AJiq7Pg^>I}}uDa9FZWWA{`O|EX=;-ugP1wGeP#1&4wVje0BU>#g5ZjVzXCrP_EAsP(ZUgrz zS=Gh@+Y=V;CD9ob8#w|SF1&OuHYc4D)rVgF<)el&?AH(X(?S*0n#iZ`oKI`plXtIk z^4>iI{Dmz;2G1rOxRp+Xnd;7nQ(IJ zJ*|h7&HAFXw@2*e87A#NTv;WK{(Z%JSWSrjGGw!Nzg`93<1Wdf6#?ZBdKuP?aNUgR z3m@|uI|8&;^)iAs4+BFKucjD%Y*W^(5UTX~amS2Lc~1fjGd_)b1E1*#UcGUmsqia^0FKLSo6}erVK9!#}|yw^dnPgD#fwyx8LxPPnESg zlGK|eaX$1>Mspn&3y`JVk8}3b$kK)_LTuj_K?6qXse9VNZ?V=NYIM8BTRQg{g7_iC~FBi?-0! zqes5Ta95{ed8hVmG2T+<)^5c>P6PKE=2b$hjx$TnG={BRh8LID^U)p4sn6fu2f}?os($Z8wt}A0F7`s0`b!vYV)_A%o2lqufO zUTntgWRUG5`${&1hwx}^PS(jazCD)IS)J=DG&91J( zQ<|VThwnE@?3;n`mf#H}AvULm!R#G9we_ts`4VEIJj_mc1O zH@1*=&R2QobD3c&c+t9Q-QH(6p>97;nS~d6vkH(=Q=!W-?D{! z`$#LRqN|$}jyWHdN)OZnfi|3E=;~S->+1gT><>KgXN4vajhYThx48A~xK^%S9u=z_ z^MQz`>?P663zd*54U&mebORv+CjAdI~!tCFM=!2|JifUWY>7f?Mz3-*n7T^EA zpnC6CUH1wOa`YZG!ty{i&94da6rBCE{7u$k9HmI5uz9kX!smK+DQP;sote zB%e}#qfVs5#qHv|hAo4aPDwZDsYbgfY_0Pb0m(r=)souGGJcnLpeRefp|g7)SlTM= zII)OYzD^LW-=yQkVqXl+(gZoZQ3~BCEzNU9_TiDmz<_Wzdo}y@{Nl!ULmbY4+{y@W zMq6P3XR$-(W_SvlsY;@<$u!kKrVnsd1Az!yfj%UP7mW)h(>xd~B4oC<0Rm=Fi4aFM zbGW&WF3po+6wIO723y!sg1sm>Dnv_Dh!BVe0GKo`2^`4uX8Ga+iI8<%JTP7}!yw>w z7p@l(a>(2Ytjp%mz-U#pDjcdG$nZx(G=;zf4wa6#(KGlC0bCIwo?NaE9tH~t2v7|` zsj@j9Fa!>VgTaw7BoYdEKz)N)Tv8yE<-30k;v0q@&6mPq_;49)7I+PlL}vSOi4X`d z5B{S*rjNP#pYSZ-?<@d(zye7=FoY@`#$>{NjPT{^`vV}~3;K@{zIMQo7G^{9W&3d` zG<|;>i@X0v2rA{zcppEG_j)^23XJAWV*;+eKvu+0DGiOyt^SNyqrii~^jVJr$o|Qb z%b@>7)=#yq&8)ZcV~0t=;+NK`5ftA>Un&_6&KvwXQE7KOG31%RtE033B1iKb4* zAfZ$;1%RNalcDM~3LJ{WAxRh#g+w94;XgoFau|RrN!~wJwFX56ps;W(7D1t+pkxe5 z9g0T7QBWL$LW8OyfSPD%bqYoeyADOA;0@RuCJ9(h29xALgZZ#L)+g2o$7@>|6Cp^| zZ&}wZR^B8o9S9&o%or@cz`tDW7)+Wim$b$v0;2}UVG(e3lsZx!j`)^nJ=%fB@dcE) zhKYcyqQ2p-EejrK1|XKSrc(gmk1p}L92$ws=Gd{>-bBb61b8iSUEW~Aw?#2x_yQh5 zYl{D_d0X1CZ&%-L0dL0o6d1fNTRe&KEr~D5pGI9z1jK!tqIi;69yH+o{w}CL?2P}g zSSSn`OGc_Ap;#In11uJm48>tdIH)>?j6kZRu~Za__A|OKo6Zd&acJ5eK&L=AfPmJ! z0V}Lisra*ZfG2H@Cpd6#!_iPA+zyGr!_jyYN*Mvi13G~H^DP1*LSWHYI23~;)1hdD z8U{)tlMzr94oQa7kXQ;GP5HU#|5J;Y?=7PLTZ;tP+Q#+Aq7q>LW7)qs{IJsj)O@o6 zJ1DRr!~WcnzjL;xlz;H`U2gwi1_1isB!7$Ff9d*{uD`{=-*Wy}UH{Vcw;1?a&i|_G z|BWu8zs__t7Vw}K0G!xbS_fVLr#(Khsi7Wdb?tqrJ}(8Z2>KW~`2v5}iLZS&fHJcr z0V6-x*j%4~f>%nISDDB2>}9|dVyvfar`~?Me?;UiSYVSCtM8&qv-Kd4;W4qiTsL#6 zsQC@`IY-ZI+pM?G*2J0}e&ry6!28%aJy77|dw5-=t-ElWYXI@S$?U~e$-#Qv6T%XZ z+>S3&p2g>}JWnOLke4W@R#FrWtD@>$HfZcs}uB0Jl!uB3^BV}!*)J&z=)>TEmcZRn=Frx`!Cv`<=iQ{ zT`j-w*=xus`}p-Pap8XcQ;PtPDkJFngey}A2vSyrr>(TYc4IfXmZuL zj_7l@<-0ca%U)#nV@zw;Drg8v8hN(tAoi{EsliP&VantaK9p>?&NHY$IcCct=AB{uKB^>WRx0HAvfjXNy*kbGfx^k7hm(P zk0=~c>v*unBQ5#SO`dWke)$$MF!7!2d9_%Lf!ihnLGfa3E@4V^P}oi_Ubo3{@J6d& z_0}KFY)Bh)0LjH&HdWr%4XU-aJLl|9+q5c8LF%Pkp4)RF*)$2&#HYDqR_z&im@7JY zXZ4Ow#YT#t0?l3Rq(*9c3oqn~zALV8=3Ujg3vsnPyY0j-*Sl8RbbSnU!W^;_jwtS^ z7_1#|=Vthv@AvVYN%1xBKIxHtWzc7Oqv2Ma@K>+Bx<8B^w^m8v{rI$JV@bV?FnYv?Zs94f3$#Uq*%2mqfhWn(mxsNcDgyfWQg=RU|odMRBh(I`u z^k}}@Z_K{9vfh6R8W_`~y6n!qxr6U;FS>tdOGVc0u}E}SWMP8Jx@)Y-p86*moW4oK z{)I6oO~)(p>Sp`*I2dP(*+qHWZE$LnZ`t`MqP6$UpvLGJ=?6b zuimxyq~<-67J0G6OX+^rc7FTmYF-Azp64?`-7+X96uX0FSLPxedqzF3Ve);!vGJ{{ z9`>lbmY%Dtp{Mw?bmLW>K@YMnY(0N0I&_4sx~>$V!aS1E_J{+6JM;WXgc{{>nI3Ds zB%XnGQ@no9vRBBxez?gp(!*3+#J*o?k23t-npFq=&F5^)vFqVCg+*@(Ck_4DRClTA zRZ?4uWnEz%z4ykE4OPs3yU1qcb8q2oA91Zp57ZAbge^v{5I3F>3d5VjcS^*yJy#R4 zQ7C*gJ-p~LAh%1CKdxA-#-!GPJ<6N!xeoS3Nxz%|XPC`?zTY3U>p0_9Uf9CYd0)l$ zC(nj7ZqNQM>Yu73{+O)rZfi|ewEx(y`wQ%fo*7Xi@w`iF6p;6t^3oJrNhCk{)VJgn1RCbYe7|HeEUjVOnWV0MQa-&7)=U!&B^Sp;jWX=+H#gC%y#mbTtH8JbDS zypY-HHh!-r_O;{|=$o7DtOcPX0f+0jF9iA!0}mLoc6v!=RIH;fj{VQ3W|-=2i8+A0hK{ zriQIM_L*Io-*v8@ANeTx22Habc4n%hU)_MeSH{KJYk4NH4D73?9JATBxVFSQN2fcp z=l-2J3ry!xs4e_g(YInj6g>aHhZ+%>*3u{sCRy0aLPS|5Y=6{ER?e0ex5Kkc&pkB$(3$!uQ{$=ZFAlrEi|Lh!)hL^lkRf3)qaMDT+;p{ zwM6uR)$u~B$$s`ldhM;q-^#rErD5AUU5~xwjv$@xCqJA>?UFC*{?o7#rO8T~QA`3Q z&ZgR`CliJ?$iF&`Q9G*qUW4pVR-nE5mSj}C9`Dp=3E`_L&GOG2Wu6LF9_ z-D{GUVC6T?NGu;3U)V9w;b@Y)YORfOc-fjKYJttHOOdAQVv_F(XV3aF)Lj!FljdTz zKPu)NbRKW`$BF4s17oE<=CTkqb&rxQxpHM&R?VT_XCt2shCG^`)h{z@M@olw&u^2EKGMJ8b1V<%+CXR@&mYh{6*)PU zGNrjtm@^!|J*rt~zeMCZQPDyo=CNa*Vk$9IkL6dbarLydsnQYjkrQ1b<27NPkH-V3 z%aaJh1>vd>cglZ>co5vkl5~W*b~nvy^@Lel9Megd^`R9bphb!=6Ey8_8QwtCpmw0S zq}(%Gbi^4~L_~ud+U;FgLz6LoNH@j0=&@eMk#e8292WPTO2;w?r}L8{&}Ulo*XorP zH~4^kRm;JN<~5z(=lgWt&RU(|k>rRlVY>INsOCDKE=BFZo4@(0(BafqTM$ZBYWCZo zTThSNU+x~{74TcTSVGNnL9vb3MPKweTiPp`mC{|(7S_KPmt~k<=RJWf81K%Axun9M zF+Ph*(5ar%?Mu73nkY7m!?ZJE2R&;>Yo8}M2N}BD^D%?U$V6u5lJcXWRl3|c&xbfm zO42g()CuQ5agtVMs#BCEr|!x(M+&;Rf9enWi5iQgxbsmDURCKhZ$I`Frn97SoGdQ} zt4_@hy|ZJOx9k@Banmoz82QMwo%PvGws)axqD`cTUNY&?u|gTfL5z5Do#R7V)kh?> z{l>BN%7{|ZAJw&+u+Kg0eyznsSsZ~yE^k3G2j0T@O|1QzIRkNd z8xU@^qD`M4TPpZn&1rSTNo2(GUL>MhT`ik}N2A}gl|Sq0&KMBlZax&JiDfyLG|H68 zD>ctvxcm5^uFU$E2VEk6n;4nwLm3PdK7F_`3$o63nyqE&c)&NSVy{hXPO;Tl59J!B z_pX$tJZZA7IoYp#VyY;$NVZ4Qg;MJldJ{XK^J$u(`*95O)L-&!_FlCEYdk7r&S^Yc zSUwrFap?fPPRdhAS?ye8&DPG;WSaiW@hVP_NW^wg>2cAMIE#_V_%Wd(9clrwXt(V`vGduG-A`Lk~q^l#Q&>-vO; z?72yab<)bGhPFUP!TF=LFLU{a$rtp>TPIt|Vt#3h$ES~UuyXBJOBOy%Z`P(}jO85UVS=}C)T;maZ@xIY}=@!vwYSuLR;PGb2DX>{$yx8V*&F1kQ zdF$707@0-U3FT1>WjLVP*l@kn;tT}iBNu1GxW$|uCC`I%KSCBX+Y zhQdCG%$f*$?i=tA5<(T6!8Z7Olu-UdOuw=ryIOqbYfyqy=%ciacb(jH6yx})r4mEA z>!RjCOjcfWt})2_<>qK%HMLd6o7#8rBf<}`Sj+iS^L*jgk2st@?N39%PW=-d*pYiV zY$uRe!TLlBi$v9r2xbF2b`Z$WB!W#O2T?=7Bx(SiX#{!G&^j39v_A#4H+79Jk1ACA&zaROin zJRT2&BVkA+6!3s@Bbgz@2q=@QDS-HbVM*nZIdpaioy7zTFo`5qXowL60@%TS)fdcm zaQFt#^0lD<7{PT14mL1XDwh??AyciwsLT+} z?;$AUZ}IF<4r8Sq3K>RaP=f(iE|3-RLrNPv2j_1Q0ty1?!R(bNfb1VUL+G@>$@-x- zfn}wg?+XFKzv2Gi{a5cR&VZML1HqC-4i%JVXK4fxn#K zK#6D+1&Sh*kXS4Ng+Y)g-$6KW=zuDTjPI)wKv4iFEF1_RQ&3Qn0TBm9qv0qh9zmu; zF$e&kipG%*FxV9+3YlQd;sg_c>7)k}1E?@IGhoFaAe><4Y-a>P>cjsYab^%hXh47w zWILT18u7QmogPee3n2>FL>OS;cq{?|M`93YJRJYGlLwW<1(YbjM8Ne?UvLG}LI9cp zh$RYi3IMD`0KE__I8F^nLBlE8w<6?uaVzf6iPoeOwG3KaiU^KR6=U!K0a z0u1_!2@GD5ErCe>l7vePqkf?ii2Gt92NIb9RN(#oDyYB4>HlG}PzEF{35i2Ou~eD? z6iud(pm;1155*af5J(&vOF^NiD{Xv3=dx%a;Y1GAECA>f=mrqbN;hDg6)JUq^bQZC z3V4D8?=~C_MZ(>Y2m%~UK%urE-~>1v0{iz{#8J>R4A}q&rI1O09^ptpk9Z0dO2T6> zcq|P^Asdi>Ec(A{(O{)T3;~Y(u|-3eVBz{}QVn7MZ`!{&e7Dj8)O;BOR#0F;hJ9O+ zzj7u}%D?#eDz|?z0|5Oq$v@)vCtW}3`bP}>Bj=y$`bpP6V&ES+|5Vrij4r9a&vaBK zu+a+#PHYkP?uY}YJu#BKjU{MV@I8IK^aL;>$+q?8fUFu_6lEVg9A#2OmN}( zaB^>EX>4U6ba`-PAZ2)IW&i+q+O1bvvV%Ab{MRYw2)1Q;IgZz=xxpNN+6MASLS6_j zp}@q8mTt*d)c*Uo(?9sJL`znDaM3x1pN%$}gqNU}Uz|^^?0hbN%sQs;`{_1d(8XwZ z`Rmg9KdBUjlh{@bF)QHy>aR*IzV&Pm})eHX)Jkp(QkvFT-5u( zR%;b*SivqcA{S;1=Ef7VK#1-uZ!!Viu9beZiywq47?cTffP+cpMMM_;6I(oaX3P_` zOIB#|brk@Jpj%*!W-y?fX3L8QDVK1T%DFjn{Lo z5#!A0#@+-XkVovnP2>c!tQEwMw*fJfgvE-86EA@zN^jT#G%9@!> zjd}jY8hXk5O@toXi44xbn6Vm+`@sML+6>NA(AJv4+~7<_Gd3~5ho z*EjBdm^;NANjSxudB+?L>b}7ofx2hj4p^(>J69%SOBN1H6(jpPyJP~T=xVCT<&t4J zD5>4B^D0Nh476IeNNLfMK%fe>rP@2Uh(9WVh+bS6?-Z=-;$}CD=mh}u##kHyqZB;v7tjAoPxHqkHmE4=J+&kh3`yd>^ht$cYSZ%6d{80X*deb|#!q=N#+7zSn zS@pBx1FefrI`a35yL9C56?f^#-z)CYk-t~mrL*+DbC=H2`_5fDOYb{(=`Yc|a~Io) zG!4Bt8p>Fy#&EZWVT~{jy!ftjY9-=mki}cwj*-VD?c(&kxb%uZr)WJ&tY0YwXG@=OvRP1PAd_S}s0V}=_ z3uK(74_kUQ@M-%a2!Yf0-V9$>Pw=IgIJ|%XYm1}4@wX#yMA99%Ph-!-(sjb_TROq) z-k)Pq0R#0zMnUvO-1vC)QDyCEeIBQi?7nU3(*=*@ZlOvK3?JYz+3oLAmtGEkn7Z@@ z<|o0_e~Z}BQXXQtUnHuRzTQ%S!zPI(~2U)#KC5er3`j$5eu6}uqKq1&7@fevj=OV z5LQbeTv1ncErU%!1ilXqfXpv!s!>I=hX6L0004lX+uL$Nkc;* zaB^>EX>4Tx04R}tkv&MmKpe$iQ)@*k4t5Z6$WWauh!t_vDionYs1;guFuC*#ni!H4 z7e~Rh;NZt%)xpJCR|i)?5c~jfa&%I3krMxx6k5c1aNLh~_a1le0HIN3n$X zocD>NtSqa<=fqx##3oJ%eXJTq!$GjqgIVzJc0N(ZyDsS!^S$5c(Hd?Dwt z%6W^kR;{ttJ^2eG1$}vm>okXv!U7f{L4<-DDyYInoK~F_3mMu^`uK-jzeFyDToo{K z%wq!@WY-V=2fw?uiirs?DUt+2FOKsu0)%&gX5DeVj~%CZ0tBCdE4}UCXaLinq}SV8 z>NhHT2N6r?E>i@^ICeN!G7xCMGwz1~{;IDG)J)K&ThI5-4G zOO(Cl^X{I`-u^w)>hA|O+j5WXY2m&A000JJOGiWi{{a60|De66lK=n!32;bRa{vGf z6951U69E94oEQKA00(qQO+^Rf1qJ{(BN=k;5dZ)HF-b&0R5;6HG&eW@&p;sv2?=3f zfiWlnB_$;cObk>4qXvx{G-}XDHHe|RyZb*i003-uOL*&0I{p9v002ovPDHLkV1j3U B_Fw=2 literal 0 HcmV?d00001 diff --git a/src/main/resources/assets/codechickenlib/textures/gui/widgets/slot_large.png b/src/main/resources/assets/codechickenlib/textures/gui/widgets/slot_large.png new file mode 100644 index 0000000000000000000000000000000000000000..61c28017fb4f9b4b83470a2fbefd48fc3e662ce6 GIT binary patch literal 5546 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 literal 0 HcmV?d00001