UNPKG

molstar

Version:

A comprehensive macromolecular library.

125 lines (124 loc) 5.61 kB
/** * Copyright (c) 2024-2025 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author David Sehnal <david.sehnal@gmail.com> * @author Cai Huiyu <szmun.caihy@gmail.com> * @author Alexander Rose <alexander.rose@weirdbyte.de> */ import { Color, ColorScale } from '../../mol-util/color'; import { ColorTheme } from '../color'; import { ParamDefinition as PD } from '../../mol-util/param-definition'; import { Grid, Volume } from '../../mol-model/volume'; import { isPositionLocation } from '../../mol-geo/util/location-iterator'; import { Vec3 } from '../../mol-math/linear-algebra'; import { clamp } from '../../mol-math/interpolate'; import { ColorThemeCategory } from './categories'; const Description = `Assigns a color based on volume value at a given vertex.`; export const ExternalVolumeColorThemeParams = { volume: PD.ValueRef((ctx) => { const volumes = ctx.state.data.selectQ(q => q.root.subtree().filter(c => { var _a; return Volume.is((_a = c.obj) === null || _a === void 0 ? void 0 : _a.data); })); return volumes.map(v => { var _a, _b; return [v.transform.ref, (_b = (_a = v.obj) === null || _a === void 0 ? void 0 : _a.label) !== null && _b !== void 0 ? _b : '<unknown>']; }); }, (ref, getData) => getData(ref)), coloring: PD.MappedStatic('absolute-value', { 'absolute-value': PD.Group({ domain: PD.MappedStatic('auto', { custom: PD.Interval([-1, 1], { step: 0.001 }), auto: PD.Group({ symmetric: PD.Boolean(false, { description: 'If true the automatic range is determined as [-|max|, |max|].' }) }) }), list: PD.ColorList('red-white-blue', { presetKind: 'scale' }) }), 'relative-value': PD.Group({ domain: PD.MappedStatic('auto', { custom: PD.Interval([-1, 1], { step: 0.001 }), auto: PD.Group({ symmetric: PD.Boolean(false, { description: 'If true the automatic range is determined as [-|max|, |max|].' }) }) }), list: PD.ColorList('red-white-blue', { presetKind: 'scale' }) }) }), defaultColor: PD.Color(Color(0xcccccc)), normalOffset: PD.Numeric(0., { min: 0, max: 20, step: 0.1 }, { description: 'Offset vertex position along its normal by given amount.' }), usePalette: PD.Boolean(false, { description: 'Use a palette to color at the pixel level.' }), }; export function ExternalVolumeColorTheme(ctx, props) { let volume; try { volume = props.volume.getValue(); } catch (_a) { // .getValue() is resolved during state reconciliation => would throw from UI } // NOTE: this will currently be slow for with GPU/texture meshes due to slow iteration // TODO: create texture to be able to do the sampling on the GPU let color; let palette; const { normalOffset, defaultColor, usePalette } = props; if (volume) { const coloring = props.coloring.params; const { stats } = volume.grid; const domain = coloring.domain.name === 'custom' ? coloring.domain.params : [stats.min, stats.max]; const isRelative = props.coloring.name === 'relative-value'; if (coloring.domain.name === 'auto' && isRelative) { domain[0] = (domain[0] - stats.mean) / stats.sigma; domain[1] = (domain[1] - stats.mean) / stats.sigma; } if (coloring.domain.name === 'auto' && coloring.domain.params.symmetric) { const max = Math.max(Math.abs(domain[0]), Math.abs(domain[1])); domain[0] = -max; domain[1] = max; } const scale = ColorScale.create({ domain, listOrName: coloring.list.colors }); const position = Vec3(); const getTrilinearlyInterpolated = Grid.makeGetTrilinearlyInterpolated(volume.grid, isRelative ? 'relative' : 'none'); color = (location) => { if (!isPositionLocation(location)) { return defaultColor; } // Offset the vertex position along its normal if (normalOffset > 0) { Vec3.scaleAndAdd(position, location.position, location.normal, normalOffset); } else { Vec3.copy(position, location.position); } const value = getTrilinearlyInterpolated(position); if (isNaN(value)) return defaultColor; if (usePalette) { return (clamp((value - domain[0]) / (domain[1] - domain[0]), 0, 1) * ColorTheme.PaletteScale); } else { return scale.color(value); } }; palette = usePalette ? { colors: coloring.list.colors.map(e => Array.isArray(e) ? e[0] : e), filter: (coloring.list.kind === 'set' ? 'nearest' : 'linear') } : undefined; } else { color = () => defaultColor; } return { factory: ExternalVolumeColorTheme, granularity: 'vertex', preferSmoothing: true, color, palette, props, description: Description, // TODO: figure out how to do legend for this }; } export const ExternalVolumeColorThemeProvider = { name: 'external-volume', label: 'External Volume', category: ColorThemeCategory.Misc, factory: ExternalVolumeColorTheme, getParams: () => ExternalVolumeColorThemeParams, defaultValues: PD.getDefaultValues(ExternalVolumeColorThemeParams), isApplicable: (ctx) => true, };