Skip to content

自定义Tooltip

字数
1634 字
阅读时间
8 分钟

在mod开发途中,为了使自己的mod更具风格化,更有特色,有时会需要自定义工具提示栏(Tooltip),本篇将会围绕几个实例,讲解neoforge1.21.1自定义Tooltip,希望能对你有所帮助

neoforge的Tooltip事件

在 NeoForge 1.21.1 中,处理工具提示(Tooltip)主要涉及三个核心事件,它们分别在工具提示生命周期的不同阶段触发。下面我将详细介绍这三个事件及其典型用法:

客户端限定:这些事件都应注册在 Dist.CLIENT​ 端

1. RenderTooltipEvent.GatherComponents

触发时机:当游戏收集工具提示组件时(在渲染之前)

主要用途

  • 添加、移除或修改工具提示内容
  • 插入自定义组件(如进度条、图标等)
  • 根据条件动态改变工具提示

2. RenderTooltipEvent.Pre

触发时机:在工具提示即将渲染前

主要用途

  • 完全接管工具提示渲染
  • 修改工具提示的位置、大小
  • 添加自定义背景或边框

3. RenderTooltipEvent.Color

触发时机:当游戏确定工具提示颜色时

主要用途

  • 修改工具提示的背景和边框颜色
  • 实现动态颜色效果
  • 根据条件改变工具提示外观

有了这几个事件,就可以方便的进行基础修改,比如改变物品提示栏的边框和背景颜色

java
    @SubscribeEvent(priority = EventPriority.LOWEST)
    @OnlyIn(Dist.CLIENT)
    public static void colorTooltip(RenderTooltipEvent.Color event) {
		event.setBorderStart(0);
        event.setBorderEnd(0);
        event.setBackgroundStart(0);
        event.setBackgroundEnd(0);
    }

像这样,我们就将Tooltip的背景和边框变成了透明

同理,我们可以查看另外三个事件的方法,进行我们的自定义修改

更进一步

往往开发者们想要更高级的自定义效果,如将边框背景渲染成图片,这又该如何实现呢?可能大家很快就能想到刚才所讲的几个事件,只要将图片渲染到Tooltip的位置就行了,但是物品提示框是动态变化,如何让一个图片就能适应所有的布局呢?

其实我们只需要用9-Slice 渲染渲染就可以了,它可以帮我们自适应边框。

现在,我们已经接近了动态边框的问题,但是,我们还有一个重要问题没有解决,就是获取Tooltip的宽度和高度,所有的事件都没有get的方法,只有获取x,y坐标,这该怎么办呢?

在高版本中ClientTooltipComponent​ 是控制工具提示(Tooltip)内容呈现的核心类。相比旧版本直接渲染文本的方式,它提供了更强大、更灵活的组件化工具提示系统。我们可以通过它来获取宽和高,至于有关ClientTooltipComponent​的内容,请看这篇文章关于ClientTooltipComponent

首先,在任意事件处,我们都可以通过getComponents()​来获取List<ClientTooltipComponent>​,

这是minecraft渲染tooltip的核心,逻辑如下

mermaid
sequenceDiagram
    participant Item as 物品/方块
    participant Event as GatherComponents事件
    participant Factory as Component工厂
    participant Render as 渲染系统
    
    Item->>Event: 提供List<Component>
    Event->>Factory: 转换Component→ClientTooltipComponent
    Factory->>Render: 生成具体组件实例
    Render->>Render: 调用getWidth/getHeight
    Render->>Render: 调用renderImage/renderText

我们可以通过判断List中最大ClientTooltipComponent​来获取长和宽

java
    public static int calculateTooltipWidth(List<ClientTooltipComponent> tooltips) {
        int maxWidth = 0;

        for (ClientTooltipComponent component : tooltips) {
            int componentWidth = component.getWidth(Minecraft.getInstance().font);
            if (componentWidth > maxWidth) {
                maxWidth = componentWidth;
            }
        }

        // 添加 padding(通常左右各4像素)
        return maxWidth + 8;
    }
