Specular map whiteness problem

Hi

I’m using a specular map (together with a normal map) in a texture. I want the maximum brightness of a pixel in the rendered image to be the colour of the corresponding pixel in the specular map, which I think seems a reasonable expectation.

However, in the rendered image the brightest pixels always come out pure white, regardless of the input specular map.

As a simple example, in the rendered sample below I’m using a pure pink specular map to make things obvious (rgb = 244, 181, 181), but although there is a pink hue to some of the specular glints, you can see that the stronger the glint is the less hue is determinable, with large areas coming out pure white.

I also tried applying an alpha to the specular map, hoping that the result might blend the white with the diffuse texture’s colours, hence reducing the brightness, but this had no effect. Adjusting the “Shininess” parameter also doesn’t prevent the glints from coming out white, it just changes which pixels come out white.

Is there a way to configure the material so that the brightest glints are not white? I find that the SpecularMap shader’s results always look far too shiny and “wet” because of this problem.

Thanks
Barney

This is with Lighting.j3md, yes? I suspect the white represents areas where the specular is saturating. Try darkening the specular map.

Hi.

I tried darkening the specular map, all that does is reduce the number/density of rendered pixels which come out white. This is using a specular map with all values rgb = (107, 107, 107), really quite dark.

Brightest areas are still pure white (255, 255, 255). I can obviously make the map even darker, but much darker than that and the specular effect is not really visible (brightest glints are still (255, 255, 255), just hardly any of them there).

So it seems that you can lessen the effect (sparser white glints), but not actually dim it (i.e. you can’t keep the glint density high but lessen the glint brightness).

Thanks,
Barney

@barney said: Hi.

I tried darkening the specular map, all that does is reduce the number/density of rendered pixels which come out white. This is using a specular map with all values rgb = (107, 107, 107), really quite dark.

You will have to post a picture because there is literally nothing in the pipeline that would cause the dithering you describe. Specular is applied to each pixel individually.

Also, to avoid confusing against the base texture you may want to make that a straight gray scale or something… just to be sure your specular isn’t interacting with the base texture in a way that is confusing.

What are the parameters of the material in question? Give us the output of

System.out.println(material.getParams());

after the material is initialized. I’m particularly interested in the value of the ‘Specular’ and ‘Shininess’ parameters.

Hmm. The output from

System.out.println(material.getParams());

is an exception (below). I don’t know if this is the root of my problem. I assume that the exception is a result of me constructing my Material “by hand” i.e. using setTexture()/setParam() rather than using a .j3m?

In any case I have constructed a self-contained example which recreates the problem, about to post it in a separate post.

Thanks

[java]
Jan 26, 2014 12:36:57 PM com.jme3.app.Application handleError
SEVERE: Uncaught exception thrown in Thread[LWJGL Renderer Thread,5,main]

java.lang.UnsupportedOperationException: The specified MatParam cannot be represented in J3M
at com.jme3.material.MatParam.getValueAsString(MatParam.java:262)
at com.jme3.material.MatParam.toString(MatParam.java:367)
at java.lang.String.valueOf(String.java:2854)
at java.lang.StringBuilder.append(StringBuilder.java:128)
at java.util.AbstractCollection.toString(AbstractCollection.java:450)
at java.lang.String.valueOf(String.java:2854)
at java.io.PrintStream.println(PrintStream.java:821)
at bumpnshine.MaterialPreview.simpleInitApp(MaterialPreview.java:373)
at com.jme3.app.SimpleApplication.initialize(SimpleApplication.java:226)
at com.jme3.system.lwjgl.LwjglAbstractDisplay.initInThread(LwjglAbstractDisplay.java:130)
at com.jme3.system.lwjgl.LwjglAbstractDisplay.run(LwjglAbstractDisplay.java:207)
at java.lang.Thread.run(Thread.java:722)[/java]

Hi

Okay, I’ve spent quite a lot of time creating a self-contained example to show you what I mean, and it seems to me that the problem I have been seeing is related to the “noisiness” of the specular maps.

