import Cartesian2 from "../Core/Cartesian2.js";
import Color from "../Core/Color.js";
import defined from "../Core/defined.js";
import destroyObject from "../Core/destroyObject.js";
import PrimitiveType from "../Core/PrimitiveType.js";
import ClearCommand from "../Renderer/ClearCommand.js";
import DrawCommand from "../Renderer/DrawCommand.js";
import FramebufferManager from "../Renderer/FramebufferManager.js";
import Pass from "../Renderer/Pass.js";
import RenderState from "../Renderer/RenderState.js";
import ShaderSource from "../Renderer/ShaderSource.js";
import BlendingState from "../Scene/BlendingState.js";
import StencilConstants from "../Scene/StencilConstants.js";
import PointCloudEyeDomeLightingShader from "../Shaders/PostProcessStages/PointCloudEyeDomeLighting.js";
/**
* Eye dome lighting. Does not support points with per-point translucency, but does allow translucent styling against the globe.
* Requires support for EXT_frag_depth and WEBGL_draw_buffers extensions in WebGL 1.0.
*
* @private
*/
function PointCloudEyeDomeLighting() {
this._framebuffer = new FramebufferManager({
colorAttachmentsLength: 2,
depth: true,
supportsDepthTexture: true,
});
this._drawCommand = undefined;
this._clearCommand = undefined;
this._strength = 1.0;
this._radius = 1.0;
}
Object.defineProperties(PointCloudEyeDomeLighting.prototype, {
framebuffer: {
get: function () {
return this._framebuffer.framebuffer;
},
},
colorGBuffer: {
get: function () {
return this._framebuffer.getColorTexture(0);
},
},
depthGBuffer: {
get: function () {
return this._framebuffer.getColorTexture(1);
},
},
});
function destroyFramebuffer(processor) {
processor._framebuffer.destroy();
processor._drawCommand = undefined;
processor._clearCommand = undefined;
}
const distanceAndEdlStrengthScratch = new Cartesian2();
function createCommands(processor, context) {
const blendFS = new ShaderSource({
defines: ["LOG_DEPTH_WRITE"],
sources: [PointCloudEyeDomeLightingShader],
});
const blendUniformMap = {
u_pointCloud_colorGBuffer: function () {
return processor.colorGBuffer;
},
u_pointCloud_depthGBuffer: function () {
return processor.depthGBuffer;
},
u_distanceAndEdlStrength: function () {
distanceAndEdlStrengthScratch.x = processor._radius;
distanceAndEdlStrengthScratch.y = processor._strength;
return distanceAndEdlStrengthScratch;
},
};
const blendRenderState = RenderState.fromCache({
blending: BlendingState.ALPHA_BLEND,
depthMask: true,
depthTest: {
enabled: true,
},
stencilTest: StencilConstants.setCesium3DTileBit(),
stencilMask: StencilConstants.CESIUM_3D_TILE_MASK,
});
processor._drawCommand = context.createViewportQuadCommand(blendFS, {
uniformMap: blendUniformMap,
renderState: blendRenderState,
pass: Pass.CESIUM_3D_TILE,
owner: processor,
});
processor._clearCommand = new ClearCommand({
framebuffer: processor.framebuffer,
color: new Color(0.0, 0.0, 0.0, 0.0),
depth: 1.0,
renderState: RenderState.fromCache(),
pass: Pass.CESIUM_3D_TILE,
owner: processor,
});
}
function createResources(processor, context) {
const width = context.drawingBufferWidth;
const height = context.drawingBufferHeight;
processor._framebuffer.update(context, width, height);
createCommands(processor, context);
}
function isSupported(context) {
return context.drawBuffers && context.fragmentDepth;
}
PointCloudEyeDomeLighting.isSupported = isSupported;
function getECShaderProgram(context, shaderProgram) {
let shader = context.shaderCache.getDerivedShaderProgram(shaderProgram, "EC");
if (!defined(shader)) {
const attributeLocations = shaderProgram._attributeLocations;
const fs = shaderProgram.fragmentShaderSource.clone();
fs.sources.splice(
0,
0,
`layout (location = 0) out vec4 out_FragData_0;\nlayout (location = 1) out vec4 out_FragData_1;`
);
fs.sources = fs.sources.map(function (source) {
source = ShaderSource.replaceMain(
source,
"czm_point_cloud_post_process_main"
);
source = source.replaceAll(/out_FragColor/g, "out_FragData_0");
return source;
});
fs.sources.push(
"void main() \n" +
"{ \n" +
" czm_point_cloud_post_process_main(); \n" +
"#ifdef LOG_DEPTH\n" +
" czm_writeLogDepth();\n" +
" out_FragData_1 = czm_packDepth(gl_FragDepth); \n" +
"#else\n" +
" out_FragData_1 = czm_packDepth(gl_FragCoord.z);\n" +
"#endif\n" +
"}"
);
shader = context.shaderCache.createDerivedShaderProgram(
shaderProgram,
"EC",
{
vertexShaderSource: shaderProgram.vertexShaderSource,
fragmentShaderSource: fs,
attributeLocations: attributeLocations,
}
);
}
return shader;
}
PointCloudEyeDomeLighting.prototype.update = function (
frameState,
commandStart,
pointCloudShading,
boundingVolume
) {
if (!isSupported(frameState.context)) {
return;
}
this._strength = pointCloudShading.eyeDomeLightingStrength;
this._radius =
pointCloudShading.eyeDomeLightingRadius * frameState.pixelRatio;
createResources(this, frameState.context);
// Hijack existing point commands to render into an offscreen FBO.
let i;
const commandList = frameState.commandList;
const commandEnd = commandList.length;
for (i = commandStart; i < commandEnd; ++i) {
const command = commandList[i];
if (
command.primitiveType !== PrimitiveType.POINTS ||
command.pass === Pass.TRANSLUCENT
) {
continue;
}
let derivedCommand;
let originalShaderProgram;
let derivedCommandObject = command.derivedCommands.pointCloudProcessor;
if (defined(derivedCommandObject)) {
derivedCommand = derivedCommandObject.command;
originalShaderProgram = derivedCommandObject.originalShaderProgram;
}
if (
!defined(derivedCommand) ||
command.dirty ||
originalShaderProgram !== command.shaderProgram ||
derivedCommand.framebuffer !== this.framebuffer
) {
// Prevent crash when tiles out-of-view come in-view during context size change or
// when the underlying shader changes while EDL is disabled
derivedCommand = DrawCommand.shallowClone(command, derivedCommand);
derivedCommand.framebuffer = this.framebuffer;
derivedCommand.shaderProgram = getECShaderProgram(
frameState.context,
command.shaderProgram
);
derivedCommand.castShadows = false;
derivedCommand.receiveShadows = false;
if (!defined(derivedCommandObject)) {
derivedCommandObject = {
command: derivedCommand,
originalShaderProgram: command.shaderProgram,
};
command.derivedCommands.pointCloudProcessor = derivedCommandObject;
}
derivedCommandObject.originalShaderProgram = command.shaderProgram;
}
commandList[i] = derivedCommand;
}
const clearCommand = this._clearCommand;
const blendCommand = this._drawCommand;
blendCommand.boundingVolume = boundingVolume;
// Blend EDL into the main FBO
commandList.push(blendCommand);
commandList.push(clearCommand);
};
/**
* Returns true if this object was destroyed; otherwise, false.
*
* If this object was destroyed, it should not be used; calling any function other than
* isDestroyed
will result in a {@link DeveloperError} exception.
*
* @returns {boolean} true
if this object was destroyed; otherwise, false
.
*
* @see PointCloudEyeDomeLighting#destroy
*/
PointCloudEyeDomeLighting.prototype.isDestroyed = function () {
return false;
};
/**
* Destroys the WebGL resources held by this object. Destroying an object allows for deterministic
* release of WebGL resources, instead of relying on the garbage collector to destroy this object.
*
* Once an object is destroyed, it should not be used; calling any function other than
* isDestroyed
will result in a {@link DeveloperError} exception. Therefore,
* assign the return value (undefined
) to the object as done in the example.
*
* @exception {DeveloperError} This object was destroyed, i.e., destroy() was called.
*
* @example
* processor = processor && processor.destroy();
*
* @see PointCloudEyeDomeLighting#isDestroyed
*/
PointCloudEyeDomeLighting.prototype.destroy = function () {
destroyFramebuffer(this);
return destroyObject(this);
};
export default PointCloudEyeDomeLighting;