diff --git a/src/main/java/net/neoforged/neoforge/client/gui/ModMismatchDisconnectedScreen.java b/src/main/java/net/neoforged/neoforge/client/gui/ModMismatchDisconnectedScreen.java index fbe90ddadf..a9ffeb5fb1 100644 --- a/src/main/java/net/neoforged/neoforge/client/gui/ModMismatchDisconnectedScreen.java +++ b/src/main/java/net/neoforged/neoforge/client/gui/ModMismatchDisconnectedScreen.java @@ -5,11 +5,14 @@ package net.neoforged.neoforge.client.gui; +import com.google.common.collect.Lists; import com.mojang.blaze3d.vertex.Tesselator; -import java.net.URL; +import com.mojang.logging.LogUtils; import java.nio.file.Path; import java.nio.file.Paths; import java.util.ArrayList; +import java.util.Comparator; +import java.util.LinkedHashMap; import java.util.List; import java.util.Map; import java.util.Optional; @@ -19,15 +22,16 @@ import net.minecraft.client.Minecraft; import net.minecraft.client.gui.GuiGraphics; import net.minecraft.client.gui.components.Button; +import net.minecraft.client.gui.components.CycleButton; +import net.minecraft.client.gui.components.MultiLineLabel; import net.minecraft.client.gui.narration.NarrationElementOutput; import net.minecraft.client.gui.screens.Screen; -import net.minecraft.network.chat.ClickEvent; import net.minecraft.network.chat.Component; -import net.minecraft.network.chat.HoverEvent; -import net.minecraft.network.chat.HoverEvent.Action; +import net.minecraft.network.chat.ComponentUtils; import net.minecraft.network.chat.MutableComponent; import net.minecraft.network.chat.Style; import net.minecraft.network.chat.TextColor; +import net.minecraft.network.chat.contents.TranslatableContents; import net.minecraft.resources.ResourceLocation; import net.minecraft.util.FormattedCharSequence; import net.neoforged.fml.ModList; @@ -35,8 +39,13 @@ import net.neoforged.neoforge.client.gui.widget.ScrollPanel; import net.neoforged.neoforge.common.I18nExtension; import org.apache.commons.lang3.tuple.Pair; +import org.slf4j.Logger; public class ModMismatchDisconnectedScreen extends Screen { + private static final Logger LOGGER = LogUtils.getLogger(); + private final Component reason; + private MultiLineLabel message = MultiLineLabel.EMPTY; + private MismatchInfoPanel scrollList; private final Screen parent; private int textHeight; private final Path modsDir; @@ -44,65 +53,81 @@ public class ModMismatchDisconnectedScreen extends Screen { private final int listHeight = 140; private final Map mismatchedChannelData; - public ModMismatchDisconnectedScreen(Screen parentScreen, Component title, Map mismatchedChannelData) { - super(title); + public ModMismatchDisconnectedScreen(Screen parentScreen, Component reason, Map mismatchedChannelData) { + super(Component.translatable("disconnect.lost")); + this.reason = reason; this.parent = parentScreen; this.modsDir = FMLPaths.MODSDIR.get(); this.logFile = FMLPaths.GAMEDIR.get().resolve(Paths.get("logs", "latest.log")); this.mismatchedChannelData = mismatchedChannelData; + this.mismatchedChannelData.replaceAll((id, r) -> { //Enhance the reason components provided by the server with the info of which mod likely owns the given channel (based on the channel's namespace), if such a mod can be found on the client + Optional modDisplayName = ModList.get().getModContainerById(id.getNamespace()).map(mod -> mod.getModInfo().getDisplayName()); + return modDisplayName.isPresent() && !(r.getContents() instanceof TranslatableContents c && c.getKey().equals("neoforge.network.negotiation.failure.mod")) ? Component.translatable("neoforge.network.negotiation.failure.mod", modDisplayName.get(), r) : r; + }); + this.mismatchedChannelData.forEach((id, r) -> LOGGER.warn("Channel [{}] failed to connect: {}", id, r.getString())); } @Override protected void init() { int listLeft = Math.max(8, this.width / 2 - 220); int listWidth = Math.min(440, this.width - 16); - int upperButtonHeight = Math.min((this.height + this.listHeight + this.textHeight) / 2 + 10, this.height - 50); - int lowerButtonHeight = Math.min((this.height + this.listHeight + this.textHeight) / 2 + 35, this.height - 25); - this.addRenderableWidget(new MismatchInfoPanel(minecraft, listWidth, listHeight, (this.height - this.listHeight) / 2, listLeft)); + + this.message = MultiLineLabel.create(this.font, this.reason, this.width - 50); + this.textHeight = this.message.getLineCount() * 9; + + int upperButtonHeight = Math.min((this.height + this.listHeight) / 2 + 25, this.height - 50); + int lowerButtonHeight = Math.min((this.height + this.listHeight) / 2 + 50, this.height - 25); + this.addRenderableWidget(this.scrollList = new MismatchInfoPanel(minecraft, listWidth, listHeight, (this.height - this.listHeight) / 2, listLeft)); int buttonWidth = Math.min(210, this.width / 2 - 20); + this.addRenderableWidget(CycleButton.onOffBuilder(true) + .create(Math.max(this.width / 4 - buttonWidth / 2, listLeft), upperButtonHeight, buttonWidth, 20, Component.translatable("fml.modmismatchscreen.simplifiedview"), (b, v) -> scrollList.toggleSimplifiedView())); this.addRenderableWidget(Button.builder(Component.literal(I18nExtension.parseMessage("fml.button.open.file", logFile.getFileName())), button -> Util.getPlatform().openFile(logFile.toFile())) - .bounds(Math.max(this.width / 4 - buttonWidth / 2, listLeft), upperButtonHeight, buttonWidth, 20) + .bounds(Math.min(this.width * 3 / 4 - buttonWidth / 2, listLeft + listWidth - buttonWidth), upperButtonHeight, buttonWidth, 20) .build()); this.addRenderableWidget(Button.builder(Component.literal(I18nExtension.parseMessage("fml.button.open.mods.folder")), button -> Util.getPlatform().openFile(modsDir.toFile())) - .bounds(Math.min(this.width * 3 / 4 - buttonWidth / 2, listLeft + listWidth - buttonWidth), upperButtonHeight, buttonWidth, 20) + .bounds(Math.max(this.width / 4 - buttonWidth / 2, listLeft), lowerButtonHeight, buttonWidth, 20) .build()); this.addRenderableWidget(Button.builder(Component.translatable("gui.toMenu"), button -> this.minecraft.setScreen(this.parent)) - .bounds((this.width - buttonWidth) / 2, lowerButtonHeight, buttonWidth, 20) + .bounds(Math.min(this.width * 3 / 4 - buttonWidth / 2, listLeft + listWidth - buttonWidth), lowerButtonHeight, buttonWidth, 20) .build()); } @Override public void render(GuiGraphics guiGraphics, int mouseX, int mouseY, float partialTicks) { - this.renderBackground(guiGraphics, mouseX, mouseY, partialTicks); - int textYOffset = 18; - guiGraphics.drawCenteredString(this.font, this.title, this.width / 2, (this.height - this.listHeight - this.textHeight) / 2 - textYOffset - 9 * 2, 0xAAAAAA); super.render(guiGraphics, mouseX, mouseY, partialTicks); + guiGraphics.drawCenteredString(this.font, this.title, this.width / 2, (this.height - this.listHeight) / 2 - this.textHeight - 9 * 4, 0xAAAAAA); + this.message.renderCentered(guiGraphics, this.width / 2, (this.height - this.listHeight) / 2 - this.textHeight - 9 * 2); } class MismatchInfoPanel extends ScrollPanel { - private final List> lineTable; - private final int contentSize; private final int nameIndent = 10; private final int tableWidth = width - border * 2 - 6 - nameIndent; - private final int nameWidth = tableWidth * 3 / 5; - private final int versionWidth = (tableWidth - nameWidth) / 2; + private final int nameWidth = tableWidth / 2; + private final int versionWidth = tableWidth - nameWidth; + private List> lineTable; + private int contentSize; + private boolean oneChannelPerEntry = true; public MismatchInfoPanel(Minecraft client, int width, int height, int top, int left) { super(client, width, height, top, left); + updateListContent(); + } - //The raw list of the strings in a table row, the components may still be too long for the final table and will be split up later. The first row element may have a style assigned to it that will be used for the whole content row. + private void updateListContent() { + Map, Component> mergedChannelData = sortAndMergeChannelData(mismatchedChannelData); record Row(MutableComponent name, MutableComponent reason) {} + //The raw list of the strings in a table row, the components may still be too long for the final table and will be split up later. The first row element may have a style assigned to it that will be used for the whole content row. List rows = new ArrayList<>(); - if (!mismatchedChannelData.isEmpty()) { - //This table section contains the mod name and both mod versions of each mod that has a mismatching client and server version + if (!mergedChannelData.isEmpty()) { + //Each table row contains the channel id(s) and the reason for the corresponding channel mismatch. rows.add(new Row(Component.translatable("fml.modmismatchscreen.table.channelname"), Component.translatable("fml.modmismatchscreen.table.reason"))); int i = 0; - for (Map.Entry modData : mismatchedChannelData.entrySet()) { - rows.add(new Row(toChannelNameComponent(modData.getKey()), modData.getValue().copy())); - if (++i >= 10) { - //If too many mismatched mod entries are present, append a line referencing how to see the full list and stop rendering any more entries - rows.add(new Row(Component.literal(""), Component.translatable("fml.modmismatchscreen.additional", mismatchedChannelData.size() - i).withStyle(ChatFormatting.ITALIC))); + for (Map.Entry, Component> channelData : mergedChannelData.entrySet()) { + rows.add(new Row(toChannelComponent(channelData.getKey(), i % 2 == 0 ? ChatFormatting.GOLD : ChatFormatting.YELLOW), channelData.getValue().copy())); + if (++i == 30 && mergedChannelData.size() > 30) { + //If too many mismatched channel entries are present, append a line referencing how to see the full list and stop rendering any more entries + rows.add(new Row(Component.literal(""), Component.translatable("fml.modmismatchscreen.additional", mergedChannelData.size() - i).withStyle(ChatFormatting.ITALIC))); break; } } @@ -111,21 +136,49 @@ record Row(MutableComponent name, MutableComponent reason) {} this.lineTable = rows.stream().flatMap(p -> splitLineToWidth(p.name(), p.reason()).stream()).collect(Collectors.toList()); this.contentSize = lineTable.size(); + this.scrollDistance = 0; + } + + /** + * Iterates over the raw channel mismatch data and merges entries with the same reason component into one channel mismatch entry each. + * Due to the reason component always containing the display name of the mod that likely owns the associated channel, this step effectively groups channels by their most likely owning mod candidate, + * so users can see more easily which mods might be the culprits of the negotiation failure that caused this screen to appear. + * + * @param mismatchedChannelData The raw mismatched channel data received from the server, which might contain entries with duplicate channel mismatch reasons + * @return A map containing channel mismatch entries with unique reasons. Each channel mismatch entry contains the list of all channels that share the same reason component, + * mapped to that reason component. + */ + private Map, Component> sortAndMergeChannelData(Map mismatchedChannelData) { + Map> channelsByReason = new LinkedHashMap<>(); + List sortedChannels = mismatchedChannelData.keySet().stream().sorted(Comparator.comparing(ResourceLocation::toString)).toList(); + for (ResourceLocation channel : sortedChannels) { + Component channelMismatchReason = mismatchedChannelData.get(channel); + if (channelsByReason.containsKey(channelMismatchReason)) + channelsByReason.get(channelMismatchReason).add(channel); + else + channelsByReason.put(channelMismatchReason, Lists.newArrayList(channel)); + } + + Map, Component> channelMismatchEntries = new LinkedHashMap<>(); + List sortedChannelEntries = channelsByReason.entrySet().stream().sorted(Comparator.comparing(entry -> entry.getValue().get(0).toString())).map(Map.Entry::getKey).toList(); + for (Component mismatchReason : sortedChannelEntries) { + channelMismatchEntries.put(channelsByReason.get(mismatchReason), mismatchReason); + } + + return channelMismatchEntries; } /** - * Splits the raw name and version strings, making them use multiple lines if needed, to fit within the table dimensions. + * Splits the raw channel namespace and mismatch reason strings, making them use multiple lines if needed, to fit within the table dimensions. * The style assigned to the name element is then applied to the entire content row. * - * @param name The first element of the content row, usually representing a table section header or the name of a mod entry - * @param reason The second element of the content row, usually representing the reason why the mod is mismatched + * @param name The first element of the content row, usually representing a table section header or a channel name entry + * @param reason The second element of the content row, usually representing the reason why the channel is mismatched * @return A list of table rows consisting of 2 elements each which consist of the same content as was given by the parameters, but split up to fit within the table dimensions. */ private List> splitLineToWidth(MutableComponent name, MutableComponent reason) { Style style = name.getStyle(); - int versionColumns = 1; - int adaptedNameWidth = nameWidth + versionWidth * (2 - versionColumns) - 4; //the name width may be expanded when the version column string is missing - List nameLines = font.split(name, adaptedNameWidth); + List nameLines = font.split(name, nameWidth - 4); List reasonLines = font.split(reason.setStyle(style), versionWidth - 4); List> splitLines = new ArrayList<>(); @@ -137,26 +190,30 @@ private List> splitLineToWidt } /** - * Adds a style information to the given mod name string. The style assigned to the returned component contains the color of the mod name, - * a hover event containing the given id, and an optional click event, which opens the homepage of mod, if present. + * Formats the given list of channel ids to a component which, depending on the current display mode of the list, will list either the first or all channel ids. + * If only one channel id is shown, the amount of channels that have the same reason component will also be displayed next to the channel id. + * The component is colored in the given color, which will be used for the whole content row. * - * @param id An id that gets displayed in the hover event. Depending on the origin it may only consist of a namespace (the mod id) or a namespace + path (a channel id associated with the mod). - * @return A component with the mod name as the main text component, and an assigned style which will be used for the whole content row. + * @param ids The list of channel ids to be formatted to the component. Depending on the current list mode, either the full list or the first entry will be used for the component text. + * @param color Defines the color of the returned component. + * @return A component with one or all entries of the channel id list as the main text component, and an assigned color which will be used for the whole content row. */ - private MutableComponent toChannelNameComponent(ResourceLocation id) { - String modId = id.getNamespace(); - - String url = ModList.get().getModContainerById(modId) - .flatMap(container -> container.getModInfo().getModURL()) - .map(URL::toString) - .orElse(""); - MutableComponent result = Component.literal(id.toString()).withStyle(ChatFormatting.YELLOW); - if (!url.isEmpty()) { - result = result.withStyle(s -> s.withHoverEvent(new HoverEvent(Action.SHOW_TEXT, Component.translatable("fml.modmismatchscreen.table.visit.mod_page", id.toString())))) - .withStyle(s -> s.withClickEvent(new ClickEvent(ClickEvent.Action.OPEN_URL, url))); - } + private MutableComponent toChannelComponent(List ids, ChatFormatting color) { + MutableComponent namespaceComponent; + if (oneChannelPerEntry) { + namespaceComponent = Component.literal(ids.get(0).toString()).withStyle(color); + + if (ids.size() > 1) + namespaceComponent.append(Component.literal("\n[+%s more]".formatted(ids.size() - 1)).withStyle(ChatFormatting.DARK_GRAY)); + } else + namespaceComponent = ComponentUtils.formatList(ids, ComponentUtils.DEFAULT_SEPARATOR, r -> Component.literal(r.toString())).withStyle(color); + + return namespaceComponent; + } - return result; + public void toggleSimplifiedView() { + this.oneChannelPerEntry = !this.oneChannelPerEntry; + updateListContent(); } @Override diff --git a/src/main/java/net/neoforged/neoforge/network/negotiation/NetworkComponentNegotiator.java b/src/main/java/net/neoforged/neoforge/network/negotiation/NetworkComponentNegotiator.java index ae22a2378f..15abfda186 100644 --- a/src/main/java/net/neoforged/neoforge/network/negotiation/NetworkComponentNegotiator.java +++ b/src/main/java/net/neoforged/neoforge/network/negotiation/NetworkComponentNegotiator.java @@ -14,6 +14,7 @@ import java.util.Optional; import net.minecraft.network.chat.Component; import net.minecraft.resources.ResourceLocation; +import net.neoforged.fml.ModList; import org.jetbrains.annotations.ApiStatus; import org.jetbrains.annotations.Nullable; import org.jetbrains.annotations.VisibleForTesting; @@ -77,13 +78,21 @@ public static NegotiationResult negotiate(List serve if (!client.isEmpty()) { final Map failureReasons = new HashMap<>(); - client.forEach(c -> failureReasons.put(c.id(), Component.translatable("neoforge.network.negotiation.failure.missing.client.server"))); + client.forEach(c -> { + Component channelFailureReason = Component.translatable("neoforge.network.negotiation.failure.missing.client.server"); + String modDisplayName = ModList.get().getModContainerById(c.id().getNamespace()).map(mc -> mc.getModInfo().getDisplayName()).orElse(""); + failureReasons.put(c.id(), modDisplayName.isEmpty() ? channelFailureReason : Component.translatable("neoforge.network.negotiation.failure.mod", modDisplayName, channelFailureReason)); + }); return new NegotiationResult(List.of(), false, failureReasons); } if (!server.isEmpty()) { final Map failureReasons = new HashMap<>(); - server.forEach(c -> failureReasons.put(c.id(), Component.translatable("neoforge.network.negotiation.failure.missing.server.client"))); + server.forEach(c -> { + Component channelFailureReason = Component.translatable("neoforge.network.negotiation.failure.missing.server.client"); + String modDisplayName = ModList.get().getModContainerById(c.id().getNamespace()).map(mc -> mc.getModInfo().getDisplayName()).orElse(""); + failureReasons.put(c.id(), modDisplayName.isEmpty() ? channelFailureReason : Component.translatable("neoforge.network.negotiation.failure.mod", modDisplayName, channelFailureReason)); + }); return new NegotiationResult(List.of(), false, failureReasons); } @@ -92,16 +101,17 @@ public static NegotiationResult negotiate(List serve for (Table.Cell match : matches.cellSet()) { final NegotiableNetworkComponent serverComponent = match.getColumnKey(); final NegotiableNetworkComponent clientComponent = match.getValue(); + final String modDisplayName = ModList.get().getModContainerById(serverComponent.id().getNamespace()).map(mc -> mc.getModInfo().getDisplayName()).orElse(""); Optional serverToClientComparison = validateComponent(serverComponent, clientComponent, "client"); if (serverToClientComparison.isPresent() && !serverToClientComparison.get().success()) { - failureReasons.put(serverComponent.id(), serverToClientComparison.get().failureReason()); + failureReasons.put(serverComponent.id(), modDisplayName.isEmpty() ? serverToClientComparison.get().failureReason() : Component.translatable("neoforge.network.negotiation.failure.mod", modDisplayName, serverToClientComparison.get().failureReason())); continue; } Optional clientToServerComparison = validateComponent(clientComponent, serverComponent, "server"); if (clientToServerComparison.isPresent() && !clientToServerComparison.get().success()) { - failureReasons.put(serverComponent.id(), clientToServerComparison.get().failureReason()); + failureReasons.put(serverComponent.id(), modDisplayName.isEmpty() ? clientToServerComparison.get().failureReason() : Component.translatable("neoforge.network.negotiation.failure.mod", modDisplayName, clientToServerComparison.get().failureReason())); continue; } diff --git a/src/main/resources/assets/neoforge/lang/en_us.json b/src/main/resources/assets/neoforge/lang/en_us.json index 1c754c1fe9..af74504ca2 100644 --- a/src/main/resources/assets/neoforge/lang/en_us.json +++ b/src/main/resources/assets/neoforge/lang/en_us.json @@ -67,6 +67,7 @@ "fml.modloading.uncaughterror":"An uncaught parallel processing error has occurred.\n\u00a77{2,exc,msg}", "fml.modloading.errorduringevent":"{0,modinfo,name} ({0,modinfo,id}) encountered an error during the {1,lower} event phase\n\u00a77{2,exc,msg}", "fml.modloading.failedtoloadforge": "Failed to load NeoForge", + "fml.modmismatchscreen.simplifiedview": "Simplified view", "fml.modloading.missingdependency": "Mod \u00a7e{4}\u00a7r requires \u00a76{3}\u00a7r \u00a7o{5,vr}\u00a7r\n\u00a77Currently, \u00a76{3}\u00a7r\u00a77 is \u00a7o{6,i18n,fml.messages.artifactversion.ornotinstalled}§r\n{7,optional,§7Reason for the dependency: §r}", "fml.modloading.missingdependency.optional": "Mod \u00a7e{4}\u00a7r only supports \u00a73{3}\u00a7r \u00a7o{5,vr}\u00a7r\n\u00a77Currently, \u00a73{3}\u00a7r\u00a77 is \u00a7o{6}", @@ -243,6 +244,7 @@ "pack.neoforge.description": "NeoForge data/resource pack", + "neoforge.network.negotiation.failure.mod": "Channel of mod \"%1$s\" failed to connect: %2$s", "neoforge.network.negotiation.failure.missing.client.server": "This channel is missing on the server side, but required on the client!", "neoforge.network.negotiation.failure.missing.server.client": "This channel is missing on the client side, but required on the server!", "neoforge.network.negotiation.failure.flow.client.missing": "The client wants the payload to flow: %s, but the server does not support it!",