I was using specular maps generated by CrazyBump (which creates specular maps derived from the diffuse texture), or in some cases custom specular maps derived from the diffuse texture using various filters. In either case the specular maps are quite “textured” by which I mean “diffuse/noisy/high frequency”, hopefully I am describing that intelligibly.

So I have created an example based on HelloMaterial which either uses a low-noise map (checkerboard of darkish green/magenta), or uses a darkened version of the diffuse texture as a specular map (high noise).

[java]
package jme3test.helloworld;

import bumpnshine.swing.;
import com.jme3.app.SimpleApplication;
import com.jme3.light.DirectionalLight;
import com.jme3.material.Material;
import com.jme3.material.RenderState.BlendMode;
import com.jme3.math.ColorRGBA;
import com.jme3.math.Vector3f;
import com.jme3.renderer.queue.RenderQueue.Bucket;
import com.jme3.scene.Geometry;
import com.jme3.scene.shape.Box;
import com.jme3.scene.shape.Sphere;
import com.jme3.texture.Texture;
import com.jme3.texture.Texture2D;
import com.jme3.texture.plugins.AWTLoader;
import com.jme3.util.TangentBinormalGenerator;
import java.awt.
;
import java.awt.image.;
import jme3tools.converters.
;

public class HelloSpecularMaterial extends SimpleApplication {

public static void main(String[] args) {
    HelloSpecularMaterial app = new HelloSpecularMaterial();
    app.start();
}
private Geometry sphereGeo;

@Override
public void simpleInitApp() {
    flyCam.setEnabled(false);

    /**
     * A bumpy rock with a shiny light effect. To make bumpy objects you must create a NormalMap.
     */
    Sphere sphereMesh = new Sphere(32, 32, 2f);
    sphereGeo = new Geometry("Shiny rock", sphereMesh);
    sphereMesh.setTextureMode(Sphere.TextureMode.Projected); // better quality on spheres
    TangentBinormalGenerator.generate(sphereMesh);           // for lighting effect
    Material sphereMat = new Material(assetManager, "Common/MatDefs/Light/Lighting.j3md");
    final Texture diffuse = assetManager.loadTexture("Textures/Terrain/Pond/Pond.jpg");
    final int width = diffuse.getImage().getWidth();
    final int height = diffuse.getImage().getHeight();

    sphereMat.setTexture("DiffuseMap", diffuse);
    sphereMat.setTexture("NormalMap", assetManager.loadTexture("Textures/Terrain/Pond/Pond_normal.png"));

    boolean patterned = false;
    if (patterned) {
        BufferedImage bufferedImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_ARGB);
        final Graphics2D graphics2D = (Graphics2D) bufferedImage.getGraphics();
        graphics2D.setColor(new Color(128, 0, 128));
        graphics2D.fillRect(0, 0, width, height);
        graphics2D.setColor(new Color(0, 128, 0));
        int patternSize = 7;
        for (int x = 0; x < width; x += patternSize) {
            for (int y = 0; y < height; y += patternSize) {
                if ((x + y) % 2 == 0) {
                    graphics2D.fillRect(x, y, patternSize, patternSize);
                }
            }
        }
        sphereMat.setTexture("SpecularMap", new Texture2D(new AWTLoader().load(bufferedImage, true)));
    } else {
        float darkness = .5f; // [0.0f to 1.0f]
        BufferedImage bufferedImage = ImageToAwt.convert(diffuse.getImage(), false, true, 0); 
        //Images.dumpDebugImage(bufferedImage, "diffuse-plain");
        final Graphics2D graphics2D = (Graphics2D) bufferedImage.getGraphics();
        graphics2D.setColor(new Color(0, 0, 0, Math.round(darkness * 255)));
        graphics2D.fillRect(0, 0, width, height);
        //Images.dumpDebugImage(bufferedImage, "diffuse-darkened");
        sphereMat.setTexture("SpecularMap", new Texture2D(new AWTLoader().load(bufferedImage, true)));
    }

