UNPKG

molstar

Version:

A comprehensive macromolecular library.

126 lines (125 loc) 6.39 kB
/** * Copyright (c) 2020 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author Alexander Rose <alexander.rose@weirdbyte.de> * @author Sebastian Bittrich <sebastian.bittrich@rcsb.org> */ import { ParamDefinition as PD } from '../../mol-util/param-definition'; import { Vec3, Mat4 } from '../../mol-math/linear-algebra'; import { Representation } from '../../mol-repr/representation'; import { Structure } from '../../mol-model/structure'; import { StructureRepresentationProvider, StructureRepresentationStateBuilder } from '../../mol-repr/structure/representation'; import { ShapeRepresentation } from '../../mol-repr/shape/representation'; import { Shape } from '../../mol-model/shape'; import { Lines } from '../../mol-geo/geometry/lines/lines'; import { Mesh } from '../../mol-geo/geometry/mesh/mesh'; import { LinesBuilder } from '../../mol-geo/geometry/lines/lines-builder'; import { Circle } from '../../mol-geo/primitive/circle'; import { transformPrimitive } from '../../mol-geo/primitive/primitive'; import { MeshBuilder } from '../../mol-geo/geometry/mesh/mesh-builder'; import { MembraneOrientationProvider } from './prop'; import { MarkerActions } from '../../mol-util/marker-action'; import { lociLabel } from '../../mol-theme/label'; import { ColorNames } from '../../mol-util/color/names'; const SharedParams = { color: PD.Color(ColorNames.lightgrey), radiusFactor: PD.Numeric(1.2, { min: 0.1, max: 3.0, step: 0.01 }, { description: 'Scale the radius of the membrane layer' }) }; const BilayerPlanesParams = { ...Mesh.Params, ...SharedParams, sectorOpacity: PD.Numeric(0.5, { min: 0, max: 1, step: 0.01 }), }; const BilayerRimsParams = { ...Lines.Params, ...SharedParams, lineSizeAttenuation: PD.Boolean(false), linesSize: PD.Numeric(0.5, { min: 0.01, max: 50, step: 0.01 }), dashedLines: PD.Boolean(false), }; const MembraneOrientationVisuals = { 'bilayer-planes': (ctx, getParams) => ShapeRepresentation(getBilayerPlanes, Mesh.Utils, { modifyState: s => ({ ...s, markerActions: MarkerActions.Highlighting }), modifyProps: p => ({ ...p, alpha: p.sectorOpacity, ignoreLight: true, doubleSided: false }) }), 'bilayer-rims': (ctx, getParams) => ShapeRepresentation(getBilayerRims, Lines.Utils, { modifyState: s => ({ ...s, markerActions: MarkerActions.Highlighting }) }) }; export const MembraneOrientationParams = { ...BilayerPlanesParams, ...BilayerRimsParams, visuals: PD.MultiSelect(['bilayer-planes', 'bilayer-rims'], PD.objectToOptions(MembraneOrientationVisuals)), }; export function getMembraneOrientationParams(ctx, structure) { return PD.clone(MembraneOrientationParams); } export function MembraneOrientationRepresentation(ctx, getParams) { return Representation.createMulti('Membrane Orientation', ctx, getParams, StructureRepresentationStateBuilder, MembraneOrientationVisuals); } export const MembraneOrientationRepresentationProvider = StructureRepresentationProvider({ name: 'membrane-orientation', label: 'Membrane Orientation', description: 'Displays a grid of points representing membrane layers.', factory: MembraneOrientationRepresentation, getParams: getMembraneOrientationParams, defaultValues: PD.getDefaultValues(MembraneOrientationParams), defaultColorTheme: { name: 'shape-group' }, defaultSizeTheme: { name: 'shape-group' }, isApplicable(structure) { return MembraneOrientationProvider.isApplicable(structure); }, ensureCustomProperties: { attach: (ctx, structure) => MembraneOrientationProvider.attach(ctx, structure, void 0, true), detach: (data) => MembraneOrientationProvider.ref(data, false) } }); function membraneLabel(data) { return `${lociLabel(Structure.Loci(data))} | Membrane Orientation`; } function getBilayerRims(ctx, data, props, shape) { const { planePoint1: p1, planePoint2: p2, centroid, radius } = MembraneOrientationProvider.get(data).value; const scaledRadius = props.radiusFactor * radius; const builder = LinesBuilder.create(128, 64, shape === null || shape === void 0 ? void 0 : shape.geometry); getLayerCircle(builder, p1, centroid, scaledRadius, props); getLayerCircle(builder, p2, centroid, scaledRadius, props); return Shape.create('Bilayer rims', data, builder.getLines(), () => props.color, () => props.linesSize, () => membraneLabel(data)); } function getLayerCircle(builder, p, centroid, radius, props, shape) { const circle = getCircle(p, centroid, radius); const { indices, vertices } = circle; for (let j = 0, jl = indices.length; j < jl; j += 3) { if (props.dashedLines && j % 2 === 1) continue; // draw every other segment to get dashes const start = indices[j] * 3; const end = indices[j + 1] * 3; const startX = vertices[start]; const startY = vertices[start + 1]; const startZ = vertices[start + 2]; const endX = vertices[end]; const endY = vertices[end + 1]; const endZ = vertices[end + 2]; builder.add(startX, startY, startZ, endX, endY, endZ, 0); } } const tmpMat = Mat4(); const tmpV = Vec3(); function getCircle(p, centroid, radius) { if (Vec3.dot(Vec3.unitY, Vec3.sub(tmpV, p, centroid)) === 0) { Mat4.targetTo(tmpMat, p, centroid, Vec3.unitY); } else { Mat4.targetTo(tmpMat, p, centroid, Vec3.unitX); } Mat4.setTranslation(tmpMat, p); Mat4.mul(tmpMat, tmpMat, Mat4.rotX90); const circle = Circle({ radius, segments: 64 }); return transformPrimitive(circle, tmpMat); } function getBilayerPlanes(ctx, data, props, shape) { const { planePoint1: p1, planePoint2: p2, centroid, radius } = MembraneOrientationProvider.get(data).value; const state = MeshBuilder.createState(128, 64, shape && shape.geometry); const scaledRadius = props.radiusFactor * radius; getLayerPlane(state, p1, centroid, scaledRadius); getLayerPlane(state, p2, centroid, scaledRadius); return Shape.create('Bilayer planes', data, MeshBuilder.getMesh(state), () => props.color, () => 1, () => membraneLabel(data)); } function getLayerPlane(state, p, centroid, radius) { const circle = getCircle(p, centroid, radius); state.currentGroup = 0; MeshBuilder.addPrimitive(state, Mat4.id, circle); MeshBuilder.addPrimitiveFlipped(state, Mat4.id, circle); }