|
@@ -6,33 +6,23 @@
|
|
|
*/
|
|
|
|
|
|
import { TMDETMembraneOrientation } from '../../extensions/tmdet/behavior';
|
|
|
-import { StateTransforms } from '../../mol-plugin-state/transforms';
|
|
|
import { createPlugin } from '../../mol-plugin-ui';
|
|
|
import { PluginUIContext } from '../../mol-plugin-ui/context';
|
|
|
import { PluginLayoutControlsDisplay } from '../../mol-plugin/layout';
|
|
|
import { DefaultPluginUISpec, PluginUISpec } from '../../mol-plugin-ui/spec';
|
|
|
import { PluginConfig } from '../../mol-plugin/config';
|
|
|
import { PluginSpec } from '../../mol-plugin/spec';
|
|
|
-import { StateBuilder, StateObject, StateObjectCell, StateObjectRef, StateTransform, StateTransformer } from '../../mol-state';
|
|
|
-import { Color } from '../../mol-util/color';
|
|
|
import '../../mol-util/polyfill';
|
|
|
import { ObjectKeys } from '../../mol-util/type-helpers';
|
|
|
import './embedded.html';
|
|
|
import './favicon.ico';
|
|
|
import './index.html';
|
|
|
-import { MolScriptBuilder as MS } from '../../mol-script/language/builder';
|
|
|
-import { createStructureRepresentationParams } from '../../mol-plugin-state/helpers/structure-representation-params';
|
|
|
-import { Expression } from '../../mol-script/language/expression';
|
|
|
-import { PluginStateObject } from '../../mol-plugin-state/objects';
|
|
|
-import { MembraneOrientation } from '../../extensions/tmdet/prop';
|
|
|
-import { MEMBRANE_STORAGE_KEY } from '../../extensions/tmdet/algorithm';
|
|
|
-import { Vec3 } from '../../mol-math/linear-algebra';
|
|
|
-import { StateObjectSelector } from "../../mol-state/object";
|
|
|
|
|
|
require('mol-plugin-ui/skin/light.scss');
|
|
|
|
|
|
export { PLUGIN_VERSION as version } from '../../mol-plugin/version';
|
|
|
export { setDebugMode, setProductionMode } from '../../mol-util/debug';
|
|
|
+export { loadWithUNITMPMembraneRepresentation } from '../../extensions/tmdet/behavior';
|
|
|
|
|
|
const Extensions = {
|
|
|
'tmdet-membrane-orientation': PluginSpec.Behavior(TMDETMembraneOrientation)
|
|
@@ -64,155 +54,10 @@ const DefaultViewerOptions = {
|
|
|
emdbProvider: PluginConfig.Download.DefaultEmdbProvider.defaultValue,
|
|
|
};
|
|
|
type ViewerOptions = typeof DefaultViewerOptions;
|
|
|
-type StructureComponentType = StateObjectSelector<
|
|
|
- PluginStateObject.Molecule.Structure,
|
|
|
- StateTransformer<StateObject<any, StateObject.Type<any>>,
|
|
|
- StateObject<any, StateObject.Type<any>>, any>
|
|
|
- > | undefined;
|
|
|
-
|
|
|
-export let membrane: MembraneOrientation;
|
|
|
|
|
|
export class Viewer {
|
|
|
plugin: PluginUIContext
|
|
|
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
- // //////////////////////////// UNITMP VIEWER PROTOTYPING SECTION
|
|
|
-
|
|
|
- async loadWithUNITMPMembraneRepresentation(params: any) {
|
|
|
- const regionDescriptors: any = await this.downloadRegionDescriptor(params);
|
|
|
-
|
|
|
- membrane = this.createMembraneOrientation(regionDescriptors);
|
|
|
-
|
|
|
- console.log('DEBUG-02', membrane.planePoint2);
|
|
|
-
|
|
|
- localStorage.setItem(MEMBRANE_STORAGE_KEY, JSON.stringify(membrane));
|
|
|
-
|
|
|
- // load structure
|
|
|
- await this.loadStructure(params, regionDescriptors);
|
|
|
-
|
|
|
- await this.createStructureRepresentation(regionDescriptors);
|
|
|
-
|
|
|
- //
|
|
|
- // reset the camera because the membranes render 1st and the structure might not be fully visible
|
|
|
- //
|
|
|
- requestAnimationFrame(() => this.plugin.canvas3d?.requestCameraReset());
|
|
|
- }
|
|
|
-
|
|
|
- private async downloadRegionDescriptor(params: any): Promise<any> {
|
|
|
- const ctx = this.plugin;
|
|
|
- // download
|
|
|
- const downloadResult: string = await ctx.runTask(ctx.fetch({ url: params.regionDescriptorUrl })) as string;
|
|
|
- const regionDescriptors: any = JSON.parse(downloadResult);
|
|
|
- return regionDescriptors;
|
|
|
- }
|
|
|
-
|
|
|
- private async createStructureRepresentation(regionDescriptors: any) {
|
|
|
- // get the first structure of the first model
|
|
|
- const structure: StateObjectRef<PluginStateObject.Molecule.Structure> = this.plugin.managers.structure.hierarchy.current.models[0].structures[0].cell;
|
|
|
- const components = await this.createStructureComponents(structure);
|
|
|
-
|
|
|
- await this.buildStructureRepresentation(components);
|
|
|
-
|
|
|
- regionDescriptors.chains.forEach((chain: any) => {
|
|
|
-
|
|
|
- for (let regionKey in chain.regions) {
|
|
|
- const regionUpdates = this.plugin.build();
|
|
|
- const region = chain.regions[regionKey];
|
|
|
- this.createRegionRepresentation(chain.chain_id, region, regionUpdates.to(structure));
|
|
|
- regionUpdates.commit();
|
|
|
- }
|
|
|
-
|
|
|
- });
|
|
|
- }
|
|
|
-
|
|
|
- private async createStructureComponents(structure: StateObjectCell<PluginStateObject.Molecule.Structure, StateTransform<StateTransformer<StateObject<any, StateObject.Type<any>>, StateObject<any, StateObject.Type<any>>, any>>>) {
|
|
|
- return {
|
|
|
- polymer: await this.plugin.builders.structure.tryCreateComponentStatic(structure, 'polymer'),
|
|
|
- ligand: await this.plugin.builders.structure.tryCreateComponentStatic(structure, 'ligand'),
|
|
|
- water: await this.plugin.builders.structure.tryCreateComponentStatic(structure, 'water'),
|
|
|
- };
|
|
|
- }
|
|
|
-
|
|
|
- private async buildStructureRepresentation(components: { polymer: StructureComponentType; ligand: StructureComponentType; water: StructureComponentType }) {
|
|
|
- const builder = this.plugin.builders.structure.representation;
|
|
|
- const update = this.plugin.build();
|
|
|
- if (components.polymer)
|
|
|
- builder.buildRepresentation(update, components.polymer, { type: 'cartoon', typeParams: { alpha: 0.5 } }, { tag: 'polymer' });
|
|
|
- if (components.ligand)
|
|
|
- builder.buildRepresentation(update, components.ligand, { type: 'ball-and-stick' }, { tag: 'ligand' });
|
|
|
- if (components.water)
|
|
|
- builder.buildRepresentation(update, components.water, { type: 'ball-and-stick', typeParams: { alpha: 0.6 } }, { tag: 'water' });
|
|
|
- await update.commit();
|
|
|
- }
|
|
|
-
|
|
|
- private createMembraneOrientation(regionDescriptors: any): MembraneOrientation {
|
|
|
- const membraneNormal: Vec3 = Vec3.fromObj(
|
|
|
- regionDescriptors['membrane-normal']
|
|
|
- );
|
|
|
- membraneNormal[0] = membraneNormal[2];
|
|
|
- membraneNormal[2] = 0;
|
|
|
-
|
|
|
- const membraneOrientation: MembraneOrientation = {
|
|
|
- planePoint1: Vec3.fromArray(Vec3.zero(), membraneNormal, 0),
|
|
|
- planePoint2: Vec3.fromArray(Vec3.zero(), membraneNormal, 0),
|
|
|
- centroid: Vec3.fromArray(
|
|
|
- Vec3.zero(), [0, 0, 0], 0
|
|
|
- ),
|
|
|
- normalVector: membraneNormal,
|
|
|
-
|
|
|
- // (NOTE: the TMDET extension calculates and sets it during applying preset)
|
|
|
- radius: regionDescriptors['radius']
|
|
|
- };
|
|
|
- membraneOrientation.planePoint2[0] *= -1;
|
|
|
- return membraneOrientation;
|
|
|
- }
|
|
|
-
|
|
|
- private async loadStructure(params: any, regionDescriptors: any) {
|
|
|
- const builders = this.plugin.builders;
|
|
|
- const data = await builders.data.download({
|
|
|
- url: params.structureUrl,
|
|
|
- label: `UniTMP: ${regionDescriptors['pdb-id']}`,
|
|
|
- isBinary: false
|
|
|
- }); // , { state: { isGhost: true } });
|
|
|
- const trajectory = await builders.structure.parseTrajectory(data, 'mmcif');
|
|
|
- // create membrane representation
|
|
|
- await builders.structure.hierarchy.applyPreset(
|
|
|
- trajectory, 'default', { representationPreset: 'preset-membrane-orientation' as any });
|
|
|
- }
|
|
|
-
|
|
|
- private createRegionRepresentation(chain: string, region: any, update: StateBuilder.To<any, any>) {
|
|
|
- const regionLabel: string = `${chain} | ${region.name}`;
|
|
|
- const color: Color = Color.fromArray(region.color, 0);
|
|
|
- const query: Expression = this.getQuery(chain, region.auth_ids as number[]);
|
|
|
-
|
|
|
- // based on https://github.com/molstar/molstar/issues/209
|
|
|
- update
|
|
|
- .apply(StateTransforms.Model.StructureSelectionFromExpression, { label: regionLabel, expression: query })
|
|
|
- .apply(StateTransforms.Representation.StructureRepresentation3D, createStructureRepresentationParams(this.plugin, update.selector.data, {
|
|
|
- color: 'uniform',
|
|
|
- colorParams: { value: color }
|
|
|
- }));
|
|
|
- }
|
|
|
-
|
|
|
- private getQuery(chainId: string, auth_array: number[]): Expression {
|
|
|
- const query: Expression =
|
|
|
- MS.struct.generator.atomGroups({
|
|
|
- 'residue-test': MS.core.set.has([MS.set( ...auth_array ), MS.ammp('auth_seq_id')]),
|
|
|
- 'chain-test': MS.core.rel.eq([chainId, MS.ammp('label_asym_id')])
|
|
|
- });
|
|
|
- return query;
|
|
|
- }
|
|
|
-
|
|
|
- // //////////////////////////// END OF PROTOTYPING SECTION
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
constructor(elementOrId: string | HTMLElement, options: Partial<ViewerOptions> = {}) {
|
|
|
const o = { ...DefaultViewerOptions, ...options };
|
|
|
const defaultSpec = DefaultPluginUISpec();
|
|
@@ -271,6 +116,7 @@ export class Viewer {
|
|
|
: elementOrId;
|
|
|
if (!element) throw new Error(`Could not get element with id '${elementOrId}'`);
|
|
|
this.plugin = createPlugin(element, spec);
|
|
|
+
|
|
|
}
|
|
|
|
|
|
handleResize() {
|