    sphereMat.setBoolean("UseMaterialColors", true);
    sphereMat.setColor("Diffuse", ColorRGBA.White);
    sphereMat.setColor("Specular", ColorRGBA.White);
    sphereMat.setFloat("Shininess", 10f); // [0,128]
    sphereGeo.setMaterial(sphereMat);
    sphereGeo.rotate(1.6f, 0, 0);          // Rotate it a bit
    sphereGeo.setLocalTranslation(0, 0, 4);
    rootNode.attachChild(sphereGeo);

    /**
     * Must add a light to make the lit object visible!
     */
    DirectionalLight sun = new DirectionalLight();
    sun.setDirection(new Vector3f(1, 0, -2).normalizeLocal());
    sun.setColor(ColorRGBA.White);
    rootNode.addLight(sun);
}

@Override
public void simpleUpdate(float tpf) {
    sphereGeo.rotate(tpf / 2f, tpf / 3f, tpf / 5f);
}

}[/java]

If you set patterned = false then it uses the darkened diffuse map. You will see that even though the specular map is fairly dark (you can adjust how dark with the “darkness” variable), the rendered glints contain large areas of pure white. (The “Specular” colour property is set to White, and I by changing that you can get the glints to contain large areas of pure e.g. Yellow, still not really right).

If you set patterned = true so it uses the checkerboard, then for the most part you do not get bright glints of pure white, the color seems to much more closely respect specular map’s colours.

But what I noticed, and I think is the key to my problem, is that if you look closely at the borders between the checker squares, you see that the specular glints there are much brighter than the squares themselves (the pattern itself is just magenta/green, but it look as if the pattern has white grid-lines between the squares).

So in short, wherever the colour in the specular map changes sharply from one pixel to the next, the resulting glint comes out much too bright.

The clincher for this is that if you set patterned = true and patternSize = 1 (so the specular map is a checkerboard of 1x1 pixel squares), then all the glints come out more-or-less pure white.

So it is not surprising that any noisy specular map (e.g. one derived from the diffuse texture) is going to have the same issue as the 1x1 checkerboard, i.e. overly bright wherever it is noisy.

Sorry if the use of noisy (high frequency) specular maps seems a bit of a niche case, though given the popularity of CrazyBump (similar tools probably also generate noisy specular maps too), it may not be all that niche.

In particular, I think that the standard way to make a specular effect look more matte (i.e. less glossy or wet) is to add noise to the specular map - but this seems to have the opposite effect in JME.

Hope I have made all of this clear!
Thanks
Barney

Just a note:
graphics2D.setColor(new Color(128, 0, 128));
graphics2D.fillRect(0, 0, width, height);
graphics2D.setColor(new Color(0, 128, 0));

Wherever the edges between this pattern are blended (due to min/mag filtering) will be pure grey because of the particular colors you picked. So, for example, if you’d used pure red and pure blue then I’d expect you to have a purple specular color. A pure white and black checkboard might be a better test.

Also, I notice you are using alpha as some kind of darkness control in the non-patterned version… but alpha is completely ignored in the specular map so your spec map is essentially black in this case.

If you are curious, you can see how the specular color is applied here:
http://code.google.com/p/jmonkeyengine/source/browse/trunk/engine/src/core-data/Common/MatDefs/Light/Lighting.frag#282

SpecularSum2.rgb * specularColor.rgb * vec3(light.y)
…where light.y is the ‘shine’ at that particular normal+location… so it’s a straight multiply between the specular lighting component and the specular color component.

For a more muted effect, you might also want to play with the shininess value.

Also, remember the color is added on top of the diffuse color. So if you continue to use a texture for the object it will make any sort of “how this works” conclusions a little harder. For example, if the texture is particularly bright (as the pond.jpg is as I recall) then most of the specular will be bright because its additive.

Either try the test again with no diffuse texture or set the diffuse color to gray to mute the diffuse coloring to better see the real effects of the specular map.

