Draft
By default, PBR shader is configured with a maximum set of lights as below :
numDirectionalLights = 2;
numPointLights = 5;
numSpotLights = 0;
Even if there are not some many lights in your scene, shader will still render them with zero intensity. So if you don't need so many, it's a good idea to properly configure them (depending on your game). These lights have significant impact on performances. A typical configuration is only one directional light. It's also a good idea to configure maximum bones per renderable as well. Here is a configuration exemple :
PBRShaderConfig colorConfig = new PBRShaderConfig();
colorConfig.numDirectionalLights = 1;
colorConfig.numPointLights = 0;
colorConfig.numSpotLights = 0;
colorConfig.maxBones = 12;
DepthShader.Config depthConfig= new DepthShader.Config();
depthConfig.maxBones = 12;
SceneManager sceneManager = new SceneManager(
new PBRShaderProvider(colorConfig),
new PBRDepthShaderProvider(depthConfig)
);
Sometimes you want ot render scenes to a FBO (Frame buffer). You can do it but you have to take some cautions: SceneManager is using FBOs internally to render shadows and transmission. So, instead of calling sceneManager.render(), you have to do something like this:
sceneManager.renderShadows();
sceneManager.renderMirror();
sceneManager.renderTransmission();
fbo.begin();
...
sceneManager.renderColors();
...
fbo.end();
Also, sometimes you want to capture depth only. You can do it like this :
fbo.begin();
sceneManager.renderDepth();
fbo.end();
You can manage the renderable provider list within a SceneManager, it allows to add arbitrary renderable providers (eg. ModelCache)
In some situations you only want to render some objects (eg. only visible objet).
sceneManager.getRenderableProviders().clear();
sceneManager.getRenderableProviders().add(scene1, scene2);
sceneManager.update(delta);
sceneManager.render();
The same way you want to exclude some objets from the rendering, you may want to exclude some objects from shadow casting. Or even having some shadow casting only objects.
In this example, we have 3 scenes :
- baseScene : both rendered and casting shadows
- shadowOnlyScene : not rendered, it only cast shadows
- renderOnlyScene : rendered but not casting shadows
// update the 3 scenes
sceneManager.getRenderableProviders().clear();
sceneManager.getRenderableProviders().add(baseScene, shadowOnlyScene, renderOnlyScene);
sceneManager.update(delta);
// render depth for shadows with only 2 scenes
sceneManager.getRenderableProviders().clear();
sceneManager.getRenderableProviders().add(baseScene, shadowOnlyScene);
sceneManager.renderShadows();
// final render with only 2 scenes
sceneManager.getRenderableProviders().clear();
sceneManager.getRenderableProviders().add(baseScene, renderOnlyScene);
sceneManager.renderColors();
Fog can be enabled like with libGDX default shader:
sceneManager.environment.set(new ColorAttribute(ColorAttribute.Fog, Color.WHITE));
Additional fog settings help to configure fog behavior :
- near and far clipping fog planes, by default same has camera near and far.
- exponent (default 1.0) can be used to tweak fog : above 1 will push the fog to back, below 1 will pull fog to front.
Note that near and far can be inverted to produce a kind of unrealistic back light effect.
sceneManager.environment.set(new FogAttribute(FogAttribute.FogEquation).set(near, far, exponent));
TODO
PBR shader is used by default which enable all glTF features and gives relatively realistic results.
However, for performance reasons you may want a simpler shader. Obviously you can create your own shader but you can use default libGDX shader as well. In this case, only few features will be supported and rendering will be less rich.
You can use default shader provider to be used with scene manager.
new SceneManager(new DefaultShaderProvider(), new DepthShaderProvider());
PBR shader supports one directional shadow light (just like default libgdx 3D).
However, it's necessary to use the DirectionalShadowLight class included in this library instead of the libgdx one
(net.mgsx.gltf.scene3d.lights.DirectionalShadowLight
).
PBR shader also supports shadow light bias in order to reduce "acne" artifacts. You can configure it by adding an attribute to the environment (PBRFloatAttribute.ShadowBias). A value of 1.0/256.0 is fine most of the time.
Additionally, PBR shader support cascade shadow maps. It provides better shadow quality without using huge depth map. Scene camera frustum is split by distance. A depth map is renderer for each area (sub frustum). Cascade shadow map only works if there is a DirectionalShadowLight in the environment. This DirectionalShadowLight is used to cover far scene area. Other extra cascades are used to cover closer scene areas.
Here is a basic setup for 3 cascades (2 extra cascades) :
csm = new CascadeShadowMap(2);
sceneManager.setCascadeShadowMap(csm);
Before rendering the scene, you should configure these cascades. You could do it manually or use a convenient method as in this example. We use a minimum depth of 1000, this value should be adjusted depending on the scene. You may want to increase this value in order to have far object out of the scene camera to be shadow casted. The split factor is set to 4 which gives good transitions between cascades. This value is tighly coupled to the cascade count and camera far plane distance. All of these and also depth maps resolution have to be tuned depending the quality and performance you want.
shadowLight = sceneManager.getFirstDirectionalShadowLight();
csm.setCascades(sceneManager.camera, shadowLight, 1000f, 4f);
...
sceneManager.update(delta);
sceneManager.render();
TODO explain DiLightEx with intensity parameter...
Point lights and Spot lights range can be exported from blender (using custom distance option). This range can be used for light frustum culling Default libgdx lights doesn't have this information, you can cast loaded lights from your GLTF files by casting them to PointLightEx or SpotLightEx to retrieve the range field.
When materials have transmission, these models produce refraction effects, which is the way to render some materials like glass. For such effect, opaque objects needs to be pre-rendered first before rendering the final pass.
By default, SceneManager doesn't perform any pre-rendering. In this case, only IBL image will be sampled which is cheaper but produce some good enough results.
If you want better results, that is, seeing deformed objects through these materials, you have to enable it explicitly like this :
sceneManager.setTransmissionSource(new TransmissionSource(shaderProvider));
You usually give the same shader provider for both TransmissionSource and normal rendering.
You can enable dynamic reflections with a planar mirror, you need to enable mirror pre-render:
sceneManager.setMirrorSource(new MirrorSource().set(0, 1, 0, -0.833f, true));
And then apply it to your plane object(s) via its material. In this case, mirror frame buffer will be sampled instead of specular cubemap for them :
material.set(MirrorAttribute.createSpecular());
Note that material with mirror reflection shouldn't be double sided. It would produce artifacts in the pre-render pass and will give wrong reflections on the back side anyway.
Note that you can change mirror framebuffer size (default is screen size) to improve performance but it will affect quality and will bias roughness.
Scene own a AnimationController to play animation on its modelInstance (regular libgdx animation player).
When your scene is an individual model (single skeleton) you typically use scene.animationController
to play
single animation and actions.
Also, Scene own a more advanced animation player providing some features like playing multiple animations.
When your scene is made of several individual models with their own animation (cinematic scene), you want to play a bunch
of animation on the same modelInstance. In this case use scene.animationsPlayer
instead of scene.animationController
.
You may want to have separated GLTF files for a model and its animations and merge them at runtime.
All you have to do is to add those extra animations from one SceneAsset to the other before creating your Scene.
modelAsset.scene.model.animations.addAll(animationsAsset.animations)
Scene scene = new Scene(modelAsset.scene)
There are several way to extends PBR shaders :
- by providing custom vertex and fragment shader code.
- by subclassing PBRShaderProvider
- by subclassing both PBRShaderProvider and PBRShader
Note that PBRDepthShader can be customized as well in the same way.
Also note that PBR shader is split into several files so you'll have to copy all of them and load them using ShaderParser
As libgdx default shader, default PBR shader use supports 1 color attribute used to multiply base color.
If you want to implement a custom shader to support more vertex color layers, you have to enable by setting PBRConfig#numVertexColors.
You can then retrieve attributes in your custom vertex shader : a_color1 for 2nd layer, a_color2 for 3rd layer and so on. For compatibility reasons, 1st layer is a_color
First you have to read the Blender GLTF exporter documentation to know how to setup your materials and what kind of objects can be exported.
Based on glTF Exporter 1.6.16 addon (Blender 2.93.1 LTS).
Here is the recommended settings and some hint about some of them :
General settings:
- Format: glTF separate is recommended for best loading time performances and compatibility.
What to include:
- Limit to: By default Blender export only enabled collections. It's helpful to exclude some objects (guides) or split into several files. You can check some boxes here to fine tune your selection.
- Cameras and Punctual lights: Check these if your want to export cameras and lights (point, sun and spot lights are supported)
- Custom properties: should be checked if you want to export shape keys names or any user defined properties, see Custom Properties section
Geometry settings:
- Normals: not mandatory but recomended, when not exported flat normals will be generated at loading time.
- Tangents: not mandatory but recomended, when not exported they are generated at loading time.
- Vertex Colors: only one vertex color attribute is supported by gdx-gltf PBR shader but you can have multiple one to be used in your custom shader or any other advanced usage.
- Looses Edges and Points: Export edges only models or vertex only models. They will be rendered as
GL_LINES
orGL_POINTS
.
Note that these attributes will only be exported when geometry have them. For example, if an object doesn't have UVs, the exported mesh won't have UVs even if the box is checked, so it doesn't hurt to have them checked.
- Compression: not supported by gdx-gltf for now.
Animation settings:
- Use current frame: You rarely want that, when unchecked it will export models in base pose regardless of current playback frame.
- Limit to playback range: You rarely want that, when unchecked it will export animations as is regardless of current playback range.
- Always Sample Animation: Required when using bone constraints (IK), set a proper Sampling Rate in this case to limit number of keyframes: too many keyframes may impact performances (loading, memory and playback).
- Group by NLA track: You need to have this checked in order to export all animations including not active ones (stashed or pushed).
- Export deformation bones only: Uncheck that if you want to export some bones used as placeholder (eg. weapon attachement).
- Include All Bone Influences: by default, exporter will limit to 4 influences (4 bones influences a vertex), you can disable this limit if you need more: gdx-gltf supports up to 8 influences, you then have to manually limit influences in your skinning (in Weight Paint mode, you can limit total weights).
- Shape Key Tangent: required if your model with shape keys uses normal maps.
glTF Extras are user defined structured data (JSON) attached to any GLTFObject, typically a key value pair.
Blender allow user defined custom properties on several entities (objects, material, etc). These custom properties are exported in related GLTF entity extras.
GLTF data structure is not kept in memory beyond loading by default to save memory usage. If you need data structure, you need to enable it either by direct loading :
new GLTFLoader.load(myFile, true)
or via AssetLoader parameters:
assetManager.load("myFile.gltf", new GLTFLoaderParameter(true));
For instance Blender GLTF exporter may save some of its own settings, you can retreive them like this:
sceneAsset.data.extras.get("glTF2ExportSettings")
glTF2ExportSettings: {
export_apply: 1
export_cameras: 1
export_copyright: mgsx
export_extras: 1
export_format: GLTF_SEPARATE
export_lights: 1
export_morph_tangent: 1
export_tangents: 1
}
here is the list of all supported custom properties (properties exported within your GLTF files) and mapping from Blender entities to GLTF entities and LibGDX entities:
- Blender Object => GLTFNode => Node
- TODO more... and test
Another example of custom properties on a Blender Object:
Because GLTF data structure differs from libGDX structure, you have to browse loaded GLTF data structure yourself, e.g.:
Array<GLTFNode> nodes = sceneAsset.data.nodes;
for(GLTFNode node : nodes){
if(node.name.equals("Cube"){
float myValue = node.extras.getFloat("myProperty");
}
}
Note that in Blender you can manually define key/value pairs (with float value) only. However you can define any JSON structure via scripting. For instance, you can copy some Blender specific settings like Empty type and size this way :
import bpy;
for object in bpy.data.objects:
if not object.data:
object["Empty"] = {
"size": object.empty_display_size,
"type": object.empty_display_type
}
You can the retreive this structured data structure from GLTF nodes in your game.
Only Principled BSDF shader is supported
TODO : blending mode, where opacity is stored...
Shape keys are converted to Morph targets in GLTF structure. It's limited to 8 morph targets, so up to 8 shape keys in addition to "Basis" shape keys.
Note that most GPU support up to 16 vertex attributes so a lot of shape keys might be a problem, especially if you have both position, normals and tangents. See Limitation section for more information.
Blender GLTF exporter export shape keys names via Custom properties. So you need to enable custom properties export in order to retrieve them.
Shape keys names are stored in Node object and can be used like this :
NodePlus nodePlus = (NodePlus)node;
int shapeKeyIndex = nodePlus.morphTargetNames.indexOf("HappyFace");
nodePlus.weights.values[index] = 0.7f;
If you didn't have shape keys names, you still can set weights but you have to know shape key index.
Current Blender exporter only export 2 vertex color layers
if you need more, you can always modify your local file in BlenderInstallDirectory/scripts/addons/io_scene_gltf2/blender/exp/gltf2_blender_extract.py and restart Blender.
Morph targets are not exported when modifiers have to be applied.
There is no workaround since modifiers can't be applied when a mesh has shape keys defined.
You should then apply modifiers (eg. subsurface) before defining your shapekeys.
- how to customize PBR shader
- how advanced fog works
- programatically create models (with all GLTF features... shape keys and such)
- all PBR shader options and static options