
Clicking with pocket computer in off-hand. New solution (help needed). [1.20.1]
MAGGen-hub opened this issue ยท 0 comments
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 =(