mirror of
https://github.com/BlueMap-Minecraft/BlueMap.git
synced 2025-01-27 18:51:21 +01:00
Fix not loading textures if they are outside the block(s) folder
This commit is contained in:
parent
bc8e52abbf
commit
ffc472ce84
@ -59,7 +59,6 @@
|
|||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
import java.nio.file.StandardOpenOption;
|
import java.nio.file.StandardOpenOption;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
import java.util.concurrent.CompletionException;
|
|
||||||
import java.util.stream.Stream;
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -329,20 +328,14 @@ public synchronized ResourcePack getResourcePack() throws ConfigurationException
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Logger.global.logInfo("Loading resources...");
|
|
||||||
resourcePack = new ResourcePack();
|
resourcePack = new ResourcePack();
|
||||||
|
|
||||||
|
List<Path> resourcePackRoots = new ArrayList<>();
|
||||||
// load from resourcepack folder
|
// load from resourcepack folder
|
||||||
try (Stream<Path> resourcepackFiles = Files.list(resourcePackFolder)) {
|
try (Stream<Path> resourcepackFiles = Files.list(resourcePackFolder)) {
|
||||||
resourcepackFiles
|
resourcepackFiles
|
||||||
.sorted(Comparator.reverseOrder())
|
.sorted(Comparator.reverseOrder())
|
||||||
.forEach(resourcepackFile -> {
|
.forEach(resourcePackRoots::add);
|
||||||
try {
|
|
||||||
resourcePack.loadResources(resourcepackFile);
|
|
||||||
} catch (IOException e) {
|
|
||||||
throw new CompletionException(e);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (configs.getCoreConfig().isScanForModResources()) {
|
if (configs.getCoreConfig().isScanForModResources()) {
|
||||||
@ -354,13 +347,7 @@ public synchronized ResourcePack getResourcePack() throws ConfigurationException
|
|||||||
resourcepackFiles
|
resourcepackFiles
|
||||||
.filter(Files::isRegularFile)
|
.filter(Files::isRegularFile)
|
||||||
.filter(file -> file.getFileName().toString().endsWith(".jar"))
|
.filter(file -> file.getFileName().toString().endsWith(".jar"))
|
||||||
.forEach(resourcepackFile -> {
|
.forEach(resourcePackRoots::add);
|
||||||
try {
|
|
||||||
resourcePack.loadResources(resourcepackFile);
|
|
||||||
} catch (IOException e) {
|
|
||||||
throw new CompletionException(e);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -370,23 +357,16 @@ public synchronized ResourcePack getResourcePack() throws ConfigurationException
|
|||||||
if (!Files.isDirectory(datapacksFolder)) continue;
|
if (!Files.isDirectory(datapacksFolder)) continue;
|
||||||
|
|
||||||
try (Stream<Path> resourcepackFiles = Files.list(worldFolder.resolve("datapacks"))) {
|
try (Stream<Path> resourcepackFiles = Files.list(worldFolder.resolve("datapacks"))) {
|
||||||
resourcepackFiles
|
resourcepackFiles.forEach(resourcePackRoots::add);
|
||||||
.forEach(resourcepackFile -> {
|
|
||||||
try {
|
|
||||||
resourcePack.loadResources(resourcepackFile);
|
|
||||||
} catch (IOException e) {
|
|
||||||
throw new CompletionException(e);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
resourcePack.loadResources(resourceExtensionsFile);
|
resourcePackRoots.add(resourceExtensionsFile);
|
||||||
resourcePack.loadResources(defaultResourceFile);
|
resourcePackRoots.add(defaultResourceFile);
|
||||||
resourcePack.bake();
|
|
||||||
Logger.global.logInfo("Resources loaded.");
|
resourcePack.loadResources(resourcePackRoots);
|
||||||
} catch (IOException | RuntimeException e) {
|
} catch (IOException | RuntimeException e) {
|
||||||
throw new ConfigurationException("Failed to parse resources!\n" +
|
throw new ConfigurationException("Failed to parse resources!\n" +
|
||||||
"Is one of your resource-packs corrupted?", e);
|
"Is one of your resource-packs corrupted?", e);
|
||||||
|
@ -4,8 +4,8 @@
|
|||||||
import com.github.benmanes.caffeine.cache.LoadingCache;
|
import com.github.benmanes.caffeine.cache.LoadingCache;
|
||||||
import com.google.gson.JsonElement;
|
import com.google.gson.JsonElement;
|
||||||
import com.google.gson.JsonObject;
|
import com.google.gson.JsonObject;
|
||||||
import de.bluecolored.bluemap.core.BlueMap;
|
|
||||||
import de.bluecolored.bluemap.api.debug.DebugDump;
|
import de.bluecolored.bluemap.api.debug.DebugDump;
|
||||||
|
import de.bluecolored.bluemap.core.BlueMap;
|
||||||
import de.bluecolored.bluemap.core.logger.Logger;
|
import de.bluecolored.bluemap.core.logger.Logger;
|
||||||
import de.bluecolored.bluemap.core.resources.BlockColorCalculatorFactory;
|
import de.bluecolored.bluemap.core.resources.BlockColorCalculatorFactory;
|
||||||
import de.bluecolored.bluemap.core.resources.BlockPropertiesConfig;
|
import de.bluecolored.bluemap.core.resources.BlockPropertiesConfig;
|
||||||
@ -13,6 +13,7 @@
|
|||||||
import de.bluecolored.bluemap.core.resources.adapter.ResourcesGson;
|
import de.bluecolored.bluemap.core.resources.adapter.ResourcesGson;
|
||||||
import de.bluecolored.bluemap.core.resources.biome.BiomeConfig;
|
import de.bluecolored.bluemap.core.resources.biome.BiomeConfig;
|
||||||
import de.bluecolored.bluemap.core.resources.resourcepack.blockmodel.BlockModel;
|
import de.bluecolored.bluemap.core.resources.resourcepack.blockmodel.BlockModel;
|
||||||
|
import de.bluecolored.bluemap.core.resources.resourcepack.blockmodel.TextureVariable;
|
||||||
import de.bluecolored.bluemap.core.resources.resourcepack.blockstate.BlockState;
|
import de.bluecolored.bluemap.core.resources.resourcepack.blockstate.BlockState;
|
||||||
import de.bluecolored.bluemap.core.resources.resourcepack.texture.Texture;
|
import de.bluecolored.bluemap.core.resources.resourcepack.texture.Texture;
|
||||||
import de.bluecolored.bluemap.core.util.Tristate;
|
import de.bluecolored.bluemap.core.util.Tristate;
|
||||||
@ -30,7 +31,9 @@
|
|||||||
import java.nio.file.Files;
|
import java.nio.file.Files;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
import java.util.concurrent.CompletableFuture;
|
import java.util.concurrent.CompletableFuture;
|
||||||
import java.util.concurrent.CompletionException;
|
import java.util.concurrent.CompletionException;
|
||||||
import java.util.regex.Pattern;
|
import java.util.regex.Pattern;
|
||||||
@ -157,18 +160,33 @@ private BlockProperties loadBlockProperties(de.bluecolored.bluemap.core.world.Bl
|
|||||||
return props.build();
|
return props.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
public synchronized void loadResources(Path root) throws IOException {
|
public synchronized void loadResources(Iterable<Path> roots) throws IOException {
|
||||||
|
Logger.global.logInfo("Loading resources...");
|
||||||
|
|
||||||
|
for (Path root : roots) {
|
||||||
Logger.global.logDebug("Loading resources from: " + root + " ...");
|
Logger.global.logDebug("Loading resources from: " + root + " ...");
|
||||||
loadResourcesInternal(root);
|
loadResourcePath(root, this::loadResources);
|
||||||
}
|
}
|
||||||
|
|
||||||
private synchronized void loadResourcesInternal(Path root) throws IOException {
|
Logger.global.logInfo("Loading textures...");
|
||||||
|
for (Path root : roots) {
|
||||||
|
Logger.global.logDebug("Loading textures from: " + root + " ...");
|
||||||
|
loadResourcePath(root, this::loadTextures);
|
||||||
|
}
|
||||||
|
|
||||||
|
Logger.global.logInfo("Baking resources...");
|
||||||
|
bake();
|
||||||
|
|
||||||
|
|
||||||
|
Logger.global.logInfo("Resources loaded.");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void loadResourcePath(Path root, PathLoader resourceLoader) throws IOException {
|
||||||
if (!Files.isDirectory(root)) {
|
if (!Files.isDirectory(root)) {
|
||||||
try (FileSystem fileSystem = FileSystems.newFileSystem(root, (ClassLoader) null)) {
|
try (FileSystem fileSystem = FileSystems.newFileSystem(root, (ClassLoader) null)) {
|
||||||
for (Path fsRoot : fileSystem.getRootDirectories()) {
|
for (Path fsRoot : fileSystem.getRootDirectories()) {
|
||||||
if (!Files.isDirectory(fsRoot)) continue;
|
if (!Files.isDirectory(fsRoot)) continue;
|
||||||
this.loadResourcesInternal(fsRoot);
|
loadResourcePath(fsRoot, resourceLoader);
|
||||||
}
|
}
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
Logger.global.logDebug("Failed to read '" + root + "': " + ex);
|
Logger.global.logDebug("Failed to read '" + root + "': " + ex);
|
||||||
@ -183,13 +201,17 @@ private synchronized void loadResourcesInternal(Path root) throws IOException {
|
|||||||
JsonObject rootElement = ResourcesGson.INSTANCE.fromJson(reader, JsonObject.class);
|
JsonObject rootElement = ResourcesGson.INSTANCE.fromJson(reader, JsonObject.class);
|
||||||
for (JsonElement element : rootElement.getAsJsonArray("jars")) {
|
for (JsonElement element : rootElement.getAsJsonArray("jars")) {
|
||||||
Path file = root.resolve(element.getAsJsonObject().get("file").getAsString());
|
Path file = root.resolve(element.getAsJsonObject().get("file").getAsString());
|
||||||
if (Files.exists(file)) loadResourcesInternal(file);
|
if (Files.exists(file)) loadResourcePath(file, resourceLoader);
|
||||||
}
|
}
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
Logger.global.logDebug("Failed to read fabric.mod.json: " + ex);
|
Logger.global.logDebug("Failed to read fabric.mod.json: " + ex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
resourceLoader.load(root);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void loadResources(Path root) throws IOException {
|
||||||
try {
|
try {
|
||||||
// do those in parallel
|
// do those in parallel
|
||||||
CompletableFuture.allOf(
|
CompletableFuture.allOf(
|
||||||
@ -226,24 +248,6 @@ private synchronized void loadResourcesInternal(Path root) throws IOException {
|
|||||||
}, blockModels));
|
}, blockModels));
|
||||||
}, BlueMap.THREAD_POOL),
|
}, BlueMap.THREAD_POOL),
|
||||||
|
|
||||||
// load textures
|
|
||||||
CompletableFuture.runAsync(() -> {
|
|
||||||
list(root.resolve("assets"))
|
|
||||||
.map(path -> path.resolve("textures"))
|
|
||||||
.flatMap(ResourcePack::list)
|
|
||||||
.filter(path -> Pattern.matches("blocks?", path.getFileName().toString()))
|
|
||||||
.filter(Files::isDirectory)
|
|
||||||
.flatMap(ResourcePack::walk)
|
|
||||||
.filter(path -> path.getFileName().toString().endsWith(".png"))
|
|
||||||
.filter(Files::isRegularFile)
|
|
||||||
.forEach(file -> loadResource(root, file, () -> {
|
|
||||||
ResourcePath<Texture> resourcePath = new ResourcePath<>(root.relativize(file));
|
|
||||||
try (InputStream in = Files.newInputStream(file)) {
|
|
||||||
return Texture.from(resourcePath, ImageIO.read(in));
|
|
||||||
}
|
|
||||||
}, textures));
|
|
||||||
}, BlueMap.THREAD_POOL),
|
|
||||||
|
|
||||||
// load colormaps
|
// load colormaps
|
||||||
CompletableFuture.runAsync(() -> {
|
CompletableFuture.runAsync(() -> {
|
||||||
walk(root.resolve("assets").resolve("minecraft").resolve("textures").resolve("colormap"))
|
walk(root.resolve("assets").resolve("minecraft").resolve("textures").resolve("colormap"))
|
||||||
@ -322,8 +326,42 @@ private synchronized void loadResourcesInternal(Path root) throws IOException {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public synchronized void bake() throws IOException {
|
private void loadTextures(Path root) throws IOException {
|
||||||
Logger.global.logDebug("Baking resources...");
|
try {
|
||||||
|
|
||||||
|
// collect all used textures
|
||||||
|
Set<ResourcePath<Texture>> usedTextures = new HashSet<>();
|
||||||
|
for (BlockModel model : blockModels.values()) {
|
||||||
|
for (TextureVariable textureVariable : model.getTextures().values()) {
|
||||||
|
if (textureVariable.isReference()) continue;
|
||||||
|
usedTextures.add(textureVariable.getTexturePath());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// load textures
|
||||||
|
list(root.resolve("assets"))
|
||||||
|
.map(path -> path.resolve("textures"))
|
||||||
|
.flatMap(ResourcePack::walk)
|
||||||
|
.filter(path -> path.getFileName().toString().endsWith(".png"))
|
||||||
|
.filter(Files::isRegularFile)
|
||||||
|
.forEach(file -> loadResource(root, file, () -> {
|
||||||
|
ResourcePath<Texture> resourcePath = new ResourcePath<>(root.relativize(file));
|
||||||
|
if (!usedTextures.contains(resourcePath)) return null; // don't load unused textures
|
||||||
|
|
||||||
|
try (InputStream in = Files.newInputStream(file)) {
|
||||||
|
return Texture.from(resourcePath, ImageIO.read(in));
|
||||||
|
}
|
||||||
|
}, textures));
|
||||||
|
|
||||||
|
} catch (RuntimeException ex) {
|
||||||
|
Throwable cause = ex.getCause();
|
||||||
|
if (cause instanceof IOException) throw (IOException) cause;
|
||||||
|
if (cause != null) throw new IOException(cause);
|
||||||
|
throw new IOException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void bake() throws IOException {
|
||||||
|
|
||||||
// fill path maps
|
// fill path maps
|
||||||
blockStates.keySet().forEach(path -> blockStatePaths.put(path.getFormatted(), path));
|
blockStates.keySet().forEach(path -> blockStatePaths.put(path.getFormatted(), path));
|
||||||
@ -352,6 +390,7 @@ public synchronized void bake() throws IOException {
|
|||||||
BufferedImage grass = new ResourcePath<BufferedImage>("minecraft:colormap/grass").getResource(colormaps::get);
|
BufferedImage grass = new ResourcePath<BufferedImage>("minecraft:colormap/grass").getResource(colormaps::get);
|
||||||
if (grass == null) throw new IOException("Failed to bake resource-pack: No grass-colormap found!");
|
if (grass == null) throw new IOException("Failed to bake resource-pack: No grass-colormap found!");
|
||||||
this.colorCalculatorFactory.setGrassMap(grass);
|
this.colorCalculatorFactory.setGrassMap(grass);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private <T> void loadResource(Path root, Path file, Loader<T> loader, Map<ResourcePath<T>, T> resultMap) {
|
private <T> void loadResource(Path root, Path file, Loader<T> loader, Map<ResourcePath<T>, T> resultMap) {
|
||||||
@ -360,6 +399,8 @@ private <T> void loadResource(Path root, Path file, Loader<T> loader, Map<Resour
|
|||||||
if (resultMap.containsKey(resourcePath)) return; // don't load already present resources
|
if (resultMap.containsKey(resourcePath)) return; // don't load already present resources
|
||||||
|
|
||||||
T resource = loader.load();
|
T resource = loader.load();
|
||||||
|
if (resource == null) return; // don't load missing resources
|
||||||
|
|
||||||
resourcePath.setResource(resource);
|
resourcePath.setResource(resource);
|
||||||
resultMap.put(resourcePath, resource);
|
resultMap.put(resourcePath, resource);
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
@ -389,4 +430,8 @@ private interface Loader<T> {
|
|||||||
T load() throws IOException;
|
T load() throws IOException;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private interface PathLoader {
|
||||||
|
void load(Path root) throws IOException;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user