完成打开样板编码终端自动填充样板功能

This commit is contained in:
GaLicn 2025-08-12 22:39:07 +08:00
parent 981d2f2b32
commit 319253d6f9
4 changed files with 187 additions and 2 deletions

View File

@ -3,7 +3,7 @@ org.gradle.jvmargs=-Xmx1G
loom.platform = forge
# Mod properties
mod_version = 1.1.3
mod_version = 1.1.4
maven_group = com.extendedae_plus
archives_name = extendedae_plus

View File

@ -0,0 +1,159 @@
package com.extendedae_plus.mixin;
import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.Shadow;
import org.spongepowered.asm.mixin.injection.At;
import org.spongepowered.asm.mixin.injection.Inject;
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
import org.jetbrains.annotations.Nullable;
import net.minecraft.world.entity.player.Inventory;
import net.minecraft.world.inventory.MenuType;
import appeng.api.inventories.InternalInventory;
import appeng.api.networking.energy.IEnergySource;
import appeng.api.storage.MEStorage;
import appeng.api.storage.StorageHelper;
import appeng.api.stacks.AEItemKey;
import appeng.api.stacks.AEKey;
import appeng.core.definitions.AEItems;
import appeng.helpers.IPatternTerminalMenuHost;
import appeng.menu.me.items.PatternEncodingTermMenu;
import appeng.api.networking.security.IActionSource;
import appeng.menu.slot.RestrictedInputSlot;
import appeng.menu.me.common.MEStorageMenu;
import com.extendedae_plus.mixin.accessor.MEStorageMenuAccessor;
@Mixin(PatternEncodingTermMenu.class)
public abstract class PatternEncodingTermMenuMixin {
// 防止重复执行
@org.spongepowered.asm.mixin.Unique
private boolean extendedae_plus$blankAutoFilled = false;
@Shadow
private RestrictedInputSlot blankPatternSlot;
@org.spongepowered.asm.mixin.Unique
private void extendedae_plus$tryFill(IPatternTerminalMenuHost host, Inventory ip) {
try {
var self = (PatternEncodingTermMenu) (Object) this;
var player = ip.player;
// 仅在服务器端执行
if (ip.player.level().isClientSide()) {
return;
}
// 必须可与网络交互
var acc = (MEStorageMenuAccessor) (Object) ((MEStorageMenu) self);
MEStorage storage = acc.getStorage();
IEnergySource power = acc.getPowerSource();
boolean hasPower = acc.getHasPower();
boolean canInteract = storage != null && power != null && hasPower; // 等价于 canInteractWithGrid()
if (!canInteract) {
return;
}
if (storage == null || power == null) {
return;
}
InternalInventory blankInv = host.getLogic().getBlankPatternInv();
var current = blankInv.getStackInSlot(0);
int limit = blankInv.getSlotLimit(0);
int space = Math.max(0, limit - current.getCount());
space = Math.min(space, AEItems.BLANK_PATTERN.asItem().getMaxStackSize());
if (space <= 0) {
return; // 已满无需填充
}
AEKey blankKey = AEItemKey.of(AEItems.BLANK_PATTERN.asItem());
long extracted = StorageHelper.poweredExtraction(power, storage, blankKey, space,
self.getActionSource());
if (extracted <= 0) {
return; // 网络无可用空白样板
}
int toInsert = (int) Math.min(extracted, space);
var stackInSlot = this.blankPatternSlot.getItem();
if (stackInSlot.isEmpty()) {
this.blankPatternSlot.set(AEItems.BLANK_PATTERN.stack(toInsert));
} else {
stackInSlot.grow(toInsert);
this.blankPatternSlot.set(stackInSlot);
}
long leftover = extracted - toInsert;
if (leftover > 0) {
StorageHelper.poweredInsert(power, storage, blankKey, leftover, self.getActionSource());
}
} catch (Throwable t) {
// swallow errors to avoid noisy logs in production
}
}
@Inject(method = "<init>(Lnet/minecraft/world/inventory/MenuType;ILnet/minecraft/world/entity/player/Inventory;Lappeng/helpers/IPatternTerminalMenuHost;Z)V",
at = @At("TAIL"))
private void extendedae_plus$autoFillBlankPattern(MenuType<?> menuType, int id, Inventory ip,
IPatternTerminalMenuHost host, boolean bindInventory,
CallbackInfo ci) {
extendedae_plus$tryFill(host, ip);
}
@Inject(method = "<init>(ILnet/minecraft/world/entity/player/Inventory;Lappeng/helpers/IPatternTerminalMenuHost;)V",
at = @At("TAIL"))
private void extendedae_plus$autoFillCtor3(int id, Inventory ip, IPatternTerminalMenuHost host, CallbackInfo ci) {
extendedae_plus$tryFill(host, ip);
}
// 在首次 broadcastChanges 后再尝试一次避免构造时网络未激活
@Inject(method = "broadcastChanges", at = @At("TAIL"))
private void extendedae_plus$retryFillAfterPower(CallbackInfo ci) {
if (this.extendedae_plus$blankAutoFilled) {
return;
}
// 仅在服务器端执行
var self = (PatternEncodingTermMenu) (Object) this;
var player = self.getPlayerInventory().player;
var acc = (MEStorageMenuAccessor) (Object) ((MEStorageMenu) self);
MEStorage storage = acc.getStorage();
IEnergySource power = acc.getPowerSource();
boolean hasPower = acc.getHasPower();
if (player.level().isClientSide()) {
return;
}
boolean canInteract = storage != null && power != null && hasPower;
if (!canInteract) {
return;
}
// 通过 host 获取 blankPatternInv
var host = ((IPatternTerminalMenuHost) self.getTarget());
InternalInventory blankInv = host.getLogic().getBlankPatternInv();
var current = blankInv.getStackInSlot(0);
int limit = blankInv.getSlotLimit(0);
int space = Math.max(0, limit - current.getCount());
space = Math.min(space, AEItems.BLANK_PATTERN.asItem().getMaxStackSize());
if (space <= 0) {
this.extendedae_plus$blankAutoFilled = true;
return;
}
AEKey blankKey = AEItemKey.of(AEItems.BLANK_PATTERN.asItem());
long extracted = StorageHelper.poweredExtraction(power, storage, blankKey, space,
self.getActionSource());
if (extracted <= 0) {
return;
}
int toInsert = (int) Math.min(extracted, space);
var stackInSlot = this.blankPatternSlot.getItem();
if (stackInSlot.isEmpty()) {
this.blankPatternSlot.set(AEItems.BLANK_PATTERN.stack(toInsert));
} else {
stackInSlot.grow(toInsert);
this.blankPatternSlot.set(stackInSlot);
}
long leftover = extracted - toInsert;
if (leftover > 0) {
StorageHelper.poweredInsert(power, storage, blankKey, leftover, self.getActionSource());
}
this.extendedae_plus$blankAutoFilled = true;
}
}

View File

@ -0,0 +1,24 @@
package com.extendedae_plus.mixin.accessor;
import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.gen.Accessor;
import org.jetbrains.annotations.Nullable;
import appeng.api.networking.energy.IEnergySource;
import appeng.api.storage.MEStorage;
import appeng.menu.me.common.MEStorageMenu;
@Mixin(MEStorageMenu.class)
public interface MEStorageMenuAccessor {
@Accessor("storage")
@Nullable
MEStorage getStorage();
@Accessor("powerSource")
@Nullable
IEnergySource getPowerSource();
@Accessor("hasPower")
boolean getHasPower();
}

View File

@ -15,7 +15,9 @@
"ContainerWirelessExPatternTerminalMixin",
"ContainerUWirelessExPatternTerminalMixin",
"TileExPatternProviderMixin",
"PartExPatternProviderMixin"
"PartExPatternProviderMixin",
"PatternEncodingTermMenuMixin",
"accessor.MEStorageMenuAccessor"
],
"injectors": {
"defaultRequire": 1