More versatile ui/settings

- The channel mask still needs work
 - Add contrast filter
This commit is contained in:
JozsefA 2021-04-28 00:01:26 -07:00
parent 70188889be
commit 54f5783489
17 changed files with 336 additions and 144 deletions

View file

@ -1154,8 +1154,17 @@
"create.gui.chromatic_projector.filter.saturate": "Saturate", "create.gui.chromatic_projector.filter.saturate": "Saturate",
"create.gui.chromatic_projector.filter.hue_shift": "Hue shift", "create.gui.chromatic_projector.filter.hue_shift": "Hue shift",
"create.gui.chromatic_projector.filter.darken": "Darken", "create.gui.chromatic_projector.filter.darken": "Darken",
"create.gui.chromatic_projector.filter.contrast": "Contrast",
"create.gui.chromatic_projector.filter.end": "End", "create.gui.chromatic_projector.filter.end": "End",
"create.gui.chromatic_projector.filter": "Filter", "create.gui.chromatic_projector.filter": "Filter",
"create.gui.chromatic_projector.surface": "Surface",
"create.gui.chromatic_projector.field": "Field",
"create.gui.chromatic_projector.strength": "Strength",
"create.gui.chromatic_projector.radius": "Radius",
"create.gui.chromatic_projector.feather": "Feather",
"create.gui.chromatic_projector.density": "Density",
"create.gui.chromatic_projector.fade": "Fade",
"create.gui.chromatic_projector.blend": "Blend",
"_": "->------------------------] Subtitles [------------------------<-", "_": "->------------------------] Subtitles [------------------------<-",
"create.subtitle.cogs": "Cogwheels rumble", "create.subtitle.cogs": "Cogwheels rumble",
"create.subtitle.slime_added": "Slime squishes", "create.subtitle.slime_added": "Slime squishes",

View file

@ -13,7 +13,7 @@ public class ChromaticProjectorInstance extends TileEntityInstance<ChromaticProj
@Override @Override
public void beginFrame() { public void beginFrame() {
Backend.effects.addSphere(tile.makeFilter()); Backend.effects.addSphere(tile.getFilter());
} }
@Override @Override

View file