1 Like

sphereMat.setColor(“Diffuse”, ColorRGBA.Black); will get rid of the diffuse color.

> Also, I notice you are using alpha as some kind of darkness control in the non-patterned version… but alpha is completely ignored in the specular map so your spec map is essentially black in this case.

Thanks. I’ll need some more time to process the rest of your post, but just a note on the above, the alpha is just used in conjunction with a solid black which I then paint over the diffuse texture with in awt, i.e. just used as a way to darken the diffuse map before using it as a specular map (quicker than writing a darken filter) … so the image used was a fully opaque darkened version of the diffuse texture. (As you can see from some commented out lines, I used a util to dump the images to disk to confirm this worked correctly.)

@barney said: If you set patterned = false then it uses the darkened diffuse map. You will see that even though the specular map is fairly dark (you can adjust how dark with the "darkness" variable), the rendered glints contain large areas of pure white. (The "Specular" colour property is set to White, and I by changing that you can get the glints to contain large areas of pure e.g. Yellow, still not really right).

What were you expecting? You’re adding 50% white to a bright diffuse textures. Anywhere that the diffuse textures is more than 50% white, you get saturation.

If you set patterned = true so it uses the checkerboard, then for the most part you do not get bright glints of pure white, the color seems to much more closely respect specular map's colours.

This is because you’re only saturating one or two color components instead of all three.

But what I noticed, and I think is the key to my problem, is that if you look closely at the borders between the checker squares, you see that the specular glints there are much brighter than the squares themselves (the pattern itself is just magenta/green, but it look as if the pattern has white grid-lines between the squares).

This is because the specular texture gets blended at the edges of the checker squares, so that all three color components become saturated.

Here’s your problem:
[java]
sphereMat.setColor(“Diffuse”, ColorRGBA.White);
sphereMat.setColor(“Specular”, ColorRGBA.White);

[/java]

If you want colorful specular reflections, you need to avoid saturation somehow, by making sure
that the diffuse and specular color components never add up to more than 100%.

Hi

Thanks for all the responses. Though I’m still sometimes struggling to get the specular effect as I want it, I now appreciate that the specular map is not treated in isolation from the Specular and Diffuse properties.

I think that coming from a strong 2d/Swing background I’m still struggling a bit with the notion of colour multiplication. I’m used to colours being added, subtracted and linearly interpolated and understand exactly what I will get, but I don’t always get what I expect when colors are multiplied (though I think I understand why multiplication is used where lighting is involved).

I guess what my original question should have boiled down to, is “is there a way to ignore the Specular colour completely, and just use colours from the SpecularMap texture?” - and I think the answer is “no”, but that I can generally achieve this by finding a suitable shade of grey for the “Specular” colour.

Cheers
Barney

If you set up your Specular color to be ColorRGBA.White, then you are basically saying to use only specular map. Same way if you don’t have specular map, you tell to use only specular color. If you map is pure dark at given pixel, material specular color won’t matter - white*black = black.

I think that is should all make perfect sense as long as you think about color components being in 0-1 float range, rather than 0-255 integer range.

Real issue is as sgold mentioned - final computation is doing diffuse + specular, which can cause oversaturation. Try putting both diffuse and specular colors of the material to 0.5f gray and see how it looks.

1 Like

Make a specular map of just fully black and fully white squares… I think you’ll see it pretty dramatically at that point.

I think so far your only confusion is what “dark” means in a specular map. Medium gray is not dark.

@barney said: I guess what my original question should have boiled down to, is "is there a way to ignore the Specular colour completely, and just use colours from the SpecularMap texture?" - and I think the answer is "no", but that I can generally achieve this by finding a suitable shade of grey for the "Specular" colour.

The Lighting.j3md material definitions compute the specular component by (among other things) multiplying a texture sample by a color. So you have choices: you can write a custom material definition which doesn’t multiply colors, or you can wrap your brain around the concept of multiplying colors. In the long run, the latter option will serve you better.