1
0
mirror of https://github.com/chylex/Minecraft-Window-Title.git synced 2025-12-18 07:58:56 +01:00

2 Commits

Author SHA1 Message Date
Kolotheegg
15b922a6f2 Merge c62d8658c4 into ba22745940 2024-07-28 21:26:24 +02:00
Kolotheegg
c62d8658c4 Reimplemented Icon Support.
Used Lambda functions for config reading
2024-07-28 21:23:37 +02:00
24 changed files with 222 additions and 310 deletions

2
.github/FUNDING.yml vendored
View File

@@ -1 +1,3 @@
github: chylex
patreon: chylex
ko_fi: chylex ko_fi: chylex

1
.idea/gradle.xml generated
View File

@@ -12,7 +12,6 @@
<option value="$PROJECT_DIR$/NeoForge" /> <option value="$PROJECT_DIR$/NeoForge" />
</set> </set>
</option> </option>
<option name="resolveExternalAnnotations" value="true" />
</GradleProjectSettings> </GradleProjectSettings>
</option> </option>
</component> </component>

View File

@@ -1,8 +1,6 @@
import net.fabricmc.loom.configuration.ide.RunConfigSettings
import org.gradle.jvm.tasks.Jar import org.gradle.jvm.tasks.Jar
val modId: String by project val modId: String by project
val modSides: String by project
val minecraftVersion: String by project val minecraftVersion: String by project
val fabricVersion: String by project val fabricVersion: String by project
@@ -26,24 +24,12 @@ loom {
ideConfigGenerated(true) ideConfigGenerated(true)
} }
fun side(name: String, configure: RunConfigSettings.() -> Unit) { named("client") {
if (modSides == "both" || modSides == name) {
named(name, configure)
}
else {
findByName(name)?.let(::remove)
}
}
side("client") {
configName = "Fabric Client" configName = "Fabric Client"
client() client()
} }
side("server") { findByName("server")?.let(::remove)
configName = "Fabric Server"
server()
}
} }
mixin { mixin {

View File

@@ -1,8 +1,10 @@
package chylex.customwindowtitle.fabric; package chylex.customwindowtitle.fabric;
import chylex.customwindowtitle.TitleChanger; import chylex.customwindowtitle.IconChanger;
import chylex.customwindowtitle.TitleConfig; import chylex.customwindowtitle.TitleConfig;
import chylex.customwindowtitle.TitleParser;
import chylex.customwindowtitle.data.CommonTokenData; import chylex.customwindowtitle.data.CommonTokenData;
import chylex.customwindowtitle.mixin.DisableVanillaTitle;
import net.fabricmc.api.ClientModInitializer; import net.fabricmc.api.ClientModInitializer;
import net.fabricmc.loader.api.FabricLoader; import net.fabricmc.loader.api.FabricLoader;
import net.minecraft.client.Minecraft; import net.minecraft.client.Minecraft;
@@ -11,12 +13,16 @@ public class CustomWindowTitle implements ClientModInitializer {
private final TitleConfig config; private final TitleConfig config;
public CustomWindowTitle() { public CustomWindowTitle() {
config = TitleConfig.load(FabricLoader.getInstance().getConfigDir().toAbsolutePath().toString()); config = TitleConfig.read(FabricLoader.getInstance().getConfigDir().toAbsolutePath().toString());
} }
@Override @Override
public void onInitializeClient() { public void onInitializeClient() {
CommonTokenData.register(new TokenProvider()); CommonTokenData.register(new TokenProvider());
Minecraft.getInstance().execute(() -> TitleChanger.setTitle(config)); Minecraft.getInstance().execute(this::updateTitle);
}
private void updateTitle() {
Minecraft.getInstance().getWindow().setTitle(TitleParser.parse(config.getTitle()));
} }
} }

View File

@@ -1,47 +1,24 @@
import net.neoforged.moddevgradle.dsl.RunModel
val modId: String by project
val modSides: String by project
val neoForgeVersion: String by project val neoForgeVersion: String by project
plugins { plugins {
id("net.neoforged.moddev") id("net.neoforged.gradle.userdev")
id("net.neoforged.gradle.mixin")
} }
neoForge { dependencies {
version = neoForgeVersion implementation("net.neoforged:neoforge:$neoForgeVersion")
}
mods { runs {
register(modId) { val runJvmArgs: Set<String> by project
sourceSet(sourceSets.main.get())
sourceSet(rootProject.sourceSets.main.get()) configureEach {
} workingDirectory = file("../run")
modSource(project.sourceSets.main.get())
jvmArguments(runJvmArgs)
} }
runs { removeIf { it.name != "client" }
val runJvmArgs: Set<String> by project
configureEach {
gameDirectory = file("../run")
jvmArguments.addAll(runJvmArgs)
}
fun side(name: String, configure: RunModel.() -> Unit) {
if (modSides == "both" || modSides == name) {
register(name, configure)
}
}
side("client") {
ideName.set("NeoForge Client")
client()
}
side("server") {
ideName.set("NeoForge Server")
server()
}
}
} }
tasks.processResources { tasks.processResources {

View File

@@ -1,8 +1,9 @@
package chylex.customwindowtitle.neoforge; package chylex.customwindowtitle.neoforge;
import chylex.customwindowtitle.TitleChanger;
import chylex.customwindowtitle.TitleConfig; import chylex.customwindowtitle.TitleConfig;
import chylex.customwindowtitle.TitleParser;
import chylex.customwindowtitle.data.CommonTokenData; import chylex.customwindowtitle.data.CommonTokenData;
import net.minecraft.client.Minecraft;
import net.neoforged.bus.api.IEventBus; import net.neoforged.bus.api.IEventBus;
import net.neoforged.bus.api.SubscribeEvent; import net.neoforged.bus.api.SubscribeEvent;
import net.neoforged.fml.common.Mod; import net.neoforged.fml.common.Mod;
@@ -15,13 +16,17 @@ public class CustomWindowTitle {
private final TitleConfig config; private final TitleConfig config;
public CustomWindowTitle(IEventBus eventBus) { public CustomWindowTitle(IEventBus eventBus) {
config = TitleConfig.load(FMLPaths.CONFIGDIR.get().toString()); config = TitleConfig.read(FMLPaths.CONFIGDIR.get().toString());
eventBus.addListener(this::onClientSetup); eventBus.addListener(this::onClientSetup);
CommonTokenData.register(new TokenProvider());
} }
@SubscribeEvent @SubscribeEvent
public void onClientSetup(final FMLClientSetupEvent e) { public void onClientSetup(final FMLClientSetupEvent e) {
CommonTokenData.register(new TokenProvider()); e.enqueueWork(this::updateTitle);
e.enqueueWork(() -> TitleChanger.setTitle(config)); }
private void updateTitle() {
Minecraft.getInstance().getWindow().setTitle(TitleParser.parse(config.getTitle()));
} }
} }

View File

@@ -14,7 +14,8 @@ To change the title or icon, navigate to the `.minecraft/config` folder, and ope
```toml ```toml
title = 'Minecraft {mcversion}' title = 'Minecraft {mcversion}'
icon = '' icon16 = ''
icon32 = ''
``` ```
Only edit text inside quotes or apostrophes. Only edit text inside quotes or apostrophes.
@@ -31,30 +32,23 @@ If any of the tokens aren't working, search the game log for **CustomWindowTitle
### Changing the Icon ### Changing the Icon
#### Minecraft 1.21+ **This feature is currently not supported in Minecraft 1.20+.**
**This feature is only available in Custom Window Title 1.4.0 and newer.** You must create two PNG images with sizes 16x16 and 32x32 pixels. The images **must be saved with transparency** even if they don't use it, **otherwise the icons will appear corrupted**. In Krita for example, you must check _Store alpha channel (transparency)_ when saving the image.
Create a square PNG image whose dimensions are a power of two, such as 32x32 or 48x48. Put the PNG file into the `.minecraft/config` folder, either directly or into a subfolder. The _icon16_ and _icon32_ configuration entries point to the PNG files relative to the `.minecraft/config` folder. For example, if you place the two icons in a folder named _customwindowtitle_ as follows:
The icon **must be saved with transparency** even if it doesn't use it, otherwise the icon may be corrupted or not appear at all. In Krita, for example, you must check _Store alpha channel (transparency)_ when saving. * `.minecraft/config/customwindowtitle-client.toml`
* `.minecraft/config/customwindowtitle/icon16.png`
* `.minecraft/config/customwindowtitle/icon32.png`
The _icon_ configuration entry points to the PNG file relative to `.minecraft/config`. Then, the two icon entries should look like this:
For example, if you placed the icon into `.minecraft/config/customwindowtitle/icon.png`, then the configuration entry should look like this:
```toml ```toml
icon = 'customwindowtitle/icon.png' icon16 = 'customwindowtitle/icon16.png'
icon32 = 'customwindowtitle/icon32.png'
``` ```
#### Minecraft 1.20
This feature is not available in Minecraft 1.20.
#### Minecraft 1.19 and older
Instead of one `icon` configuration entry, there are two configuration entries `icon16` and `icon32` for icons with dimensions 16x16 and 32x32.
## Screenshots ## Screenshots
These screenshots were taken using the following example configuration: These screenshots were taken using the following example configuration:

View File

@@ -116,10 +116,6 @@ allprojects {
inputs.property("minimumMinecraftVersion", minimumMinecraftVersion) inputs.property("minimumMinecraftVersion", minimumMinecraftVersion)
inputs.property("minimumNeoForgeVersion", minimumNeoForgeVersion) inputs.property("minimumNeoForgeVersion", minimumNeoForgeVersion)
inputs.property("minimumFabricVersion", minimumFabricVersion) inputs.property("minimumFabricVersion", minimumFabricVersion)
from(rootProject.file("logo.png")) {
into("assets/$modId")
}
} }
tasks.withType<AbstractArchiveTask>().configureEach { tasks.withType<AbstractArchiveTask>().configureEach {
@@ -137,8 +133,10 @@ subprojects {
archivesName.set("$modNameStripped-${project.name}") archivesName.set("$modNameStripped-${project.name}")
} }
tasks.compileJava { listOf("compileJava", "compileTestJava").forEach {
source({ rootProject.sourceSets.main.get().allSource }) tasks.named<JavaCompile>(it) {
source({ rootProject.sourceSets.main.get().allSource })
}
} }
tasks.processResources { tasks.processResources {

View File

@@ -17,7 +17,6 @@ icon32 = ''
``` ```
Only edit text inside quotes or apostrophes. Only edit text inside quotes or apostrophes.
### Changing the Title ### Changing the Title
You can use the following special tokens in the _title_ configuration entry: You can use the following special tokens in the _title_ configuration entry:
@@ -30,32 +29,24 @@ If any of the tokens aren't working, search the game log for **CustomWindowTitle
### Changing the Icon ### Changing the Icon
#### Minecraft 1.21+ **This feature is currently not supported in Minecraft 1.20+.**
**This feature is only available in Custom Window Title 1.4.0 and newer.** You must create two PNG images with sizes 16x16 and 32x32 pixels. The images **must be saved with transparency** even if they don't use it, **otherwise the icons will appear corrupted**. In Krita for example, you must check _Store alpha channel (transparency)_ when saving the image.
Create a square PNG image whose dimensions are a power of two, such as 32x32 or 48x48. Put the PNG file into the `.minecraft/config` folder, either directly or into a subfolder. The _icon16_ and _icon32_ configuration entries point to the PNG files relative to the `.minecraft/config` folder. For example, if you place the two icons in a folder named _customwindowtitle_ as follows:
The icon **must be saved with transparency** even if it doesn't use it, otherwise the icon may be corrupted or not appear at all. In Krita, for example, you must check _Store alpha channel (transparency)_ when saving. * `.minecraft/config/customwindowtitle-client.toml`
* `.minecraft/config/customwindowtitle/icon16.png`
* `.minecraft/config/customwindowtitle/icon32.png`
The _icon_ configuration entry points to the PNG file relative to `.minecraft/config`. Then, the two icon entries should look like this:
For example, if you placed the icon into `.minecraft/config/customwindowtitle/icon.png`, then the configuration entry should look like this:
```toml ```toml
icon = 'customwindowtitle/icon.png' icon16 = 'customwindowtitle/icon16.png'
icon32 = 'customwindowtitle/icon32.png'
``` ```
#### Minecraft 1.20
This feature is not available in Minecraft 1.20.
#### Minecraft 1.19 and older
Instead of one `icon` configuration entry, there are two configuration entries `icon16` and `icon32` for icons with dimensions 16x16 and 32x32.
## Screenshots ## Screenshots
These screenshots were taken using the following example configuration: These screenshots were taken using the following example configuration:
```toml ```toml

View File

@@ -17,7 +17,6 @@ icon32 = ''
``` ```
Only edit text inside quotes or apostrophes. Only edit text inside quotes or apostrophes.
### Changing the Title ### Changing the Title
You can use the following special tokens in the _title_ configuration entry: You can use the following special tokens in the _title_ configuration entry:
@@ -30,32 +29,24 @@ If any of the tokens aren't working, search the game log for **CustomWindowTitle
### Changing the Icon ### Changing the Icon
#### Minecraft 1.21+ **This feature is currently not supported in Minecraft 1.20+.**
**This feature is only available in Custom Window Title 1.4.0 and newer.** You must create two PNG images with sizes 16x16 and 32x32 pixels. The images **must be saved with transparency** even if they don't use it, **otherwise the icons will appear corrupted**. In Krita for example, you must check _Store alpha channel (transparency)_ when saving the image.
Create a square PNG image whose dimensions are a power of two, such as 32x32 or 48x48. Put the PNG file into the `.minecraft/config` folder, either directly or into a subfolder. The _icon16_ and _icon32_ configuration entries point to the PNG files relative to the `.minecraft/config` folder. For example, if you place the two icons in a folder named _customwindowtitle_ as follows:
The icon **must be saved with transparency** even if it doesn't use it, otherwise the icon may be corrupted or not appear at all. In Krita, for example, you must check _Store alpha channel (transparency)_ when saving. * `.minecraft/config/customwindowtitle-client.toml`
* `.minecraft/config/customwindowtitle/icon16.png`
* `.minecraft/config/customwindowtitle/icon32.png`
The _icon_ configuration entry points to the PNG file relative to `.minecraft/config`. Then, the two icon entries should look like this:
For example, if you placed the icon into `.minecraft/config/customwindowtitle/icon.png`, then the configuration entry should look like this:
```toml ```toml
icon = 'customwindowtitle/icon.png' icon16 = 'customwindowtitle/icon16.png'
icon32 = 'customwindowtitle/icon32.png'
``` ```
#### Minecraft 1.20
This feature is not available in Minecraft 1.20.
#### Minecraft 1.19 and older
Instead of one `icon` configuration entry, there are two configuration entries `icon16` and `icon32` for icons with dimensions 16x16 and 32x32.
## Screenshots ## Screenshots
These screenshots were taken using the following example configuration: These screenshots were taken using the following example configuration:
```toml ```toml

View File

@@ -1,31 +1,30 @@
# Mod # Mod
modId=customwindowtitle modId=customwindowtitle
modName=Custom Window Title modName=Custom Window Title
modDescription=Customize window title and icon. modDescription=Customize window title.
modAuthor=chylex modAuthor=chylex
modVersion=1.4.3 modVersion=1.3.0
modLicense=Unlicense modLicense=Unlicense
modSourcesURL=https://github.com/chylex/Minecraft-Window-Title modSourcesURL=https://github.com/chylex/Minecraft-Window-Title
modIssuesURL=https://github.com/chylex/Minecraft-Window-Title/issues modIssuesURL=https://github.com/chylex/Minecraft-Window-Title/issues
modSides=client modSides=client
# Dependencies # Dependencies
minecraftVersion=1.21.9 minecraftVersion=1.21
neoForgeVersion=21.9.2-beta neoForgeVersion=21.0.0-beta
neoModDevVersion=2.0.110 neoGradleVersion=7.0.152
fabricVersion=0.17.2 fabricVersion=0.15.11
loomVersion=1.10 loomVersion=1.7
mixinVersion=0.12.5+mixin.0.8.5 mixinVersion=0.12.5+mixin.0.8.5
# https://projects.neoforged.net/neoforged/neoforge # https://projects.neoforged.net/neoforged/neogradle/
# https://projects.neoforged.net/neoforged/moddevgradle
# https://fabricmc.net/develop/ # https://fabricmc.net/develop/
# https://github.com/FabricMC/fabric-loom/releases # https://github.com/FabricMC/fabric-loom/releases
# Constraints # Constraints
minimumMinecraftVersion=1.21.9 minimumMinecraftVersion=1.21
minimumNeoForgeVersion=21.9.0-beta minimumNeoForgeVersion=21.0.0-beta
minimumFabricVersion=0.15.0 minimumFabricVersion=0.12.3
# Gradle # Gradle
org.gradle.jvmargs=-Xmx3G org.gradle.jvmargs=-Xmx3G

Binary file not shown.

View File

@@ -1,6 +1,6 @@
distributionBase=GRADLE_USER_HOME distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-8.12.1-bin.zip distributionUrl=https\://services.gradle.org/distributions/gradle-8.8-bin.zip
networkTimeout=10000 networkTimeout=10000
validateDistributionUrl=true validateDistributionUrl=true
zipStoreBase=GRADLE_USER_HOME zipStoreBase=GRADLE_USER_HOME

4
gradlew vendored
View File

@@ -15,8 +15,6 @@
# See the License for the specific language governing permissions and # See the License for the specific language governing permissions and
# limitations under the License. # limitations under the License.
# #
# SPDX-License-Identifier: Apache-2.0
#
############################################################################## ##############################################################################
# #
@@ -86,7 +84,7 @@ done
# shellcheck disable=SC2034 # shellcheck disable=SC2034
APP_BASE_NAME=${0##*/} APP_BASE_NAME=${0##*/}
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036) # Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s\n' "$PWD" ) || exit APP_HOME=$( cd "${APP_HOME:-./}" > /dev/null && pwd -P ) || exit
# Use the maximum available, or set MAX_FD != -1 to use that value. # Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD=maximum MAX_FD=maximum

2
gradlew.bat vendored
View File

@@ -13,8 +13,6 @@
@rem See the License for the specific language governing permissions and @rem See the License for the specific language governing permissions and
@rem limitations under the License. @rem limitations under the License.
@rem @rem
@rem SPDX-License-Identifier: Apache-2.0
@rem
@if "%DEBUG%"=="" @echo off @if "%DEBUG%"=="" @echo off
@rem ########################################################################## @rem ##########################################################################

View File

@@ -8,9 +8,10 @@ pluginManagement {
} }
plugins { plugins {
val neoModDevVersion = settings.extra.get("neoModDevVersion") as? String val neoGradleVersion = settings.extra.get("neoGradleVersion") as? String
if (neoModDevVersion != null) { if (neoGradleVersion != null) {
id("net.neoforged.moddev") version neoModDevVersion id("net.neoforged.gradle.userdev") version neoGradleVersion
id("net.neoforged.gradle.mixin") version neoGradleVersion
} }
val loomVersion = settings.extra.get("loomVersion") as? String val loomVersion = settings.extra.get("loomVersion") as? String

View File

@@ -1,65 +1,65 @@
package chylex.customwindowtitle; package chylex.customwindowtitle;
import net.minecraft.client.Minecraft; import net.minecraft.client.Minecraft;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.lwjgl.glfw.GLFW;
import org.lwjgl.glfw.GLFWImage; import org.lwjgl.glfw.GLFWImage;
import org.lwjgl.stb.STBImage; import org.lwjgl.stb.STBImage;
import org.lwjgl.system.MemoryStack; import org.lwjgl.system.MemoryStack;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream;
import java.nio.ByteBuffer; import java.nio.ByteBuffer;
import java.nio.IntBuffer; import java.nio.IntBuffer;
import java.nio.file.Files; import java.nio.file.Files;
import java.nio.file.Path; import java.nio.file.Path;
public final class IconChanger { public class IconChanger {
private static final Logger LOGGER = LogManager.getLogger("CustomWindowTitle");
private IconChanger() {} public static void setIcon(final Path iconPath) {
final long windowHandle = Minecraft.getInstance().getWindow().getWindow();
setWindowIcon(windowHandle, iconPath);
}
public static void setIcon(Path iconPath) { private static void setWindowIcon(final long windowHandle, final Path iconPath) {
long windowHandle = Minecraft.getInstance().getWindow().handle(); try (final MemoryStack stack = MemoryStack.stackPush()) {
setWindowIcon(windowHandle, iconPath); final IntBuffer w = stack.mallocInt(1);
} final IntBuffer h = stack.mallocInt(1);
final IntBuffer channels = stack.mallocInt(1);
private static void setWindowIcon(long windowHandle, Path iconPath) { final ByteBuffer icon = loadIcon(iconPath, w, h, channels);
ByteBuffer icon = null; if (icon != null) {
try (MemoryStack stack = MemoryStack.stackPush()) { final GLFWImage glfwImage1 = GLFWImage.malloc();
IntBuffer w = stack.mallocInt(1); glfwImage1.set(w.get(0), h.get(0), icon);
IntBuffer h = stack.mallocInt(1); final GLFWImage glfwImage2 = GLFWImage.malloc();
IntBuffer channels = stack.mallocInt(1); glfwImage2.set(w.get(0), h.get(0), icon);
icon = loadIcon(iconPath, w, h, channels); final GLFWImage.Buffer icons = GLFWImage.malloc(2);
if (icon == null) { icons.put(0, glfwImage1);
return; icons.put(1, glfwImage2);
}
try (GLFWImage.Buffer icons = GLFWImage.malloc(1)) { org.lwjgl.glfw.GLFW.glfwSetWindowIcon(windowHandle, icons);
GLFWImage iconImage = icons.get(0);
iconImage.set(w.get(0), h.get(0), icon);
GLFW.glfwSetWindowIcon(windowHandle, icons); icons.free();
} glfwImage1.free();
} catch (Exception e) { glfwImage2.free();
LOGGER.error("Failed to set window icon from path: {}", iconPath, e); } else {
} finally { System.err.println("Failed to load icon: " + iconPath);
if (icon != null) { }
STBImage.stbi_image_free(icon); }
} }
}
}
private static ByteBuffer loadIcon(Path path, IntBuffer w, IntBuffer h, IntBuffer channels) throws IOException { private static ByteBuffer loadIcon(final Path path, final IntBuffer w, final IntBuffer h, final IntBuffer channels) {
byte[] iconBytes = Files.readAllBytes(path); try (final InputStream inputStream = Files.newInputStream(path)) {
final byte[] iconBytes = inputStream.readAllBytes();
ByteBuffer buffer = ByteBuffer.allocateDirect(iconBytes.length).put(iconBytes).flip(); final ByteBuffer buffer = ByteBuffer.allocateDirect(iconBytes.length).put(iconBytes).flip();
ByteBuffer icon = STBImage.stbi_load_from_memory(buffer, w, h, channels, 4); final ByteBuffer icon = STBImage.stbi_load_from_memory(buffer, w, h, channels, 4);
if (icon == null) {
if (icon == null) { System.err.println("Failed to load image from memory for: " + path + " - " + STBImage.stbi_failure_reason());
LOGGER.error("Failed to load image from path: {} - {}", path, STBImage.stbi_failure_reason()); }
} return icon;
} catch (final IOException e) {
return icon; e.printStackTrace();
} return null;
}
}
} }

View File

@@ -1,11 +0,0 @@
package chylex.customwindowtitle;
import net.minecraft.client.Minecraft;
public final class TitleChanger {
private TitleChanger() {}
public static void setTitle(TitleConfig config) {
Minecraft.getInstance().getWindow().setTitle(TitleParser.parse(config.getTitle()));
}
}

View File

@@ -1,91 +1,75 @@
package chylex.customwindowtitle; package chylex.customwindowtitle;
import com.google.common.collect.ImmutableMap;
import java.io.IOException; import java.io.IOException;
import java.nio.charset.StandardCharsets; import java.io.InputStream;
import java.nio.file.Files; import java.nio.file.Files;
import java.nio.file.Path; import java.nio.file.Path;
import java.nio.file.Paths; import java.nio.file.Paths;
import java.nio.charset.StandardCharsets;
import java.util.Collections;
import java.util.LinkedHashMap; import java.util.LinkedHashMap;
import java.util.Map; import java.util.Map;
import java.util.stream.Collectors; import java.util.stream.Collectors;
public final class TitleConfig { public final class TitleConfig {
private static final String KEY_TITLE = "title"; private static final Map<String, String> DEFAULTS;
private static final String KEY_ICON = "icon"; private static TitleConfig instance;
private static final ImmutableMap<String, String> DEFAULTS = ImmutableMap.<String, String>builder() static {
.put(KEY_TITLE, "Minecraft {mcversion}") final Map<String, String> defaults = new LinkedHashMap<>();
.put(KEY_ICON, "") defaults.put("title", "Minecraft {mcversion}");
.buildOrThrow(); defaults.put("squareIcon", "");
DEFAULTS = Collections.unmodifiableMap(defaults);
private static volatile TitleConfig instance;
public static TitleConfig getInstance() {
return instance;
} }
public static TitleConfig load(String folder) { public static TitleConfig read(final String folder) {
if (instance != null) {
throw new IllegalStateException("TitleConfig has already been loaded and cannot be loaded again.");
}
if (instance == null) { if (instance == null) {
synchronized(TitleConfig.class) { final Path configFile = Paths.get(folder, "customwindowtitle-client.toml");
if (instance == null) { final Map<String, String> config = new LinkedHashMap<>(DEFAULTS);
instance = loadImpl(folder);
try {
if (!Files.exists(configFile)) {
Files.write(configFile, config.entrySet().stream()
.map(entry -> String.format("%s = '%s'", entry.getKey(), entry.getValue()))
.collect(Collectors.toList()), StandardCharsets.UTF_8);
} else {
Files.readAllLines(configFile, StandardCharsets.UTF_8).stream()
.map(String::trim)
.filter(line -> !line.isEmpty())
.forEach(line -> {
final String[] split = line.split("=", 2);
if (split.length != 2) {
throw new RuntimeException("CustomWindowTitle configuration has an invalid line: " + line);
}
final String key = split[0].trim();
final String value = parseTrimmedValue(split[1].trim());
if (config.containsKey(key)) {
config.put(key, value);
} else {
throw new RuntimeException("CustomWindowTitle configuration has an invalid key: " + key);
}
});
} }
} catch (final IOException e) {
throw new RuntimeException("CustomWindowTitle configuration error", e);
} }
}
final String iconPath = config.get("squareIcon");
final Path pathIcon = iconPath.isEmpty() ? null : Paths.get(folder, iconPath);
if (pathIcon != null && Files.notExists(pathIcon)) {
throw new RuntimeException("CustomWindowTitle icon not found: " + pathIcon);
}
instance = new TitleConfig(config.get("title"), pathIcon);
}
return instance; return instance;
} }
private static TitleConfig loadImpl(String folder) {
Path configFile = Paths.get(folder, "customwindowtitle-client.toml");
Map<String, String> config = new LinkedHashMap<>(DEFAULTS);
try {
if (!Files.exists(configFile)) {
Files.write(configFile, config.entrySet().stream()
.map(entry -> String.format("%s = '%s'", entry.getKey(), entry.getValue()))
.collect(Collectors.toList()), StandardCharsets.UTF_8);
}
else {
Files.readAllLines(configFile, StandardCharsets.UTF_8).stream()
.map(String::trim)
.filter(line -> !line.isEmpty())
.forEach(line -> {
String[] split = line.split("=", 2);
if (split.length != 2) {
throw new RuntimeException("CustomWindowTitle configuration has an invalid line: " + line);
}
String key = split[0].trim();
String value = parseTrimmedValue(split[1].trim());
if (config.containsKey(key)) {
config.put(key, value);
}
else {
throw new RuntimeException("CustomWindowTitle configuration has an invalid key: " + key);
}
});
}
} catch (IOException e) {
throw new RuntimeException("CustomWindowTitle configuration error", e);
}
String iconPathStr = config.get(KEY_ICON);
Path iconPath = iconPathStr.isEmpty() ? null : Paths.get(folder, iconPathStr);
if (iconPath != null && Files.notExists(iconPath)) {
throw new RuntimeException("CustomWindowTitle configuration points to a missing icon: " + iconPath);
}
return new TitleConfig(config.get(KEY_TITLE), iconPath);
}
private static String parseTrimmedValue(String value) { private static String parseTrimmedValue(String value) {
if (value.isEmpty()) { if (value.isEmpty()) {
return value; return value;
@@ -113,6 +97,10 @@ public final class TitleConfig {
this.icon = icon; this.icon = icon;
} }
public static TitleConfig getInstance() {
return instance;
}
public String getTitle() { public String getTitle() {
return title; return title;
} }

View File

@@ -7,7 +7,7 @@ import org.apache.logging.log4j.Logger;
import java.util.regex.Matcher; import java.util.regex.Matcher;
import java.util.regex.Pattern; import java.util.regex.Pattern;
final class TitleParser { public final class TitleParser {
private static final Pattern tokenRegex = Pattern.compile("\\{([a-z]+)(?::([^}]+))?}"); private static final Pattern tokenRegex = Pattern.compile("\\{([a-z]+)(?::([^}]+))?}");
private static final Logger logger = LogManager.getLogger("CustomWindowTitle"); private static final Logger logger = LogManager.getLogger("CustomWindowTitle");

View File

@@ -5,10 +5,10 @@ import net.minecraft.client.Minecraft;
public interface CommonTokenProvider { public interface CommonTokenProvider {
default String getMinecraftVersion() { default String getMinecraftVersion() {
return SharedConstants.getCurrentVersion().name(); return SharedConstants.getCurrentVersion().getName();
} }
String getModVersion(String modId); String getModVersion(final String modId);
default String getUsername() { default String getUsername() {
return Minecraft.getInstance().getUser().getName(); return Minecraft.getInstance().getUser().getName();

View File

@@ -1,5 +1,7 @@
package chylex.customwindowtitle.mixin; package chylex.customwindowtitle.mixin;
import chylex.customwindowtitle.IconChanger;
import chylex.customwindowtitle.TitleConfig;
import net.minecraft.client.Minecraft; import net.minecraft.client.Minecraft;
import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.injection.At; import org.spongepowered.asm.mixin.injection.At;
@@ -8,8 +10,17 @@ import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
@Mixin(Minecraft.class) @Mixin(Minecraft.class)
public final class DisableVanillaTitle { public final class DisableVanillaTitle {
@Inject(method = "updateTitle()V", at = @At("HEAD"), cancellable = true) @Inject(method = "updateTitle()V", at = @At("HEAD"), cancellable = true)
private void updateTitle(final CallbackInfo info) { private void updateTitle(final CallbackInfo info) {
info.cancel(); info.cancel();
} }
@Inject(method = "onResourceLoadFinished", at = @At("HEAD"))
private void onFinishedLoading(final CallbackInfo callbackInfo) {
final TitleConfig config = TitleConfig.getInstance();
if (config != null && config.hasIcon()) {
IconChanger.setIcon(config.getIconPath());
}
}
} }

View File

@@ -1,20 +0,0 @@
package chylex.customwindowtitle.mixin;
import chylex.customwindowtitle.IconChanger;
import chylex.customwindowtitle.TitleConfig;
import net.minecraft.client.Minecraft;
import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.injection.At;
import org.spongepowered.asm.mixin.injection.Inject;
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
@Mixin(Minecraft.class)
public final class InitializeCustomIcon {
@Inject(method = "onResourceLoadFinished", at = @At("HEAD"))
private void onFinishedLoading(CallbackInfo callbackInfo) {
TitleConfig config = TitleConfig.getInstance();
if (config != null && config.hasIcon()) {
IconChanger.setIcon(config.getIconPath());
}
}
}

View File

@@ -5,8 +5,7 @@
"refmap": "customwindowtitle.refmap.json", "refmap": "customwindowtitle.refmap.json",
"compatibilityLevel": "JAVA_17", "compatibilityLevel": "JAVA_17",
"client": [ "client": [
"DisableVanillaTitle", "DisableVanillaTitle"
"InitializeCustomIcon"
], ],
"injectors": { "injectors": {
"defaultRequire": 1 "defaultRequire": 1