4 Commits

Author SHA1 Message Date
Claude
785efe24b9 v1.2.1 — add icon, drop contact homepage
- Embed music quiz icon at assets/chat_answer/icon.png; reference from
  fabric.mod.json so ModMenu shows it.
- Remove contact.homepage (private Gitea — owner-only).
2026-05-14 02:28:20 +09:00
Claude
e4bfda783a v1.2.0 — target MC 26.1.2
User reported NoClassDefFoundError: net/minecraft/class_7471 on MC 26.1.2.
Root cause: v1.1.1 was built for 1.21.6 intermediary, which uses
class_NNNN obfuscated names. MC 26.x ships an unobfuscated server jar
with Mojang names directly, so intermediary lookups for class_7471
(=PlayerChatMessage in 1.21.6) fail at runtime.

Build retargeted to 26.1.2:
- minecraft 26.1.2 / loader 0.19.2 / fabric-api 0.148.2+26.1.2
- Loom 1.16-SNAPSHOT, Shadow 9.4.1 (Java 25 bytecode support)
- Gradle 9.5.1, JDK 25 toolchain
- Drop officialMojangMappings(); use intermediary:0.0.0 identity
  (Mojang stopped publishing proguard mappings for 26.x).
- Adapt code: ServerPlayer.getServer() removed in 26.1.2 → use
  player.level().getServer() (ServerPlayer.level() returns ServerLevel).
- NeoForge dropped from this build — moddev plugin can't parse 26.1.2
  yet, falls back to 1.21.5.
- 1.2.0 = Fabric only; 1.21.6 users stay on 1.1.1.
2026-05-14 01:55:02 +09:00
Claude
fd9d17e818 v1.1.1 — make mod load on client (single-player) too
fabric.mod.json 의 environment 가 "server" 라 client 설치에선 Fabric Loader
가 모드를 건너뛰어 ModMenu 에 안 나타나는 문제. "*" 로 바꿔서 client (싱글
플레이 integrated server 포함) + dedicated server 양쪽에서 로드되게 함.
NeoForge 쪽도 side = "BOTH" 로 변경. 실제 로직은 변함없이 server 측 채팅
이벤트만 hook 하므로 client 단독에서는 no-op.
2026-05-14 00:46:11 +09:00
Claude
464762589c Add player-join hook to signal mod presence to datapack (v1.1.0)
ChatAnswerCore.onPlayerJoin 이 storage chat_answer:status active=1b 를 set.
Fabric 측은 ServerPlayConnectionEvents.JOIN, NeoForge 측은 PlayerLoggedInEvent
에서 호출. 데이터팩이 mq:load 단계에서 0b 로 clear 해 두므로, 모드가 빠진
환경에선 이 hook 이 일어나지 않아 0b 가 유지되고, 모드가 있으면 첫 로그인
직후 1b 로 갱신되어 데이터팩이 활성 상태를 분기할 수 있다.
2026-05-13 22:20:00 +09:00
12 changed files with 66 additions and 57 deletions

View File

@@ -9,11 +9,11 @@ allprojects {
version = project.mod_version version = project.mod_version
java { java {
toolchain.languageVersion = JavaLanguageVersion.of(21) toolchain.languageVersion = JavaLanguageVersion.of(25)
} }
tasks.withType(JavaCompile).configureEach { tasks.withType(JavaCompile).configureEach {
options.release = 21 options.release = 25
options.encoding = 'UTF-8' options.encoding = 'UTF-8'
} }
@@ -24,30 +24,7 @@ allprojects {
} }
} }
// ───── merged jar ────────────────────────────────────────────────────────── // MC 26.1.2 배포물은 Fabric jar 단일 빌드. buildAll = fabric remapJar 결과물.
// fabric + neoforge 각각의 remapJar 결과물을 한 jar 안에 압축해서 단일 배포물 생성.
// 같은 클래스(common 코드)는 한 번만 포함. 각 로더는 자신의 mod metadata
// (fabric.mod.json / META-INF/neoforge.mods.toml) 만 인식해서 자기 쪽 진입점만 로드.
tasks.register('mergedJar', Jar) {
dependsOn ':fabric:relocatedJar', ':neoforge:jar'
archiveBaseName = project.mod_id
archiveVersion = project.mod_version
archiveClassifier = 'all'
destinationDirectory = file('build/libs')
duplicatesStrategy = DuplicatesStrategy.EXCLUDE
// Fabric: Shadow(relocatedJar) 가 common 패키지를 kr.tkrmagid.chatanswer.fabric.core 로 옮긴 jar
// NeoForge: common 은 그대로 kr.tkrmagid.chatanswer.core 에 위치
// → 같은 클래스명 다른 매핑이라도 패키지 경로가 달라서 공존 가능
from(zipTree(project(':fabric').tasks.named('relocatedJar').flatMap { it.archiveFile }))
from(zipTree(project(':neoforge').tasks.named('jar').flatMap { it.archiveFile })) {
// META-INF/MANIFEST.MF 는 Fabric 측 것을 그대로 사용 (둘 다 단순 manifest)
exclude 'META-INF/MANIFEST.MF'
}
}
tasks.register('buildAll') { tasks.register('buildAll') {
dependsOn 'mergedJar' dependsOn ':fabric:remapJar'
} }

