Materials and You: Volume 1

- Add MaterialShaders
  - MaterialShaders must be registered
  - Materials are no longer registered
  - Material now returns a MaterialShaders instead of the locations of
each shader
- Rename MaterialIndices to MaterialShaderIndicies
- Improve MaterialShaderIndicies and move it to the backend package
- Use RegisterClientReloadListenersEvent instead of adding listeners
manually
This commit is contained in:
PepperCode1 2023-11-30 15:38:14 -08:00
parent aa75e7c276
commit 3a581bac79
17 changed files with 195 additions and 234 deletions

View file

@ -8,6 +8,7 @@ import org.slf4j.Logger;
import com.jozufozu.flywheel.api.event.EndClientResourceReloadEvent; import com.jozufozu.flywheel.api.event.EndClientResourceReloadEvent;
import com.jozufozu.flywheel.api.visualization.VisualizationManager; import com.jozufozu.flywheel.api.visualization.VisualizationManager;
import com.jozufozu.flywheel.backend.Backends; import com.jozufozu.flywheel.backend.Backends;
import com.jozufozu.flywheel.backend.MaterialShaderIndices;
import com.jozufozu.flywheel.backend.compile.FlwPrograms; import com.jozufozu.flywheel.backend.compile.FlwPrograms;
import com.jozufozu.flywheel.backend.compile.Pipelines; import com.jozufozu.flywheel.backend.compile.Pipelines;
import com.jozufozu.flywheel.backend.engine.UniformBuffer; import com.jozufozu.flywheel.backend.engine.UniformBuffer;
@ -22,8 +23,7 @@ import com.jozufozu.flywheel.impl.visualization.VisualizationEventHandler;
import com.jozufozu.flywheel.lib.context.Contexts; import com.jozufozu.flywheel.lib.context.Contexts;
import com.jozufozu.flywheel.lib.instance.InstanceTypes; import com.jozufozu.flywheel.lib.instance.InstanceTypes;
import com.jozufozu.flywheel.lib.light.LightUpdater; import com.jozufozu.flywheel.lib.light.LightUpdater;
import com.jozufozu.flywheel.lib.material.MaterialIndices; import com.jozufozu.flywheel.lib.material.StandardMaterialShaders;
import com.jozufozu.flywheel.lib.material.Materials;
import com.jozufozu.flywheel.lib.memory.FlwMemoryTracker; import com.jozufozu.flywheel.lib.memory.FlwMemoryTracker;
import com.jozufozu.flywheel.lib.model.ModelCache; import com.jozufozu.flywheel.lib.model.ModelCache;
import com.jozufozu.flywheel.lib.model.ModelHolder; import com.jozufozu.flywheel.lib.model.ModelHolder;
@ -42,6 +42,7 @@ import net.minecraft.core.Vec3i;
import net.minecraft.resources.ResourceLocation; import net.minecraft.resources.ResourceLocation;
import net.minecraftforge.api.distmarker.Dist; import net.minecraftforge.api.distmarker.Dist;
import net.minecraftforge.client.event.CustomizeGuiOverlayEvent; import net.minecraftforge.client.event.CustomizeGuiOverlayEvent;
import net.minecraftforge.client.event.RegisterClientReloadListenersEvent;
import net.minecraftforge.common.MinecraftForge; import net.minecraftforge.common.MinecraftForge;
import net.minecraftforge.event.level.LevelEvent; import net.minecraftforge.event.level.LevelEvent;
import net.minecraftforge.eventbus.api.IEventBus; import net.minecraftforge.eventbus.api.IEventBus;
@ -101,6 +102,10 @@ public class Flywheel {
forgeEventBus.addListener(LightUpdater::onClientTick); forgeEventBus.addListener(LightUpdater::onClientTick);
forgeEventBus.addListener((LevelEvent.Unload e) -> LevelAttached.onUnloadLevel(e)); forgeEventBus.addListener((LevelEvent.Unload e) -> LevelAttached.onUnloadLevel(e));
// forgeEventBus.addListener(ExampleEffect::tick);
// forgeEventBus.addListener(ExampleEffect::onReload);
modEventBus.addListener(Flywheel::registerClientReloadListeners);
modEventBus.addListener(Flywheel::onClientSetup); modEventBus.addListener(Flywheel::onClientSetup);
modEventBus.addListener(BackendManagerImpl::onEndClientResourceReload); modEventBus.addListener(BackendManagerImpl::onEndClientResourceReload);
@ -111,25 +116,25 @@ public class Flywheel {
modEventBus.addListener(PartialModel::onModelRegistry); modEventBus.addListener(PartialModel::onModelRegistry);
modEventBus.addListener(PartialModel::onModelBake); modEventBus.addListener(PartialModel::onModelBake);
// forgeEventBus.addListener(ExampleEffect::tick);
// forgeEventBus.addListener(ExampleEffect::onReload);
BackendManagerImpl.init(); BackendManagerImpl.init();
ShadersModHandler.init();
Pipelines.init(); Pipelines.init();
Backends.init(); Backends.init();
FlwPrograms.ResourceReloadListener.register(); }
ShadersModHandler.init(); private static void registerClientReloadListeners(RegisterClientReloadListenersEvent event) {
event.registerReloadListener(FlwPrograms.ResourceReloadListener.INSTANCE);
} }
private static void onClientSetup(FMLClientSetupEvent event) { private static void onClientSetup(FMLClientSetupEvent event) {
VertexTypes.init(); VertexTypes.init();
InstanceTypes.init(); InstanceTypes.init();
Materials.init(); StandardMaterialShaders.init();
Contexts.init(); Contexts.init();
MaterialIndices.init(); MaterialShaderIndices.init();
VanillaVisuals.init(); VanillaVisuals.init();

View file

@ -1,22 +1,14 @@
package com.jozufozu.flywheel.api.material; package com.jozufozu.flywheel.api.material;
import com.jozufozu.flywheel.api.registry.Registry;
import com.jozufozu.flywheel.impl.RegistryImpl;
import net.minecraft.client.renderer.RenderType; import net.minecraft.client.renderer.RenderType;
import net.minecraft.resources.ResourceLocation;
public interface Material { public interface Material {
static Registry<Material> REGISTRY = RegistryImpl.create();
ResourceLocation vertexShader();
ResourceLocation fragmentShader();
void setup(); void setup();
void clear(); void clear();
MaterialShaders shaders();
RenderType getFallbackRenderType(); RenderType getFallbackRenderType();
MaterialVertexTransformer getVertexTransformer(); MaterialVertexTransformer getVertexTransformer();

View file

@ -0,0 +1,14 @@
package com.jozufozu.flywheel.api.material;
import com.jozufozu.flywheel.api.registry.Registry;
import com.jozufozu.flywheel.impl.RegistryImpl;
import net.minecraft.resources.ResourceLocation;
public interface MaterialShaders {
static Registry<MaterialShaders> REGISTRY = RegistryImpl.create();
ResourceLocation vertexShader();
ResourceLocation fragmentShader();
}

View file

@ -1,11 +1,10 @@
package com.jozufozu.flywheel.lib.material; package com.jozufozu.flywheel.backend;
import java.util.List; import java.util.List;
import org.jetbrains.annotations.ApiStatus;
import org.jetbrains.annotations.Unmodifiable; import org.jetbrains.annotations.Unmodifiable;
import com.jozufozu.flywheel.api.material.Material; import com.jozufozu.flywheel.api.material.MaterialShaders;
import it.unimi.dsi.fastutil.objects.Object2IntMap; import it.unimi.dsi.fastutil.objects.Object2IntMap;
import it.unimi.dsi.fastutil.objects.Object2IntMaps; import it.unimi.dsi.fastutil.objects.Object2IntMaps;
@ -15,83 +14,58 @@ import it.unimi.dsi.fastutil.objects.ObjectList;
import it.unimi.dsi.fastutil.objects.ObjectLists; import it.unimi.dsi.fastutil.objects.ObjectLists;
import it.unimi.dsi.fastutil.objects.ObjectOpenHashSet; import it.unimi.dsi.fastutil.objects.ObjectOpenHashSet;
import it.unimi.dsi.fastutil.objects.ObjectSet; import it.unimi.dsi.fastutil.objects.ObjectSet;
import it.unimi.dsi.fastutil.objects.Reference2IntMap;
import it.unimi.dsi.fastutil.objects.Reference2IntMaps;
import it.unimi.dsi.fastutil.objects.Reference2IntOpenHashMap;
import net.minecraft.resources.ResourceLocation; import net.minecraft.resources.ResourceLocation;
// TODO: add messages to exceptions public final class MaterialShaderIndices {
public final class MaterialIndices {
private static Reference2IntMap<Material> materialIndices;
private static Object2IntMap<ResourceLocation> vertexShaderIndices; private static Object2IntMap<ResourceLocation> vertexShaderIndices;
private static Object2IntMap<ResourceLocation> fragmentShaderIndices; private static Object2IntMap<ResourceLocation> fragmentShaderIndices;
private static ObjectList<Material> materialsByIndex;
private static ObjectList<ResourceLocation> vertexShadersByIndex; private static ObjectList<ResourceLocation> vertexShadersByIndex;
private static ObjectList<ResourceLocation> fragmentShadersByIndex; private static ObjectList<ResourceLocation> fragmentShadersByIndex;
private static boolean initialized; private static boolean initialized;
public static int getMaterialIndex(Material material) { private MaterialShaderIndices() {
if (!initialized) {
throw new IllegalStateException();
}
return materialIndices.getInt(material);
} }
public static int getVertexShaderIndex(ResourceLocation vertexShader) { public static int getVertexShaderIndex(ResourceLocation vertexShader) {
if (!initialized) { if (!initialized) {
throw new IllegalStateException(); throw new IllegalStateException("Not initialized!");
} }
return vertexShaderIndices.getInt(vertexShader); return vertexShaderIndices.getInt(vertexShader);
} }
public static int getFragmentShaderIndex(ResourceLocation fragmentShader) { public static int getFragmentShaderIndex(ResourceLocation fragmentShader) {
if (!initialized) { if (!initialized) {
throw new IllegalStateException(); throw new IllegalStateException("Not initialized!");
} }
return fragmentShaderIndices.getInt(fragmentShader); return fragmentShaderIndices.getInt(fragmentShader);
} }
public static int getVertexShaderIndex(Material material) { public static int getVertexShaderIndex(MaterialShaders shaders) {
return getVertexShaderIndex(material.vertexShader()); return getVertexShaderIndex(shaders.vertexShader());
} }
public static int getFragmentShaderIndex(Material material) { public static int getFragmentShaderIndex(MaterialShaders shaders) {
return getFragmentShaderIndex(material.fragmentShader()); return getFragmentShaderIndex(shaders.fragmentShader());
}
public static Material getMaterial(int index) {
if (!initialized) {
throw new IllegalStateException();
}
return materialsByIndex.get(index);
} }
public static ResourceLocation getVertexShader(int index) { public static ResourceLocation getVertexShader(int index) {
if (!initialized) { if (!initialized) {
throw new IllegalStateException(); throw new IllegalStateException("Not initialized!");
} }
return vertexShadersByIndex.get(index); return vertexShadersByIndex.get(index);
} }
public static ResourceLocation getFragmentShader(int index) { public static ResourceLocation getFragmentShader(int index) {
if (!initialized) { if (!initialized) {
throw new IllegalStateException(); throw new IllegalStateException("Not initialized!");
} }
return fragmentShadersByIndex.get(index); return fragmentShadersByIndex.get(index);
} }
@Unmodifiable
public static List<Material> getAllMaterials() {
if (!initialized) {
throw new IllegalStateException();
}
return materialsByIndex;
}
@Unmodifiable @Unmodifiable
public static List<ResourceLocation> getAllVertexShaders() { public static List<ResourceLocation> getAllVertexShaders() {
if (!initialized) { if (!initialized) {
throw new IllegalStateException(); throw new IllegalStateException("Not initialized!");
} }
return vertexShadersByIndex; return vertexShadersByIndex;
} }
@ -99,38 +73,34 @@ public final class MaterialIndices {
@Unmodifiable @Unmodifiable
public static List<ResourceLocation> getAllFragmentShaders() { public static List<ResourceLocation> getAllFragmentShaders() {
if (!initialized) { if (!initialized) {
throw new IllegalStateException(); throw new IllegalStateException("Not initialized!");
} }
return fragmentShadersByIndex; return fragmentShadersByIndex;
} }
private static void initInner() { private static void initInner() {
int amount = Material.REGISTRY.getAll().size(); int amount = MaterialShaders.REGISTRY.getAll().size();
Reference2IntMap<Material> materialIndices = new Reference2IntOpenHashMap<>();
Object2IntMap<ResourceLocation> vertexShaderIndices = new Object2IntOpenHashMap<>(); Object2IntMap<ResourceLocation> vertexShaderIndices = new Object2IntOpenHashMap<>();
vertexShaderIndices.defaultReturnValue(-1);
Object2IntMap<ResourceLocation> fragmentShaderIndices = new Object2IntOpenHashMap<>(); Object2IntMap<ResourceLocation> fragmentShaderIndices = new Object2IntOpenHashMap<>();
ObjectList<Material> materialsByIndex = new ObjectArrayList<>(amount); fragmentShaderIndices.defaultReturnValue(-1);
ObjectList<ResourceLocation> vertexShadersByIndex = new ObjectArrayList<>(amount); ObjectList<ResourceLocation> vertexShadersByIndex = new ObjectArrayList<>(amount);
ObjectList<ResourceLocation> fragmentShadersByIndex = new ObjectArrayList<>(amount); ObjectList<ResourceLocation> fragmentShadersByIndex = new ObjectArrayList<>(amount);
ObjectSet<ResourceLocation> allVertexShaders = new ObjectOpenHashSet<>(); ObjectSet<ResourceLocation> allVertexShaders = new ObjectOpenHashSet<>();
ObjectSet<ResourceLocation> allFragmentShaders = new ObjectOpenHashSet<>(); ObjectSet<ResourceLocation> allFragmentShaders = new ObjectOpenHashSet<>();
int materialIndex = 0;
int vertexShaderIndex = 0; int vertexShaderIndex = 0;
int fragmentShaderIndex = 0; int fragmentShaderIndex = 0;
for (Material material : Material.REGISTRY) { for (MaterialShaders shaders : MaterialShaders.REGISTRY) {
materialIndices.put(material, materialIndex); ResourceLocation vertexShader = shaders.vertexShader();
materialsByIndex.add(material);
materialIndex++;
ResourceLocation vertexShader = material.vertexShader();
if (allVertexShaders.add(vertexShader)) { if (allVertexShaders.add(vertexShader)) {
vertexShaderIndices.put(vertexShader, vertexShaderIndex); vertexShaderIndices.put(vertexShader, vertexShaderIndex);
vertexShadersByIndex.add(vertexShader); vertexShadersByIndex.add(vertexShader);
vertexShaderIndex++; vertexShaderIndex++;
} }
ResourceLocation fragmentShader = material.fragmentShader(); ResourceLocation fragmentShader = shaders.fragmentShader();
if (allFragmentShaders.add(fragmentShader)) { if (allFragmentShaders.add(fragmentShader)) {
fragmentShaderIndices.put(fragmentShader, fragmentShaderIndex); fragmentShaderIndices.put(fragmentShader, fragmentShaderIndex);
fragmentShadersByIndex.add(fragmentShader); fragmentShadersByIndex.add(fragmentShader);
@ -138,18 +108,15 @@ public final class MaterialIndices {
} }
} }
MaterialIndices.materialIndices = Reference2IntMaps.unmodifiable(materialIndices); MaterialShaderIndices.vertexShaderIndices = Object2IntMaps.unmodifiable(vertexShaderIndices);
MaterialIndices.vertexShaderIndices = Object2IntMaps.unmodifiable(vertexShaderIndices); MaterialShaderIndices.fragmentShaderIndices = Object2IntMaps.unmodifiable(fragmentShaderIndices);
MaterialIndices.fragmentShaderIndices = Object2IntMaps.unmodifiable(fragmentShaderIndices); MaterialShaderIndices.vertexShadersByIndex = ObjectLists.unmodifiable(vertexShadersByIndex);
MaterialIndices.materialsByIndex = ObjectLists.unmodifiable(materialsByIndex); MaterialShaderIndices.fragmentShadersByIndex = ObjectLists.unmodifiable(fragmentShadersByIndex);
MaterialIndices.vertexShadersByIndex = ObjectLists.unmodifiable(vertexShadersByIndex);
MaterialIndices.fragmentShadersByIndex = ObjectLists.unmodifiable(fragmentShadersByIndex);
initialized = true; initialized = true;
} }
@ApiStatus.Internal
public static void init() { public static void init() {
Material.REGISTRY.addFreezeCallback(MaterialIndices::initInner); MaterialShaders.REGISTRY.addFreezeCallback(MaterialShaderIndices::initInner);
} }
} }

View file

@ -6,16 +6,14 @@ import com.jozufozu.flywheel.api.context.Context;
import com.jozufozu.flywheel.api.instance.InstanceType; import com.jozufozu.flywheel.api.instance.InstanceType;
import com.jozufozu.flywheel.api.uniform.ShaderUniforms; import com.jozufozu.flywheel.api.uniform.ShaderUniforms;
import com.jozufozu.flywheel.api.vertex.VertexType; import com.jozufozu.flywheel.api.vertex.VertexType;
import com.jozufozu.flywheel.backend.MaterialShaderIndices;
import com.jozufozu.flywheel.backend.compile.component.MaterialAdapterComponent; import com.jozufozu.flywheel.backend.compile.component.MaterialAdapterComponent;
import com.jozufozu.flywheel.backend.compile.component.UniformComponent; import com.jozufozu.flywheel.backend.compile.component.UniformComponent;
import com.jozufozu.flywheel.backend.compile.core.CompilerStats; import com.jozufozu.flywheel.backend.compile.core.CompilerStats;
import com.jozufozu.flywheel.glsl.ShaderSources; import com.jozufozu.flywheel.glsl.ShaderSources;
import com.jozufozu.flywheel.glsl.generate.FnSignature; import com.jozufozu.flywheel.glsl.generate.FnSignature;
import com.jozufozu.flywheel.glsl.generate.GlslExpr; import com.jozufozu.flywheel.glsl.generate.GlslExpr;
import com.jozufozu.flywheel.lib.material.MaterialIndices;
import net.minecraft.client.Minecraft;
import net.minecraft.server.packs.resources.ReloadableResourceManager;
import net.minecraft.server.packs.resources.ResourceManager; import net.minecraft.server.packs.resources.ResourceManager;
import net.minecraft.server.packs.resources.ResourceManagerReloadListener; import net.minecraft.server.packs.resources.ResourceManagerReloadListener;
@ -44,7 +42,7 @@ public class FlwPrograms {
private static MaterialAdapterComponent createFragmentMaterialComponent(SourceLoader loadChecker) { private static MaterialAdapterComponent createFragmentMaterialComponent(SourceLoader loadChecker) {
return MaterialAdapterComponent.builder(Flywheel.rl("fragment_material_adapter")) return MaterialAdapterComponent.builder(Flywheel.rl("fragment_material_adapter"))
.materialSources(MaterialIndices.getAllFragmentShaders()) .materialSources(MaterialShaderIndices.getAllFragmentShaders())
.adapt(FnSignature.ofVoid("flw_materialFragment")) .adapt(FnSignature.ofVoid("flw_materialFragment"))
.adapt(FnSignature.create() .adapt(FnSignature.create()
.returnType("bool") .returnType("bool")
@ -62,7 +60,7 @@ public class FlwPrograms {
private static MaterialAdapterComponent createVertexMaterialComponent(SourceLoader loadChecker) { private static MaterialAdapterComponent createVertexMaterialComponent(SourceLoader loadChecker) {
return MaterialAdapterComponent.builder(Flywheel.rl("vertex_material_adapter")) return MaterialAdapterComponent.builder(Flywheel.rl("vertex_material_adapter"))
.materialSources(MaterialIndices.getAllVertexShaders()) .materialSources(MaterialShaderIndices.getAllVertexShaders())
.adapt(FnSignature.ofVoid("flw_materialVertex")) .adapt(FnSignature.ofVoid("flw_materialVertex"))
.switchOn(GlslExpr.variable("_flw_materialVertexID")) .switchOn(GlslExpr.variable("_flw_materialVertexID"))
.build(loadChecker); .build(loadChecker);
@ -100,17 +98,5 @@ public class FlwPrograms {
public void onResourceManagerReload(ResourceManager manager) { public void onResourceManagerReload(ResourceManager manager) {
FlwPrograms.reload(manager); FlwPrograms.reload(manager);
} }
public static void register() {
// Can be null when running data generators due to the unfortunate time we call this
Minecraft minecraft = Minecraft.getInstance();
if (minecraft == null) {
return;
}
if (minecraft.getResourceManager() instanceof ReloadableResourceManager reloadable) {
reloadable.registerReloadListener(INSTANCE);
}
}
} }
} }

View file

@ -20,7 +20,7 @@ import net.minecraft.client.renderer.RenderType;
import net.minecraft.client.resources.model.ModelBakery; import net.minecraft.client.resources.model.ModelBakery;
public class BatchedCrumbling { public class BatchedCrumbling {
static void renderCrumbling(List<Engine.CrumblingBlock> crumblingBlocks, BatchContext batchContext, BatchedDrawTracker drawTracker) { public static void render(List<Engine.CrumblingBlock> crumblingBlocks, BatchContext batchContext, BatchedDrawTracker drawTracker) {
var instancesPerType = doCrumblingSort(crumblingBlocks); var instancesPerType = doCrumblingSort(crumblingBlocks);
for (var entry : instancesPerType.entrySet()) { for (var entry : instancesPerType.entrySet()) {
@ -82,7 +82,7 @@ public class BatchedCrumbling {
return out; return out;
} }
public static <I extends Instance> int bufferOne(BatchedInstancer<I> batchedInstancer, int baseVertex, ReusableVertexList vertexList, DrawBuffer drawBuffer, Instance instance) { private static <I extends Instance> int bufferOne(BatchedInstancer<I> batchedInstancer, int baseVertex, ReusableVertexList vertexList, DrawBuffer drawBuffer, Instance instance) {
int totalVertices = 0; int totalVertices = 0;
for (TransformCall<I> transformCall : batchedInstancer.getTransformCalls()) { for (TransformCall<I> transformCall : batchedInstancer.getTransformCalls()) {
@ -101,9 +101,8 @@ public class BatchedCrumbling {
return totalVertices; return totalVertices;
} }
static final class CrumblingBucket { private static class CrumblingBucket {
private int vertexCount; private int vertexCount;
private final List<Pair<Instance, BatchedInstancer<?>>> instances = new ArrayList<>(); private final List<Pair<Instance, BatchedInstancer<?>>> instances = new ArrayList<>();
} }
} }

View file

@ -44,6 +44,7 @@ class BatchedDrawManager extends InstancerStorage<BatchedInstancer<?>> {
return new BatchedInstancer<>(type); return new BatchedInstancer<>(type);
} }
@SuppressWarnings({ "rawtypes", "unchecked" })
@Override @Override
protected <I extends Instance> void add(InstancerKey<I> key, BatchedInstancer<?> instancer, Model model, RenderStage stage) { protected <I extends Instance> void add(InstancerKey<I> key, BatchedInstancer<?> instancer, Model model, RenderStage stage) {
var stagePlan = stagePlans.computeIfAbsent(stage, renderStage -> new BatchedStagePlan(renderStage, drawTracker)); var stagePlan = stagePlans.computeIfAbsent(stage, renderStage -> new BatchedStagePlan(renderStage, drawTracker));
@ -51,8 +52,10 @@ class BatchedDrawManager extends InstancerStorage<BatchedInstancer<?>> {
for (var entry : meshes.entrySet()) { for (var entry : meshes.entrySet()) {
var material = entry.getKey(); var material = entry.getKey();
RenderType renderType = material.getFallbackRenderType(); RenderType renderType = material.getFallbackRenderType();
var transformCall = new TransformCall<>(instancer, material, alloc(entry.getValue(), renderType.format())); var mesh = alloc(entry.getValue(), renderType.format());
var transformCall = new TransformCall<>(instancer, material, mesh);
stagePlan.put(renderType, transformCall); stagePlan.put(renderType, transformCall);
instancer.addTransformCall((TransformCall) transformCall);
} }
} }

View file

@ -48,7 +48,7 @@ public class BatchingEngine extends AbstractEngine {
executor.syncUntil(flushFlag::isRaised); executor.syncUntil(flushFlag::isRaised);
var batchContext = BatchContext.create(context, this.renderOrigin); var batchContext = BatchContext.create(context, this.renderOrigin);
BatchedCrumbling.renderCrumbling(crumblingBlocks, batchContext, this.drawManager.drawTracker); BatchedCrumbling.render(crumblingBlocks, batchContext, this.drawManager.drawTracker);
} }
@Override @Override

View file

@ -30,7 +30,6 @@ public class TransformCall<I extends Instance> {
this.instancer = instancer; this.instancer = instancer;
this.material = material; this.material = material;
this.mesh = mesh; this.mesh = mesh;
instancer.addTransformCall(this);
InstanceVertexTransformer<I> instanceVertexTransformer = instancer.type.getVertexTransformer(); InstanceVertexTransformer<I> instanceVertexTransformer = instancer.type.getVertexTransformer();
InstanceBoundingSphereTransformer<I> boundingSphereTransformer = instancer.type.getBoundingSphereTransformer(); InstanceBoundingSphereTransformer<I> boundingSphereTransformer = instancer.type.getBoundingSphereTransformer();

View file

@ -5,7 +5,7 @@ import org.lwjgl.system.MemoryUtil;
import com.jozufozu.flywheel.api.event.RenderStage; import com.jozufozu.flywheel.api.event.RenderStage;
import com.jozufozu.flywheel.api.instance.Instance; import com.jozufozu.flywheel.api.instance.Instance;
import com.jozufozu.flywheel.api.material.Material; import com.jozufozu.flywheel.api.material.Material;
import com.jozufozu.flywheel.lib.material.MaterialIndices; import com.jozufozu.flywheel.backend.MaterialShaderIndices;
public class IndirectDraw<I extends Instance> { public class IndirectDraw<I extends Instance> {
private final IndirectInstancer<I> instancer; private final IndirectInstancer<I> instancer;
@ -25,8 +25,8 @@ public class IndirectDraw<I extends Instance> {
this.mesh = mesh; this.mesh = mesh;
this.stage = stage; this.stage = stage;
this.vertexMaterialID = MaterialIndices.getVertexShaderIndex(material); this.vertexMaterialID = MaterialShaderIndices.getVertexShaderIndex(material.shaders());
this.fragmentMaterialID = MaterialIndices.getFragmentShaderIndex(material); this.fragmentMaterialID = MaterialShaderIndices.getFragmentShaderIndex(material.shaders());
} }
public IndirectInstancer<I> instancer() { public IndirectInstancer<I> instancer() {

View file

@ -13,7 +13,6 @@ import java.util.Map;
import com.jozufozu.flywheel.api.event.RenderStage; import com.jozufozu.flywheel.api.event.RenderStage;
import com.jozufozu.flywheel.api.instance.Instance; import com.jozufozu.flywheel.api.instance.Instance;
import com.jozufozu.flywheel.api.material.Material; import com.jozufozu.flywheel.api.material.Material;
import com.jozufozu.flywheel.lib.material.MaterialIndices;
public class IndirectDrawSet<I extends Instance> { public class IndirectDrawSet<I extends Instance> {
@ -49,7 +48,7 @@ public class IndirectDrawSet<I extends Instance> {
multiDraws.clear(); multiDraws.clear();
// sort by stage, then material // sort by stage, then material
indirectDraws.sort(Comparator.comparing(IndirectDraw<I>::stage) indirectDraws.sort(Comparator.comparing(IndirectDraw<I>::stage)
.thenComparing(draw -> MaterialIndices.getMaterialIndex(draw.material()))); .thenComparing(draw -> draw.material().hashCode()));
for (int start = 0, i = 0; i < indirectDraws.size(); i++) { for (int start = 0, i = 0; i < indirectDraws.size(); i++) {
var draw = indirectDraws.get(i); var draw = indirectDraws.get(i);

View file

@ -23,42 +23,6 @@ import it.unimi.dsi.fastutil.ints.Int2ObjectMap;
import net.minecraft.client.resources.model.ModelBakery; import net.minecraft.client.resources.model.ModelBakery;
public class InstancedCrumbling { public class InstancedCrumbling {
@NotNull
public static Map<ShaderState, Int2ObjectMap<List<Runnable>>> doCrumblingSort(List<Engine.CrumblingBlock> instances) {
Map<ShaderState, Int2ObjectMap<List<Runnable>>> out = new HashMap<>();
for (Engine.CrumblingBlock triple : instances) {
int progress = triple.progress();
if (progress < 0 || progress >= ModelBakery.DESTROY_TYPES.size()) {
continue;
}
for (Instance instance : triple.instances()) {
// Filter out instances that weren't created by this engine.
// If all is well, we probably shouldn't take the `continue`
// branches but better to do checked casts.
if (!(instance.handle() instanceof InstanceHandleImpl impl)) {
continue;
}
if (!(impl.instancer instanceof InstancedInstancer<?> instancer)) {
continue;
}
List<DrawCall> draws = instancer.drawCalls();
draws.removeIf(DrawCall::isInvalid);
for (DrawCall draw : draws) {
out.computeIfAbsent(draw.shaderState, $ -> new Int2ObjectArrayMap<>())
.computeIfAbsent(progress, $ -> new ArrayList<>())
.add(() -> draw.renderOne(impl));
}
}
}
return out;
}
public static void render(List<Engine.CrumblingBlock> crumblingBlocks) { public static void render(List<Engine.CrumblingBlock> crumblingBlocks) {
// Sort draw calls into buckets, so we don't have to do as many shader binds. // Sort draw calls into buckets, so we don't have to do as many shader binds.
var byShaderState = doCrumblingSort(crumblingBlocks); var byShaderState = doCrumblingSort(crumblingBlocks);
@ -108,6 +72,42 @@ public class InstancedCrumbling {
} }
} }
@NotNull
private static Map<ShaderState, Int2ObjectMap<List<Runnable>>> doCrumblingSort(List<Engine.CrumblingBlock> instances) {
Map<ShaderState, Int2ObjectMap<List<Runnable>>> out = new HashMap<>();
for (Engine.CrumblingBlock triple : instances) {
int progress = triple.progress();
if (progress < 0 || progress >= ModelBakery.DESTROY_TYPES.size()) {
continue;
}
for (Instance instance : triple.instances()) {
// Filter out instances that weren't created by this engine.
// If all is well, we probably shouldn't take the `continue`
// branches but better to do checked casts.
if (!(instance.handle() instanceof InstanceHandleImpl impl)) {
continue;
}
if (!(impl.instancer instanceof InstancedInstancer<?> instancer)) {
continue;
}
List<DrawCall> draws = instancer.drawCalls();
draws.removeIf(DrawCall::isInvalid);
for (DrawCall draw : draws) {
out.computeIfAbsent(draw.shaderState, $ -> new Int2ObjectArrayMap<>())
.computeIfAbsent(progress, $ -> new ArrayList<>())
.add(() -> draw.renderOne(impl));
}
}
}
return out;
}
private static int getDiffuseTexture(Material material) { private static int getDiffuseTexture(Material material) {
material.setup(); material.setup();

View file

@ -9,6 +9,7 @@ import com.jozufozu.flywheel.api.event.RenderStage;
import com.jozufozu.flywheel.api.material.Material; import com.jozufozu.flywheel.api.material.Material;
import com.jozufozu.flywheel.api.task.Plan; import com.jozufozu.flywheel.api.task.Plan;
import com.jozufozu.flywheel.api.task.TaskExecutor; import com.jozufozu.flywheel.api.task.TaskExecutor;
import com.jozufozu.flywheel.backend.MaterialShaderIndices;
import com.jozufozu.flywheel.backend.compile.InstancingPrograms; import com.jozufozu.flywheel.backend.compile.InstancingPrograms;
import com.jozufozu.flywheel.backend.engine.AbstractEngine; import com.jozufozu.flywheel.backend.engine.AbstractEngine;
import com.jozufozu.flywheel.backend.engine.AbstractInstancer; import com.jozufozu.flywheel.backend.engine.AbstractInstancer;
@ -18,7 +19,6 @@ import com.jozufozu.flywheel.gl.GlStateTracker;
import com.jozufozu.flywheel.gl.GlTextureUnit; import com.jozufozu.flywheel.gl.GlTextureUnit;
import com.jozufozu.flywheel.gl.shader.GlProgram; import com.jozufozu.flywheel.gl.shader.GlProgram;
import com.jozufozu.flywheel.lib.context.Contexts; import com.jozufozu.flywheel.lib.context.Contexts;
import com.jozufozu.flywheel.lib.material.MaterialIndices;
import com.jozufozu.flywheel.lib.task.Flag; import com.jozufozu.flywheel.lib.task.Flag;
import com.jozufozu.flywheel.lib.task.NamedFlag; import com.jozufozu.flywheel.lib.task.NamedFlag;
import com.jozufozu.flywheel.lib.task.SyncedPlan; import com.jozufozu.flywheel.lib.task.SyncedPlan;
@ -125,8 +125,8 @@ public class InstancingEngine extends AbstractEngine {
public static void uploadMaterialIDUniform(GlProgram program, Material material) { public static void uploadMaterialIDUniform(GlProgram program, Material material) {
int materialIDUniform = program.getUniformLocation("_flw_materialID_instancing"); int materialIDUniform = program.getUniformLocation("_flw_materialID_instancing");
int vertexID = MaterialIndices.getVertexShaderIndex(material); int vertexID = MaterialShaderIndices.getVertexShaderIndex(material.shaders());
int fragmentID = MaterialIndices.getFragmentShaderIndex(material); int fragmentID = MaterialShaderIndices.getFragmentShaderIndex(material.shaders());
GL32.glUniform2ui(materialIDUniform, vertexID, fragmentID); GL32.glUniform2ui(materialIDUniform, vertexID, fragmentID);
} }
} }

View file

@ -1,8 +1,5 @@
package com.jozufozu.flywheel.lib.material; package com.jozufozu.flywheel.lib.material;
import org.jetbrains.annotations.ApiStatus;
import com.jozufozu.flywheel.Flywheel;
import com.jozufozu.flywheel.api.material.Material; import com.jozufozu.flywheel.api.material.Material;
import com.jozufozu.flywheel.api.material.MaterialVertexTransformer; import com.jozufozu.flywheel.api.material.MaterialVertexTransformer;
import com.jozufozu.flywheel.gl.GlTextureUnit; import com.jozufozu.flywheel.gl.GlTextureUnit;
@ -37,109 +34,92 @@ public final class Materials {
private static final ResourceLocation MINECART_LOCATION = new ResourceLocation("textures/entity/minecart.png"); private static final ResourceLocation MINECART_LOCATION = new ResourceLocation("textures/entity/minecart.png");
public static final Material CHUNK_SOLID_SHADED = SimpleMaterial.builder() public static final Material CHUNK_SOLID_SHADED = SimpleMaterial.builder()
.vertexShader(Files.SHADED_VERTEX)
.fragmentShader(Files.DEFAULT_FRAGMENT)
.addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true)) .addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true))
.shaders(StandardMaterialShaders.SHADED)
.fallbackRenderType(RenderType.solid()) .fallbackRenderType(RenderType.solid())
.vertexTransformer(SHADING_TRANSFORMER) .vertexTransformer(SHADING_TRANSFORMER)
.register(); .build();
public static final Material CHUNK_SOLID_UNSHADED = SimpleMaterial.builder() public static final Material CHUNK_SOLID_UNSHADED = SimpleMaterial.builder()
.vertexShader(Files.DEFAULT_VERTEX)
.fragmentShader(Files.DEFAULT_FRAGMENT)
.addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true)) .addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true))
.fallbackRenderType(RenderType.solid()) .fallbackRenderType(RenderType.solid())
.register(); .build();
public static final Material CHUNK_CUTOUT_MIPPED_SHADED = SimpleMaterial.builder() public static final Material CHUNK_CUTOUT_MIPPED_SHADED = SimpleMaterial.builder()
.vertexShader(Files.SHADED_VERTEX)
.fragmentShader(Files.CUTOUT_FRAGMENT)
.addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true)) .addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true))
.shaders(StandardMaterialShaders.SHADED_CUTOUT)
.fallbackRenderType(RenderType.cutoutMipped()) .fallbackRenderType(RenderType.cutoutMipped())
.vertexTransformer(SHADING_TRANSFORMER) .vertexTransformer(SHADING_TRANSFORMER)
.register(); .build();
public static final Material CHUNK_CUTOUT_MIPPED_UNSHADED = SimpleMaterial.builder() public static final Material CHUNK_CUTOUT_MIPPED_UNSHADED = SimpleMaterial.builder()
.vertexShader(Files.DEFAULT_VERTEX)
.fragmentShader(Files.CUTOUT_FRAGMENT)
.addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true)) .addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true))
.shaders(StandardMaterialShaders.CUTOUT)
.fallbackRenderType(RenderType.cutoutMipped()) .fallbackRenderType(RenderType.cutoutMipped())
.register(); .build();
public static final Material CHUNK_CUTOUT_SHADED = SimpleMaterial.builder() public static final Material CHUNK_CUTOUT_SHADED = SimpleMaterial.builder()
.vertexShader(Files.SHADED_VERTEX)
.fragmentShader(Files.CUTOUT_FRAGMENT)
.addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, false)) .addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, false))
.shaders(StandardMaterialShaders.SHADED_CUTOUT)
.fallbackRenderType(RenderType.cutout()) .fallbackRenderType(RenderType.cutout())
.vertexTransformer(SHADING_TRANSFORMER) .vertexTransformer(SHADING_TRANSFORMER)
.register(); .build();
public static final Material CHUNK_CUTOUT_UNSHADED = SimpleMaterial.builder() public static final Material CHUNK_CUTOUT_UNSHADED = SimpleMaterial.builder()
.vertexShader(Files.DEFAULT_VERTEX)
.fragmentShader(Files.CUTOUT_FRAGMENT)
.addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, false)) .addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, false))
.shaders(StandardMaterialShaders.CUTOUT)
.fallbackRenderType(RenderType.cutout()) .fallbackRenderType(RenderType.cutout())
.register(); .build();
public static final Material CHUNK_TRANSLUCENT_SHADED = SimpleMaterial.builder() public static final Material CHUNK_TRANSLUCENT_SHADED = SimpleMaterial.builder()
.vertexShader(Files.SHADED_VERTEX)
.fragmentShader(Files.DEFAULT_FRAGMENT)
.addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true)) .addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true))
.addShard(Shards.TRANSLUCENT_TRANSPARENCY) .addShard(Shards.TRANSLUCENT_TRANSPARENCY)
.shaders(StandardMaterialShaders.SHADED)
.fallbackRenderType(RenderType.translucent()) .fallbackRenderType(RenderType.translucent())
.vertexTransformer(SHADING_TRANSFORMER) .vertexTransformer(SHADING_TRANSFORMER)
.register(); .build();
public static final Material CHUNK_TRANSLUCENT_UNSHADED = SimpleMaterial.builder() public static final Material CHUNK_TRANSLUCENT_UNSHADED = SimpleMaterial.builder()
.vertexShader(Files.DEFAULT_VERTEX)
.fragmentShader(Files.DEFAULT_FRAGMENT)
.addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true)) .addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true))
.addShard(Shards.TRANSLUCENT_TRANSPARENCY) .addShard(Shards.TRANSLUCENT_TRANSPARENCY)
.fallbackRenderType(RenderType.translucent()) .fallbackRenderType(RenderType.translucent())
.register(); .build();
public static final Material CHUNK_TRIPWIRE_SHADED = SimpleMaterial.builder() public static final Material CHUNK_TRIPWIRE_SHADED = SimpleMaterial.builder()
.vertexShader(Files.SHADED_VERTEX)
.fragmentShader(Files.CUTOUT_FRAGMENT)
.addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true)) .addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true))
.addShard(Shards.TRANSLUCENT_TRANSPARENCY) .addShard(Shards.TRANSLUCENT_TRANSPARENCY)
.shaders(StandardMaterialShaders.SHADED_CUTOUT)
.fallbackRenderType(RenderType.tripwire()) .fallbackRenderType(RenderType.tripwire())
.vertexTransformer(SHADING_TRANSFORMER) .vertexTransformer(SHADING_TRANSFORMER)
.register(); .build();
public static final Material CHUNK_TRIPWIRE_UNSHADED = SimpleMaterial.builder() public static final Material CHUNK_TRIPWIRE_UNSHADED = SimpleMaterial.builder()
.vertexShader(Files.DEFAULT_VERTEX)
.fragmentShader(Files.CUTOUT_FRAGMENT)
.addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true)) .addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, true))
.addShard(Shards.TRANSLUCENT_TRANSPARENCY) .addShard(Shards.TRANSLUCENT_TRANSPARENCY)
.shaders(StandardMaterialShaders.CUTOUT)
.fallbackRenderType(RenderType.tripwire()) .fallbackRenderType(RenderType.tripwire())
.register(); .build();
public static final Material CHEST = SimpleMaterial.builder() public static final Material CHEST = SimpleMaterial.builder()
.vertexShader(Files.SHADED_VERTEX)
.addShard(Shards.diffuseTex(Sheets.CHEST_SHEET, false, false)) .addShard(Shards.diffuseTex(Sheets.CHEST_SHEET, false, false))
.shaders(StandardMaterialShaders.SHADED)
.fallbackRenderType(Sheets.chestSheet()) .fallbackRenderType(Sheets.chestSheet())
.register(); .build();
public static final Material SHULKER = SimpleMaterial.builder() public static final Material SHULKER = SimpleMaterial.builder()
.vertexShader(Files.SHADED_VERTEX)
.fragmentShader(Files.CUTOUT_FRAGMENT)
.addShard(Shards.diffuseTex(Sheets.SHULKER_SHEET, false, false)) .addShard(Shards.diffuseTex(Sheets.SHULKER_SHEET, false, false))
.addShard(Shards.DISABLE_CULL) .addShard(Shards.DISABLE_CULL)
.shaders(StandardMaterialShaders.SHADED_CUTOUT)
.fallbackRenderType(Sheets.shulkerBoxSheet()) .fallbackRenderType(Sheets.shulkerBoxSheet())
.register(); .build();
public static final Material BELL = SimpleMaterial.builder() public static final Material BELL = SimpleMaterial.builder()
.vertexShader(Files.SHADED_VERTEX)
.addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, false)) .addShard(Shards.diffuseTex(InventoryMenu.BLOCK_ATLAS, false, false))
.shaders(StandardMaterialShaders.SHADED)
.fallbackRenderType(Sheets.solidBlockSheet()) .fallbackRenderType(Sheets.solidBlockSheet())
.register(); .build();
public static final Material MINECART = SimpleMaterial.builder() public static final Material MINECART = SimpleMaterial.builder()
.vertexShader(Files.SHADED_VERTEX)
.addShard(Shards.diffuseTex(MINECART_LOCATION, false, false)) .addShard(Shards.diffuseTex(MINECART_LOCATION, false, false))
.shaders(StandardMaterialShaders.SHADED)
.fallbackRenderType(RenderType.entitySolid(MINECART_LOCATION)) .fallbackRenderType(RenderType.entitySolid(MINECART_LOCATION))
.register(); .build();
private Materials() { private Materials() {
} }
@ApiStatus.Internal
public static void init() {
}
public static final class Shards { public static final class Shards {
public static final GlStateShard DISABLE_CULL = new GlStateShard( public static final GlStateShard DISABLE_CULL = new GlStateShard(
() -> { () -> {
@ -176,17 +156,4 @@ public final class Materials {
); );
} }
} }
public static final class Files {
public static final ResourceLocation DEFAULT_VERTEX = Names.DEFAULT.withSuffix(".vert");
public static final ResourceLocation SHADED_VERTEX = Names.SHADED.withSuffix(".vert");
public static final ResourceLocation DEFAULT_FRAGMENT = Names.DEFAULT.withSuffix(".frag");
public static final ResourceLocation CUTOUT_FRAGMENT = Names.CUTOUT.withSuffix(".frag");
}
public static final class Names {
public static final ResourceLocation DEFAULT = Flywheel.rl("material/default");
public static final ResourceLocation CUTOUT = Flywheel.rl("material/cutout");
public static final ResourceLocation SHADED = Flywheel.rl("material/shaded");
}
} }

