aboutsummaryrefslogtreecommitdiff
path: root/src/main/kotlin/moe/nea/firmament/features/texturepack/CustomSkyBlockTextures.kt
blob: f37e3ea9accdc7a9da28b86734a18c3da8a8b10b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
/*
 * SPDX-FileCopyrightText: 2023 Linnea Gräf <nea@nea.moe>
 * SPDX-FileCopyrightText: 2024 Linnea Gräf <nea@nea.moe>
 *
 * SPDX-License-Identifier: GPL-3.0-or-later
 */

package moe.nea.firmament.features.texturepack

import com.mojang.authlib.minecraft.MinecraftProfileTexture
import com.mojang.authlib.properties.Property
import org.spongepowered.asm.mixin.injection.callback.CallbackInfoReturnable
import net.minecraft.block.SkullBlock
import net.minecraft.client.MinecraftClient
import net.minecraft.client.render.RenderLayer
import net.minecraft.client.util.ModelIdentifier
import net.minecraft.component.type.ProfileComponent
import net.minecraft.util.Identifier
import moe.nea.firmament.annotations.Subscribe
import moe.nea.firmament.events.BakeExtraModelsEvent
import moe.nea.firmament.events.CustomItemModelEvent
import moe.nea.firmament.events.TickEvent
import moe.nea.firmament.features.FirmamentFeature
import moe.nea.firmament.gui.config.ManagedConfig
import moe.nea.firmament.util.IdentityCharacteristics
import moe.nea.firmament.util.item.decodeProfileTextureProperty
import moe.nea.firmament.util.skyBlockId

object CustomSkyBlockTextures : FirmamentFeature {
    override val identifier: String
        get() = "custom-skyblock-textures"

    object TConfig : ManagedConfig(identifier) {
        val enabled by toggle("enabled") { true }
        val skullsEnabled by toggle("skulls-enabled") { true }
        val cacheDuration by integer("cache-duration", 0, 20) { 1 }
        val enableModelOverrides by toggle("model-overrides") { true }
        val enableArmorOverrides by toggle("armor-overrides") { true }
    }

    override val config: ManagedConfig
        get() = TConfig

    @Subscribe
    fun onTick(it: TickEvent) {
        if (TConfig.cacheDuration < 1 || it.tickCount % TConfig.cacheDuration == 0) {
            // TODO: unify all of those caches somehow
            CustomItemModelEvent.clearCache()
            skullTextureCache.clear()
            CustomGlobalTextures.overrideCache.clear()
            CustomGlobalArmorOverrides.overrideCache.clear()
        }
    }

    @Subscribe
    fun bakeCustomFirmModels(event: BakeExtraModelsEvent) {
        val resources =
            MinecraftClient.getInstance().resourceManager.findResources("models/item"
            ) { it: Identifier ->
                "firmskyblock" == it.namespace && it.path
                    .endsWith(".json")
            }
        for (identifier in resources.keys) {
            val modelId = ModelIdentifier.ofInventoryVariant(
                Identifier.of(
                    "firmskyblock",
                    identifier.path.substring(
                        "models/item/".length,
                        identifier.path.length - ".json".length),
                ))
            event.addModel(modelId)
        }
    }

    @Subscribe
    fun onCustomModelId(it: CustomItemModelEvent) {
        if (!TConfig.enabled) return
        val id = it.itemStack.skyBlockId ?: return
        it.overrideModel = ModelIdentifier.ofInventoryVariant(Identifier.of("firmskyblock", id.identifier.path))
    }

    private val skullTextureCache = mutableMapOf<IdentityCharacteristics<ProfileComponent>, Any>()
    private val sentinelPresentInvalid = Object()

    private val mcUrlRegex = "https?://textures.minecraft.net/texture/([a-fA-F0-9]+)".toRegex()

    fun getSkullId(textureProperty: Property): String? {
        val texture = decodeProfileTextureProperty(textureProperty) ?: return null
        val textureUrl =
            texture.textures[MinecraftProfileTexture.Type.SKIN]?.url ?: return null
        val mcUrlData = mcUrlRegex.matchEntire(textureUrl) ?: return null
        return mcUrlData.groupValues[1]
    }

    fun getSkullTexture(profile: ProfileComponent): Identifier? {
        val id = getSkullId(profile.properties["textures"].firstOrNull() ?: return null) ?: return null
        return Identifier.of("firmskyblock", "textures/placedskull/$id.png")
    }

    fun modifySkullTexture(
        type: SkullBlock.SkullType?,
        component: ProfileComponent?,
        cir: CallbackInfoReturnable<RenderLayer>
    ) {
        if (type != SkullBlock.Type.PLAYER) return
        if (!TConfig.skullsEnabled) return
        if (component == null) return
        val ic = IdentityCharacteristics(component)

        val n = skullTextureCache.getOrPut(ic) {
            val id = getSkullTexture(component) ?: return@getOrPut sentinelPresentInvalid
            if (!MinecraftClient.getInstance().resourceManager.getResource(id).isPresent) {
                return@getOrPut sentinelPresentInvalid
            }
            return@getOrPut id
        }
        if (n === sentinelPresentInvalid) return
        cir.returnValue = RenderLayer.getEntityTranslucent(n as Identifier)
    }
}