java
    public static int calculateTextTooltipHeight( List<ClientTooltipComponent> tooltipLines) {
        // 每行高度(字体高度 + 行间距)
        int lineHeight = Minecraft.getInstance().font.lineHeight + 2; // 通常有2像素的行间距

        // 总高度 = 行数 × 每行高度 + 上下padding(通常各4像素)
        return tooltipLines.size() * lineHeight + 8;
    }

好了,作者的文笔很差,请见谅,更多更详细的内容,可以关注作者的b站,查看视频讲解

视频源码

java
@EventBusSubscriber
public class Tooltip {

    //边框宽度
    private static final int BORDER = 4;
    //图片大小
    private static final int TEXTURE_SIZE = 32;
    //图片路径
    static final ResourceLocation CUSTOM_TOOLTIP_BG = getPath("tooltip");



    //获取图片路径
    private static ResourceLocation getPath(String part) {
        return ResourceLocation.parse(ExampleMod.MODID +":textures/tooltip/" + part + ".png");
    }


    @SubscribeEvent(priority = EventPriority.LOWEST)
    @OnlyIn(Dist.CLIENT)
    public static void colorTooltip(RenderTooltipEvent.Color event) {
        int height = calculateTextTooltipHeight(event.getComponents());
        int width = calculateTooltipWidth(event.getComponents());
        int x = event.getX()-4;
        int y = event.getY()-4;
        renderTooltipBorder(event,0);
        if (!isFromMod(event.getItemStack().getItem(), ExampleMod.MODID)){
            Rarity rarity = event.getItemStack().getRarity();
            if (rarity.equals(Rarity.COMMON)){
                renderTooltipBorder(event,0xFFFFFFFF);
            }else if (rarity.equals(Rarity.UNCOMMON)){
                renderTooltipBorder(event,0xFFFFFF00);
            } else if (rarity.equals(Rarity.RARE)) {
                renderTooltipBorder(event, 0xFF00FFFF);
            }else if (rarity.equals(Rarity.EPIC)){
                renderTooltipBorder(event,0xFFFF00FF);
            }
        }
        event.setBackgroundStart(0);
        event.setBackgroundEnd(0);
        if (Minecraft.getInstance().screen != null ){
            renderItem(event.getItemStack(),event.getGraphics());
        }
        if (isFromMod(event.getItemStack().getItem(),ExampleMod.MODID)){
            PoseStack poseStack = event.getGraphics().pose();
            poseStack.pushPose();
            poseStack.translate(0, 0, 400); // 确保在正确层级渲染


            // 9-slice 渲染
            renderNineSlicedBg(event.getGraphics(),x,y,width,height);

            poseStack.popPose();

        }
    }

