mirror of
https://github.com/Jozufozu/Flywheel.git
synced 2025-01-14 16:26:07 +01:00
Taking textures back out of context
- Contexts shouldn't be responsible for binding the api defined textures - Move light and overlay binding to TextureBinder - Clean up & standardize material/texture setup/teardown calls - Revert removal of MaterialRenderState texture stuff - Add SimpleContext + Builder - Rename Textures -> TextureSource - Add API methods to set basic float uniforms, more to come later - Throw errors if a context tries to set uniforms/textures with reserved names.
This commit is contained in:
parent
1028ca4633
commit
6dfd5ed563
16 changed files with 303 additions and 101 deletions
|
@ -10,7 +10,7 @@ public interface Context {
|
||||||
*
|
*
|
||||||
* @param material The material about to be rendered.
|
* @param material The material about to be rendered.
|
||||||
* @param shader The shader to prepare.
|
* @param shader The shader to prepare.
|
||||||
* @param textures Source of the textures to use.
|
* @param textureSource Source of the textures to use.
|
||||||
*/
|
*/
|
||||||
void prepare(Material material, Shader shader, Textures textures);
|
void prepare(Material material, Shader shader, TextureSource textureSource);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,20 @@
|
||||||
package com.jozufozu.flywheel.api.context;
|
package com.jozufozu.flywheel.api.context;
|
||||||
|
|
||||||
|
import org.joml.Matrix4fc;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.BackendImplemented;
|
import com.jozufozu.flywheel.api.BackendImplemented;
|
||||||
|
|
||||||
@BackendImplemented
|
@BackendImplemented
|
||||||
public interface Shader {
|
public interface Shader {
|
||||||
void setTexture(String glslName, Texture texture);
|
void setTexture(String glslName, Texture texture);
|
||||||
|
|
||||||
|
void setFloat(String glslName, float value);
|
||||||
|
|
||||||
|
void setVec2(String glslName, float x, float y);
|
||||||
|
|
||||||
|
void setVec3(String glslName, float x, float y, float z);
|
||||||
|
|
||||||
|
void setVec4(String glslName, float x, float y, float z, float w);
|
||||||
|
|
||||||
|
void setMat4(String glslName, Matrix4fc matrix);
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,7 +5,7 @@ import com.jozufozu.flywheel.api.BackendImplemented;
|
||||||
import net.minecraft.resources.ResourceLocation;
|
import net.minecraft.resources.ResourceLocation;
|
||||||
|
|
||||||
@BackendImplemented
|
@BackendImplemented
|
||||||
public interface Textures {
|
public interface TextureSource {
|
||||||
/**
|
/**
|
||||||
* Get a built-in texture by its resource location.
|
* Get a built-in texture by its resource location.
|
||||||
*
|
*
|
|
@ -5,6 +5,7 @@ import java.util.List;
|
||||||
import com.jozufozu.flywheel.backend.InternalVertex;
|
import com.jozufozu.flywheel.backend.InternalVertex;
|
||||||
import com.jozufozu.flywheel.backend.compile.core.CompilationHarness;
|
import com.jozufozu.flywheel.backend.compile.core.CompilationHarness;
|
||||||
import com.jozufozu.flywheel.backend.compile.core.Compile;
|
import com.jozufozu.flywheel.backend.compile.core.Compile;
|
||||||
|
import com.jozufozu.flywheel.backend.gl.shader.GlProgram;
|
||||||
import com.jozufozu.flywheel.backend.gl.shader.ShaderType;
|
import com.jozufozu.flywheel.backend.gl.shader.ShaderType;
|
||||||
import com.jozufozu.flywheel.backend.glsl.ShaderSources;
|
import com.jozufozu.flywheel.backend.glsl.ShaderSources;
|
||||||
import com.jozufozu.flywheel.backend.glsl.SourceComponent;
|
import com.jozufozu.flywheel.backend.glsl.SourceComponent;
|
||||||
|
@ -52,6 +53,14 @@ public class PipelineCompiler {
|
||||||
.then((key, program) -> {
|
.then((key, program) -> {
|
||||||
program.setUniformBlockBinding("_FlwFrameUniforms", 0);
|
program.setUniformBlockBinding("_FlwFrameUniforms", 0);
|
||||||
program.setUniformBlockBinding("_FlwFogUniforms", 1);
|
program.setUniformBlockBinding("_FlwFogUniforms", 1);
|
||||||
|
|
||||||
|
program.bind();
|
||||||
|
|
||||||
|
program.setSamplerBinding("_flw_diffuseTex", 0);
|
||||||
|
program.setSamplerBinding("_flw_overlayTex", 1);
|
||||||
|
program.setSamplerBinding("_flw_lightTex", 2);
|
||||||
|
|
||||||
|
GlProgram.unbind();
|
||||||
})
|
})
|
||||||
.harness(pipeline.compilerMarker(), sources);
|
.harness(pipeline.compilerMarker(), sources);
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,9 +8,13 @@ import com.jozufozu.flywheel.api.material.DepthTest;
|
||||||
import com.jozufozu.flywheel.api.material.Material;
|
import com.jozufozu.flywheel.api.material.Material;
|
||||||
import com.jozufozu.flywheel.api.material.Transparency;
|
import com.jozufozu.flywheel.api.material.Transparency;
|
||||||
import com.jozufozu.flywheel.api.material.WriteMask;
|
import com.jozufozu.flywheel.api.material.WriteMask;
|
||||||
|
import com.jozufozu.flywheel.backend.gl.GlTextureUnit;
|
||||||
import com.mojang.blaze3d.platform.GlStateManager;
|
import com.mojang.blaze3d.platform.GlStateManager;
|
||||||
import com.mojang.blaze3d.systems.RenderSystem;
|
import com.mojang.blaze3d.systems.RenderSystem;
|
||||||
|
|
||||||
|
import net.minecraft.client.Minecraft;
|
||||||
|
import net.minecraft.client.renderer.texture.AbstractTexture;
|
||||||
|
|
||||||
public final class MaterialRenderState {
|
public final class MaterialRenderState {
|
||||||
public static final Comparator<Material> COMPARATOR = Comparator.comparing(Material::texture)
|
public static final Comparator<Material> COMPARATOR = Comparator.comparing(Material::texture)
|
||||||
.thenComparing(Material::blur)
|
.thenComparing(Material::blur)
|
||||||
|
@ -25,6 +29,7 @@ public final class MaterialRenderState {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void setup(Material material) {
|
public static void setup(Material material) {
|
||||||
|
setupTexture(material);
|
||||||
setupBackfaceCulling(material.backfaceCulling());
|
setupBackfaceCulling(material.backfaceCulling());
|
||||||
setupPolygonOffset(material.polygonOffset());
|
setupPolygonOffset(material.polygonOffset());
|
||||||
setupDepthTest(material.depthTest());
|
setupDepthTest(material.depthTest());
|
||||||
|
@ -32,6 +37,17 @@ public final class MaterialRenderState {
|
||||||
setupWriteMask(material.writeMask());
|
setupWriteMask(material.writeMask());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void setupTexture(Material material) {
|
||||||
|
GlTextureUnit.T0.makeActive();
|
||||||
|
AbstractTexture texture = Minecraft.getInstance()
|
||||||
|
.getTextureManager()
|
||||||
|
.getTexture(material.texture());
|
||||||
|
texture.setFilter(material.blur(), material.mipmap());
|
||||||
|
var textureId = texture.getId();
|
||||||
|
RenderSystem.setShaderTexture(0, textureId);
|
||||||
|
RenderSystem.bindTexture(textureId);
|
||||||
|
}
|
||||||
|
|
||||||
private static void setupBackfaceCulling(boolean backfaceCulling) {
|
private static void setupBackfaceCulling(boolean backfaceCulling) {
|
||||||
if (backfaceCulling) {
|
if (backfaceCulling) {
|
||||||
RenderSystem.enableCull();
|
RenderSystem.enableCull();
|
||||||
|
@ -125,6 +141,7 @@ public final class MaterialRenderState {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void reset() {
|
public static void reset() {
|
||||||
|
resetTexture();
|
||||||
resetBackfaceCulling();
|
resetBackfaceCulling();
|
||||||
resetPolygonOffset();
|
resetPolygonOffset();
|
||||||
resetDepthTest();
|
resetDepthTest();
|
||||||
|
@ -132,6 +149,11 @@ public final class MaterialRenderState {
|
||||||
resetWriteMask();
|
resetWriteMask();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void resetTexture() {
|
||||||
|
GlTextureUnit.T0.makeActive();
|
||||||
|
RenderSystem.setShaderTexture(0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
private static void resetBackfaceCulling() {
|
private static void resetBackfaceCulling() {
|
||||||
RenderSystem.enableCull();
|
RenderSystem.enableCull();
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@ import com.jozufozu.flywheel.api.model.Model;
|
||||||
import com.jozufozu.flywheel.backend.compile.IndirectPrograms;
|
import com.jozufozu.flywheel.backend.compile.IndirectPrograms;
|
||||||
import com.jozufozu.flywheel.backend.engine.MaterialRenderState;
|
import com.jozufozu.flywheel.backend.engine.MaterialRenderState;
|
||||||
import com.jozufozu.flywheel.backend.engine.textures.TextureBinder;
|
import com.jozufozu.flywheel.backend.engine.textures.TextureBinder;
|
||||||
import com.jozufozu.flywheel.backend.engine.textures.TexturesImpl;
|
import com.jozufozu.flywheel.backend.engine.textures.TextureSourceImpl;
|
||||||
import com.jozufozu.flywheel.backend.engine.uniform.Uniforms;
|
import com.jozufozu.flywheel.backend.engine.uniform.Uniforms;
|
||||||
import com.jozufozu.flywheel.backend.gl.Driver;
|
import com.jozufozu.flywheel.backend.gl.Driver;
|
||||||
import com.jozufozu.flywheel.backend.gl.GlCompat;
|
import com.jozufozu.flywheel.backend.gl.GlCompat;
|
||||||
|
@ -185,7 +185,7 @@ public class IndirectCullingGroup<I extends Instance> {
|
||||||
hasNewDraws = true;
|
hasNewDraws = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void submit(RenderStage stage, TexturesImpl textures) {
|
public void submit(RenderStage stage, TextureSourceImpl textures) {
|
||||||
if (nothingToDo(stage)) {
|
if (nothingToDo(stage)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,8 @@ import com.jozufozu.flywheel.backend.engine.InstancerKey;
|
||||||
import com.jozufozu.flywheel.backend.engine.InstancerStorage;
|
import com.jozufozu.flywheel.backend.engine.InstancerStorage;
|
||||||
import com.jozufozu.flywheel.backend.engine.MaterialRenderState;
|
import com.jozufozu.flywheel.backend.engine.MaterialRenderState;
|
||||||
import com.jozufozu.flywheel.backend.engine.textures.TextureBinder;
|
import com.jozufozu.flywheel.backend.engine.textures.TextureBinder;
|
||||||
import com.jozufozu.flywheel.backend.engine.textures.TexturesImpl;
|
import com.jozufozu.flywheel.backend.engine.textures.TextureSourceImpl;
|
||||||
|
import com.jozufozu.flywheel.backend.gl.GlStateTracker;
|
||||||
import com.jozufozu.flywheel.backend.gl.buffer.GlBuffer;
|
import com.jozufozu.flywheel.backend.gl.buffer.GlBuffer;
|
||||||
import com.jozufozu.flywheel.backend.gl.buffer.GlBufferType;
|
import com.jozufozu.flywheel.backend.gl.buffer.GlBufferType;
|
||||||
import com.jozufozu.flywheel.lib.context.ContextShaders;
|
import com.jozufozu.flywheel.lib.context.ContextShaders;
|
||||||
|
@ -39,7 +40,7 @@ import net.minecraft.client.resources.model.ModelBakery;
|
||||||
public class IndirectDrawManager extends InstancerStorage<IndirectInstancer<?>> {
|
public class IndirectDrawManager extends InstancerStorage<IndirectInstancer<?>> {
|
||||||
private final IndirectPrograms programs;
|
private final IndirectPrograms programs;
|
||||||
private final StagingBuffer stagingBuffer;
|
private final StagingBuffer stagingBuffer;
|
||||||
private final TexturesImpl textures = new TexturesImpl();
|
private final TextureSourceImpl textures = new TextureSourceImpl();
|
||||||
private final Map<GroupKey<?>, IndirectCullingGroup<?>> cullingGroups = new HashMap<>();
|
private final Map<GroupKey<?>, IndirectCullingGroup<?>> cullingGroups = new HashMap<>();
|
||||||
private final GlBuffer crumblingDrawBuffer = new GlBuffer();
|
private final GlBuffer crumblingDrawBuffer = new GlBuffer();
|
||||||
|
|
||||||
|
@ -71,9 +72,14 @@ public class IndirectDrawManager extends InstancerStorage<IndirectInstancer<?>>
|
||||||
}
|
}
|
||||||
|
|
||||||
public void renderStage(RenderStage stage) {
|
public void renderStage(RenderStage stage) {
|
||||||
|
TextureBinder.bindLightAndOverlay();
|
||||||
|
|
||||||
for (var group : cullingGroups.values()) {
|
for (var group : cullingGroups.values()) {
|
||||||
group.submit(stage, textures);
|
group.submit(stage, textures);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MaterialRenderState.reset();
|
||||||
|
TextureBinder.resetLightAndOverlay();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -117,50 +123,56 @@ public class IndirectDrawManager extends InstancerStorage<IndirectInstancer<?>>
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var crumblingMaterial = SimpleMaterial.builder();
|
try (var state = GlStateTracker.getRestoreState()) {
|
||||||
|
TextureBinder.bindLightAndOverlay();
|
||||||
|
|
||||||
// Scratch memory for writing draw commands.
|
var crumblingMaterial = SimpleMaterial.builder();
|
||||||
var block = MemoryBlock.malloc(IndirectBuffers.DRAW_COMMAND_STRIDE);
|
|
||||||
|
|
||||||
GlBufferType.DRAW_INDIRECT_BUFFER.bind(crumblingDrawBuffer.handle());
|
// Scratch memory for writing draw commands.
|
||||||
glBindBufferRange(GL_SHADER_STORAGE_BUFFER, IndirectBuffers.DRAW_INDEX, crumblingDrawBuffer.handle(), 0, IndirectBuffers.DRAW_COMMAND_STRIDE);
|
var block = MemoryBlock.malloc(IndirectBuffers.DRAW_COMMAND_STRIDE);
|
||||||
|
|
||||||
for (var instanceTypeEntry : byType.entrySet()) {
|
GlBufferType.DRAW_INDIRECT_BUFFER.bind(crumblingDrawBuffer.handle());
|
||||||
var byProgress = instanceTypeEntry.getValue();
|
glBindBufferRange(GL_SHADER_STORAGE_BUFFER, IndirectBuffers.DRAW_INDEX, crumblingDrawBuffer.handle(), 0, IndirectBuffers.DRAW_COMMAND_STRIDE);
|
||||||
|
|
||||||
// Set up the crumbling program buffers. Nothing changes here between draws.
|
for (var instanceTypeEntry : byType.entrySet()) {
|
||||||
var program = cullingGroups.get(instanceTypeEntry.getKey())
|
var byProgress = instanceTypeEntry.getValue();
|
||||||
.bindWithContextShader(ContextShaders.CRUMBLING);
|
|
||||||
|
|
||||||
for (var progressEntry : byProgress.int2ObjectEntrySet()) {
|
// Set up the crumbling program buffers. Nothing changes here between draws.
|
||||||
for (var instanceHandlePair : progressEntry.getValue()) {
|
var program = cullingGroups.get(instanceTypeEntry.getKey())
|
||||||
IndirectInstancer<?> instancer = instanceHandlePair.first();
|
.bindWithContextShader(ContextShaders.CRUMBLING);
|
||||||
int instanceIndex = instanceHandlePair.second().index;
|
|
||||||
|
|
||||||
for (IndirectDraw draw : instancer.draws()) {
|
for (var progressEntry : byProgress.int2ObjectEntrySet()) {
|
||||||
|
for (var instanceHandlePair : progressEntry.getValue()) {
|
||||||
|
IndirectInstancer<?> instancer = instanceHandlePair.first();
|
||||||
|
int instanceIndex = instanceHandlePair.second().index;
|
||||||
|
|
||||||
// Transform the material to be suited for crumbling.
|
for (IndirectDraw draw : instancer.draws()) {
|
||||||
CommonCrumbling.applyCrumblingProperties(crumblingMaterial, draw.material());
|
|
||||||
Contexts.CRUMBLING.get(progressEntry.getIntKey())
|
|
||||||
.prepare(crumblingMaterial, program, textures);
|
|
||||||
|
|
||||||
// Set up gl state for the draw.
|
// Transform the material to be suited for crumbling.
|
||||||
MaterialRenderState.setup(crumblingMaterial);
|
CommonCrumbling.applyCrumblingProperties(crumblingMaterial, draw.material());
|
||||||
|
var context = Contexts.CRUMBLING.get(progressEntry.getIntKey());
|
||||||
|
|
||||||
// Upload the draw command.
|
context.prepare(crumblingMaterial, program, textures);
|
||||||
draw.writeWithOverrides(block.ptr(), instanceIndex, crumblingMaterial);
|
MaterialRenderState.setup(crumblingMaterial);
|
||||||
crumblingDrawBuffer.upload(block);
|
|
||||||
|
|
||||||
// Submit! Everything is already bound by here.
|
// Upload the draw command.
|
||||||
glDrawElementsIndirect(GL_TRIANGLES, GL_UNSIGNED_INT, 0);
|
draw.writeWithOverrides(block.ptr(), instanceIndex, crumblingMaterial);
|
||||||
|
crumblingDrawBuffer.upload(block);
|
||||||
|
|
||||||
TextureBinder.resetTextureBindings();
|
// Submit! Everything is already bound by here.
|
||||||
|
glDrawElementsIndirect(GL_TRIANGLES, GL_UNSIGNED_INT, 0);
|
||||||
|
|
||||||
|
TextureBinder.resetTextureBindings();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
block.free();
|
MaterialRenderState.reset();
|
||||||
|
TextureBinder.resetLightAndOverlay();
|
||||||
|
|
||||||
|
block.free();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Map<GroupKey<?>, Int2ObjectMap<List<Pair<IndirectInstancer<?>, InstanceHandleImpl>>>> doCrumblingSort(List<Engine.CrumblingBlock> crumblingBlocks) {
|
private static Map<GroupKey<?>, Int2ObjectMap<List<Pair<IndirectInstancer<?>, InstanceHandleImpl>>>> doCrumblingSort(List<Engine.CrumblingBlock> crumblingBlocks) {
|
||||||
|
|
|
@ -10,7 +10,6 @@ import com.jozufozu.flywheel.backend.compile.IndirectPrograms;
|
||||||
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;
|
||||||
import com.jozufozu.flywheel.backend.engine.InstancerStorage;
|
import com.jozufozu.flywheel.backend.engine.InstancerStorage;
|
||||||
import com.jozufozu.flywheel.backend.engine.MaterialRenderState;
|
|
||||||
import com.jozufozu.flywheel.backend.engine.uniform.Uniforms;
|
import com.jozufozu.flywheel.backend.engine.uniform.Uniforms;
|
||||||
import com.jozufozu.flywheel.backend.gl.GlStateTracker;
|
import com.jozufozu.flywheel.backend.gl.GlStateTracker;
|
||||||
import com.jozufozu.flywheel.lib.task.Flag;
|
import com.jozufozu.flywheel.lib.task.Flag;
|
||||||
|
@ -55,8 +54,6 @@ public class IndirectEngine extends AbstractEngine {
|
||||||
|
|
||||||
try (var restoreState = GlStateTracker.getRestoreState()) {
|
try (var restoreState = GlStateTracker.getRestoreState()) {
|
||||||
drawManager.renderStage(stage);
|
drawManager.renderStage(stage);
|
||||||
|
|
||||||
MaterialRenderState.reset();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -64,11 +61,7 @@ public class IndirectEngine extends AbstractEngine {
|
||||||
public void renderCrumbling(TaskExecutor executor, RenderContext context, List<CrumblingBlock> crumblingBlocks) {
|
public void renderCrumbling(TaskExecutor executor, RenderContext context, List<CrumblingBlock> crumblingBlocks) {
|
||||||
executor.syncUntil(flushFlag::isRaised);
|
executor.syncUntil(flushFlag::isRaised);
|
||||||
|
|
||||||
try (var restoreState = GlStateTracker.getRestoreState()) {
|
drawManager.renderCrumbling(crumblingBlocks);
|
||||||
drawManager.renderCrumbling(crumblingBlocks);
|
|
||||||
|
|
||||||
MaterialRenderState.reset();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -6,7 +6,7 @@ import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.backend.Engine;
|
import com.jozufozu.flywheel.api.backend.Engine;
|
||||||
import com.jozufozu.flywheel.api.context.Textures;
|
import com.jozufozu.flywheel.api.context.TextureSource;
|
||||||
import com.jozufozu.flywheel.api.instance.Instance;
|
import com.jozufozu.flywheel.api.instance.Instance;
|
||||||
import com.jozufozu.flywheel.backend.compile.InstancingPrograms;
|
import com.jozufozu.flywheel.backend.compile.InstancingPrograms;
|
||||||
import com.jozufozu.flywheel.backend.engine.CommonCrumbling;
|
import com.jozufozu.flywheel.backend.engine.CommonCrumbling;
|
||||||
|
@ -24,7 +24,7 @@ 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 {
|
||||||
public static void render(List<Engine.CrumblingBlock> crumblingBlocks, InstancingPrograms programs, Textures textures) {
|
public static void render(List<Engine.CrumblingBlock> crumblingBlocks, InstancingPrograms programs, TextureSource textureSource) {
|
||||||
// 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);
|
||||||
|
|
||||||
|
@ -35,6 +35,8 @@ public class InstancedCrumbling {
|
||||||
var crumblingMaterial = SimpleMaterial.builder();
|
var crumblingMaterial = SimpleMaterial.builder();
|
||||||
|
|
||||||
try (var state = GlStateTracker.getRestoreState()) {
|
try (var state = GlStateTracker.getRestoreState()) {
|
||||||
|
TextureBinder.bindLightAndOverlay();
|
||||||
|
|
||||||
for (var shaderStateEntry : byShaderState.entrySet()) {
|
for (var shaderStateEntry : byShaderState.entrySet()) {
|
||||||
var byProgress = shaderStateEntry.getValue();
|
var byProgress = shaderStateEntry.getValue();
|
||||||
|
|
||||||
|
@ -61,8 +63,8 @@ public class InstancedCrumbling {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
Contexts.CRUMBLING.get(progressEntry.getIntKey())
|
var context = Contexts.CRUMBLING.get(progressEntry.getIntKey());
|
||||||
.prepare(crumblingMaterial, program, textures);
|
context.prepare(crumblingMaterial, program, textureSource);
|
||||||
|
|
||||||
drawCalls.forEach(Runnable::run);
|
drawCalls.forEach(Runnable::run);
|
||||||
|
|
||||||
|
@ -71,6 +73,7 @@ public class InstancedCrumbling {
|
||||||
}
|
}
|
||||||
|
|
||||||
MaterialRenderState.reset();
|
MaterialRenderState.reset();
|
||||||
|
TextureBinder.resetLightAndOverlay();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -17,7 +17,7 @@ import com.jozufozu.flywheel.backend.engine.InstancerStorage;
|
||||||
import com.jozufozu.flywheel.backend.engine.MaterialEncoder;
|
import com.jozufozu.flywheel.backend.engine.MaterialEncoder;
|
||||||
import com.jozufozu.flywheel.backend.engine.MaterialRenderState;
|
import com.jozufozu.flywheel.backend.engine.MaterialRenderState;
|
||||||
import com.jozufozu.flywheel.backend.engine.textures.TextureBinder;
|
import com.jozufozu.flywheel.backend.engine.textures.TextureBinder;
|
||||||
import com.jozufozu.flywheel.backend.engine.textures.TexturesImpl;
|
import com.jozufozu.flywheel.backend.engine.textures.TextureSourceImpl;
|
||||||
import com.jozufozu.flywheel.backend.engine.uniform.Uniforms;
|
import com.jozufozu.flywheel.backend.engine.uniform.Uniforms;
|
||||||
import com.jozufozu.flywheel.backend.gl.GlStateTracker;
|
import com.jozufozu.flywheel.backend.gl.GlStateTracker;
|
||||||
import com.jozufozu.flywheel.backend.gl.shader.GlProgram;
|
import com.jozufozu.flywheel.backend.gl.shader.GlProgram;
|
||||||
|
@ -27,7 +27,7 @@ import com.jozufozu.flywheel.lib.task.SyncedPlan;
|
||||||
|
|
||||||
public class InstancingEngine extends AbstractEngine {
|
public class InstancingEngine extends AbstractEngine {
|
||||||
private final InstancingPrograms programs;
|
private final InstancingPrograms programs;
|
||||||
private final TexturesImpl textures = new TexturesImpl();
|
private final TextureSourceImpl textures = new TextureSourceImpl();
|
||||||
private final InstancedDrawManager drawManager = new InstancedDrawManager();
|
private final InstancedDrawManager drawManager = new InstancedDrawManager();
|
||||||
private final Flag flushFlag = new NamedFlag("flushed");
|
private final Flag flushFlag = new NamedFlag("flushed");
|
||||||
|
|
||||||
|
@ -88,6 +88,8 @@ public class InstancingEngine extends AbstractEngine {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void render(InstancedDrawManager.DrawSet drawSet) {
|
private void render(InstancedDrawManager.DrawSet drawSet) {
|
||||||
|
TextureBinder.bindLightAndOverlay();
|
||||||
|
|
||||||
for (var entry : drawSet) {
|
for (var entry : drawSet) {
|
||||||
var shader = entry.getKey();
|
var shader = entry.getKey();
|
||||||
var drawCalls = entry.getValue();
|
var drawCalls = entry.getValue();
|
||||||
|
@ -98,16 +100,17 @@ public class InstancingEngine extends AbstractEngine {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
var program = programs.get(shader.instanceType(), shader.context()
|
var context = shader.context();
|
||||||
.contextShader());
|
var material = shader.material();
|
||||||
|
|
||||||
|
var program = programs.get(shader.instanceType(), context.contextShader());
|
||||||
program.bind();
|
program.bind();
|
||||||
|
|
||||||
Uniforms.bindForDraw();
|
Uniforms.bindForDraw();
|
||||||
uploadMaterialUniform(program, shader.material());
|
uploadMaterialUniform(program, material);
|
||||||
|
|
||||||
shader.context()
|
context.prepare(material, program, textures);
|
||||||
.prepare(shader.material(), program, textures);
|
MaterialRenderState.setup(material);
|
||||||
MaterialRenderState.setup(shader.material());
|
|
||||||
|
|
||||||
for (var drawCall : drawCalls) {
|
for (var drawCall : drawCalls) {
|
||||||
drawCall.render();
|
drawCall.render();
|
||||||
|
@ -116,6 +119,7 @@ public class InstancingEngine extends AbstractEngine {
|
||||||
}
|
}
|
||||||
|
|
||||||
MaterialRenderState.reset();
|
MaterialRenderState.reset();
|
||||||
|
TextureBinder.resetLightAndOverlay();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void uploadMaterialUniform(GlProgram program, Material material) {
|
public static void uploadMaterialUniform(GlProgram program, Material material) {
|
||||||
|
|
|
@ -2,16 +2,23 @@ package com.jozufozu.flywheel.backend.engine.textures;
|
||||||
|
|
||||||
import static org.lwjgl.opengl.GL13.GL_TEXTURE0;
|
import static org.lwjgl.opengl.GL13.GL_TEXTURE0;
|
||||||
|
|
||||||
|
import com.jozufozu.flywheel.backend.gl.GlTextureUnit;
|
||||||
import com.mojang.blaze3d.systems.RenderSystem;
|
import com.mojang.blaze3d.systems.RenderSystem;
|
||||||
|
|
||||||
import it.unimi.dsi.fastutil.ints.Int2IntArrayMap;
|
import it.unimi.dsi.fastutil.ints.Int2IntArrayMap;
|
||||||
import it.unimi.dsi.fastutil.ints.Int2IntMap;
|
import it.unimi.dsi.fastutil.ints.Int2IntMap;
|
||||||
|
import net.minecraft.client.Minecraft;
|
||||||
|
|
||||||
public class TextureBinder {
|
public class TextureBinder {
|
||||||
// TODO: some kind of cache eviction when the program changes
|
// TODO: some kind of cache eviction when the program changes
|
||||||
// so we don't always reset and bind the light and overlay textures?
|
// so we don't always reset and bind the light and overlay textures?
|
||||||
private static final Int2IntMap texturesToSamplerUnits = new Int2IntArrayMap();
|
private static final Int2IntMap texturesToSamplerUnits = new Int2IntArrayMap();
|
||||||
private static int nextSamplerUnit = 0;
|
// 0 is reserved for diffuse
|
||||||
|
// 1 is overlay
|
||||||
|
// 2 is light
|
||||||
|
// 3..n are for whatever else the context needs
|
||||||
|
private static final int baseSamplerUnit = 3;
|
||||||
|
private static int nextSamplerUnit = baseSamplerUnit;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Binds the given texture to the next available texture unit, returning the unit it was bound to.
|
* Binds the given texture to the next available texture unit, returning the unit it was bound to.
|
||||||
|
@ -29,7 +36,7 @@ public class TextureBinder {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void resetTextureBindings() {
|
public static void resetTextureBindings() {
|
||||||
nextSamplerUnit = 0;
|
nextSamplerUnit = baseSamplerUnit;
|
||||||
|
|
||||||
for (Int2IntMap.Entry entry : texturesToSamplerUnits.int2IntEntrySet()) {
|
for (Int2IntMap.Entry entry : texturesToSamplerUnits.int2IntEntrySet()) {
|
||||||
RenderSystem.activeTexture(GL_TEXTURE0 + entry.getIntValue());
|
RenderSystem.activeTexture(GL_TEXTURE0 + entry.getIntValue());
|
||||||
|
@ -38,4 +45,26 @@ public class TextureBinder {
|
||||||
|
|
||||||
texturesToSamplerUnits.clear();
|
texturesToSamplerUnits.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static void bindLightAndOverlay() {
|
||||||
|
var gameRenderer = Minecraft.getInstance().gameRenderer;
|
||||||
|
gameRenderer.overlayTexture()
|
||||||
|
.setupOverlayColor();
|
||||||
|
gameRenderer.lightTexture()
|
||||||
|
.turnOnLightLayer();
|
||||||
|
|
||||||
|
GlTextureUnit.T1.makeActive();
|
||||||
|
RenderSystem.bindTexture(RenderSystem.getShaderTexture(1));
|
||||||
|
GlTextureUnit.T2.makeActive();
|
||||||
|
RenderSystem.bindTexture(RenderSystem.getShaderTexture(2));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void resetLightAndOverlay() {
|
||||||
|
var gameRenderer = Minecraft.getInstance().gameRenderer;
|
||||||
|
|
||||||
|
gameRenderer.overlayTexture()
|
||||||
|
.teardownOverlayColor();
|
||||||
|
gameRenderer.lightTexture()
|
||||||
|
.turnOffLightLayer();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@ import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.context.Texture;
|
import com.jozufozu.flywheel.api.context.Texture;
|
||||||
import com.jozufozu.flywheel.api.context.Textures;
|
import com.jozufozu.flywheel.api.context.TextureSource;
|
||||||
import com.jozufozu.flywheel.backend.mixin.LightTextureAccessor;
|
import com.jozufozu.flywheel.backend.mixin.LightTextureAccessor;
|
||||||
import com.jozufozu.flywheel.backend.mixin.OverlayTextureAccessor;
|
import com.jozufozu.flywheel.backend.mixin.OverlayTextureAccessor;
|
||||||
|
|
||||||
|
@ -12,13 +12,13 @@ import net.minecraft.client.Minecraft;
|
||||||
import net.minecraft.client.renderer.texture.AbstractTexture;
|
import net.minecraft.client.renderer.texture.AbstractTexture;
|
||||||
import net.minecraft.resources.ResourceLocation;
|
import net.minecraft.resources.ResourceLocation;
|
||||||
|
|
||||||
public class TexturesImpl implements Textures {
|
public class TextureSourceImpl implements TextureSource {
|
||||||
|
|
||||||
private final DirectTexture lightTexture;
|
private final DirectTexture lightTexture;
|
||||||
private final DirectTexture overlayTexture;
|
private final DirectTexture overlayTexture;
|
||||||
private final Map<ResourceLocation, WrappedTexture> wrappers = new HashMap<>();
|
private final Map<ResourceLocation, WrappedTexture> wrappers = new HashMap<>();
|
||||||
|
|
||||||
public TexturesImpl() {
|
public TextureSourceImpl() {
|
||||||
var gameRenderer = Minecraft.getInstance().gameRenderer;
|
var gameRenderer = Minecraft.getInstance().gameRenderer;
|
||||||
|
|
||||||
this.lightTexture = new DirectTexture(((LightTextureAccessor) gameRenderer.lightTexture()).flywheel$texture()
|
this.lightTexture = new DirectTexture(((LightTextureAccessor) gameRenderer.lightTexture()).flywheel$texture()
|
|
@ -2,11 +2,17 @@ package com.jozufozu.flywheel.backend.gl.shader;
|
||||||
|
|
||||||
import static org.lwjgl.opengl.GL20.glDeleteProgram;
|
import static org.lwjgl.opengl.GL20.glDeleteProgram;
|
||||||
import static org.lwjgl.opengl.GL20.glGetUniformLocation;
|
import static org.lwjgl.opengl.GL20.glGetUniformLocation;
|
||||||
|
import static org.lwjgl.opengl.GL20.glUniform1f;
|
||||||
import static org.lwjgl.opengl.GL20.glUniform1i;
|
import static org.lwjgl.opengl.GL20.glUniform1i;
|
||||||
|
import static org.lwjgl.opengl.GL20.glUniform2f;
|
||||||
|
import static org.lwjgl.opengl.GL20.glUniform3f;
|
||||||
|
import static org.lwjgl.opengl.GL20.glUniform4f;
|
||||||
|
import static org.lwjgl.opengl.GL20.glUniformMatrix4fv;
|
||||||
import static org.lwjgl.opengl.GL31.GL_INVALID_INDEX;
|
import static org.lwjgl.opengl.GL31.GL_INVALID_INDEX;
|
||||||
import static org.lwjgl.opengl.GL31.glGetUniformBlockIndex;
|
import static org.lwjgl.opengl.GL31.glGetUniformBlockIndex;
|
||||||
import static org.lwjgl.opengl.GL31.glUniformBlockBinding;
|
import static org.lwjgl.opengl.GL31.glUniformBlockBinding;
|
||||||
|
|
||||||
|
import org.joml.Matrix4fc;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.context.Shader;
|
import com.jozufozu.flywheel.api.context.Shader;
|
||||||
|
@ -39,6 +45,8 @@ public class GlProgram extends GlObject implements Shader {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void setTexture(String glslName, Texture texture) {
|
public void setTexture(String glslName, Texture texture) {
|
||||||
|
throwIfReserved(glslName);
|
||||||
|
|
||||||
if (!(texture instanceof IdentifiedTexture identified)) {
|
if (!(texture instanceof IdentifiedTexture identified)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -55,6 +63,71 @@ public class GlProgram extends GlObject implements Shader {
|
||||||
glUniform1i(uniform, binding);
|
glUniform1i(uniform, binding);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setFloat(String glslName, float value) {
|
||||||
|
throwIfReserved(glslName);
|
||||||
|
|
||||||
|
int uniform = getUniformLocation(glslName);
|
||||||
|
|
||||||
|
if (uniform < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
glUniform1f(uniform, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setVec2(String glslName, float x, float y) {
|
||||||
|
throwIfReserved(glslName);
|
||||||
|
|
||||||
|
int uniform = getUniformLocation(glslName);
|
||||||
|
|
||||||
|
if (uniform < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
glUniform2f(uniform, x, y);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setVec3(String glslName, float x, float y, float z) {
|
||||||
|
throwIfReserved(glslName);
|
||||||
|
|
||||||
|
int uniform = getUniformLocation(glslName);
|
||||||
|
|
||||||
|
if (uniform < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
glUniform3f(uniform, x, y, z);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setVec4(String glslName, float x, float y, float z, float w) {
|
||||||
|
throwIfReserved(glslName);
|
||||||
|
|
||||||
|
int uniform = getUniformLocation(glslName);
|
||||||
|
|
||||||
|
if (uniform < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
glUniform4f(uniform, x, y, z, w);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setMat4(String glslName, Matrix4fc matrix) {
|
||||||
|
throwIfReserved(glslName);
|
||||||
|
|
||||||
|
int uniform = getUniformLocation(glslName);
|
||||||
|
|
||||||
|
if (uniform < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
glUniformMatrix4fv(uniform, false, matrix.get(new float[16]));
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieves the index of the uniform with the given name.
|
* Retrieves the index of the uniform with the given name.
|
||||||
*
|
*
|
||||||
|
@ -101,4 +174,18 @@ public class GlProgram extends GlObject implements Shader {
|
||||||
protected void deleteInternal(int handle) {
|
protected void deleteInternal(int handle) {
|
||||||
glDeleteProgram(handle);
|
glDeleteProgram(handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static void throwIfReserved(String glslName) {
|
||||||
|
if (glslName.startsWith("flw_")) {
|
||||||
|
throw new IllegalArgumentException("Uniform names starting with flw_are reserved");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (glslName.startsWith("_flw_")) {
|
||||||
|
throw new IllegalArgumentException("Uniform names starting with _flw_ are reserved for internal use");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (glslName.startsWith("gl_")) {
|
||||||
|
throw new IllegalArgumentException("Uniform names cannot start with gl_");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,6 @@
|
||||||
package com.jozufozu.flywheel.lib.context;
|
package com.jozufozu.flywheel.lib.context;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.context.Context;
|
import com.jozufozu.flywheel.api.context.Context;
|
||||||
import com.jozufozu.flywheel.api.context.ContextShader;
|
|
||||||
import com.jozufozu.flywheel.api.context.Shader;
|
|
||||||
import com.jozufozu.flywheel.api.context.Textures;
|
|
||||||
import com.jozufozu.flywheel.api.material.Material;
|
|
||||||
|
|
||||||
import it.unimi.dsi.fastutil.ints.Int2ObjectArrayMap;
|
import it.unimi.dsi.fastutil.ints.Int2ObjectArrayMap;
|
||||||
import it.unimi.dsi.fastutil.ints.Int2ObjectMap;
|
import it.unimi.dsi.fastutil.ints.Int2ObjectMap;
|
||||||
|
@ -12,22 +8,8 @@ import it.unimi.dsi.fastutil.ints.Int2ObjectMaps;
|
||||||
import net.minecraft.client.resources.model.ModelBakery;
|
import net.minecraft.client.resources.model.ModelBakery;
|
||||||
|
|
||||||
public final class Contexts {
|
public final class Contexts {
|
||||||
public static final Context DEFAULT = new Context() {
|
public static final Context DEFAULT = SimpleContext.builder(ContextShaders.DEFAULT)
|
||||||
@Override
|
.build();
|
||||||
public ContextShader contextShader() {
|
|
||||||
return ContextShaders.DEFAULT;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void prepare(Material material, Shader shader, Textures textures) {
|
|
||||||
var texture = textures.byName(material.texture());
|
|
||||||
texture.filter(material.blur(), material.mipmap());
|
|
||||||
shader.setTexture("_flw_diffuseTex", texture);
|
|
||||||
|
|
||||||
shader.setTexture("_flw_overlayTex", textures.overlay());
|
|
||||||
shader.setTexture("_flw_lightTex", textures.light());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
public static final Int2ObjectMap<Context> CRUMBLING;
|
public static final Int2ObjectMap<Context> CRUMBLING;
|
||||||
|
|
||||||
|
@ -35,7 +17,12 @@ public final class Contexts {
|
||||||
var map = new Int2ObjectArrayMap<Context>();
|
var map = new Int2ObjectArrayMap<Context>();
|
||||||
|
|
||||||
for (int i = 0; i < ModelBakery.BREAKING_LOCATIONS.size(); i++) {
|
for (int i = 0; i < ModelBakery.BREAKING_LOCATIONS.size(); i++) {
|
||||||
map.put(i, new Crumbling(i));
|
var crumblingLocation = ModelBakery.BREAKING_LOCATIONS.get(i);
|
||||||
|
map.put(i, SimpleContext.builder(ContextShaders.CRUMBLING)
|
||||||
|
.preparation((material, shader, textureSource) -> {
|
||||||
|
shader.setTexture("crumblingTex", textureSource.byName(crumblingLocation));
|
||||||
|
})
|
||||||
|
.build());
|
||||||
}
|
}
|
||||||
|
|
||||||
CRUMBLING = Int2ObjectMaps.unmodifiable(map);
|
CRUMBLING = Int2ObjectMaps.unmodifiable(map);
|
||||||
|
@ -43,21 +30,4 @@ public final class Contexts {
|
||||||
|
|
||||||
private Contexts() {
|
private Contexts() {
|
||||||
}
|
}
|
||||||
|
|
||||||
private record Crumbling(int stage) implements Context {
|
|
||||||
@Override
|
|
||||||
public ContextShader contextShader() {
|
|
||||||
return ContextShaders.CRUMBLING;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void prepare(Material material, Shader shader, Textures textures) {
|
|
||||||
var texture = textures.byName(material.texture());
|
|
||||||
texture.filter(material.blur(), material.mipmap());
|
|
||||||
shader.setTexture("_flw_diffuseTex", texture);
|
|
||||||
|
|
||||||
var crumblingTexture = textures.byName(ModelBakery.BREAKING_LOCATIONS.get(stage));
|
|
||||||
shader.setTexture("_flw_crumblingTex", crumblingTexture);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,61 @@
|
||||||
|
package com.jozufozu.flywheel.lib.context;
|
||||||
|
|
||||||
|
import org.jetbrains.annotations.Nullable;
|
||||||
|
|
||||||
|
import com.jozufozu.flywheel.api.context.Context;
|
||||||
|
import com.jozufozu.flywheel.api.context.ContextShader;
|
||||||
|
import com.jozufozu.flywheel.api.context.Shader;
|
||||||
|
import com.jozufozu.flywheel.api.context.TextureSource;
|
||||||
|
import com.jozufozu.flywheel.api.material.Material;
|
||||||
|
|
||||||
|
public class SimpleContext implements Context {
|
||||||
|
private final ContextShader contextShader;
|
||||||
|
private final Preparation preparation;
|
||||||
|
|
||||||
|
public SimpleContext(ContextShader contextShader, Preparation preparation) {
|
||||||
|
this.contextShader = contextShader;
|
||||||
|
this.preparation = preparation;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Builder builder(ContextShader contextShader) {
|
||||||
|
return new Builder(contextShader);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ContextShader contextShader() {
|
||||||
|
return contextShader;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void prepare(Material material, Shader shader, TextureSource textureSource) {
|
||||||
|
preparation.prepare(material, shader, textureSource);
|
||||||
|
}
|
||||||
|
|
||||||
|
@FunctionalInterface
|
||||||
|
public interface Preparation {
|
||||||
|
void prepare(Material material, Shader shader, TextureSource textureSource);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class Builder {
|
||||||
|
private final ContextShader contextShader;
|
||||||
|
@Nullable
|
||||||
|
private Preparation preparation;
|
||||||
|
|
||||||
|
public Builder(ContextShader contextShader) {
|
||||||
|
this.contextShader = contextShader;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Builder preparation(Preparation preparation) {
|
||||||
|
this.preparation = preparation;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public SimpleContext build() {
|
||||||
|
if (preparation == null) {
|
||||||
|
preparation = (material, shader, textureSource) -> {
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return new SimpleContext(contextShader, preparation);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,4 @@
|
||||||
uniform sampler2D _flw_crumblingTex;
|
uniform sampler2D crumblingTex;
|
||||||
|
|
||||||
in vec2 crumblingTexCoord;
|
in vec2 crumblingTexCoord;
|
||||||
|
|
||||||
|
@ -8,7 +8,7 @@ void flw_beginFragment() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void flw_endFragment() {
|
void flw_endFragment() {
|
||||||
crumblingSampleColor = texture(_flw_crumblingTex, crumblingTexCoord);
|
crumblingSampleColor = texture(crumblingTex, crumblingTexCoord);
|
||||||
|
|
||||||
// Make the crumbling overlay transparent when the fragment color after the material shader is transparent.
|
// Make the crumbling overlay transparent when the fragment color after the material shader is transparent.
|
||||||
flw_fragColor.rgb = crumblingSampleColor.rgb;
|
flw_fragColor.rgb = crumblingSampleColor.rgb;
|
||||||
|
|
Loading…
Reference in a new issue