View file

@ -1,25 +1,23 @@
package com.jozufozu.flywheel.lib.material; package com.jozufozu.flywheel.lib.material;
import com.jozufozu.flywheel.api.material.Material; import com.jozufozu.flywheel.api.material.Material;
import com.jozufozu.flywheel.api.material.MaterialShaders;
import com.jozufozu.flywheel.api.material.MaterialVertexTransformer; import com.jozufozu.flywheel.api.material.MaterialVertexTransformer;
import net.minecraft.client.renderer.RenderStateShard; import net.minecraft.client.renderer.RenderStateShard;
import net.minecraft.client.renderer.RenderType; import net.minecraft.client.renderer.RenderType;
import net.minecraft.resources.ResourceLocation;
public class SimpleMaterial implements Material { public class SimpleMaterial implements Material {
protected final ResourceLocation vertexShader;
protected final ResourceLocation fragmentShader;
protected final Runnable setup; protected final Runnable setup;
protected final Runnable clear; protected final Runnable clear;
protected final MaterialShaders shaders;
protected final RenderType fallbackRenderType; protected final RenderType fallbackRenderType;
protected final MaterialVertexTransformer vertexTransformer; protected final MaterialVertexTransformer vertexTransformer;
public SimpleMaterial(ResourceLocation vertexShader, ResourceLocation fragmentShader, Runnable setup, Runnable clear, RenderType fallbackRenderType, MaterialVertexTransformer vertexTransformer) { public SimpleMaterial(Runnable setup, Runnable clear, MaterialShaders shaders, RenderType fallbackRenderType, MaterialVertexTransformer vertexTransformer) {
this.vertexShader = vertexShader;
this.fragmentShader = fragmentShader;
this.setup = setup; this.setup = setup;
this.clear = clear; this.clear = clear;
this.shaders = shaders;
this.fallbackRenderType = fallbackRenderType; this.fallbackRenderType = fallbackRenderType;
this.vertexTransformer = vertexTransformer; this.vertexTransformer = vertexTransformer;
} }
@ -28,16 +26,6 @@ public class SimpleMaterial implements Material {
return new Builder(); return new Builder();
} }
@Override
public ResourceLocation vertexShader() {
return vertexShader;
}
@Override
public ResourceLocation fragmentShader() {
return fragmentShader;
}
@Override @Override
public void setup() { public void setup() {
setup.run(); setup.run();
@ -48,6 +36,11 @@ public class SimpleMaterial implements Material {
clear.run(); clear.run();
} }
@Override
public MaterialShaders shaders() {
return shaders;
}
@Override @Override
public RenderType getFallbackRenderType() { public RenderType getFallbackRenderType() {
return fallbackRenderType; return fallbackRenderType;
@ -59,26 +52,15 @@ public class SimpleMaterial implements Material {
} }
public static class Builder { public static class Builder {
protected ResourceLocation vertexShader = Materials.Files.DEFAULT_VERTEX;
protected ResourceLocation fragmentShader = Materials.Files.DEFAULT_FRAGMENT;
protected Runnable setup = () -> {}; protected Runnable setup = () -> {};
protected Runnable clear = () -> {}; protected Runnable clear = () -> {};
protected MaterialShaders shaders = StandardMaterialShaders.DEFAULT;
protected RenderType fallbackRenderType = RenderType.solid(); protected RenderType fallbackRenderType = RenderType.solid();
protected MaterialVertexTransformer vertexTransformer = (vertexList, level) -> {}; protected MaterialVertexTransformer vertexTransformer = (vertexList, level) -> {};
public Builder() { public Builder() {
} }
public Builder vertexShader(ResourceLocation vertexShader) {
this.vertexShader = vertexShader;
return this;
}
public Builder fragmentShader(ResourceLocation fragmentShader) {
this.fragmentShader = fragmentShader;
return this;
}
public Builder addSetup(Runnable setup) { public Builder addSetup(Runnable setup) {
this.setup = chain(this.setup, setup); this.setup = chain(this.setup, setup);
return this; return this;
@ -95,6 +77,11 @@ public class SimpleMaterial implements Material {
return this; return this;
} }
public Builder shaders(MaterialShaders shaders) {
this.shaders = shaders;
return this;
}
public Builder fallbackRenderType(RenderType type) { public Builder fallbackRenderType(RenderType type) {
this.fallbackRenderType = type; this.fallbackRenderType = type;
return this; return this;
@ -105,8 +92,8 @@ public class SimpleMaterial implements Material {
return this; return this;
} }
public SimpleMaterial register() { public SimpleMaterial build() {
return Material.REGISTRY.registerAndGet(new SimpleMaterial(vertexShader, fragmentShader, setup, clear, fallbackRenderType, vertexTransformer)); return new SimpleMaterial(setup, clear, shaders, fallbackRenderType, vertexTransformer);
} }
private static Runnable chain(Runnable runnable1, Runnable runnable2) { private static Runnable chain(Runnable runnable1, Runnable runnable2) {

View file

@ -0,0 +1,8 @@
package com.jozufozu.flywheel.lib.material;
import com.jozufozu.flywheel.api.material.MaterialShaders;
import net.minecraft.resources.ResourceLocation;
public record SimpleMaterialShaders(ResourceLocation vertexShader, ResourceLocation fragmentShader) implements MaterialShaders {
}

View file

@ -0,0 +1,35 @@
package com.jozufozu.flywheel.lib.material;
import org.jetbrains.annotations.ApiStatus;
import com.jozufozu.flywheel.Flywheel;
import com.jozufozu.flywheel.api.material.MaterialShaders;
import net.minecraft.resources.ResourceLocation;
public final class StandardMaterialShaders {
public static final MaterialShaders DEFAULT = MaterialShaders.REGISTRY.registerAndGet(new SimpleMaterialShaders(Files.DEFAULT_VERTEX, Files.DEFAULT_FRAGMENT));
public static final MaterialShaders SHADED = MaterialShaders.REGISTRY.registerAndGet(new SimpleMaterialShaders(Files.SHADED_VERTEX, Files.DEFAULT_FRAGMENT));
public static final MaterialShaders CUTOUT = MaterialShaders.REGISTRY.registerAndGet(new SimpleMaterialShaders(Files.DEFAULT_VERTEX, Files.CUTOUT_FRAGMENT));
public static final MaterialShaders SHADED_CUTOUT = MaterialShaders.REGISTRY.registerAndGet(new SimpleMaterialShaders(Files.SHADED_VERTEX, Files.CUTOUT_FRAGMENT));
private StandardMaterialShaders() {
}
@ApiStatus.Internal
public static void init() {
}
public static final class Files {
public static final ResourceLocation DEFAULT_VERTEX = Names.DEFAULT.withSuffix(".vert");
public static final ResourceLocation SHADED_VERTEX = Names.SHADED.withSuffix(".vert");
public static final ResourceLocation DEFAULT_FRAGMENT = Names.DEFAULT.withSuffix(".frag");
public static final ResourceLocation CUTOUT_FRAGMENT = Names.CUTOUT.withSuffix(".frag");
}
public static final class Names {
public static final ResourceLocation DEFAULT = Flywheel.rl("material/default");
public static final ResourceLocation SHADED = Flywheel.rl("material/shaded");
public static final ResourceLocation CUTOUT = Flywheel.rl("material/cutout");
}
}