Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Metadata picking preparation #12075

Merged
merged 73 commits into from
Sep 30, 2024
Merged
Show file tree
Hide file tree
Changes from 45 commits
Commits
Show all changes
73 commits
Select commit Hold shift + click to select a range
656c638
Removed some unused properties
javagl Jul 9, 2024
247102f
Draft of metadata picking
javagl Jul 9, 2024
e561dec
Fix typo
javagl Jul 9, 2024
cc1b826
Try to avoid recompilation
javagl Jul 15, 2024
a28f62d
Do not modify the bounding sphere...
javagl Jul 16, 2024
122c4e8
Remove debug stuff
javagl Jul 16, 2024
a44ac07
Draft for warnings for unsupported property types
javagl Jul 17, 2024
e0c57f6
Draft for encoding metadata values for picking
javagl Jul 17, 2024
dc7552c
Update matrix only when command is defined
javagl Jul 19, 2024
609e277
Minor comment fix
javagl Jul 19, 2024
927d01b
Draft for metadata value decoding
javagl Jul 19, 2024
461821b
Minor consolidation
javagl Jul 20, 2024
261dbba
Fix JSDoc type
javagl Jul 25, 2024
3817389
Do not duplicate full command - use derived command
javagl Jul 25, 2024
a183fd4
Merge remote-tracking branch 'origin/main' into metadata-picking-prep…
javagl Jul 25, 2024
7848f5d
Minor consolidation
javagl Jul 30, 2024
66b3252
Update ModelRuntimePrimitiveSpec
javagl Jul 30, 2024
016a1a8
Fix import file extension
javagl Jul 30, 2024
d575317
Trying to make JSDoc happier
javagl Jul 31, 2024
4d0b722
Merge remote-tracking branch 'origin/main' into metadata-picking-prep…
javagl Jul 31, 2024
2f29019
Remove debug log
javagl Aug 18, 2024
a1989c0
First pass of PR feedback
javagl Aug 18, 2024
88500ec
Removed debug log
javagl Aug 18, 2024
fe4c2bd
Replace repeated strings with constants
javagl Aug 18, 2024
f8bac8f
Generalizations and cleanups for metadata decoding
javagl Aug 23, 2024
8a24da2
Hacky spec experiments
javagl Aug 28, 2024
ebddfe1
Revert some experiments. Cleanups.
javagl Aug 29, 2024
c0eaf7b
Remove debug log
javagl Aug 29, 2024
e732550
Cleanups for metadata picking specs
javagl Aug 29, 2024
775113d
Basic specs and debugging
javagl Aug 30, 2024
2056da0
Extend metadata picking specs
javagl Aug 31, 2024
0e6b7f6
Remove debug output
javagl Aug 31, 2024
45727e8
Remove inlined image creation
javagl Sep 7, 2024
2a948cf
Merge remote-tracking branch 'origin/main' into metadata-picking-prep…
javagl Sep 8, 2024
ddd5fb8
Wrap up specs for metadata picking
javagl Sep 10, 2024
4347a23
Merge remote-tracking branch 'origin/main' into metadata-picking-prep…
javagl Sep 10, 2024
d813dba
Update spec for new pipeline stage
javagl Sep 10, 2024
0ad2bc5
Revive parts of the metadata picking specs
javagl Sep 10, 2024
aa75922
Try declaring metadata picking specs as not-WebGL
javagl Sep 10, 2024
9c3c081
Classify metadata picking spec as WebGL
javagl Sep 11, 2024
c854235
Create WebGL 1 context for metadata picking specs
javagl Sep 11, 2024
384b2fa
Skip metadata picking specs on WebGL stubs
javagl Sep 11, 2024
3c1b497
Make sure that scene is defined in specs
javagl Sep 11, 2024
a005521
Do not try to remove the primitives
javagl Sep 11, 2024
30e80b1
Do not try to keep the scene
javagl Sep 11, 2024
9007e85
Remove debug log
javagl Sep 27, 2024
2a92b94
Cleaner handling for dirty commands
javagl Sep 27, 2024
8b02ec4
Clarification for undefined schema ID
javagl Sep 27, 2024
7735111
Optional chaining. Fix from removed debug log.
javagl Sep 27, 2024
d816418
Optional chaining. Comment cleanups.
javagl Sep 27, 2024
7d34d0d
Return type in JSDoc
javagl Sep 27, 2024
8af87b2
Pull PickedMetadataInfo into class
javagl Sep 27, 2024
6054373
Add type information in JSDoc
javagl Sep 27, 2024
e1e2ae6
Fix return type in JSDoc
javagl Sep 27, 2024
6f7f162
Added return types in JSDoc
javagl Sep 27, 2024
651a2b2
Add return types in JSDoc
javagl Sep 27, 2024
a7a0ece
Make functions local, as per review comment...
javagl Sep 27, 2024
0e47cfb
Merge tag 'pre-prettier-v3' into metadata-picking-preparation-prettier
javagl Sep 28, 2024
21129e2
Merge tag 'post-prettier-v3' into metadata-picking-preparation-prettier
javagl Sep 28, 2024
777183f
Merge remote-tracking branch 'origin/main' into metadata-picking-prep…
javagl Sep 28, 2024
942a178
Update CHANGES.md
ggetz Sep 30, 2024
efe3510
Removed obsolete comment
javagl Sep 30, 2024
35bdc13
Throw error for invalid component types
javagl Sep 30, 2024
93b994a
Use predefined constants
javagl Sep 30, 2024
17bc38b
Update comment in Picking.js via review suggestion
javagl Sep 30, 2024
048771f
Merge branch 'metadata-picking-preparation' of https://github.com/Ces…
javagl Sep 30, 2024
4f99d1b
Update comment in Scene.js from review suggestion
javagl Sep 30, 2024
7300ae1
Update comment from review suggestion
javagl Sep 30, 2024
bd84faa
Merge branch 'metadata-picking-preparation' of https://github.com/Ces…
javagl Sep 30, 2024
eb406b5
Merge remote-tracking branch 'origin/main' into metadata-picking-prep…
javagl Sep 30, 2024
59b4787
Do not declare required parameters as optional
javagl Sep 30, 2024
2ee6366
Do not declare parameters as optional
javagl Sep 30, 2024
280e8d9
Move metadata picking specs into SceneSpec
javagl Sep 30, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
44 changes: 43 additions & 1 deletion packages/engine/Source/Renderer/DrawCommand.js
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,8 @@ function DrawCommand(options) {
this._owner = options.owner;
this._debugOverlappingFrustums = 0;
this._pickId = options.pickId;
this._pickMetadataAllowed = options.pickMetadataAllowed === true;
this._pickedMetadataInfo = undefined;

// Set initial flags.
this._flags = 0;
Expand Down Expand Up @@ -511,7 +513,7 @@ Object.defineProperties(DrawCommand.prototype, {
* during the pick pass.
*
* @memberof DrawCommand.prototype
* @type {string}
* @type {string|undefined}
* @default undefined
*/
pickId: {
Expand All @@ -525,6 +527,44 @@ Object.defineProperties(DrawCommand.prototype, {
}
},
},

/**
* Whether metadata picking is allowed.
*
* This is essentially only set to `true` for draw commands that are
* part of a `ModelDrawCommand`, to check whether a derived command
* for metadata picking has to be created.
*
* @memberof DrawCommand.prototype
* @type {boolean}
* @default undefined
* @private
*/
pickMetadataAllowed: {
get: function () {
return this._pickMetadataAllowed;
},
},

/**
* Information about picked metadata.
*
* @memberof DrawCommand.prototype
* @type {PickedMetadataInfo|undefined}
* @default undefined
*/
pickedMetadataInfo: {
get: function () {
return this._pickedMetadataInfo;
},
set: function (value) {
if (this._pickedMetadataInfo !== value) {
this._pickedMetadataInfo = value;
this.dirty = true;
}
},
},

/**
* Whether this command should be executed in the pick pass only.
*
Expand Down Expand Up @@ -590,6 +630,8 @@ DrawCommand.shallowClone = function (command, result) {
result._owner = command._owner;
result._debugOverlappingFrustums = command._debugOverlappingFrustums;
result._pickId = command._pickId;
result._pickMetadataAllowed = command._pickMetadataAllowed;
result._pickedMetadataInfo = command._pickedMetadataInfo;
result._flags = command._flags;

result.dirty = true;
Expand Down
223 changes: 223 additions & 0 deletions packages/engine/Source/Scene/DerivedCommand.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@ import defined from "../Core/defined.js";
import DrawCommand from "../Renderer/DrawCommand.js";
import RenderState from "../Renderer/RenderState.js";
import ShaderSource from "../Renderer/ShaderSource.js";
import MetadataType from "./MetadataType.js";
import MetadataPickingPipelineStage from "./Model/MetadataPickingPipelineStage.js";

/**
* @private
Expand Down Expand Up @@ -375,6 +377,227 @@ DerivedCommand.createPickDerivedCommand = function (
return result;
};

/**
* Replaces the value of the specified 'define' directive identifier
* with the given value.
*
* The given defines are the parts of the define directives that are
* stored in the `ShaderSource`. For example, the defines may be
* `["EXAMPLE", "EXAMPLE_VALUE 123"]`
*
* Calling `replaceDefine(defines, "EXAMPLE", 999)` will result in
* the defines being
* `["EXAMPLE 999", "EXAMPLE_VALUE 123"]`
*
* @param {string[]} defines The define directive identifiers
* @param {string} defineName The name (identifier) of the define directive
* @param {any} newDefineValue The new value whose string representation
* will become the token string for the define directive
* @private
*/
function replaceDefine(defines, defineName, newDefineValue) {
const n = defines.length;
for (let i = 0; i < n; i++) {
const define = defines[i];
const tokens = define.trimStart().split(/\s+/);
if (tokens[0] === defineName) {
defines[i] = `${defineName} ${newDefineValue}`;
}
}
}

/**
* Returns the component count for the given class property, or
* its array length if it is an array.
*
* This will be
* `[1, 2, 3, 4]` for `[SCALAR, VEC2, VEC3, VEC4`] types,
* or the array length if it is an array.
*
* @param {MetadataClassProperty} classProperty The class property
* @returns The component count
javagl marked this conversation as resolved.
Show resolved Hide resolved
* @private
*/
function getComponentCount(classProperty) {
if (!classProperty.isArray) {
return MetadataType.getComponentCount(classProperty.type);
}
return classProperty.arrayLength;
}

/**
* Returns the type that the given class property has in a GLSL shader.
*
* It returns the same string as `PropertyTextureProperty.prototype.getGlslType`
* for a property texture property with the given class property
*
* @param {MetadataClassProperty} classProperty The class property
* @returns The GLSL shader type string for the property
javagl marked this conversation as resolved.
Show resolved Hide resolved
*/
function getGlslType(classProperty) {
const componentCount = getComponentCount(classProperty);
if (classProperty.normalized) {
if (componentCount === 1) {
return "float";
}
return `vec${componentCount}`;
}
if (componentCount === 1) {
return "int";
}
return `ivec${componentCount}`;
}

/**
* Creates a new `ShaderProgram` from the given input that renders metadata
* values into the frame buffer, according to the given picked metadata info.
*
* This will update the `defines` of the fragment shader of the given shader
* program, by setting `METADATA_PICKING_ENABLED`, and updating the
* `METADATA_PICKING_VALUE_*` defines so that they reflect the components
* of the metadata that should be written into the RGBA (vec4) that
* ends up as the 'color' in the frame buffer.
*
* The RGBA values will eventually be converted back into an actual metadata
* value in `Picking.js`, by calling `MetadataPicking.decodeMetadataValues`.
*
* @param {Context} context The context
* @param {ShaderProgram} shaderProgram The shader program
* @param {PickedMetadataInfo} pickedMetadataInfo The picked metadata info
* @returns The new shader program
javagl marked this conversation as resolved.
Show resolved Hide resolved
* @private
*/
function getPickMetadataShaderProgram(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This would be easier to read with some types in the doc. I'm assuming shaderProgram is a ShaderProgram ? Some confirmation would help me read it more confidently & quickly.

Copy link
Contributor Author

@javagl javagl Aug 18, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would a @see getPickShaderProgram be enough here? 😁

Seriously: I'm a HUGE fan of the "Boy Scout Rule" of programming:

Leave your code better than you found it.

Derived from the rule "Leave the campground cleaner that you found it". Now one could argue in how far this applies when boy scouts visit something that isn't a "campground" to begin with.

However, I added the information that the shaderProgram is a ShaderProgram. (This should be self-evident, but isn't (hint: a drawCommand is not always a DrawCommand)). I'd like to explain more, but explaining requires a depth of understanding that I won't claim to have.

context,
shaderProgram,
pickedMetadataInfo
) {
const schemaId = pickedMetadataInfo.schemaId;
const className = pickedMetadataInfo.className;
const propertyName = pickedMetadataInfo.propertyName;
const keyword = `pickMetadata-${schemaId}-${className}-${propertyName}`;
const shader = context.shaderCache.getDerivedShaderProgram(
shaderProgram,
keyword
);
if (defined(shader)) {
return shader;
}

const classProperty = pickedMetadataInfo.classProperty;
const glslType = getGlslType(classProperty);

// Define the components that will go into the output `metadataValues`.
// By default, all of them are 0.0.
const sourceValueStrings = ["0.0", "0.0", "0.0", "0.0"];
const componentCount = getComponentCount(classProperty);
if (componentCount === 1) {
// When the property is a scalar, store its value directly
// in `metadataValues.x`
sourceValueStrings[0] = `float(value)`;
} else {
// When the property is an array, store the array elements
// in `metadataValues.x/y/z/w`
const components = ["x", "y", "z", "w"];
for (let i = 0; i < componentCount; i++) {
const component = components[i];
const valueString = `value.${component}`;
sourceValueStrings[i] = `float(${valueString})`;
}
}

// Make sure that the `metadataValues` components are all in
// the range [0, 1] (which will result in RGBA components
// in [0, 255] during rendering)
if (!classProperty.normalized) {
for (let i = 0; i < componentCount; i++) {
sourceValueStrings[i] += " / 255.0";
}
}

const newDefines = shaderProgram.fragmentShaderSource.defines.slice();
newDefines.push(MetadataPickingPipelineStage.METADATA_PICKING_ENABLED);

// Replace the defines of the shader, using the type, property
// access, and value components that have been determined
replaceDefine(
newDefines,
MetadataPickingPipelineStage.METADATA_PICKING_VALUE_TYPE,
glslType
);
replaceDefine(
newDefines,
MetadataPickingPipelineStage.METADATA_PICKING_VALUE_STRING,
`metadata.${propertyName}`
);
replaceDefine(
newDefines,
MetadataPickingPipelineStage.METADATA_PICKING_VALUE_COMPONENT_X,
sourceValueStrings[0]
);
replaceDefine(
newDefines,
MetadataPickingPipelineStage.METADATA_PICKING_VALUE_COMPONENT_Y,
sourceValueStrings[1]
);
replaceDefine(
newDefines,
MetadataPickingPipelineStage.METADATA_PICKING_VALUE_COMPONENT_Z,
sourceValueStrings[2]
);
replaceDefine(
newDefines,
MetadataPickingPipelineStage.METADATA_PICKING_VALUE_COMPONENT_W,
sourceValueStrings[3]
);

const newFragmentShaderSource = new ShaderSource({
sources: shaderProgram.fragmentShaderSource.sources,
defines: newDefines,
});
const newShader = context.shaderCache.createDerivedShaderProgram(
shaderProgram,
keyword,
{
vertexShaderSource: shaderProgram.vertexShaderSource,
fragmentShaderSource: newFragmentShaderSource,
attributeLocations: shaderProgram._attributeLocations,
}
);
return newShader;
}

/**
* @private
*/
DerivedCommand.createPickMetadataDerivedCommand = function (
scene,
command,
context,
result
) {
if (!defined(result)) {
result = {};
}
result.pickMetadataCommand = DrawCommand.shallowClone(
command,
result.pickMetadataCommand
);

result.pickMetadataCommand.shaderProgram = getPickMetadataShaderProgram(
context,
command.shaderProgram,
command.pickedMetadataInfo
);
result.pickMetadataCommand.renderState = getPickRenderState(
scene,
command.renderState
);
result.shaderProgramId = command.shaderProgram.id;

return result;
};

function getHdrShaderProgram(context, shaderProgram) {
let shader = context.shaderCache.getDerivedShaderProgram(
shaderProgram,
Expand Down
31 changes: 31 additions & 0 deletions packages/engine/Source/Scene/FrameState.js
Original file line number Diff line number Diff line change
Expand Up @@ -421,6 +421,37 @@ function FrameState(context, creditDisplay, jobScheduler) {
* @default 0.0
*/
this.minimumTerrainHeight = 0.0;

/**
* Whether metadata picking is currently in progress.
*
* This is set to `true` in the `Picking.pickMetadata` function,
* immediately before updating and executing the draw commands,
* and set back to `false` immediately afterwards. It will be
* used to determine whether the metadata picking draw commands
* should be executed, in the `Scene.executeCommand` function.
*
* @type {boolean}
* @default false
*/
this.pickingMetadata = false;

/**
* Metadata picking information.
*
* This describes the metadata property that is supposed to be picked
* in a `Picking.pickMetadata` call.
*
* This is stored in the frame state and in the metadata picking draw
* commands. In the `Scene.updateDerivedCommands` call, it will be
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What if a refactor of Scene makes these details obsolete? Would it be safer to be more general here?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd love these details to become obsolete 🙂 One could accuse me of overshooting with this comment. But ... to be honest: I did spend a considerable amount of time with reading code that I should not have to read. Figuring out what scene3DOnly is (beyond doing a full-text search and reading, reading, reading) is not so much fun. There's a light in the FrameState. Is this even used anywhere? (I don't think so...). So ... we could open an issue to just clean it up and add documentation, but that would hardly be considered something actionable.

If you insist, I remove the comments (or replace them with /** @private */ 😶 )

* checked whether the instance that is stored in the frame state
* is different from the one in the draw command, and if necessary,
* the derived commands for metadata picking will be updated based
* on this information.
*
* @type {PickedMetadataInfo|undefined}
*/
this.pickedMetadataInfo = undefined;
}

/**
Expand Down
Loading