molstar
Version:
A comprehensive macromolecular library.
526 lines (525 loc) • 26.3 kB
JavaScript
/**
* Copyright (c) 2018-2024 mol* contributors, licensed under MIT, See LICENSE file for more info.
*
* @author Alexander Rose <alexander.rose@weirdbyte.de>
* @author Gianluca Tomasello <giagitom@gmail.com>
*/
Object.defineProperty(exports, "__esModule", { value: true });
exports.UnitsImageParams = exports.UnitsTextureMeshParams = exports.UnitsDirectVolumeParams = exports.UnitsTextParams = exports.UnitsLinesParams = exports.UnitsPointsParams = exports.UnitsCylindersParams = exports.UnitsSpheresParams = exports.UnitsMeshParams = void 0;
exports.UnitsVisual = UnitsVisual;
exports.UnitsMeshVisual = UnitsMeshVisual;
exports.UnitsSpheresVisual = UnitsSpheresVisual;
exports.UnitsCylindersVisual = UnitsCylindersVisual;
exports.UnitsPointsVisual = UnitsPointsVisual;
exports.UnitsLinesVisual = UnitsLinesVisual;
exports.UnitsTextVisual = UnitsTextVisual;
exports.UnitsDirectVolumeVisual = UnitsDirectVolumeVisual;
exports.UnitsTextureMeshVisual = UnitsTextureMeshVisual;
exports.UnitsImageVisual = UnitsImageVisual;
const structure_1 = require("../../mol-model/structure");
const visual_1 = require("../visual");
const geometry_1 = require("../../mol-geo/geometry/geometry");
const theme_1 = require("../../mol-theme/theme");
const common_1 = require("./visual/util/common");
const render_object_1 = require("../../mol-gl/render-object");
const loci_1 = require("../../mol-model/loci");
const int_1 = require("../../mol-data/int");
const util_1 = require("../util");
const color_1 = require("../../mol-theme/color");
const marker_data_1 = require("../../mol-geo/geometry/marker-data");
const marker_action_1 = require("../../mol-util/marker-action");
const mol_util_1 = require("../../mol-util");
const size_data_1 = require("../../mol-geo/geometry/size-data");
const color_data_1 = require("../../mol-geo/geometry/color-data");
const mesh_1 = require("../../mol-geo/geometry/mesh/mesh");
const size_1 = require("../../mol-theme/size");
const spheres_1 = require("../../mol-geo/geometry/spheres/spheres");
const cylinders_1 = require("../../mol-geo/geometry/cylinders/cylinders");
const points_1 = require("../../mol-geo/geometry/points/points");
const lines_1 = require("../../mol-geo/geometry/lines/lines");
const text_1 = require("../../mol-geo/geometry/text/text");
const direct_volume_1 = require("../../mol-geo/geometry/direct-volume/direct-volume");
const texture_mesh_1 = require("../../mol-geo/geometry/texture-mesh/texture-mesh");
const image_1 = require("../../mol-geo/geometry/image/image");
const params_1 = require("./params");
const type_helpers_1 = require("../../mol-util/type-helpers");
function createUnitsRenderObject(structureGroup, geometry, locationIt, theme, props, materialId) {
const { createValues, createRenderableState } = geometry_1.Geometry.getUtils(geometry);
const transform = (0, common_1.createUnitsTransform)(structureGroup, props.includeParent, geometry.boundingSphere, props.cellSize, props.batchSize);
const values = createValues(geometry, transform, locationIt, theme, props);
const state = createRenderableState(props);
return (0, render_object_1.createRenderObject)(geometry.kind, values, state, materialId);
}
function UnitsVisual(builder, materialId) {
const { defaultProps, createGeometry, createLocationIterator, getLoci, eachLocation, setUpdateState, initUpdateState, mustRecreate, processValues, dispose } = builder;
const { createEmpty: createEmptyGeometry, updateValues, updateBoundingSphere, updateRenderableState, createPositionIterator } = builder.geometryUtils;
const updateState = util_1.VisualUpdateState.create();
const previousMark = { loci: loci_1.EmptyLoci, action: marker_action_1.MarkerAction.None, status: -1 };
let renderObject;
let newProps = Object.assign({}, defaultProps);
let newTheme = theme_1.Theme.createEmpty();
let newStructureGroup;
let currentProps;
let currentTheme;
let currentStructureGroup;
let geometry;
let geometryVersion = -1;
let locationIt;
let positionIt;
function prepareUpdate(theme, props, structureGroup) {
if (!structureGroup && !currentStructureGroup) {
throw new Error('missing structureGroup');
}
newProps = props;
newTheme = theme;
newStructureGroup = structureGroup;
util_1.VisualUpdateState.reset(updateState);
if (!renderObject || !currentStructureGroup) {
initUpdateState === null || initUpdateState === void 0 ? void 0 : initUpdateState(updateState, newProps, newTheme, newStructureGroup);
// console.log('create new');
updateState.createNew = true;
updateState.createGeometry = true;
return;
}
setUpdateState(updateState, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup);
if (!structure_1.Structure.areHierarchiesEqual(currentStructureGroup.structure, newStructureGroup.structure)) {
// console.log('new hierarchy');
updateState.updateTransform = true;
updateState.updateColor = true;
updateState.updateSize = true;
}
if (!color_1.ColorTheme.areEqual(newTheme.color, currentTheme.color)) {
// console.log('new colorTheme');
updateState.updateColor = true;
}
if (!size_1.SizeTheme.areEqual(newTheme.size, currentTheme.size)) {
// console.log('new sizeTheme');
updateState.updateSize = true;
}
if (currentStructureGroup.structure.child !== newStructureGroup.structure.child) {
// console.log('new child');
updateState.createGeometry = true;
updateState.updateTransform = true;
}
if (newProps.instanceGranularity !== currentProps.instanceGranularity || newProps.cellSize !== currentProps.cellSize || newProps.batchSize !== currentProps.batchSize) {
updateState.updateTransform = true;
}
if (!(0, mol_util_1.deepEqual)(newProps.unitKinds, currentProps.unitKinds)) {
// console.log('new unitKinds');
updateState.createGeometry = true;
}
if (newStructureGroup.group.transformHash !== currentStructureGroup.group.transformHash) {
// console.log('new transformHash');
if (newStructureGroup.group.units.length !== currentStructureGroup.group.units.length || updateState.updateColor) {
updateState.updateTransform = true;
}
else {
updateState.updateMatrix = true;
}
}
// check if the operator or conformation of unit has changed
const newUnit = newStructureGroup.group.units[0];
const currentUnit = currentStructureGroup.group.units[0];
if (!structure_1.Unit.areOperatorsEqual(newUnit, currentUnit)) {
// console.log('new operators');
updateState.updateTransform = true;
}
if (!structure_1.Unit.areConformationsEqual(newUnit, currentUnit)) {
// console.log('new conformation');
updateState.createGeometry = true;
}
if (updateState.updateTransform) {
updateState.updateMatrix = true;
}
if (updateState.updateSize && !('uSize' in renderObject.values)) {
updateState.createGeometry = true;
}
if (updateState.createGeometry || updateState.updateTransform) {
if (currentStructureGroup.structure.hashCode !== newStructureGroup.structure.hashCode) {
// console.log('new hashCode');
updateState.updateColor = true;
updateState.updateSize = true;
}
if (newTheme.color.granularity.startsWith('vertex') ||
renderObject.values.dColorType.ref.value.startsWith('vertex') ||
newTheme.color.granularity.startsWith('volume') ||
renderObject.values.dColorType.ref.value.startsWith('volume')) {
updateState.updateColor = true;
}
}
}
function update(newGeometry) {
if (updateState.createNew) {
locationIt = createLocationIterator(newStructureGroup, newProps);
if (newGeometry) {
renderObject = createUnitsRenderObject(newStructureGroup, newGeometry, locationIt, newTheme, newProps, materialId);
positionIt = createPositionIterator(newGeometry, renderObject.values);
}
else {
throw new Error('expected geometry to be given');
}
}
else {
if (!renderObject) {
throw new Error('expected renderObject to be available');
}
if (updateState.updateColor || updateState.updateSize || updateState.updateTransform) {
// console.log('update locationIterator');
locationIt = createLocationIterator(newStructureGroup, newProps);
}
if (updateState.updateTransform) {
// console.log('update transform');
const { instanceCount, groupCount } = locationIt;
if (newProps.instanceGranularity) {
(0, marker_data_1.createMarkers)(instanceCount, 'instance', renderObject.values);
}
else {
(0, marker_data_1.createMarkers)(instanceCount * groupCount, 'groupInstance', renderObject.values);
}
}
if (updateState.updateMatrix) {
// console.log('update matrix');
(0, common_1.createUnitsTransform)(newStructureGroup, newProps.includeParent, (newGeometry === null || newGeometry === void 0 ? void 0 : newGeometry.boundingSphere) || renderObject.values.invariantBoundingSphere.ref.value, newProps.cellSize, newProps.batchSize, renderObject.values);
if ('lodLevels' in renderObject.values) {
// to trigger `uLod` update in `renderable.cull`
mol_util_1.ValueCell.update(renderObject.values.lodLevels, renderObject.values.lodLevels.ref.value);
}
}
if (updateState.createGeometry) {
// console.log('update geometry');
if (newGeometry) {
mol_util_1.ValueCell.updateIfChanged(renderObject.values.drawCount, geometry_1.Geometry.getDrawCount(newGeometry));
mol_util_1.ValueCell.updateIfChanged(renderObject.values.uVertexCount, geometry_1.Geometry.getVertexCount(newGeometry));
mol_util_1.ValueCell.updateIfChanged(renderObject.values.uGroupCount, geometry_1.Geometry.getGroupCount(newGeometry));
}
else {
throw new Error('expected geometry to be given');
}
}
if (updateState.updateTransform || updateState.createGeometry) {
updateBoundingSphere(renderObject.values, newGeometry || geometry);
positionIt = createPositionIterator(newGeometry || geometry, renderObject.values);
}
if (updateState.updateSize) {
// not all geometries have size data, so check here
if ('uSize' in renderObject.values) {
// console.log('update size');
(0, size_data_1.createSizes)(locationIt, newTheme.size, renderObject.values);
}
}
if (updateState.updateColor) {
// console.log('update color');
(0, color_data_1.createColors)(locationIt, positionIt, newTheme.color, renderObject.values);
}
updateValues(renderObject.values, newProps);
updateRenderableState(renderObject.state, newProps);
}
currentProps = newProps;
currentTheme = newTheme;
currentStructureGroup = newStructureGroup;
if (newGeometry) {
geometry = newGeometry;
geometryVersion += 1;
}
}
function _createGeometry(ctx, unit, structure, theme, props, geometry) {
return (0, common_1.includesUnitKind)(props.unitKinds, unit)
? createGeometry(ctx, unit, structure, theme, props, geometry)
: createEmptyGeometry(geometry);
}
function lociIsSuperset(loci) {
if ((0, loci_1.isEveryLoci)(loci))
return true;
if (structure_1.Structure.isLoci(loci) && structure_1.Structure.areRootsEquivalent(loci.structure, currentStructureGroup.structure))
return true;
if (structure_1.StructureElement.Loci.is(loci) && structure_1.Structure.areRootsEquivalent(loci.structure, currentStructureGroup.structure)) {
if (structure_1.StructureElement.Loci.isWholeStructure(loci))
return true;
}
return false;
}
function eachInstance(loci, structureGroup, apply) {
let changed = false;
if (structure_1.Bond.isLoci(loci)) {
const { structure, group } = structureGroup;
if (!structure_1.Structure.areEquivalent(loci.structure, structure))
return false;
for (const b of loci.bonds) {
if (b.aUnit !== b.bUnit)
continue;
const unitIdx = group.unitIndexMap.get(b.aUnit.id);
if (unitIdx !== undefined) {
if (apply(int_1.Interval.ofSingleton(unitIdx)))
changed = true;
}
}
}
else if (structure_1.StructureElement.Loci.is(loci)) {
const { structure, group } = structureGroup;
if (!structure_1.Structure.areEquivalent(loci.structure, structure))
return false;
for (const e of loci.elements) {
const unitIdx = group.unitIndexMap.get(e.unit.id);
if (unitIdx !== undefined) {
if (apply(int_1.Interval.ofSingleton(unitIdx)))
changed = true;
}
}
}
return changed;
}
function lociApply(loci, apply, isMarking) {
if (lociIsSuperset(loci)) {
if (currentProps.instanceGranularity) {
return apply(int_1.Interval.ofBounds(0, locationIt.instanceCount));
}
else {
return apply(int_1.Interval.ofBounds(0, locationIt.groupCount * locationIt.instanceCount));
}
}
else {
if (currentProps.instanceGranularity) {
return eachInstance(loci, currentStructureGroup, apply);
}
else {
return eachLocation(loci, currentStructureGroup, apply, isMarking);
}
}
}
function finalize(ctx) {
if (renderObject) {
processValues === null || processValues === void 0 ? void 0 : processValues(renderObject.values, geometry, currentProps, currentTheme, ctx.webgl);
}
}
return {
get groupCount() { return locationIt ? locationIt.count : 0; },
get renderObject() { return locationIt && locationIt.count ? renderObject : undefined; },
get geometryVersion() { return geometryVersion; },
createOrUpdate(ctx, theme, props, structureGroup) {
prepareUpdate(theme, props, structureGroup || currentStructureGroup);
if (updateState.createGeometry) {
const newGeometry = _createGeometry(ctx, newStructureGroup.group.units[0], newStructureGroup.structure, newTheme, newProps, geometry);
if ((0, type_helpers_1.isPromiseLike)(newGeometry)) {
return newGeometry.then(g => {
update(g);
finalize(ctx);
});
}
update(newGeometry);
}
else {
update();
}
finalize(ctx);
},
getLoci(pickingId) {
return renderObject ? getLoci(pickingId, currentStructureGroup, renderObject.id) : loci_1.EmptyLoci;
},
eachLocation(cb) {
locationIt.reset();
while (locationIt.hasNext) {
const { location, isSecondary } = locationIt.move();
cb(location, isSecondary);
}
},
mark(loci, action) {
let hasInvariantId = true;
if (structure_1.StructureElement.Loci.is(loci)) {
hasInvariantId = false;
const { invariantId } = currentStructureGroup.group.units[0];
for (const e of loci.elements) {
if (e.unit.invariantId === invariantId) {
hasInvariantId = true;
break;
}
}
}
return hasInvariantId ? visual_1.Visual.mark(renderObject, loci, action, lociApply, previousMark) : false;
},
setVisibility(visible) {
visual_1.Visual.setVisibility(renderObject, visible);
},
setAlphaFactor(alphaFactor) {
visual_1.Visual.setAlphaFactor(renderObject, alphaFactor);
},
setPickable(pickable) {
visual_1.Visual.setPickable(renderObject, pickable);
},
setColorOnly(colorOnly) {
visual_1.Visual.setColorOnly(renderObject, colorOnly);
},
setTransform(matrix, instanceMatrices) {
visual_1.Visual.setTransform(renderObject, matrix, instanceMatrices);
},
setOverpaint(overpaint, webgl) {
const smoothing = { geometry, props: currentProps, webgl };
visual_1.Visual.setOverpaint(renderObject, overpaint, lociApply, true, smoothing);
},
setTransparency(transparency, webgl) {
const smoothing = { geometry, props: currentProps, webgl };
visual_1.Visual.setTransparency(renderObject, transparency, lociApply, true, smoothing);
},
setEmissive(emissive, webgl) {
const smoothing = { geometry, props: currentProps, webgl };
visual_1.Visual.setEmissive(renderObject, emissive, lociApply, true, smoothing);
},
setSubstance(substance, webgl) {
const smoothing = { geometry, props: currentProps, webgl };
visual_1.Visual.setSubstance(renderObject, substance, lociApply, true, smoothing);
},
setClipping(clipping) {
visual_1.Visual.setClipping(renderObject, clipping, lociApply, true);
},
setThemeStrength(strength) {
visual_1.Visual.setThemeStrength(renderObject, strength);
},
destroy() {
dispose === null || dispose === void 0 ? void 0 : dispose(geometry);
if (renderObject) {
renderObject.state.disposed = true;
renderObject = undefined;
}
},
mustRecreate
};
}
// mesh
exports.UnitsMeshParams = { ...params_1.StructureMeshParams, ...params_1.StructureParams };
function UnitsMeshVisual(builder, materialId) {
return UnitsVisual({
...builder,
setUpdateState: (state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup) => {
builder.setUpdateState(state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup);
if (!size_1.SizeTheme.areEqual(newTheme.size, currentTheme.size))
state.createGeometry = true;
},
geometryUtils: mesh_1.Mesh.Utils
}, materialId);
}
// spheres
exports.UnitsSpheresParams = { ...params_1.StructureSpheresParams, ...params_1.StructureParams };
function UnitsSpheresVisual(builder, materialId) {
return UnitsVisual({
...builder,
setUpdateState: (state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup) => {
builder.setUpdateState(state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup);
if (!size_1.SizeTheme.areEqual(newTheme.size, currentTheme.size))
state.updateSize = true;
},
geometryUtils: spheres_1.Spheres.Utils
}, materialId);
}
// cylinders
exports.UnitsCylindersParams = { ...params_1.StructureCylindersParams, ...params_1.StructureParams };
function UnitsCylindersVisual(builder, materialId) {
return UnitsVisual({
...builder,
setUpdateState: (state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup) => {
builder.setUpdateState(state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup);
if (!size_1.SizeTheme.areEqual(newTheme.size, currentTheme.size))
state.updateSize = true;
},
geometryUtils: cylinders_1.Cylinders.Utils
}, materialId);
}
// points
exports.UnitsPointsParams = { ...params_1.StructurePointsParams, ...params_1.StructureParams };
function UnitsPointsVisual(builder, materialId) {
return UnitsVisual({
...builder,
setUpdateState: (state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup) => {
builder.setUpdateState(state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup);
if (!size_1.SizeTheme.areEqual(newTheme.size, currentTheme.size))
state.updateSize = true;
},
geometryUtils: points_1.Points.Utils
}, materialId);
}
// lines
exports.UnitsLinesParams = { ...params_1.StructureLinesParams, ...params_1.StructureParams };
function UnitsLinesVisual(builder, materialId) {
return UnitsVisual({
...builder,
setUpdateState: (state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup) => {
builder.setUpdateState(state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup);
if (!size_1.SizeTheme.areEqual(newTheme.size, currentTheme.size))
state.updateSize = true;
},
geometryUtils: lines_1.Lines.Utils
}, materialId);
}
// text
exports.UnitsTextParams = { ...params_1.StructureTextParams, ...params_1.StructureParams };
function UnitsTextVisual(builder, materialId) {
return UnitsVisual({
...builder,
setUpdateState: (state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup) => {
builder.setUpdateState(state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup);
if (!size_1.SizeTheme.areEqual(newTheme.size, currentTheme.size))
state.updateSize = true;
if (newProps.background !== currentProps.background)
state.createGeometry = true;
if (newProps.backgroundMargin !== currentProps.backgroundMargin)
state.createGeometry = true;
if (newProps.tether !== currentProps.tether)
state.createGeometry = true;
if (newProps.tetherLength !== currentProps.tetherLength)
state.createGeometry = true;
if (newProps.tetherBaseWidth !== currentProps.tetherBaseWidth)
state.createGeometry = true;
if (newProps.attachment !== currentProps.attachment)
state.createGeometry = true;
if (newProps.fontFamily !== currentProps.fontFamily)
state.createGeometry = true;
if (newProps.fontQuality !== currentProps.fontQuality)
state.createGeometry = true;
if (newProps.fontStyle !== currentProps.fontStyle)
state.createGeometry = true;
if (newProps.fontVariant !== currentProps.fontVariant)
state.createGeometry = true;
if (newProps.fontWeight !== currentProps.fontWeight)
state.createGeometry = true;
},
geometryUtils: text_1.Text.Utils
}, materialId);
}
// direct-volume
exports.UnitsDirectVolumeParams = { ...params_1.StructureDirectVolumeParams, ...params_1.StructureParams };
function UnitsDirectVolumeVisual(builder, materialId) {
return UnitsVisual({
...builder,
setUpdateState: (state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup) => {
builder.setUpdateState(state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup);
if (!size_1.SizeTheme.areEqual(newTheme.size, currentTheme.size))
state.createGeometry = true;
},
geometryUtils: direct_volume_1.DirectVolume.Utils
}, materialId);
}
// texture-mesh
exports.UnitsTextureMeshParams = { ...params_1.StructureTextureMeshParams, ...params_1.StructureParams };
function UnitsTextureMeshVisual(builder, materialId) {
return UnitsVisual({
...builder,
setUpdateState: (state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup) => {
builder.setUpdateState(state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup);
if (!size_1.SizeTheme.areEqual(newTheme.size, currentTheme.size))
state.createGeometry = true;
},
geometryUtils: texture_mesh_1.TextureMesh.Utils
}, materialId);
}
// image
exports.UnitsImageParams = { ...params_1.StructureImageParams, ...params_1.StructureParams };
function UnitsImageVisual(builder, materialId) {
return UnitsVisual({
...builder,
setUpdateState: (state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup) => {
builder.setUpdateState(state, newProps, currentProps, newTheme, currentTheme, newStructureGroup, currentStructureGroup);
if (!size_1.SizeTheme.areEqual(newTheme.size, currentTheme.size))
state.createGeometry = true;
},
geometryUtils: image_1.Image.Utils
}, materialId);
}
;