@ -25,10 +25,12 @@ public class ChromaticProjectorScreen extends AbstractSimiScreen {
private final ItemStack renderedItem = AllBlocks.CHROMATIC_PROJECTOR.asStack(); private final ItemStack renderedItem = AllBlocks.CHROMATIC_PROJECTOR.asStack();
private final AllGuiTextures background = AllGuiTextures.PROJECTOR; private final AllGuiTextures background = AllGuiTextures.PROJECTOR;
private final int guiBottom = guiTop + background.height;
private final int guiRight = guiLeft + background.width;
private IconButton confirmButton; private IconButton confirmButton;
private final ITextComponent title = Lang.translate("gui.chromatic_projector.title"); private final ITextComponent title = Lang.translate("gui.chromatic_projector.title");
private Vector<FilterStep> stages; private final Vector<FilterStep> stages;
private Vector<Vector<ScrollInput>> inputs; private Vector<Vector<ScrollInput>> inputs;
@ -39,6 +41,15 @@ public class ChromaticProjectorScreen extends AbstractSimiScreen {
private ScrollInput feather; private ScrollInput feather;
private ScrollInput fade; private ScrollInput fade;
private IconButton blend;
private ScrollInput strength;
private IconButton fieldEffect;
private IconButton rChannel;
private IconButton gChannel;
private IconButton bChannel;
public ChromaticProjectorScreen(ChromaticProjectorTileEntity te) { public ChromaticProjectorScreen(ChromaticProjectorTileEntity te) {
this.tile = te; this.tile = te;
this.stages = te.stages; this.stages = te.stages;
@ -62,35 +73,73 @@ public class ChromaticProjectorScreen extends AbstractSimiScreen {
for (int row = 0; row < stages.size(); row++) for (int row = 0; row < stages.size(); row++)
initInputsOfRow(row); initInputsOfRow(row);
int guiBottom = guiTop + background.height;
int guiRight = guiLeft + background.width;
confirmButton = confirmButton =
new IconButton(guiLeft + background.width - 33, guiTop + background.height - 24, AllIcons.I_CONFIRM); new IconButton(guiRight - 33, guiBottom - 26, AllIcons.I_CONFIRM);
widgets.add(confirmButton); widgets.add(confirmButton);
int xRight = guiLeft + 53; initEffectSettings();
int xLeft = guiLeft + 93; initMetaSettings();
int yTop = guiTop + 117; }
int yBottom = guiTop + 139;
radius = new ScrollInput(xRight, yTop, 28, 18) private void initMetaSettings() {
.titled(new StringTextComponent("Radius")) int guiBottom = guiTop + background.height;
int y = guiBottom - 23;
blend = new IconButton(guiLeft + 16, y, AllIcons.I_FX_BLEND);
blend.setToolTip(Lang.translate("gui.chromatic_projector.blend"));
int channelX = guiLeft + 39;
rChannel = new IconButton(channelX, y, AllIcons.I_FX_BLEND);
rChannel.setToolTip(new StringTextComponent("R"));
channelX += 18;
gChannel = new IconButton(channelX, y, AllIcons.I_FX_BLEND);
gChannel.setToolTip(new StringTextComponent("G"));
channelX += 18;
bChannel = new IconButton(channelX, y, AllIcons.I_FX_BLEND);
bChannel.setToolTip(new StringTextComponent("B"));
fieldEffect = new IconButton(guiLeft + 135, y, tile.field ? AllIcons.I_FX_FIELD_ON : AllIcons.I_FX_FIELD_OFF);
fieldEffect.setToolTip(Lang.translate("gui.chromatic_projector.field"));
strength = new ScrollInput(guiLeft + 159, y, 25, 18)
.titled(Lang.translate("gui.chromatic_projector.strength"))
.withStepFunction(ctx -> step(ctx, 5))
.calling(tile::setStrength)
.withRange(0, 101)
.setState((int) (tile.strength * 100));
Collections.addAll(widgets, blend, rChannel, gChannel, bChannel, fieldEffect, strength);
}
private void initEffectSettings() {
int x = guiLeft + 188;
int y = guiTop + 40;
radius = new ScrollInput(x, y, 28, 18)
.titled(Lang.translate("gui.chromatic_projector.radius"))
.withStepFunction(ctx -> step(ctx, 2)) .withStepFunction(ctx -> step(ctx, 2))
.calling(tile::setRadius) .calling(tile::setRadius)
.withRange(0, 201) .withRange(0, 201)
.setState((int) (tile.radius * 2)); .setState((int) (tile.radius * 2));
feather = new ScrollInput(xRight, yBottom, 28, 18) y += 22;
.titled(new StringTextComponent("Feather")) feather = new ScrollInput(x, y, 28, 18)
.titled(Lang.translate("gui.chromatic_projector.feather"))
.withStepFunction(ctx -> step(ctx, 5)) .withStepFunction(ctx -> step(ctx, 5))
.calling(tile::setFeather) .calling(tile::setFeather)
.withRange(0, 201) .withRange(0, 201)
.setState((int) (tile.feather * 10)); .setState((int) (tile.feather * 10));
density = new ScrollInput(xLeft, yTop, 28, 18) y += 22;
.titled(new StringTextComponent("Density")) density = new ScrollInput(x, y, 28, 18)
.titled(Lang.translate("gui.chromatic_projector.density"))
.withStepFunction(ctx -> step(ctx, 10)) .withStepFunction(ctx -> step(ctx, 10))
.calling(tile::setDensity) .calling(tile::setDensity)
.withRange(0, 401) .withRange(0, 401)
.setState((int) (tile.density * 100)); .setState((int) (tile.density * 100));
fade = new ScrollInput(xLeft, yBottom, 28, 18) y += 22;
.titled(new StringTextComponent("Fade")) fade = new ScrollInput(x, y, 28, 18)
.titled(Lang.translate("gui.chromatic_projector.fade"))
.withStepFunction(ctx -> step(ctx, 1)) .withStepFunction(ctx -> step(ctx, 1))
.calling(tile::setFade) .calling(tile::setFade)
.withRange(0, 51) .withRange(0, 51)
@ -112,7 +161,7 @@ public class ChromaticProjectorScreen extends AbstractSimiScreen {
ScrollInput type = ScrollInput type =
new SelectionScrollInput(x, y + rowHeight * row, 86, 18) new SelectionScrollInput(x, y + rowHeight * row, 86, 18)
.forOptions(ColorEffect.getOptions()) .forOptions(ColorEffect.getOptions())
.calling(state -> instructionUpdated(row, state)) .calling(state -> stageUpdated(row, state))
.setState(filter.filter.id) .setState(filter.filter.id)
.titled(Lang.translate("gui.chromatic_projector.filter")); .titled(Lang.translate("gui.chromatic_projector.filter"));
ScrollInput value = ScrollInput value =
@ -174,6 +223,8 @@ public class ChromaticProjectorScreen extends AbstractSimiScreen {
renderScroll(matrixStack, feather, 10f); renderScroll(matrixStack, feather, 10f);
renderScroll(matrixStack, fade, 10f); renderScroll(matrixStack, fade, 10f);
renderScroll(matrixStack, strength, 100f);
textRenderer.drawWithShadow(matrixStack, title, guiLeft - 3 + (background.width - textRenderer.getWidth(title)) / 2, guiTop + 3, textRenderer.drawWithShadow(matrixStack, title, guiLeft - 3 + (background.width - textRenderer.getWidth(title)) / 2, guiTop + 3,
0xffffff); 0xffffff);
@ -188,7 +239,7 @@ public class ChromaticProjectorScreen extends AbstractSimiScreen {
String text = String.valueOf(input.getState() / divisor); String text = String.valueOf(input.getState() / divisor);
int stringWidth = textRenderer.getStringWidth(text); int stringWidth = textRenderer.getStringWidth(text);
textRenderer.drawWithShadow(matrixStack, text, input.x + (12 - stringWidth / 2), input.y + 5, 0xFFFFEE); textRenderer.drawWithShadow(matrixStack, text, input.x + 2, input.y + 5, 0xFFFFEE);
} }
private void label(MatrixStack matrixStack, int x, int y, ITextComponent text) { private void label(MatrixStack matrixStack, int x, int y, ITextComponent text) {
@ -204,7 +255,7 @@ public class ChromaticProjectorScreen extends AbstractSimiScreen {
sendPacket(); sendPacket();
} }
private void instructionUpdated(int index, int state) { private void stageUpdated(int index, int state) {
ColorEffect newValue = ColorEffect.all.get(state); ColorEffect newValue = ColorEffect.all.get(state);
stages.get(index).filter = newValue; stages.get(index).filter = newValue;
stages.get(index).value = newValue.defaultValue; stages.get(index).value = newValue.defaultValue;
@ -231,6 +282,33 @@ public class ChromaticProjectorScreen extends AbstractSimiScreen {
return true; return true;
} }
if (blend.isHovered()) {
tile.blend = !tile.blend;
return true;
}
if (fieldEffect.isHovered()) {
tile.field = !tile.field;
fieldEffect.setIcon(tile.field ? AllIcons.I_FX_FIELD_ON : AllIcons.I_FX_FIELD_OFF);
return fieldEffect.mouseClicked(x, y, button);
}
if (rChannel.isHovered()) {
tile.rMask = !tile.rMask;
return true;
}
if (gChannel.isHovered()) {
tile.gMask = !tile.gMask;
return true;
}
if (bChannel.isHovered()) {
tile.bMask = !tile.bMask;
return true;
}
return super.mouseClicked(x, y, button); return super.mouseClicked(x, y, button);
} }

View file

@ -10,7 +10,6 @@ import net.minecraft.block.BlockState;
import net.minecraft.nbt.CompoundNBT; import net.minecraft.nbt.CompoundNBT;
import net.minecraft.tileentity.TileEntityType; import net.minecraft.tileentity.TileEntityType;
import net.minecraft.util.math.BlockPos; import net.minecraft.util.math.BlockPos;
import net.minecraft.util.math.vector.Matrix4f;
import net.minecraftforge.common.util.Constants; import net.minecraftforge.common.util.Constants;
public class ChromaticProjectorTileEntity extends SyncedTileEntity implements IInstanceRendered { public class ChromaticProjectorTileEntity extends SyncedTileEntity implements IInstanceRendered {
@ -18,27 +17,49 @@ public class ChromaticProjectorTileEntity extends SyncedTileEntity implements II
Vector<FilterStep> stages = FilterStep.createDefault(); Vector<FilterStep> stages = FilterStep.createDefault();
float radius = 3f; float radius = 3f;
float density = 1;
float feather = 1; float feather = 1;
float density = 1;
float fade = 1; float fade = 1;
boolean blend = true; boolean blend = true;
public boolean surface = true;
public boolean field = true;
public float strength = 1;
public boolean rMask = true;
public boolean gMask = true;
public boolean bMask = true;
public ChromaticProjectorTileEntity(TileEntityType<?> te) { public ChromaticProjectorTileEntity(TileEntityType<?> te) {
super(te); super(te);
} }
public FilterSphere makeFilter() { public FilterSphere getFilter() {
Matrix4f filter = FilterStep.fold(stages);
BlockPos pos = getPos(); BlockPos pos = getPos();
return new FilterSphere() FilterSphere sphere = new FilterSphere();
.setFilter(filter)
.setCenter(pos.getX() + 0.5, pos.getY() + 0.5, pos.getZ() + 0.5) sphere.x = (float) (pos.getX() + 0.5);
.setRadius(radius) sphere.y = (float) (pos.getY() + 0.5);
.setDensity(density) sphere.z = (float) (pos.getZ() + 0.5);
.setFeather(feather) sphere.radius = radius;
.setBlendOver(true)
.setFade(fade); sphere.feather = feather;
sphere.density = density;
sphere.fade = fade;
sphere.blend = blend;
sphere.surface = surface;
sphere.field = field;
sphere.strength = strength;
sphere.rMask = rMask;
sphere.gMask = gMask;
sphere.bMask = bMask;
sphere.filter = FilterStep.fold(stages);
return sphere;
} }
public ChromaticProjectorTileEntity setRadius(int radius) { public ChromaticProjectorTileEntity setRadius(int radius) {
@ -66,6 +87,11 @@ public class ChromaticProjectorTileEntity extends SyncedTileEntity implements II
return this; return this;
} }
public ChromaticProjectorTileEntity setStrength(int strength) {
this.strength = strength / 100f;
return this;
}
@Override @Override
public CompoundNBT write(CompoundNBT tag) { public CompoundNBT write(CompoundNBT tag) {
super.write(tag); super.write(tag);
@ -73,9 +99,19 @@ public class ChromaticProjectorTileEntity extends SyncedTileEntity implements II
tag.put("filters", FilterStep.writeAll(stages)); tag.put("filters", FilterStep.writeAll(stages));
tag.putFloat("radius", radius); tag.putFloat("radius", radius);
tag.putFloat("density", density);
tag.putFloat("feather", feather); tag.putFloat("feather", feather);
tag.putFloat("density", density);
tag.putFloat("fade", fade); tag.putFloat("fade", fade);
tag.putBoolean("blend", blend);
tag.putBoolean("surface", surface);
tag.putBoolean("field", field);
tag.putFloat("strength", strength);
tag.putBoolean("rMask", rMask);
tag.putBoolean("gMask", gMask);
tag.putBoolean("bMask", bMask);
return tag; return tag;
} }
@ -87,8 +123,18 @@ public class ChromaticProjectorTileEntity extends SyncedTileEntity implements II
stages = FilterStep.readAll(tag.getList("filters", Constants.NBT.TAG_COMPOUND)); stages = FilterStep.readAll(tag.getList("filters", Constants.NBT.TAG_COMPOUND));
radius = tag.getFloat("radius"); radius = tag.getFloat("radius");
density = tag.getFloat("density");
feather = tag.getFloat("feather"); feather = tag.getFloat("feather");
density = tag.getFloat("density");
fade = tag.getFloat("fade"); fade = tag.getFloat("fade");
blend = tag.getBoolean("blend");
surface = tag.getBoolean("surface");
field = tag.getBoolean("field");
strength = tag.getFloat("strength");
rMask = tag.getBoolean("rMask");
gMask = tag.getBoolean("gMask");
bMask = tag.getBoolean("bMask");
} }
} }

View file

@ -19,12 +19,13 @@ public class ColorEffect {
static final HashMap<String, ColorEffect> lookup = new HashMap<>(); static final HashMap<String, ColorEffect> lookup = new HashMap<>();
private static int nextId = 0; private static int nextId = 0;
public static final ColorEffect INVERT = create("invert", ColorMatrices::invert);
public static final ColorEffect SEPIA = create("sepia", ColorMatrices::sepia); public static final ColorEffect SEPIA = create("sepia", ColorMatrices::sepia);
public static final ColorEffect GRAYSCALE = create("grayscale", ColorMatrices::grayscale); public static final ColorEffect GRAYSCALE = create("grayscale", ColorMatrices::grayscale);
public static final ColorEffect DARKEN = create("darken", ColorMatrices::darken).setDefaultValue(20); public static final ColorEffect DARKEN = create("darken", ColorMatrices::darken).setDefaultValue(20);
public static final ColorEffect CONTRAST = create("contrast", ColorMatrices::contrast).setRange(0, 200).setDefaultValue(100);
public static final ColorEffect SATURATE = create("saturate", ColorMatrices::saturate).setRange(0, 200); public static final ColorEffect SATURATE = create("saturate", ColorMatrices::saturate).setRange(0, 200);
public static final ColorEffect HUE_SHIFT = create("hue_shift", ColorMatrices::hueShift).setRange(0, 360).setDivisor(1f).setDefaultValue(120); public static final ColorEffect HUE_SHIFT = create("hue_shift", ColorMatrices::hueShift).setRange(0, 360).setDivisor(1f).setDefaultValue(120);
public static final ColorEffect INVERT = create("invert", ColorMatrices::invert);
public static final ColorEffect END = create("end", ColorMatrices::identity).setBackground(AllGuiTextures.PROJECTOR_END); public static final ColorEffect END = create("end", ColorMatrices::identity).setBackground(AllGuiTextures.PROJECTOR_END);
boolean hasParameter; boolean hasParameter;

View file

@ -12,9 +12,19 @@ public class ConfigureProjectorPacket extends TileEntityConfigurationPacket<Chro
Vector<CompoundNBT> stages; Vector<CompoundNBT> stages;
float radius; float radius;
float density;
float feather; float feather;
float density;
float fade; float fade;
boolean blend;
public boolean surface;
public boolean field;
public float strength;
public boolean rMask;
public boolean gMask;
public boolean bMask;
public ConfigureProjectorPacket(PacketBuffer buffer) { public ConfigureProjectorPacket(PacketBuffer buffer) {
super(buffer); super(buffer);
@ -26,18 +36,38 @@ public class ConfigureProjectorPacket extends TileEntityConfigurationPacket<Chro
stages = tile.stages.stream() stages = tile.stages.stream()
.map(FilterStep::write) .map(FilterStep::write)
.collect(Collectors.toCollection(Vector::new)); .collect(Collectors.toCollection(Vector::new));
this.radius = tile.radius; radius = tile.radius;
this.density = tile.density;
this.feather = tile.feather; feather = tile.feather;
this.fade = tile.fade; density = tile.density;
fade = tile.fade;
blend = tile.blend;
surface = tile.surface;
field = tile.field;
strength = tile.strength;
rMask = tile.rMask;
gMask = tile.gMask;
bMask = tile.bMask;
} }
@Override @Override
protected void writeSettings(PacketBuffer buffer) { protected void writeSettings(PacketBuffer buffer) {
buffer.writeFloat(radius); buffer.writeFloat(radius);
buffer.writeFloat(density);
buffer.writeFloat(feather); buffer.writeFloat(feather);
buffer.writeFloat(density);
buffer.writeFloat(fade); buffer.writeFloat(fade);
buffer.writeBoolean(blend);
buffer.writeBoolean(surface);
buffer.writeBoolean(field);
buffer.writeFloat(strength);
buffer.writeBoolean(rMask);
buffer.writeBoolean(gMask);
buffer.writeBoolean(bMask);
buffer.writeInt(stages.size()); buffer.writeInt(stages.size());
for (CompoundNBT stage : stages) { for (CompoundNBT stage : stages) {
@ -48,9 +78,19 @@ public class ConfigureProjectorPacket extends TileEntityConfigurationPacket<Chro
@Override @Override
protected void readSettings(PacketBuffer buffer) { protected void readSettings(PacketBuffer buffer) {
radius = buffer.readFloat(); radius = buffer.readFloat();
density = buffer.readFloat();
feather = buffer.readFloat(); feather = buffer.readFloat();
density = buffer.readFloat();
fade = buffer.readFloat(); fade = buffer.readFloat();
blend = buffer.readBoolean();
surface = buffer.readBoolean();
field = buffer.readBoolean();
strength = buffer.readFloat();
rMask = buffer.readBoolean();
gMask = buffer.readBoolean();
bMask = buffer.readBoolean();
int count = buffer.readInt(); int count = buffer.readInt();
stages = new Vector<>(FilterStep.MAX_STEPS); stages = new Vector<>(FilterStep.MAX_STEPS);
@ -66,10 +106,20 @@ public class ConfigureProjectorPacket extends TileEntityConfigurationPacket<Chro
.map(FilterStep::new) .map(FilterStep::new)
.collect(Collectors.toCollection(Vector::new)); .collect(Collectors.toCollection(Vector::new));
tile.radius = this.radius; tile.radius = radius;
tile.density = this.density;
tile.feather = this.feather; tile.feather = feather;
tile.fade = this.fade; tile.density = density;
tile.fade = fade;
tile.blend = blend;
tile.surface = surface;
tile.field = field;
tile.strength = strength;
tile.rMask = rMask;
tile.gMask = gMask;
tile.bMask = bMask;
tile.sendData(); tile.sendData();
} }

View file

@ -11,7 +11,7 @@ import net.minecraft.util.math.vector.Matrix4f;
public class FilterStep { public class FilterStep {
public static final int MAX_STEPS = 4; public static final int MAX_STEPS = 6;
ColorEffect filter; ColorEffect filter;
int value; int value;

View file

@ -56,7 +56,7 @@ public enum AllGuiTextures implements IScreenRenderable {
SEQUENCER_EMPTY("sequencer.png", 0, 102, 162, 22), SEQUENCER_EMPTY("sequencer.png", 0, 102, 162, 22),
SEQUENCER_AWAIT("sequencer.png", 0, 160, 162, 22), SEQUENCER_AWAIT("sequencer.png", 0, 160, 162, 22),
PROJECTOR("projector.png", 173, 194), PROJECTOR("projector.png", 235, 185),
PROJECTOR_FILTER_STRENGTH("projector.png", 0, 14, 162, 22), PROJECTOR_FILTER_STRENGTH("projector.png", 0, 14, 162, 22),
PROJECTOR_FILTER("projector.png", 0, 36, 162, 22), PROJECTOR_FILTER("projector.png", 0, 36, 162, 22),
PROJECTOR_END("projector.png", 0, 58, 162, 22), PROJECTOR_END("projector.png", 0, 58, 162, 22),

View file

@ -60,7 +60,10 @@ public class AllIcons implements IScreenRenderable {
I_SCHEMATIC = newRow(), I_SCHEMATIC = newRow(),
I_MTD_LEFT = newRow(), I_MTD_CLOSE = next(), I_MTD_RIGHT = next(), I_MTD_SCAN = next(), I_MTD_REPLAY = next(), I_MTD_LEFT = newRow(), I_MTD_CLOSE = next(), I_MTD_RIGHT = next(), I_MTD_SCAN = next(), I_MTD_REPLAY = next(),
I_MTD_USER_MODE = next(), I_MTD_SLOW_MODE = next(); I_MTD_USER_MODE = next(), I_MTD_SLOW_MODE = next(),
I_FX_SURFACE_OFF = newRow(), I_FX_SURFACE_ON = next(), I_FX_FIELD_OFF = next(), I_FX_FIELD_ON = next(),
I_FX_BLEND = next(), I_FX_BLEND_OFF = next();
public AllIcons(int x, int y) { public AllIcons(int x, int y) {
iconX = x * 16; iconX = x * 16;

View file

@ -50,4 +50,7 @@ public class IconButton extends AbstractSimiWidget {
toolTip.add(text); toolTip.add(text);
} }
public void setIcon(AllIcons icon) {
this.icon = icon;
}
} }

View file

@ -16,13 +16,21 @@ public class RenderUtil {
return b == 0 && n != 0; return b == 0 && n != 0;
} }
public static double lengthSqr(double x, double y, double z) {
return x * x + y * y + z * z;
}
public static double length(double x, double y, double z) {
return Math.sqrt(lengthSqr(x, y, z));
}
public static float[] writeMatrixStack(MatrixStack stack) { public static float[] writeMatrixStack(MatrixStack stack) {
return writeMatrixStack(stack.peek().getModel(), stack.peek().getNormal()); return writeMatrixStack(stack.peek().getModel(), stack.peek().getNormal());
} }
// GPUs want matrices in column major order. // GPUs want matrices in column major order.
public static float[] writeMatrixStack(Matrix4f model, Matrix3f normal) { public static float[] writeMatrixStack(Matrix4f model, Matrix3f normal) {
return new float[] { return new float[]{
model.a00, model.a00,
model.a10, model.a10,
model.a20, model.a20,

View file

@ -93,13 +93,32 @@ public class ColorMatrices {
} }
public static Matrix4f darken(float amount) { public static Matrix4f darken(float amount) {
Matrix4f darken = new Matrix4f(); Matrix4f mat = new Matrix4f();
darken.loadIdentity(); mat.loadIdentity();
darken.multiply(1f - amount); mat.multiply(1f - amount);
darken.a03 = amount; return mat;
darken.a13 = amount; }
darken.a23 = amount;
return darken; public static Matrix4f brightness(float amount) {
Matrix4f mat = new Matrix4f();
mat.loadIdentity();
mat.a03 = amount;
mat.a13 = amount;
mat.a23 = amount;
return mat;
}
public static Matrix4f contrast(float amount) {
Matrix4f sub = new Matrix4f();
sub.a00 = amount;
sub.a11 = amount;
sub.a22 = amount;
sub.a33 = 1;
sub.a30 = 0.5f - amount * 0.5f;
sub.a31 = 0.5f - amount * 0.5f;
sub.a32 = 0.5f - amount * 0.5f;
return sub;
} }
public static Matrix4f identity() { public static Matrix4f identity() {

View file

@ -9,6 +9,7 @@ import org.lwjgl.opengl.GL30;
import com.simibubi.create.foundation.render.AllProgramSpecs; import com.simibubi.create.foundation.render.AllProgramSpecs;
import com.simibubi.create.foundation.render.backend.Backend; import com.simibubi.create.foundation.render.backend.Backend;
import com.simibubi.create.foundation.render.backend.RenderUtil;
import com.simibubi.create.foundation.render.backend.gl.GlBuffer; import com.simibubi.create.foundation.render.backend.gl.GlBuffer;
import com.simibubi.create.foundation.render.backend.gl.GlPrimitiveType; import com.simibubi.create.foundation.render.backend.gl.GlPrimitiveType;
import com.simibubi.create.foundation.render.backend.gl.GlVertexArray; import com.simibubi.create.foundation.render.backend.gl.GlVertexArray;
@ -116,10 +117,16 @@ public class EffectsHandler {
program.setCameraPos(cameraPos.inverse()); program.setCameraPos(cameraPos.inverse());
for (FilterSphere sphere : spheres) { for (FilterSphere sphere : spheres) {
sphere.center = sphere.center.subtract(cameraPos); sphere.x -= cameraPos.x;
sphere.y -= cameraPos.y;
sphere.z -= cameraPos.z;
} }
spheres.sort((o1, o2) -> (int) Math.signum(o2.center.length() - o1.center.length())); spheres.sort((o1, o2) -> {
double l1 = RenderUtil.length(o1.x, o1.y, o1.z);
double l2 = RenderUtil.length(o2.x, o2.y, o2.z);
return (int) Math.signum(l2 - l1);
});
program.uploadFilters(spheres); program.uploadFilters(spheres);

View file

@ -5,87 +5,49 @@ import java.nio.FloatBuffer;
import com.simibubi.create.foundation.render.backend.RenderUtil; import com.simibubi.create.foundation.render.backend.RenderUtil;
import net.minecraft.util.math.vector.Matrix4f; import net.minecraft.util.math.vector.Matrix4f;
import net.minecraft.util.math.vector.Vector3d;
public class FilterSphere { public class FilterSphere {
public Vector3d center; public float x;
public float y;
public float z;
public float radius; public float radius;
public float feather; public float feather;
public float fade; public float fade;
public float density = 2; public float density = 2;
public float strength = 1; public boolean blend = false;
public boolean blendOver = false;
public float r; public boolean surface = true;
public float g; public boolean field = true;
public float b; public float strength = 1;
public float colorFeather;
public boolean rMask;
public boolean gMask;
public boolean bMask;
public Matrix4f filter; public Matrix4f filter;
public FilterSphere setCenter(Vector3d center) {
this.center = center;
return this;
}
public FilterSphere setCenter(double x, double y, double z) {
this.center = new Vector3d(x, y, z);
return this;
}
public FilterSphere setRadius(float radius) {
this.radius = radius;
return this;
}
public FilterSphere setFeather(float feather) {
this.feather = feather;
return this;
}
public FilterSphere setFade(float fade) {
this.fade = fade;
return this;
}
public FilterSphere setDensity(float density) {
this.density = density;
return this;
}
public FilterSphere setStrength(float strength) {
this.strength = strength;
return this;
}
public FilterSphere setFilter(Matrix4f filter) {
this.filter = filter;
return this;
}
public FilterSphere setBlendOver(boolean blendOver) {
this.blendOver = blendOver;
return this;
}
public void write(FloatBuffer buf) { public void write(FloatBuffer buf) {
buf.put(new float[]{ buf.put(new float[]{
(float) center.x, x,
(float) center.y, y,
(float) center.z, z,
radius, radius,
feather, feather,
fade, fade,
density, density,
blendOver ? 1f : 0f, blend ? 1 : 0,
1f,
1f, surface ? 1 : 0,
0f, field ? 1 : 0,
0f, strength,
0.5f, //r, 0, // padding
0.1f, //g,
0.1f, //b, rMask ? 1 : 0,
0f, //colorFeather, gMask ? 1 : 0,
bMask ? 1 : 0,
0, // padding
}); });
buf.put(RenderUtil.writeMatrix(filter)); buf.put(RenderUtil.writeMatrix(filter));

View file

@ -17,8 +17,8 @@ uniform vec3 uCameraPos;
struct SphereFilter { struct SphereFilter {
vec4 sphere;// <vec3 position, float radius> vec4 sphere;// <vec3 position, float radius>
vec4 d1;// <float feather, float fade, float density, float blend mode> vec4 d1;// <float feather, float fade, float density, float blend mode>
vec4 d2;// <float surfaceStrength, float bubbleStrength> vec4 strength;// <float surfaceStrength, float bubbleStrength, float strength>
vec4 colorMask;// <vec3 rgb, float feather> vec4 channelMask;// <vec3 rgb>
mat4 colorOp; mat4 colorOp;
}; };
@ -69,6 +69,12 @@ vec2 raySphere(vec3 worldDir, vec3 position, float radius) {
return vec2(discriminant, hitDepth); return vec2(discriminant, hitDepth);
} }
// if i == 0 return s
// if i == 1 return 1 - s
float invert(float s, float i) {
return i - 2*i*s + s;
}
float bubbleFilterStrength(vec3 worldDir, float depth, vec4 sphere, float feather, float density) { float bubbleFilterStrength(vec3 worldDir, float depth, vec4 sphere, float feather, float density) {
vec3 position = sphere.xyz; vec3 position = sphere.xyz;
@ -96,18 +102,18 @@ float filterStrength(vec3 worldDir, float depth, inout SphereFilter f) {
vec4 data = f.d1; vec4 data = f.d1;
float feather = data.x; float feather = data.x;
float strength = 0.; float strength;
// transition effect // transition effect
float transitionRadius = sphere.w + feather; float transitionRadius = sphere.w + feather;
strength += 1. - smoothstep(transitionRadius, transitionRadius + data.y, length(sphere.xyz)); strength = 1. - smoothstep(transitionRadius, transitionRadius + max(0.5, data.y), length(sphere.xyz));
// bubble effect // bubble effect
strength += bubbleFilterStrength(worldDir, depth, sphere, feather, data.z); strength = max(strength, bubbleFilterStrength(worldDir, depth, sphere, feather, data.z));
strength *= f.d2.y; strength *= f.strength.y;
// surface effect // surface effect
strength += surfaceFilterStrength(worldDir * depth, sphere, feather) * f.d2.x; strength = max(strength, surfaceFilterStrength(worldDir * depth, sphere, feather) * f.strength.x);
return strength; return strength * f.strength.z;
} }
vec3 applyFilters(vec3 worldDir, float depth, vec3 diffuse) { vec3 applyFilters(vec3 worldDir, float depth, vec3 diffuse) {
@ -127,13 +133,13 @@ vec3 applyFilters(vec3 worldDir, float depth, vec3 diffuse) {
vec3 filtered = filterColor(s.colorOp, baseColor); vec3 filtered = filterColor(s.colorOp, baseColor);
vec3 baseHsv = rgb2hsv(baseColor); // vec3 baseHsv = rgb2hsv(baseColor);
vec3 maskHsv = rgb2hsv(s.colorMask.rgb); // vec3 maskHsv = rgb2hsv(s.colorMask.rgb);
vec3 diff = abs(baseHsv - maskHsv) * vec3(1., 1.1, 0.1); // float diff = dot(abs(baseHsv - maskHsv), vec3(1., 1.1, 0.1));
float colorMask = step(s.colorMask.w, length(diff)); // float colorMask = step(s.colorMask.w, diff);
float mixing = clamp(strength, 0., 1.); float mixing = clamp(strength, 0., 1.);
accum = mix(accum, filtered, mixing * colorMask); accum = mix(accum, filtered, mixing * s.channelMask.xyz);
//accum = vec3(colorMask); //accum = vec3(colorMask);
} }
} }

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.9 KiB

After

Width:  |  Height:  |  Size: 4.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.3 KiB

After

Width:  |  Height:  |  Size: 4.4 KiB