CC: Tweaked

CC: Tweaked

64M Downloads

Clicking with pocket computer in off-hand. New solution (help needed). [1.20.1]

MAGGen-hub opened this issue ยท 0 comments

commented

Hello there! It's me the creator of off-hand clicking feature for pocket computers: #918

Well... It's very sad that Minecraft devs removed passEvents field from their code... #1471 (c45fc94)
But it's still posible to fix.

Currently I came up with next solution:
Class net.minecraft.client.Minecraft contains method handleKeybinds that used to handle all input events from all keys. In versions < 1.19.2 it can be runned with passEvents for our screen with benefits but in >1.20.1 we must disable the screen it self to force it to work. Also that class contains missTime field that affects attack function. Any gui screen sets that field to 10000 so we can't break blocks or actualy do anything with Left Click when GUI enabled. Setting values <0 to it, allow us to use items with left click.

So the code for dan200.computercraft.client.gui.NoTermComputerScreen looks like this.
And works perfectly, (at least on fabric [see issue below]):

// SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers
//
// SPDX-License-Identifier: MPL-2.0

package dan200.computercraft.client.gui;

import com.mojang.blaze3d.platform.InputConstants;
import dan200.computercraft.client.gui.widgets.TerminalWidget;
import dan200.computercraft.core.terminal.Terminal;
import dan200.computercraft.core.util.Nullability;
import dan200.computercraft.shared.computer.inventory.AbstractComputerMenu;
import net.minecraft.client.KeyMapping;
import net.minecraft.client.Minecraft;
import net.minecraft.client.gui.GuiGraphics;
import net.minecraft.client.gui.screens.Screen;
import net.minecraft.client.gui.screens.inventory.MenuAccess;
import net.minecraft.network.chat.Component;
import net.minecraft.world.entity.player.Inventory;
import org.jspecify.annotations.Nullable;
import org.lwjgl.glfw.GLFW;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Objects;

import java.lang.reflect.Field;

import static dan200.computercraft.core.util.Nullability.assertNonNull;

/**
 * The GUI for off-hand computers. This accepts keyboard input, but does not render a terminal.
 *
 * @param <T> The concrete type of the associated menu.
 */
public class NoTermComputerScreen<T extends AbstractComputerMenu> extends Screen implements MenuAccess<T> {
    private final T menu;
    private final Terminal terminalData;
    private @Nullable TerminalWidget terminal;


    public NoTermComputerScreen(T menu, Inventory player, Component title) {
        super(title);
        this.menu = menu;
        terminalData = menu.getTerminal();
    }

    @Override
    public T getMenu() {
        return menu;
    }

    @Override
    protected void init() {
        // First ensure we're still grabbing the mouse, so the user can look around. Then reset bits of state that
        // grabbing unsets.

        minecraft().mouseHandler.grabMouse();
        minecraft().screen = this;
        KeyMapping.releaseAll();

        super.init();

        terminal = addWidget(new TerminalWidget(terminalData, new ClientInputHandler(menu), 0, 0));
        terminal.visible = false;
        terminal.active = false;
        setFocused(terminal);
    }

    @Override
    public final void tick() {
        super.tick();
        assertNonNull(terminal).update();
        base_override();//update mouse keys
    }

    @Override
    public boolean mouseScrolled(double pMouseX, double pMouseY, double pDelta) {
        Objects.requireNonNull(minecraft().player).getInventory().swapPaint(pDelta);
        return super.mouseScrolled(pMouseX, pMouseY, pDelta);
    }
    @Override
    public boolean mouseReleased(double mouseX, double mouseY, int button) {
        KeyMapping.set(InputConstants.Type.MOUSE.getOrCreate(button), false);// set key_held false
        return super.mouseReleased(mouseX,mouseY,button);
    }

    @Override
    public boolean mouseClicked(double mouseX, double mouseY, int button) {
        KeyMapping.set(InputConstants.Type.MOUSE.getOrCreate(button), true);// set key_held -> true
        KeyMapping.click(InputConstants.Type.MOUSE.getOrCreate(button));// click!
        base_override();//apply click (click can't be detected from `tick`)
        return super.mouseClicked(mouseX,mouseY,button);
    }

    @Override
    public void onClose() {
        KeyMapping.releaseAll();//FREE ALL KEYS
        Objects.requireNonNull(minecraft().player).closeContainer();
        super.onClose();
    }

    private void base_override(){
        // Disable screen (Emulate passEvents condition form 1.19.2)
        Screen s = minecraft().screen;
        minecraft().screen = null;

        try { //can't update access transformes/wideners... HELP!!!!
            Field f = minecraft().getClass().getDeclaredField("missTime");
            f.setAccessible(true);
            f.set(minecraft(),-10);
            Method m = minecraft().getClass().getDeclaredMethod("handleKeybinds");
            m.setAccessible(true);
            m.invoke(minecraft());
        } catch (NoSuchMethodException | InvocationTargetException | IllegalAccessException | NoSuchFieldException e) {
            //Logger.getGlobal().log(Level.WARNING,"Aboba bebra!");
        }

        // Recover pocket computer screen
        minecraft().screen = s;
    }

    @Override
    public boolean isPauseScreen() {
        return false;
    }

    @Override
    public final boolean keyPressed(int key, int scancode, int modifiers) {
        // Forward the tab key to the terminal, rather than moving between controls.
        if (key == GLFW.GLFW_KEY_TAB && getFocused() != null && getFocused() == terminal) {
            return getFocused().keyPressed(key, scancode, modifiers);
        }

        return super.keyPressed(key, scancode, modifiers);
    }

    @Override
    public void render(GuiGraphics graphics, int mouseX, int mouseY, float partialTicks) {
        super.render(graphics, mouseX, mouseY, partialTicks);

        var font = minecraft().font;
        var lines = font.split(Component.translatable("gui.computercraft.pocket_computer_overlay"), (int) (width * 0.8));
        var y = 10;
        for (var line : lines) {
            graphics.drawString(font, line, (width / 2) - (font.width(line) / 2), y, 0xFFFFFF, true);
            y += 9;
        }
    }

    private Minecraft minecraft() {
        return Nullability.assertNonNull(minecraft);
    }

}

Changed methods: onClose, mouseClick
Added methods: mouseRelease, base_override

Known issues:
With this solution you can't attack mobs on minecraft Forge.
I tried to figure out the reason, but IntelliJ Idea IDE tells me that bytecode is different from sourcecode so... I can't find out what blocks startAttack and continueAttack methods from working... And this is very stragne, because net.minecraft.client.Minecraft is not a part of forge or fabric... Maybe issue is in ticking system or something...
It can be a good idea for Forge version to invoke thouse methods from NoTermComputerScreen directly, or even recreate them inside your NoTermComputerScreen class, but it will probably cause issues with other mods, where items have custom behaviour...

HELP REQUEST:
Well I tried almost EVERYTHING!!! But still have NO ACCESS!
I changed: projects/common/src/main/resources/computercraft.accesswidener
I changed: projects/common/src/main/resources/computercraft-common.accesswidener
I changed: projects/forge/src/main/resources/META-INF/accesstransformer.cfg
I used: gradlew clean then gradlew build for forge and fabric separatly! NO EFFECT!
WHAT I need to do to update thouse???

Probably that because of lack of experience in using gradlew, but I really tired from seaching the correct way to get that access without java reflection...

I can't create pull request until I figure out how to "fix" this =(