mirror of
https://github.com/FabricMC/fabric.git
synced 2024-11-22 23:58:02 -05:00
add sprite atlas path hooks, path-specific events (#195)
This commit is contained in:
parent
200eb5c2ac
commit
d36c5fcf8e
7 changed files with 183 additions and 45 deletions
|
@ -1,5 +1,5 @@
|
||||||
archivesBaseName = "fabric-textures-v0"
|
archivesBaseName = "fabric-textures-v0"
|
||||||
version = getSubprojectVersion(project, "0.1.0")
|
version = getSubprojectVersion(project, "0.1.1")
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
compile project(path: ':fabric-api-base', configuration: 'dev')
|
compile project(path: ':fabric-api-base', configuration: 'dev')
|
||||||
|
|
|
@ -17,32 +17,43 @@
|
||||||
package net.fabricmc.fabric.api.event.client;
|
package net.fabricmc.fabric.api.event.client;
|
||||||
|
|
||||||
import net.fabricmc.fabric.api.event.Event;
|
import net.fabricmc.fabric.api.event.Event;
|
||||||
import net.fabricmc.fabric.api.event.EventFactory;
|
import net.fabricmc.fabric.impl.client.texture.SpriteRegistryCallbackHolder;
|
||||||
import net.minecraft.client.MinecraftClient;
|
|
||||||
import net.minecraft.client.texture.Sprite;
|
import net.minecraft.client.texture.Sprite;
|
||||||
import net.minecraft.client.texture.SpriteAtlasTexture;
|
import net.minecraft.client.texture.SpriteAtlasTexture;
|
||||||
import net.minecraft.util.Identifier;
|
import net.minecraft.util.Identifier;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.function.Consumer;
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
public interface ClientSpriteRegistryCallback {
|
public interface ClientSpriteRegistryCallback {
|
||||||
public static final Event<ClientSpriteRegistryCallback> EVENT = EventFactory.createArrayBacked(ClientSpriteRegistryCallback.class,
|
/**
|
||||||
(listeners) -> (atlasTexture, registry) -> {
|
* @deprecated Use the {@link ClientSpriteRegistryCallback#event(Identifier)} registration method. Since 1.14
|
||||||
for (ClientSpriteRegistryCallback callback : listeners) {
|
* started making use of multiple sprite atlases, it is unwise to register sprites to *all* of them.
|
||||||
callback.registerSprites(atlasTexture, registry);
|
*/
|
||||||
}
|
@Deprecated
|
||||||
}
|
public static final Event<ClientSpriteRegistryCallback> EVENT = SpriteRegistryCallbackHolder.EVENT_GLOBAL;
|
||||||
);
|
|
||||||
|
|
||||||
void registerSprites(SpriteAtlasTexture atlasTexture, Registry registry);
|
void registerSprites(SpriteAtlasTexture atlasTexture, Registry registry);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an event instance for a given atlas path.
|
||||||
|
*
|
||||||
|
* @param atlasId The atlas texture ID you want to register to.
|
||||||
|
* @return The event for a given atlas path.
|
||||||
|
*
|
||||||
|
* @since 0.1.1
|
||||||
|
*/
|
||||||
|
static Event<ClientSpriteRegistryCallback> event(Identifier atlasId) {
|
||||||
|
return SpriteRegistryCallbackHolder.eventLocal(atlasId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Use the {@link ClientSpriteRegistryCallback#event(Identifier)} registration method.
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
static void registerBlockAtlas(ClientSpriteRegistryCallback callback) {
|
static void registerBlockAtlas(ClientSpriteRegistryCallback callback) {
|
||||||
EVENT.register((atlasTexture, registry) -> {
|
event(SpriteAtlasTexture.BLOCK_ATLAS_TEX).register(callback);
|
||||||
if (atlasTexture == MinecraftClient.getInstance().getSpriteAtlas()) {
|
|
||||||
callback.registerSprites(atlasTexture, registry);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class Registry {
|
public static class Registry {
|
||||||
|
@ -69,7 +80,7 @@ public interface ClientSpriteRegistryCallback {
|
||||||
* @param sprite The sprite to be added.
|
* @param sprite The sprite to be added.
|
||||||
*/
|
*/
|
||||||
public void register(Sprite sprite) {
|
public void register(Sprite sprite) {
|
||||||
this.spriteMap.put(sprite.getId(), sprite);
|
spriteMap.put(sprite.getId(), sprite);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,23 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2016, 2017, 2018, 2019 FabricMC
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package net.fabricmc.fabric.impl.client.texture;
|
||||||
|
|
||||||
|
import net.minecraft.util.Identifier;
|
||||||
|
|
||||||
|
public interface SpriteAtlasTextureHooks {
|
||||||
|
void onRegisteredAs(Identifier id);
|
||||||
|
}
|
|
@ -0,0 +1,48 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2016, 2017, 2018, 2019 FabricMC
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package net.fabricmc.fabric.impl.client.texture;
|
||||||
|
|
||||||
|
import net.fabricmc.fabric.api.event.Event;
|
||||||
|
import net.fabricmc.fabric.api.event.EventFactory;
|
||||||
|
import net.fabricmc.fabric.api.event.client.ClientSpriteRegistryCallback;
|
||||||
|
import net.minecraft.util.Identifier;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public final class SpriteRegistryCallbackHolder {
|
||||||
|
public static final Event<ClientSpriteRegistryCallback> EVENT_GLOBAL = createEvent();
|
||||||
|
private static final Map<Identifier, Event<ClientSpriteRegistryCallback>> eventMap = new HashMap<>();
|
||||||
|
|
||||||
|
private SpriteRegistryCallbackHolder() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Event<ClientSpriteRegistryCallback> eventLocal(Identifier key) {
|
||||||
|
return eventMap.computeIfAbsent(key, (a) -> createEvent());
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Event<ClientSpriteRegistryCallback> createEvent() {
|
||||||
|
return EventFactory.createArrayBacked(ClientSpriteRegistryCallback.class,
|
||||||
|
(listeners) -> (atlasTexture, registry) -> {
|
||||||
|
for (ClientSpriteRegistryCallback callback : listeners) {
|
||||||
|
callback.registerSprites(atlasTexture, registry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -20,6 +20,8 @@ import com.google.common.base.Joiner;
|
||||||
import net.fabricmc.fabric.api.client.texture.*;
|
import net.fabricmc.fabric.api.client.texture.*;
|
||||||
import net.fabricmc.fabric.api.event.client.ClientSpriteRegistryCallback;
|
import net.fabricmc.fabric.api.event.client.ClientSpriteRegistryCallback;
|
||||||
import net.fabricmc.fabric.impl.client.texture.FabricSprite;
|
import net.fabricmc.fabric.impl.client.texture.FabricSprite;
|
||||||
|
import net.fabricmc.fabric.impl.client.texture.SpriteAtlasTextureHooks;
|
||||||
|
import net.fabricmc.fabric.impl.client.texture.SpriteRegistryCallbackHolder;
|
||||||
import net.minecraft.client.resource.metadata.AnimationResourceMetadata;
|
import net.minecraft.client.resource.metadata.AnimationResourceMetadata;
|
||||||
import net.minecraft.client.texture.Sprite;
|
import net.minecraft.client.texture.Sprite;
|
||||||
import net.minecraft.client.texture.SpriteAtlasTexture;
|
import net.minecraft.client.texture.SpriteAtlasTexture;
|
||||||
|
@ -42,7 +44,7 @@ import java.io.IOException;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
|
||||||
@Mixin(SpriteAtlasTexture.class)
|
@Mixin(SpriteAtlasTexture.class)
|
||||||
public abstract class MixinSpriteAtlasTexture {
|
public abstract class MixinSpriteAtlasTexture implements SpriteAtlasTextureHooks {
|
||||||
@Shadow
|
@Shadow
|
||||||
private static Logger LOGGER;
|
private static Logger LOGGER;
|
||||||
@Shadow
|
@Shadow
|
||||||
|
@ -51,33 +53,38 @@ public abstract class MixinSpriteAtlasTexture {
|
||||||
@Shadow
|
@Shadow
|
||||||
public abstract Sprite getSprite(Identifier id);
|
public abstract Sprite getSprite(Identifier id);
|
||||||
|
|
||||||
|
private final Set<Identifier> fabric_localIds = new HashSet<>();
|
||||||
|
|
||||||
|
// EVENT/HOOKS LOGIC
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRegisteredAs(Identifier id) {
|
||||||
|
fabric_localIds.add(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
// INJECTION LOGIC
|
||||||
|
|
||||||
private Map<Identifier, Sprite> fabric_injectedSprites;
|
private Map<Identifier, Sprite> fabric_injectedSprites;
|
||||||
|
|
||||||
/**
|
// Loads in custom sprite object injections.
|
||||||
* The purpose of this patch is to allow injecting sprites at the stage of Sprite instantiation, such as
|
@Inject(at = @At("RETURN"), method = "loadSprites")
|
||||||
* Sprites with CustomSpriteLoaders.
|
private void afterLoadSprites(ResourceManager resourceManager_1, Set<Identifier> set_1, CallbackInfoReturnable<Collection<Sprite>> info) {
|
||||||
* <p>
|
if (fabric_injectedSprites != null) {
|
||||||
* FabricSprite is a red herring. It's only use to go around Sprite's constructors being protected.
|
info.getReturnValue().addAll(fabric_injectedSprites.values());
|
||||||
* <p>
|
fabric_injectedSprites = null;
|
||||||
* method_18160 is a lambda used in runAsync.
|
|
||||||
*/
|
|
||||||
@SuppressWarnings("JavaDoc")
|
|
||||||
|
|
||||||
@Redirect(method = "method_18160", at = @At(value = "NEW", target = "net/minecraft/client/texture/Sprite"))
|
|
||||||
public Sprite newSprite(Identifier id, PngFile pngFile, AnimationResourceMetadata animationMetadata) {
|
|
||||||
if (fabric_injectedSprites.containsKey(id)) {
|
|
||||||
return fabric_injectedSprites.get(id);
|
|
||||||
} else {
|
|
||||||
return new FabricSprite(id, pngFile, animationMetadata);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Handles DependentSprite + custom sprite object injections.
|
||||||
@ModifyVariable(at = @At(value = "INVOKE", target = "Lnet/minecraft/client/texture/SpriteAtlasTexture;loadSprites(Lnet/minecraft/resource/ResourceManager;Ljava/util/Set;)Ljava/util/Collection;"), method = "stitch")
|
@ModifyVariable(at = @At(value = "INVOKE", target = "Lnet/minecraft/client/texture/SpriteAtlasTexture;loadSprites(Lnet/minecraft/resource/ResourceManager;Ljava/util/Set;)Ljava/util/Collection;"), method = "stitch")
|
||||||
public Set<Identifier> setHook(Set<Identifier> set) {
|
public Set<Identifier> beforeSpriteLoad(Set<Identifier> set) {
|
||||||
fabric_injectedSprites = new HashMap<>();
|
fabric_injectedSprites = new HashMap<>();
|
||||||
ClientSpriteRegistryCallback.Registry registry = new ClientSpriteRegistryCallback.Registry(fabric_injectedSprites, set::add);
|
ClientSpriteRegistryCallback.Registry registry = new ClientSpriteRegistryCallback.Registry(fabric_injectedSprites, set::add);
|
||||||
//noinspection ConstantConditions
|
//noinspection ConstantConditions
|
||||||
ClientSpriteRegistryCallback.EVENT.invoker().registerSprites((SpriteAtlasTexture) (Object) this, registry);
|
for (Identifier id : fabric_localIds) {
|
||||||
|
SpriteRegistryCallbackHolder.eventLocal(id).invoker().registerSprites((SpriteAtlasTexture) (Object) this, registry);
|
||||||
|
}
|
||||||
|
SpriteRegistryCallbackHolder.EVENT_GLOBAL.invoker().registerSprites((SpriteAtlasTexture) (Object) this, registry);
|
||||||
|
|
||||||
// TODO: Unoptimized.
|
// TODO: Unoptimized.
|
||||||
Set<DependentSprite> dependentSprites = new HashSet<>();
|
Set<DependentSprite> dependentSprites = new HashSet<>();
|
||||||
|
@ -90,8 +97,14 @@ public abstract class MixinSpriteAtlasTexture {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Set<Identifier> result = set;
|
||||||
|
boolean isResultNew = false;
|
||||||
|
|
||||||
if (!dependentSprites.isEmpty()) {
|
if (!dependentSprites.isEmpty()) {
|
||||||
Set<Identifier> result = new LinkedHashSet<>();
|
if (!isResultNew) {
|
||||||
|
result = new LinkedHashSet<>();
|
||||||
|
isResultNew = true;
|
||||||
|
}
|
||||||
|
|
||||||
for (Identifier id : set) {
|
for (Identifier id : set) {
|
||||||
if (!dependentSpriteIds.contains(id)) {
|
if (!dependentSpriteIds.contains(id)) {
|
||||||
|
@ -123,13 +136,23 @@ public abstract class MixinSpriteAtlasTexture {
|
||||||
}
|
}
|
||||||
throw new CrashException(report);
|
throw new CrashException(report);
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
|
||||||
} else {
|
|
||||||
return set;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!fabric_injectedSprites.isEmpty()) {
|
||||||
|
if (!isResultNew) {
|
||||||
|
result = new LinkedHashSet<>(set);
|
||||||
|
isResultNew = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
result.removeAll(fabric_injectedSprites.keySet());
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handles CustomSpriteLoader.
|
||||||
|
*/
|
||||||
@Inject(at = @At("HEAD"), method = "loadSprite", cancellable = true)
|
@Inject(at = @At("HEAD"), method = "loadSprite", cancellable = true)
|
||||||
public void loadSprite(ResourceManager manager, Sprite sprite, CallbackInfoReturnable<Boolean> info) {
|
public void loadSprite(ResourceManager manager, Sprite sprite, CallbackInfoReturnable<Boolean> info) {
|
||||||
// refer SpriteAtlasTexture.loadSprite
|
// refer SpriteAtlasTexture.loadSprite
|
||||||
|
@ -137,24 +160,20 @@ public abstract class MixinSpriteAtlasTexture {
|
||||||
try {
|
try {
|
||||||
if (!((CustomSpriteLoader) sprite).load(manager, mipLevel)) {
|
if (!((CustomSpriteLoader) sprite).load(manager, mipLevel)) {
|
||||||
info.setReturnValue(false);
|
info.setReturnValue(false);
|
||||||
info.cancel();
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} catch (RuntimeException | IOException e) {
|
} catch (RuntimeException | IOException e) {
|
||||||
LOGGER.error("Unable to load custom sprite {}: {}", sprite.getId(), e);
|
LOGGER.error("Unable to load custom sprite {}: {}", sprite.getId(), e);
|
||||||
info.setReturnValue(false);
|
info.setReturnValue(false);
|
||||||
info.cancel();
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
sprite.generateMipmaps(this.mipLevel);
|
sprite.generateMipmaps(this.mipLevel);
|
||||||
info.setReturnValue(true);
|
info.setReturnValue(true);
|
||||||
info.cancel();
|
|
||||||
} catch (Throwable e) {
|
} catch (Throwable e) {
|
||||||
LOGGER.error("Unable to apply mipmap to custom sprite {}: {}", sprite.getId(), e);
|
LOGGER.error("Unable to apply mipmap to custom sprite {}: {}", sprite.getId(), e);
|
||||||
info.setReturnValue(false);
|
info.setReturnValue(false);
|
||||||
info.cancel();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,36 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2016, 2017, 2018, 2019 FabricMC
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package net.fabricmc.fabric.mixin.client.texture;
|
||||||
|
|
||||||
|
import net.fabricmc.fabric.impl.client.texture.SpriteAtlasTextureHooks;
|
||||||
|
import net.minecraft.client.texture.Texture;
|
||||||
|
import net.minecraft.client.texture.TextureManager;
|
||||||
|
import net.minecraft.util.Identifier;
|
||||||
|
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.CallbackInfoReturnable;
|
||||||
|
|
||||||
|
@Mixin(TextureManager.class)
|
||||||
|
public class MixinTextureManager {
|
||||||
|
@Inject(at = @At("RETURN"), method = "registerTexture")
|
||||||
|
private void afterRegisterTexture(Identifier identifier, Texture texture, CallbackInfoReturnable<Boolean> info) {
|
||||||
|
if (texture instanceof SpriteAtlasTextureHooks) {
|
||||||
|
((SpriteAtlasTextureHooks) texture).onRegisteredAs(identifier);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -3,7 +3,8 @@
|
||||||
"package": "net.fabricmc.fabric.mixin.client.texture",
|
"package": "net.fabricmc.fabric.mixin.client.texture",
|
||||||
"compatibilityLevel": "JAVA_8",
|
"compatibilityLevel": "JAVA_8",
|
||||||
"client": [
|
"client": [
|
||||||
"MixinSpriteAtlasTexture"
|
"MixinSpriteAtlasTexture",
|
||||||
|
"MixinTextureManager"
|
||||||
],
|
],
|
||||||
"injectors": {
|
"injectors": {
|
||||||
"defaultRequire": 1
|
"defaultRequire": 1
|
||||||
|
|
Loading…
Reference in a new issue