View File

@@ -31,11 +31,31 @@ public final class ChatAnswerCore {
private ChatAnswerCore() {} private ChatAnswerCore() {}
/**
* 플레이어 로그인 직후 호출. 데이터팩이 "모드 살아있음" 신호로 쓸 수 있게
* storage chat_answer:status 에 active=1b 를 세팅한다. 데이터팩의 mq:load 가
* 매 /reload 와 서버 시작 시 이 값을 0b 로 clear 하므로, 모드가 없으면 이
* 호출이 일어나지 않아 0b 로 유지되고, 모드가 있으면 첫 로그인 직후 1b 로 갱신.
*/
public static void onPlayerJoin(ServerPlayer player) {
MinecraftServer server = player.level().getServer();
if (server == null) return;
CommandSourceStack source = server.createCommandSourceStack().withSuppressedOutput();
try {
server.getCommands().performPrefixedCommand(
source,
"data modify storage chat_answer:status active set value 1b"
);
} catch (Exception e) {
LOG.debug("[{}] failed to set active flag: {}", MOD_ID, e.toString());
}
}
/** /**
* @return true = 채팅을 평소처럼 broadcast / false = 채팅 차단 (이미 정답 제출 처리됨) * @return true = 채팅을 평소처럼 broadcast / false = 채팅 차단 (이미 정답 제출 처리됨)
*/ */
public static boolean handleChat(ServerPlayer sender, String rawText) { public static boolean handleChat(ServerPlayer sender, String rawText) {
MinecraftServer server = sender.getServer(); MinecraftServer server = sender.level().getServer();
if (server == null) return true; if (server == null) return true;
if (!isAcceptingAnswer(server)) return true; if (!isAcceptingAnswer(server)) return true;

View File

@@ -1,9 +1,9 @@
plugins { plugins {
id 'fabric-loom' version '1.10-SNAPSHOT' id 'fabric-loom' version '1.16-SNAPSHOT'
id 'com.gradleup.shadow' version '8.3.5' id 'com.gradleup.shadow' version '9.4.1'
} }
archivesBaseName = "${project.mod_id}-fabric" base.archivesName = "${project.mod_id}-fabric"
// common/ 디렉토리의 로더 비종속 소스를 fabric 컴파일에 포함 (Mojang 매핑으로 컴파일) // common/ 디렉토리의 로더 비종속 소스를 fabric 컴파일에 포함 (Mojang 매핑으로 컴파일)
sourceSets { sourceSets {
@@ -15,14 +15,11 @@ sourceSets {
} }
dependencies { dependencies {
// MC 26.x: server jar 가 unobfuscated. intermediary 0.0.0 = identity mapping.
minecraft "com.mojang:minecraft:${project.minecraft_version}" minecraft "com.mojang:minecraft:${project.minecraft_version}"
mappings loom.officialMojangMappings() mappings "net.fabricmc:intermediary:0.0.0:v2"
modImplementation "net.fabricmc:fabric-loader:${project.loader_version}" implementation "net.fabricmc:fabric-loader:${project.loader_version}"
modImplementation "net.fabricmc.fabric-api:fabric-api:${project.fabric_version}" implementation "net.fabricmc.fabric-api:fabric-api:${project.fabric_version}"
}
loom {
serverOnlyMinecraftJar()
} }
processResources { processResources {
@@ -30,6 +27,8 @@ processResources {
inputs.property "mod_id", project.mod_id inputs.property "mod_id", project.mod_id
inputs.property "mod_name", project.mod_name inputs.property "mod_name", project.mod_name
filteringCharset = 'UTF-8'
filesMatching("fabric.mod.json") { filesMatching("fabric.mod.json") {
expand( expand(
"version": project.version, "version": project.version,

View File

@@ -3,6 +3,7 @@ package kr.tkrmagid.chatanswer.fabric;
import kr.tkrmagid.chatanswer.core.ChatAnswerCore; import kr.tkrmagid.chatanswer.core.ChatAnswerCore;
import net.fabricmc.api.ModInitializer; import net.fabricmc.api.ModInitializer;
import net.fabricmc.fabric.api.message.v1.ServerMessageEvents; import net.fabricmc.fabric.api.message.v1.ServerMessageEvents;
import net.fabricmc.fabric.api.networking.v1.ServerPlayConnectionEvents;
public final class ChatAnswerFabric implements ModInitializer { public final class ChatAnswerFabric implements ModInitializer {
@Override @Override
@@ -10,5 +11,8 @@ public final class ChatAnswerFabric implements ModInitializer {
ServerMessageEvents.ALLOW_CHAT_MESSAGE.register((message, sender, params) -> ServerMessageEvents.ALLOW_CHAT_MESSAGE.register((message, sender, params) ->
ChatAnswerCore.handleChat(sender, message.signedContent()) ChatAnswerCore.handleChat(sender, message.signedContent())
); );
ServerPlayConnectionEvents.JOIN.register((handler, sender, server) ->
ChatAnswerCore.onPlayerJoin(handler.player)
);
} }
} }

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.4 KiB

View File

@@ -2,20 +2,18 @@
"schemaVersion": 1, "schemaVersion": 1,
"id": "${mod_id}", "id": "${mod_id}",
"version": "${version}", "version": "${version}",
"name": "${mod_name}", "name": "채팅정답",
"description": "음악퀴즈(mq) 데이터팩이 정답 입력을 받는 상태(init=5)에서 채팅을 가로채 mq:answer/submit 함수로 전달합니다.", "description": "음악퀴즈(mq) 데이터팩이 정답 입력을 받는 상태(init=5)에서 채팅을 가로채 mq:answer/submit 함수로 전달합니다.",
"authors": [ "tkrmagid" ], "authors": [ "tkrmagid" ],
"contact": {
"homepage": "https://git.tkrmagid.kr/tkrmagid/mc_chat_answer_mod"
},
"license": "MIT", "license": "MIT",
"environment": "server", "icon": "assets/chat_answer/icon.png",
"environment": "*",
"entrypoints": { "entrypoints": {
"main": [ "kr.tkrmagid.chatanswer.fabric.ChatAnswerFabric" ] "main": [ "kr.tkrmagid.chatanswer.fabric.ChatAnswerFabric" ]
}, },
"depends": { "depends": {
"fabricloader": ">=0.16.0", "fabricloader": ">=0.19.0",
"minecraft": ">=1.21.6", "minecraft": ">=26.1.2",
"java": ">=21", "java": ">=21",
"fabric-api": "*" "fabric-api": "*"
} }

View File

@@ -2,19 +2,19 @@ org.gradle.jvmargs=-Xmx3G
org.gradle.parallel=true org.gradle.parallel=true
# ───── target Minecraft / loader versions ─────────────────────────────────── # ───── target Minecraft / loader versions ───────────────────────────────────
# 1.21.6 = dialog system 최초 버전 = 음악퀴즈 데이터팩 최소 요구 버전 # MC 26.1.2 = 사용자 환경. 1.21.6 과는 intermediary 매핑이 달라서 동일 jar 로 양쪽
minecraft_version=1.21.6 # 지원 불가 → 26.1.2 전용 빌드.
minecraft_version=26.1.2
# Fabric # Fabric
yarn_mappings=1.21.6+build.1 loader_version=0.19.2
loader_version=0.16.10 fabric_version=0.148.2+26.1.2
fabric_version=0.128.2+1.21.6
# NeoForge # NeoForge
neoforge_version=21.6.20-beta neoforge_version=26.1.2.36-beta
# ───── mod metadata ───────────────────────────────────────────────────────── # ───── mod metadata ─────────────────────────────────────────────────────────
mod_id=chat_answer mod_id=chat_answer
mod_version=1.0.0 mod_version=1.2.1
mod_group=kr.tkrmagid.chatanswer mod_group=kr.tkrmagid.chatanswer
mod_name=채팅정답 mod_name=채팅정답

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-bin.zip distributionUrl=https\://services.gradle.org/distributions/gradle-9.5.1-bin.zip
networkTimeout=10000 networkTimeout=10000
validateDistributionUrl=true validateDistributionUrl=true
zipStoreBase=GRADLE_USER_HOME zipStoreBase=GRADLE_USER_HOME

View File

@@ -2,7 +2,7 @@ plugins {
id 'net.neoforged.moddev' version '2.0.97' id 'net.neoforged.moddev' version '2.0.97'
} }
archivesBaseName = "${project.mod_id}-neoforge" base.archivesName = "${project.mod_id}-neoforge"
sourceSets { sourceSets {
main { main {

View File

@@ -1,16 +1,19 @@
package kr.tkrmagid.chatanswer.neoforge; package kr.tkrmagid.chatanswer.neoforge;
import kr.tkrmagid.chatanswer.core.ChatAnswerCore; import kr.tkrmagid.chatanswer.core.ChatAnswerCore;
import net.minecraft.server.level.ServerPlayer;
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;
import net.neoforged.neoforge.common.NeoForge; import net.neoforged.neoforge.common.NeoForge;
import net.neoforged.neoforge.event.ServerChatEvent; import net.neoforged.neoforge.event.ServerChatEvent;
import net.neoforged.neoforge.event.entity.player.PlayerEvent;
@Mod(ChatAnswerCore.MOD_ID) @Mod(ChatAnswerCore.MOD_ID)
public final class ChatAnswerNeoForge { public final class ChatAnswerNeoForge {
public ChatAnswerNeoForge(IEventBus modBus) { public ChatAnswerNeoForge(IEventBus modBus) {
NeoForge.EVENT_BUS.addListener(ChatAnswerNeoForge::onServerChat); NeoForge.EVENT_BUS.addListener(ChatAnswerNeoForge::onServerChat);
NeoForge.EVENT_BUS.addListener(ChatAnswerNeoForge::onPlayerLogin);
} }
@SubscribeEvent @SubscribeEvent
@@ -20,4 +23,11 @@ public final class ChatAnswerNeoForge {
event.setCanceled(true); event.setCanceled(true);
} }
} }
@SubscribeEvent
public static void onPlayerLogin(PlayerEvent.PlayerLoggedInEvent event) {
if (event.getEntity() instanceof ServerPlayer player) {
ChatAnswerCore.onPlayerJoin(player);
}
}
} }

View File

@@ -16,11 +16,11 @@ modId = "neoforge"
type = "required" type = "required"
versionRange = "[${neoforge_version},)" versionRange = "[${neoforge_version},)"
ordering = "NONE" ordering = "NONE"
side = "SERVER" side = "BOTH"
[[dependencies.${mod_id}]] [[dependencies.${mod_id}]]
modId = "minecraft" modId = "minecraft"
type = "required" type = "required"
versionRange = "[${minecraft_version},)" versionRange = "[${minecraft_version},)"
ordering = "NONE" ordering = "NONE"
side = "SERVER" side = "BOTH"

View File

@@ -8,4 +8,5 @@ pluginManagement {
} }
rootProject.name = 'chat_answer' rootProject.name = 'chat_answer'
include 'fabric', 'neoforge' // NeoForge moddev plugin 이 MC 26.1.2 를 파싱하지 못함(1.21.5 로 fallback) → 26.x 정식 지원 전까지 제외.
include 'fabric'