|
@@ -8,7 +8,7 @@ import { RuntimeContext } from 'mol-task'
|
|
|
import { GraphicsRenderObject } from 'mol-gl/render-object'
|
|
|
import { PickingId } from '../mol-geo/geometry/picking';
|
|
|
import { Loci } from 'mol-model/loci';
|
|
|
-import { MarkerAction } from '../mol-geo/geometry/marker-data';
|
|
|
+import { MarkerAction, applyMarkerAction } from '../mol-geo/geometry/marker-data';
|
|
|
import { ParamDefinition as PD } from 'mol-util/param-definition';
|
|
|
import { WebGLContext } from 'mol-gl/webgl/context';
|
|
|
import { Theme } from 'mol-theme/theme';
|
|
@@ -17,6 +17,8 @@ import { updateTransformData, fillIdentityTransform } from 'mol-geo/geometry/tra
|
|
|
import { calculateTransformBoundingSphere } from 'mol-gl/renderable/util';
|
|
|
import { ValueCell } from 'mol-util';
|
|
|
import { Overpaint } from 'mol-theme/overpaint';
|
|
|
+import { createOverpaint, clearOverpaint, applyOverpaintColor } from 'mol-geo/geometry/overpaint-data';
|
|
|
+import { Interval } from 'mol-data/int';
|
|
|
|
|
|
export interface VisualContext {
|
|
|
readonly runtime: RuntimeContext
|
|
@@ -36,10 +38,12 @@ interface Visual<D, P extends PD.Params> {
|
|
|
setAlphaFactor: (alphaFactor: number) => void
|
|
|
setPickable: (pickable: boolean) => void
|
|
|
setTransform: (matrix?: Mat4, instanceMatrices?: Float32Array | null) => void
|
|
|
- setOverpaint: (layers: Overpaint.Layers, clear?: boolean) => void
|
|
|
+ setOverpaint: (overpaint: Overpaint) => void
|
|
|
destroy: () => void
|
|
|
}
|
|
|
namespace Visual {
|
|
|
+ export type LociApply = (loci: Loci, apply: (interval: Interval) => boolean) => boolean
|
|
|
+
|
|
|
export function setVisibility(renderObject: GraphicsRenderObject | undefined, visible: boolean) {
|
|
|
if (renderObject) renderObject.state.visible = visible
|
|
|
}
|
|
@@ -52,23 +56,63 @@ namespace Visual {
|
|
|
if (renderObject) renderObject.state.pickable = pickable
|
|
|
}
|
|
|
|
|
|
- export function setTransform(renderObject: GraphicsRenderObject | undefined, transform?: Mat4, instanceTransforms?: Float32Array | null) {
|
|
|
- if (renderObject && (transform || instanceTransforms)) {
|
|
|
- const { values } = renderObject
|
|
|
- if (transform) {
|
|
|
- Mat4.copy(values.matrix.ref.value, transform)
|
|
|
- ValueCell.update(values.matrix, values.matrix.ref.value)
|
|
|
- }
|
|
|
- if (instanceTransforms) {
|
|
|
- values.extraTransform.ref.value.set(instanceTransforms)
|
|
|
- ValueCell.update(values.extraTransform, values.extraTransform.ref.value)
|
|
|
- } else if (instanceTransforms === null) {
|
|
|
- fillIdentityTransform(values.extraTransform.ref.value, values.instanceCount.ref.value)
|
|
|
- ValueCell.update(values.extraTransform, values.extraTransform.ref.value)
|
|
|
+ export function mark(renderObject: GraphicsRenderObject | undefined, loci: Loci, action: MarkerAction, lociApply: LociApply) {
|
|
|
+ if (!renderObject) return false
|
|
|
+
|
|
|
+ const { tMarker } = renderObject.values
|
|
|
+
|
|
|
+ function apply(interval: Interval) {
|
|
|
+ const start = Interval.start(interval)
|
|
|
+ const end = Interval.end(interval)
|
|
|
+ return applyMarkerAction(tMarker.ref.value.array, start, end, action)
|
|
|
+ }
|
|
|
+
|
|
|
+ const changed = lociApply(loci, apply)
|
|
|
+ if (changed) ValueCell.update(tMarker, tMarker.ref.value)
|
|
|
+ return changed
|
|
|
+ }
|
|
|
+
|
|
|
+ export function setOverpaint(renderObject: GraphicsRenderObject | undefined, overpaint: Overpaint, lociApply: LociApply, clear: boolean) {
|
|
|
+ if (!renderObject) return
|
|
|
+
|
|
|
+ const { tOverpaint, uGroupCount, instanceCount } = renderObject.values
|
|
|
+ const count = uGroupCount.ref.value * instanceCount.ref.value
|
|
|
+
|
|
|
+ // ensure texture has right size
|
|
|
+ createOverpaint(overpaint.layers.length ? count : 0, renderObject.values)
|
|
|
+
|
|
|
+ // clear if requested
|
|
|
+ if (clear) clearOverpaint(tOverpaint.ref.value.array, 0, count)
|
|
|
+
|
|
|
+ for (let i = 0, il = overpaint.layers.length; i < il; ++i) {
|
|
|
+ const { loci, color } = overpaint.layers[i]
|
|
|
+ const apply = (interval: Interval) => {
|
|
|
+ const start = Interval.start(interval)
|
|
|
+ const end = Interval.end(interval)
|
|
|
+ return applyOverpaintColor(tOverpaint.ref.value.array, start, end, color, overpaint.alpha)
|
|
|
}
|
|
|
- updateTransformData(values)
|
|
|
- const boundingSphere = calculateTransformBoundingSphere(values.invariantBoundingSphere.ref.value, values.aTransform.ref.value, values.instanceCount.ref.value)
|
|
|
- ValueCell.update(values.boundingSphere, boundingSphere)
|
|
|
+ lociApply(loci, apply)
|
|
|
+ }
|
|
|
+ ValueCell.update(tOverpaint, tOverpaint.ref.value)
|
|
|
+ }
|
|
|
+
|
|
|
+ export function setTransform(renderObject: GraphicsRenderObject | undefined, transform?: Mat4, instanceTransforms?: Float32Array | null) {
|
|
|
+ if (!renderObject || (!transform && !instanceTransforms)) return
|
|
|
+
|
|
|
+ const { values } = renderObject
|
|
|
+ if (transform) {
|
|
|
+ Mat4.copy(values.matrix.ref.value, transform)
|
|
|
+ ValueCell.update(values.matrix, values.matrix.ref.value)
|
|
|
+ }
|
|
|
+ if (instanceTransforms) {
|
|
|
+ values.extraTransform.ref.value.set(instanceTransforms)
|
|
|
+ ValueCell.update(values.extraTransform, values.extraTransform.ref.value)
|
|
|
+ } else if (instanceTransforms === null) {
|
|
|
+ fillIdentityTransform(values.extraTransform.ref.value, values.instanceCount.ref.value)
|
|
|
+ ValueCell.update(values.extraTransform, values.extraTransform.ref.value)
|
|
|
}
|
|
|
+ updateTransformData(values)
|
|
|
+ const boundingSphere = calculateTransformBoundingSphere(values.invariantBoundingSphere.ref.value, values.aTransform.ref.value, values.instanceCount.ref.value)
|
|
|
+ ValueCell.update(values.boundingSphere, boundingSphere)
|
|
|
}
|
|
|
}
|