123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236 |
- /**
- * Copyright (c) 2020 mol* contributors, licensed under MIT, See LICENSE file for more info.
- *
- * @author David Sehnal <david.sehnal@gmail.com>
- */
- import { PluginStateObject, PluginStateTransform } from '../../mol-plugin-state/objects';
- import { createSphericalCollocationGrid } from './orbitals';
- import { ParamDefinition as PD } from '../../mol-util/param-definition';
- import { Task } from '../../mol-task';
- import { CustomProperties } from '../../mol-model/custom-property';
- import { SphericalBasisOrder } from './spherical-functions';
- import { Volume } from '../../mol-model/volume';
- import { PluginContext } from '../../mol-plugin/context';
- import { ColorNames } from '../../mol-util/color/names';
- import { createVolumeRepresentationParams } from '../../mol-plugin-state/helpers/volume-representation-params';
- import { StateTransformer } from '../../mol-state';
- import { Theme } from '../../mol-theme/theme';
- import { VolumeRepresentation3DHelpers } from '../../mol-plugin-state/transforms/representation';
- import { AlphaOrbital, Basis, CubeGrid } from './data-model';
- import { createSphericalCollocationDensityGrid } from './density';
- import { Tensor } from '../../mol-math/linear-algebra';
- export class BasisAndOrbitals extends PluginStateObject.Create<{ basis: Basis, order: SphericalBasisOrder, orbitals: AlphaOrbital[] }>({ name: 'Basis', typeClass: 'Object' }) { }
- export const StaticBasisAndOrbitals = PluginStateTransform.BuiltIn({
- name: 'static-basis-and-orbitals',
- display: 'Basis and Orbitals',
- from: PluginStateObject.Root,
- to: BasisAndOrbitals,
- params: {
- label: PD.Text('Orbital Data', { isHidden: true }),
- basis: PD.Value<Basis>(void 0 as any, { isHidden: true }),
- order: PD.Text<SphericalBasisOrder>('gaussian' as SphericalBasisOrder, { isHidden: true }),
- orbitals: PD.Value<AlphaOrbital[]>([], { isHidden: true })
- },
- })({
- apply({ params }) {
- return new BasisAndOrbitals({ basis: params.basis, order: params.order, orbitals: params.orbitals }, { label: params.label });
- }
- });
- const CreateOrbitalVolumeParamBase = {
- cutoffThreshold: PD.Numeric(0.0015, { min: 0, max: 0.1, step: 0.0001 }),
- boxExpand: PD.Numeric(4.5, { min: 0, max: 7, step: 0.1 }),
- gridSpacing: PD.ObjectList({ atomCount: PD.Numeric(0), spacing: PD.Numeric(0.35, { min: 0.1, max: 2, step: 0.01 }) }, e => `Atoms ${e.atomCount}: ${e.spacing}`, {
- defaultValue: [
- { atomCount: 55, spacing: 0.5 },
- { atomCount: 40, spacing: 0.45 },
- { atomCount: 25, spacing: 0.4 },
- { atomCount: 0, spacing: 0.35 },
- ]
- }),
- clampValues: PD.MappedStatic('off', {
- off: PD.EmptyGroup(),
- on: PD.Group({
- sigma: PD.Numeric(8, { min: 1, max: 20 }, { description: 'Clamp values to range [sigma * negIsoValue, sigma * posIsoValue].' })
- })
- })
- };
- function clampData(matrix: Tensor.Data, min: number, max: number) {
- for (let i = 0, _i = matrix.length; i < _i; i++) {
- const v = matrix[i];
- if (v < min) matrix[i] = min;
- else if (v > max) matrix[i] = max;
- }
- }
- function clampGrid(data: CubeGrid, v: number) {
- const grid = data.grid;
- const min = (data.isovalues?.negative ?? data.grid.stats.min) * v;
- const max = (data.isovalues?.positive ?? data.grid.stats.max) * v;
- // clamp values for better direct volume resolution
- // current implementation uses Byte array for values
- // if this is not enough, update mol* to use float
- // textures instead
- if (grid.stats.min < min || grid.stats.max > max) {
- clampData(data.grid.cells.data, min, max);
- if (grid.stats.min < min) {
- (grid.stats.min as number) = min;
- }
- if (grid.stats.max > max) {
- (grid.stats.max as number) = max;
- }
- }
- }
- export const CreateOrbitalVolume = PluginStateTransform.BuiltIn({
- name: 'create-orbital-volume',
- display: 'Orbital Volume',
- from: BasisAndOrbitals,
- to: PluginStateObject.Volume.Data,
- params: (a) => {
- if (!a) {
- return { index: PD.Numeric(0), ...CreateOrbitalVolumeParamBase };
- }
- return {
- index: PD.Select(0, a.data.orbitals.map((o, i) => [i, `[${i + 1}] ${o.energy.toFixed(4)}`])),
- ...CreateOrbitalVolumeParamBase
- };
- }
- })({
- apply({ a, params }, plugin: PluginContext) {
- return Task.create('Orbital Volume', async ctx => {
- const data = await createSphericalCollocationGrid({
- basis: a.data.basis,
- cutoffThreshold: params.cutoffThreshold,
- sphericalOrder: a.data.order,
- boxExpand: params.boxExpand,
- gridSpacing: params.gridSpacing.map(e => [e.atomCount, e.spacing] as [number, number])
- }, a.data.orbitals[params.index], plugin.canvas3d?.webgl).runInContext(ctx);
- const volume: Volume = {
- grid: data.grid,
- sourceData: { name: 'custom grid', kind: 'alpha-orbitals', data },
- customProperties: new CustomProperties(),
- _propertyData: Object.create(null),
- };
- if (params.clampValues?.name === 'on') {
- clampGrid(data, params.clampValues?.params?.sigma ?? 8);
- }
- return new PluginStateObject.Volume.Data(volume, { label: 'Orbital Volume' });
- });
- }
- });
- export const CreateOrbitalDensityVolume = PluginStateTransform.BuiltIn({
- name: 'create-orbital-density-volume',
- display: 'Orbital Density Volume',
- from: BasisAndOrbitals,
- to: PluginStateObject.Volume.Data,
- params: CreateOrbitalVolumeParamBase
- })({
- apply({ a, params }, plugin: PluginContext) {
- return Task.create('Orbital Volume', async ctx => {
- const data = await createSphericalCollocationDensityGrid({
- basis: a.data.basis,
- cutoffThreshold: params.cutoffThreshold,
- sphericalOrder: a.data.order,
- boxExpand: params.boxExpand,
- gridSpacing: params.gridSpacing.map(e => [e.atomCount, e.spacing] as [number, number])
- }, a.data.orbitals, plugin.canvas3d?.webgl).runInContext(ctx);
- const volume: Volume = {
- grid: data.grid,
- sourceData: { name: 'custom grid', kind: 'alpha-orbitals', data },
- customProperties: new CustomProperties(),
- _propertyData: Object.create(null),
- };
- if (params.clampValues?.name === 'on') {
- clampGrid(data, params.clampValues?.params?.sigma ?? 8);
- }
- return new PluginStateObject.Volume.Data(volume, { label: 'Orbital Volume' });
- });
- }
- });
- export const CreateOrbitalRepresentation3D = PluginStateTransform.BuiltIn({
- name: 'create-orbital-representation-3d',
- display: 'Orbital Representation 3D',
- from: PluginStateObject.Volume.Data,
- to: PluginStateObject.Volume.Representation3D,
- params: {
- directVolume: PD.Boolean(false),
- relativeIsovalue: PD.Numeric(1, { min: 0.01, max: 5, step: 0.01 }),
- kind: PD.Select<'positive' | 'negative'>('positive', [['positive', 'Positive'], ['negative', 'Negative']]),
- color: PD.Color(ColorNames.blue),
- alpha: PD.Numeric(1, { min: 0, max: 1, step: 0.01 }),
- xrayShaded: PD.Boolean(false),
- pickable: PD.Boolean(true)
- }
- })({
- canAutoUpdate() {
- return true;
- },
- apply({ a, params: srcParams }, plugin: PluginContext) {
- return Task.create('Orbitals Representation 3D', async ctx => {
- const params = volumeParams(plugin, a, srcParams);
- const propertyCtx = { runtime: ctx, assetManager: plugin.managers.asset };
- const provider = plugin.representation.volume.registry.get(params.type.name);
- if (provider.ensureCustomProperties) await provider.ensureCustomProperties.attach(propertyCtx, a.data);
- const props = params.type.params || {};
- const repr = provider.factory({ webgl: plugin.canvas3d?.webgl, ...plugin.representation.volume.themes }, provider.getParams);
- repr.setTheme(Theme.create(plugin.representation.volume.themes, { volume: a.data }, params));
- await repr.createOrUpdate(props, a.data).runInContext(ctx);
- repr.setState({ pickable: srcParams.pickable });
- return new PluginStateObject.Volume.Representation3D({ repr, source: a }, { label: provider.label, description: VolumeRepresentation3DHelpers.getDescription(props) });
- });
- },
- update({ a, b, newParams: srcParams }, plugin: PluginContext) {
- return Task.create('Orbitals Representation 3D', async ctx => {
- const newParams = volumeParams(plugin, a, srcParams);
- const props = { ...b.data.repr.props, ...newParams.type.params };
- b.data.repr.setTheme(Theme.create(plugin.representation.volume.themes, { volume: a.data }, newParams));
- await b.data.repr.createOrUpdate(props, a.data).runInContext(ctx);
- b.data.repr.setState({ pickable: srcParams.pickable });
- b.description = VolumeRepresentation3DHelpers.getDescription(props);
- return StateTransformer.UpdateResult.Updated;
- });
- }
- });
- function volumeParams(plugin: PluginContext, volume: PluginStateObject.Volume.Data, params: StateTransformer.Params<typeof CreateOrbitalRepresentation3D>) {
- if (volume.data.sourceData.kind !== 'alpha-orbitals') throw new Error('Invalid data source kind.');
- const { isovalues } = volume.data.sourceData.data as CubeGrid;
- if (!isovalues) throw new Error('Isovalues are not computed.');
- const value = isovalues[params.kind];
- return createVolumeRepresentationParams(plugin, volume.data, params.directVolume ? {
- type: 'direct-volume',
- typeParams: {
- alpha: params.alpha,
- renderMode: {
- name: 'isosurface',
- params: { isoValue: { kind: 'absolute', absoluteValue: (value ?? 1000) * params.relativeIsovalue }, singleLayer: false }
- },
- xrayShaded: params.xrayShaded
- },
- color: 'uniform',
- colorParams: { value: params.color }
- } : {
- type: 'isosurface',
- typeParams: { isoValue: { kind: 'absolute', absoluteValue: (value ?? 1000) * params.relativeIsovalue }, alpha: params.alpha, xrayShaded: params.xrayShaded },
- color: 'uniform',
- colorParams: { value: params.color }
- });
- }
|