    public static boolean isFromMod(Item item, String expectedModId) {
        ResourceLocation registryName = BuiltInRegistries.ITEM.getKey(item);
        return expectedModId.equals(registryName.getNamespace());
    }
    private static void renderNineSlicedBg(GuiGraphics guiGraphics, int x, int y, int width, int height) {
        /* 四个角 */
        // 左上
        guiGraphics.blit(CUSTOM_TOOLTIP_BG, x, y,
                0, 0, BORDER, BORDER, TEXTURE_SIZE, TEXTURE_SIZE);
        // 右上
        guiGraphics.blit(CUSTOM_TOOLTIP_BG, x + width - BORDER, y,
                TEXTURE_SIZE - BORDER, 0, BORDER, BORDER, TEXTURE_SIZE, TEXTURE_SIZE);
        // 左下
        guiGraphics.blit(CUSTOM_TOOLTIP_BG, x, y + height - BORDER,
                0, TEXTURE_SIZE - BORDER, BORDER, BORDER, TEXTURE_SIZE, TEXTURE_SIZE);
        // 右下
        guiGraphics.blit(CUSTOM_TOOLTIP_BG, x + width - BORDER, y + height - BORDER,
                TEXTURE_SIZE - BORDER, TEXTURE_SIZE - BORDER, BORDER, BORDER, TEXTURE_SIZE, TEXTURE_SIZE);

        /* 四条边 */
        // 上边
        guiGraphics.blit(CUSTOM_TOOLTIP_BG, x + BORDER, y, width - 2 * BORDER, BORDER,
                BORDER, 0, TEXTURE_SIZE - 2 * BORDER, BORDER, TEXTURE_SIZE, TEXTURE_SIZE);
        // 下边
        guiGraphics.blit(CUSTOM_TOOLTIP_BG, x + BORDER, y + height - BORDER, width - 2 * BORDER, BORDER,
                BORDER, TEXTURE_SIZE - BORDER, TEXTURE_SIZE - 2 * BORDER, BORDER, TEXTURE_SIZE, TEXTURE_SIZE);
        // 左边
        guiGraphics.blit(CUSTOM_TOOLTIP_BG, x, y + BORDER, BORDER, height - 2 * BORDER,
                0, BORDER, BORDER, TEXTURE_SIZE - 2 * BORDER, TEXTURE_SIZE, TEXTURE_SIZE);
        // 右边
        guiGraphics.blit(CUSTOM_TOOLTIP_BG, x + width - BORDER, y + BORDER, BORDER, height - 2 * BORDER,
                TEXTURE_SIZE - BORDER, BORDER, BORDER, TEXTURE_SIZE - 2 * BORDER, TEXTURE_SIZE, TEXTURE_SIZE);

        /* 中间 */
        guiGraphics.blit(CUSTOM_TOOLTIP_BG, x + BORDER, y + BORDER, width - 2 * BORDER, height - 2 * BORDER,
                BORDER, BORDER, TEXTURE_SIZE - 2 * BORDER, TEXTURE_SIZE - 2 * BORDER, TEXTURE_SIZE, TEXTURE_SIZE);
    }

    public static int calculateTooltipWidth(List<ClientTooltipComponent> tooltips) {
        int maxWidth = 0;

        for (ClientTooltipComponent component : tooltips) {
            int componentWidth = component.getWidth(Minecraft.getInstance().font);
            if (componentWidth > maxWidth) {
                maxWidth = componentWidth;
            }
        }

        // 添加 padding(通常左右各4像素)
        return maxWidth + 8;
    }

    public static int calculateTextTooltipHeight( List<ClientTooltipComponent> tooltipLines) {
        // 每行高度(字体高度 + 行间距)
        int lineHeight = Minecraft.getInstance().font.lineHeight + 2; // 通常有2像素的行间距

        // 总高度 = 行数 × 每行高度 + 上下padding(通常各4像素)
        return tooltipLines.size() * lineHeight + 8;
    }

    private static void renderTooltipBorder(RenderTooltipEvent.Color event,int color) {
        event.setBorderStart(color);
        event.setBorderEnd(color);
    }

    private static void renderItem(ItemStack itemStack,  GuiGraphics guiGraphicsContext) {
        Minecraft minecraft = Minecraft.getInstance();
        BakedModel bakedmodel = minecraft.getItemRenderer().getModel(itemStack, minecraft.level, null, 0);
        int screenWidth = minecraft.getWindow().getGuiScaledWidth();  // 获取缩放后的GUI宽度
        int screenHeight = minecraft.getWindow().getGuiScaledHeight();
        //调整大小
        guiGraphicsContext.pose().pushPose();
        guiGraphicsContext.pose().translate((float)(screenWidth /4), (float)(screenHeight/2), 400f);
        guiGraphicsContext.pose().mulPose((new Matrix4f()).scaling(1.0F, -1.0F, 1.0F));
        guiGraphicsContext.pose().scale(64.0f, 64.0f, 1.0f);
        Lighting.setupForFlatItems();
        minecraft.getItemRenderer().render(
                itemStack,
                ItemDisplayContext.GUI,
                false, guiGraphicsContext.pose(),
                guiGraphicsContext.bufferSource(),
                15728880,
                OverlayTexture.NO_OVERLAY,
                bakedmodel);
        guiGraphicsContext.flush();

        guiGraphicsContext.pose().popPose();
    }

贡献者

The avatar of contributor named as 小飘 小飘

文件历史