|
@@ -31,124 +31,124 @@ import { CellPackInfoProvider } from './property';
|
|
|
import { CellPackColorThemeProvider } from './color';
|
|
|
|
|
|
function getCellPackModelUrl(fileName: string, baseUrl: string) {
|
|
|
- return `${baseUrl}/results/${fileName}`
|
|
|
+ return `${baseUrl}/results/${fileName}`;
|
|
|
}
|
|
|
|
|
|
async function getModel(id: string, model_id: number, baseUrl: string, file?: File) {
|
|
|
let model: Model;
|
|
|
if (file) {
|
|
|
- const text = await file.text()
|
|
|
+ const text = await file.text();
|
|
|
if (file.name.endsWith('.cif')) {
|
|
|
- const cif = (await parseCif(text)).blocks[0]
|
|
|
- model = (await trajectoryFromMmCIF(cif).run())[model_id]
|
|
|
+ const cif = (await parseCif(text)).blocks[0];
|
|
|
+ model = (await trajectoryFromMmCIF(cif).run())[model_id];
|
|
|
} else if (file.name.endsWith('.pdb')) {
|
|
|
- const pdb = await parsePDBfile(text, id)
|
|
|
+ const pdb = await parsePDBfile(text, id);
|
|
|
|
|
|
- model = (await trajectoryFromPDB(pdb).run())[model_id]
|
|
|
+ model = (await trajectoryFromPDB(pdb).run())[model_id];
|
|
|
} else {
|
|
|
- throw new Error(`unsupported file type '${file.name}'`)
|
|
|
+ throw new Error(`unsupported file type '${file.name}'`);
|
|
|
}
|
|
|
} else if (id.match(/^[1-9][a-zA-Z0-9]{3,3}$/i)) {
|
|
|
- const cif = await getFromPdb(id)
|
|
|
- model = (await trajectoryFromMmCIF(cif).run())[model_id]
|
|
|
+ const cif = await getFromPdb(id);
|
|
|
+ model = (await trajectoryFromMmCIF(cif).run())[model_id];
|
|
|
} else {
|
|
|
- const pdb = await getFromCellPackDB(id, baseUrl)
|
|
|
- model = (await trajectoryFromPDB(pdb).run())[model_id]
|
|
|
+ const pdb = await getFromCellPackDB(id, baseUrl);
|
|
|
+ model = (await trajectoryFromPDB(pdb).run())[model_id];
|
|
|
}
|
|
|
- return model
|
|
|
+ return model;
|
|
|
}
|
|
|
|
|
|
async function getStructure(model: Model, source: IngredientSource, props: { assembly?: string } = {}) {
|
|
|
- let structure = Structure.ofModel(model)
|
|
|
- const { assembly } = props
|
|
|
+ let structure = Structure.ofModel(model);
|
|
|
+ const { assembly } = props;
|
|
|
|
|
|
if (assembly) {
|
|
|
- structure = await StructureSymmetry.buildAssembly(structure, assembly).run()
|
|
|
+ structure = await StructureSymmetry.buildAssembly(structure, assembly).run();
|
|
|
}
|
|
|
let query;
|
|
|
if (source.selection){
|
|
|
- const asymIds: string[] = source.selection.replace(' :', '').split(' or')
|
|
|
+ const asymIds: string[] = source.selection.replace(' :', '').split(' or');
|
|
|
query = MS.struct.modifier.union([
|
|
|
MS.struct.generator.atomGroups({
|
|
|
'entity-test': MS.core.rel.eq([MS.ammp('entityType'), 'polymer']),
|
|
|
'chain-test': MS.core.set.has([MS.set(...asymIds), MS.ammp('auth_asym_id')])
|
|
|
})
|
|
|
- ])
|
|
|
+ ]);
|
|
|
} else {
|
|
|
query = MS.struct.modifier.union([
|
|
|
MS.struct.generator.atomGroups({
|
|
|
'entity-test': MS.core.rel.eq([MS.ammp('entityType'), 'polymer'])
|
|
|
})
|
|
|
- ])
|
|
|
+ ]);
|
|
|
}
|
|
|
|
|
|
- const compiled = compile<StructureSelection>(query)
|
|
|
- const result = compiled(new QueryContext(structure))
|
|
|
- structure = StructureSelection.unionStructure(result)
|
|
|
+ const compiled = compile<StructureSelection>(query);
|
|
|
+ const result = compiled(new QueryContext(structure));
|
|
|
+ structure = StructureSelection.unionStructure(result);
|
|
|
|
|
|
- return structure
|
|
|
+ return structure;
|
|
|
}
|
|
|
|
|
|
function getTransformLegacy(trans: Vec3, rot: Quat) {
|
|
|
- const q: Quat = Quat.create(-rot[3], rot[0], rot[1], rot[2])
|
|
|
- const m: Mat4 = Mat4.fromQuat(Mat4.zero(), q)
|
|
|
- Mat4.transpose(m, m)
|
|
|
- Mat4.scale(m, m, Vec3.create(-1.0, 1.0, -1.0))
|
|
|
- Mat4.setTranslation(m, trans)
|
|
|
- return m
|
|
|
+ const q: Quat = Quat.create(-rot[3], rot[0], rot[1], rot[2]);
|
|
|
+ const m: Mat4 = Mat4.fromQuat(Mat4.zero(), q);
|
|
|
+ Mat4.transpose(m, m);
|
|
|
+ Mat4.scale(m, m, Vec3.create(-1.0, 1.0, -1.0));
|
|
|
+ Mat4.setTranslation(m, trans);
|
|
|
+ return m;
|
|
|
}
|
|
|
|
|
|
function getTransform(trans: Vec3, rot: Quat) {
|
|
|
- const q: Quat = Quat.create(rot[0], rot[1], rot[2], rot[3])
|
|
|
- const m: Mat4 = Mat4.fromQuat(Mat4.zero(), q)
|
|
|
- const p: Vec3 = Vec3.create(trans[0], trans[1], trans[2])
|
|
|
- Mat4.setTranslation(m, p)
|
|
|
- return m
|
|
|
+ const q: Quat = Quat.create(rot[0], rot[1], rot[2], rot[3]);
|
|
|
+ const m: Mat4 = Mat4.fromQuat(Mat4.zero(), q);
|
|
|
+ const p: Vec3 = Vec3.create(trans[0], trans[1], trans[2]);
|
|
|
+ Mat4.setTranslation(m, p);
|
|
|
+ return m;
|
|
|
}
|
|
|
|
|
|
|
|
|
function getResultTransforms(results: Ingredient['results'], legacy: boolean) {
|
|
|
- if (legacy) return results.map((r: Ingredient['results'][0]) => getTransformLegacy(r[0], r[1]))
|
|
|
- else return results.map((r: Ingredient['results'][0]) => getTransform(r[0], r[1]))
|
|
|
+ if (legacy) return results.map((r: Ingredient['results'][0]) => getTransformLegacy(r[0], r[1]));
|
|
|
+ else return results.map((r: Ingredient['results'][0]) => getTransform(r[0], r[1]));
|
|
|
}
|
|
|
|
|
|
function getCurveTransforms(ingredient: Ingredient) {
|
|
|
- const n = ingredient.nbCurve || 0
|
|
|
- const instances: Mat4[] = []
|
|
|
+ const n = ingredient.nbCurve || 0;
|
|
|
+ const instances: Mat4[] = [];
|
|
|
const segmentLength = ingredient.radii
|
|
|
? (ingredient.radii[0].radii
|
|
|
? ingredient.radii[0].radii[0] * 2.0
|
|
|
: 3.4)
|
|
|
: 3.4;
|
|
|
for (let i = 0; i < n; ++i) {
|
|
|
- const cname = `curve${i}`
|
|
|
+ const cname = `curve${i}`;
|
|
|
if (!(cname in ingredient)) {
|
|
|
// console.warn(`Expected '${cname}' in ingredient`)
|
|
|
- continue
|
|
|
+ continue;
|
|
|
}
|
|
|
- const _points = ingredient[cname] as Vec3[]
|
|
|
+ const _points = ingredient[cname] as Vec3[];
|
|
|
if (_points.length <= 2) {
|
|
|
// TODO handle curve with 2 or less points
|
|
|
- continue
|
|
|
+ continue;
|
|
|
}
|
|
|
- const points = new Float32Array(_points.length * 3)
|
|
|
- for (let i = 0, il = _points.length; i < il; ++i) Vec3.toArray(_points[i], points, i * 3)
|
|
|
- const newInstances = getMatFromResamplePoints(points, segmentLength)
|
|
|
- instances.push(...newInstances)
|
|
|
+ const points = new Float32Array(_points.length * 3);
|
|
|
+ for (let i = 0, il = _points.length; i < il; ++i) Vec3.toArray(_points[i], points, i * 3);
|
|
|
+ const newInstances = getMatFromResamplePoints(points, segmentLength);
|
|
|
+ instances.push(...newInstances);
|
|
|
}
|
|
|
|
|
|
- return instances
|
|
|
+ return instances;
|
|
|
}
|
|
|
|
|
|
function getAssembly(transforms: Mat4[], structure: Structure) {
|
|
|
- const builder = Structure.Builder()
|
|
|
+ const builder = Structure.Builder();
|
|
|
const { units } = structure;
|
|
|
|
|
|
for (let i = 0, il = transforms.length; i < il; ++i) {
|
|
|
- const id = `${i + 1}`
|
|
|
- const op = SymmetryOperator.create(id, transforms[i], { assembly: { id, operId: i, operList: [ id ] } })
|
|
|
+ const id = `${i + 1}`;
|
|
|
+ const op = SymmetryOperator.create(id, transforms[i], { assembly: { id, operId: i, operList: [ id ] } });
|
|
|
for (const unit of units) {
|
|
|
- builder.addWithOperator(unit, op)
|
|
|
+ builder.addWithOperator(unit, op);
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -156,51 +156,51 @@ function getAssembly(transforms: Mat4[], structure: Structure) {
|
|
|
}
|
|
|
|
|
|
function getCifCurve(name: string, transforms: Mat4[], model: Model) {
|
|
|
- if (!MmcifFormat.is(model.sourceData)) throw new Error('mmcif source data needed')
|
|
|
+ if (!MmcifFormat.is(model.sourceData)) throw new Error('mmcif source data needed');
|
|
|
|
|
|
- const { db } = model.sourceData.data
|
|
|
- const d = db.atom_site
|
|
|
- const n = d._rowCount
|
|
|
- const rowCount = n * transforms.length
|
|
|
+ const { db } = model.sourceData.data;
|
|
|
+ const d = db.atom_site;
|
|
|
+ const n = d._rowCount;
|
|
|
+ const rowCount = n * transforms.length;
|
|
|
|
|
|
- const { offsets, count } = model.atomicHierarchy.chainAtomSegments
|
|
|
+ const { offsets, count } = model.atomicHierarchy.chainAtomSegments;
|
|
|
|
|
|
- const x = d.Cartn_x.toArray()
|
|
|
- const y = d.Cartn_y.toArray()
|
|
|
- const z = d.Cartn_z.toArray()
|
|
|
+ const x = d.Cartn_x.toArray();
|
|
|
+ const y = d.Cartn_y.toArray();
|
|
|
+ const z = d.Cartn_z.toArray();
|
|
|
|
|
|
- const Cartn_x = new Float32Array(rowCount)
|
|
|
- const Cartn_y = new Float32Array(rowCount)
|
|
|
- const Cartn_z = new Float32Array(rowCount)
|
|
|
- const map = new Uint32Array(rowCount)
|
|
|
- const seq = new Int32Array(rowCount)
|
|
|
- let offset = 0
|
|
|
+ const Cartn_x = new Float32Array(rowCount);
|
|
|
+ const Cartn_y = new Float32Array(rowCount);
|
|
|
+ const Cartn_z = new Float32Array(rowCount);
|
|
|
+ const map = new Uint32Array(rowCount);
|
|
|
+ const seq = new Int32Array(rowCount);
|
|
|
+ let offset = 0;
|
|
|
for (let c = 0; c < count; ++c) {
|
|
|
- const cStart = offsets[c]
|
|
|
- const cEnd = offsets[c + 1]
|
|
|
- const cLength = cEnd - cStart
|
|
|
+ const cStart = offsets[c];
|
|
|
+ const cEnd = offsets[c + 1];
|
|
|
+ const cLength = cEnd - cStart;
|
|
|
for (let t = 0, tl = transforms.length; t < tl; ++t) {
|
|
|
- const m = transforms[t]
|
|
|
+ const m = transforms[t];
|
|
|
for (let j = cStart; j < cEnd; ++j) {
|
|
|
- const i = offset + j - cStart
|
|
|
- const xj = x[j], yj = y[j], zj = z[j]
|
|
|
- Cartn_x[i] = m[0] * xj + m[4] * yj + m[8] * zj + m[12]
|
|
|
- Cartn_y[i] = m[1] * xj + m[5] * yj + m[9] * zj + m[13]
|
|
|
- Cartn_z[i] = m[2] * xj + m[6] * yj + m[10] * zj + m[14]
|
|
|
- map[i] = j
|
|
|
- seq[i] = t + 1
|
|
|
+ const i = offset + j - cStart;
|
|
|
+ const xj = x[j], yj = y[j], zj = z[j];
|
|
|
+ Cartn_x[i] = m[0] * xj + m[4] * yj + m[8] * zj + m[12];
|
|
|
+ Cartn_y[i] = m[1] * xj + m[5] * yj + m[9] * zj + m[13];
|
|
|
+ Cartn_z[i] = m[2] * xj + m[6] * yj + m[10] * zj + m[14];
|
|
|
+ map[i] = j;
|
|
|
+ seq[i] = t + 1;
|
|
|
}
|
|
|
- offset += cLength
|
|
|
+ offset += cLength;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function multColumn<T>(column: Column<T>) {
|
|
|
- const array = column.toArray()
|
|
|
+ const array = column.toArray();
|
|
|
return Column.ofLambda({
|
|
|
value: row => array[map[row]],
|
|
|
areValuesEqual: (rowA, rowB) => map[rowA] === map[rowB] || array[map[rowA]] === array[map[rowB]],
|
|
|
rowCount, schema: column.schema
|
|
|
- })
|
|
|
+ });
|
|
|
}
|
|
|
|
|
|
const _atom_site: CifCategory.SomeFields<mmCIF_Schema['atom_site']> = {
|
|
@@ -233,13 +233,13 @@ function getCifCurve(name: string, transforms: Mat4[], model: Model) {
|
|
|
|
|
|
pdbx_PDB_ins_code: CifField.ofColumn(Column.ofConst('', rowCount, Column.Schema.str)),
|
|
|
pdbx_PDB_model_num: CifField.ofColumn(Column.ofConst(1, rowCount, Column.Schema.int)),
|
|
|
- }
|
|
|
+ };
|
|
|
|
|
|
const categories = {
|
|
|
entity: CifCategory.ofTable('entity', db.entity),
|
|
|
chem_comp: CifCategory.ofTable('chem_comp', db.chem_comp),
|
|
|
atom_site: CifCategory.ofFields('atom_site', _atom_site)
|
|
|
- }
|
|
|
+ };
|
|
|
|
|
|
return {
|
|
|
header: name,
|
|
@@ -249,125 +249,125 @@ function getCifCurve(name: string, transforms: Mat4[], model: Model) {
|
|
|
}
|
|
|
|
|
|
async function getCurve(name: string, ingredient: Ingredient, transforms: Mat4[], model: Model) {
|
|
|
- const cif = getCifCurve(name, transforms, model)
|
|
|
+ const cif = getCifCurve(name, transforms, model);
|
|
|
|
|
|
const curveModelTask = Task.create('Curve Model', async ctx => {
|
|
|
- const format = MmcifFormat.fromFrame(cif)
|
|
|
- const models = await createModels(format.data.db, format, ctx)
|
|
|
- return models[0]
|
|
|
- })
|
|
|
+ const format = MmcifFormat.fromFrame(cif);
|
|
|
+ const models = await createModels(format.data.db, format, ctx);
|
|
|
+ return models[0];
|
|
|
+ });
|
|
|
|
|
|
- const curveModel = await curveModelTask.run()
|
|
|
- return getStructure(curveModel, ingredient.source)
|
|
|
+ const curveModel = await curveModelTask.run();
|
|
|
+ return getStructure(curveModel, ingredient.source);
|
|
|
}
|
|
|
|
|
|
async function getIngredientStructure(ingredient: Ingredient, baseUrl: string, ingredientFiles: IngredientFiles) {
|
|
|
- const { name, source, results, nbCurve } = ingredient
|
|
|
- if (source.pdb === 'None') return
|
|
|
+ const { name, source, results, nbCurve } = ingredient;
|
|
|
+ if (source.pdb === 'None') return;
|
|
|
|
|
|
- const file = ingredientFiles[source.pdb]
|
|
|
+ const file = ingredientFiles[source.pdb];
|
|
|
if (!file) {
|
|
|
// TODO can these be added to the library?
|
|
|
- if (name === 'HIV1_CAhex_0_1_0') return
|
|
|
- if (name === 'HIV1_CAhexCyclophilA_0_1_0') return
|
|
|
- if (name === 'iLDL') return
|
|
|
- if (name === 'peptides') return
|
|
|
- if (name === 'lypoglycane') return
|
|
|
+ if (name === 'HIV1_CAhex_0_1_0') return;
|
|
|
+ if (name === 'HIV1_CAhexCyclophilA_0_1_0') return;
|
|
|
+ if (name === 'iLDL') return;
|
|
|
+ if (name === 'peptides') return;
|
|
|
+ if (name === 'lypoglycane') return;
|
|
|
}
|
|
|
|
|
|
// model id in case structure is NMR
|
|
|
const model_id = (ingredient.source.model) ? parseInt(ingredient.source.model) : 0;
|
|
|
- const model = await getModel(source.pdb || name, model_id, baseUrl, file)
|
|
|
- if (!model) return
|
|
|
+ const model = await getModel(source.pdb || name, model_id, baseUrl, file);
|
|
|
+ if (!model) return;
|
|
|
|
|
|
if (nbCurve) {
|
|
|
- return getCurve(name, ingredient, getCurveTransforms(ingredient), model)
|
|
|
+ return getCurve(name, ingredient, getCurveTransforms(ingredient), model);
|
|
|
} else {
|
|
|
let bu: string|undefined = source.bu ? source.bu : undefined;
|
|
|
if (bu){
|
|
|
if (bu === 'AU') {
|
|
|
bu = undefined;
|
|
|
} else {
|
|
|
- bu = bu.slice(2)
|
|
|
+ bu = bu.slice(2);
|
|
|
}
|
|
|
}
|
|
|
- let structure = await getStructure(model, source, { assembly: bu })
|
|
|
+ let structure = await getStructure(model, source, { assembly: bu });
|
|
|
// transform with offset and pcp
|
|
|
- let legacy: boolean = true
|
|
|
+ let legacy: boolean = true;
|
|
|
if (ingredient.offset || ingredient.principalAxis){
|
|
|
// center the structure
|
|
|
- legacy = false
|
|
|
- const boundary = structure.boundary
|
|
|
- let structureCenter: Vec3 = Vec3.zero()
|
|
|
- Vec3.negate(structureCenter, boundary.sphere.center)
|
|
|
- const m1: Mat4 = Mat4.identity()
|
|
|
- Mat4.setTranslation(m1, structureCenter)
|
|
|
- structure = Structure.transform(structure, m1)
|
|
|
+ legacy = false;
|
|
|
+ const boundary = structure.boundary;
|
|
|
+ let structureCenter: Vec3 = Vec3.zero();
|
|
|
+ Vec3.negate(structureCenter, boundary.sphere.center);
|
|
|
+ const m1: Mat4 = Mat4.identity();
|
|
|
+ Mat4.setTranslation(m1, structureCenter);
|
|
|
+ structure = Structure.transform(structure, m1);
|
|
|
if (ingredient.offset){
|
|
|
if (!Vec3.exactEquals(ingredient.offset, Vec3.zero())){
|
|
|
const m: Mat4 = Mat4.identity();
|
|
|
- Mat4.setTranslation(m, ingredient.offset)
|
|
|
+ Mat4.setTranslation(m, ingredient.offset);
|
|
|
structure = Structure.transform(structure, m);
|
|
|
}
|
|
|
}
|
|
|
if (ingredient.principalAxis){
|
|
|
if (!Vec3.exactEquals(ingredient.principalAxis, Vec3.unitZ)){
|
|
|
const q: Quat = Quat.identity();
|
|
|
- Quat.rotationTo(q, ingredient.principalAxis, Vec3.unitZ)
|
|
|
- const m: Mat4 = Mat4.fromQuat(Mat4.zero(), q)
|
|
|
+ Quat.rotationTo(q, ingredient.principalAxis, Vec3.unitZ);
|
|
|
+ const m: Mat4 = Mat4.fromQuat(Mat4.zero(), q);
|
|
|
structure = Structure.transform(structure, m);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
- return getAssembly(getResultTransforms(results, legacy), structure)
|
|
|
+ return getAssembly(getResultTransforms(results, legacy), structure);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
export function createStructureFromCellPack(packing: CellPacking, baseUrl: string, ingredientFiles: IngredientFiles) {
|
|
|
return Task.create('Create Packing Structure', async ctx => {
|
|
|
- const { ingredients, name } = packing
|
|
|
- const structures: Structure[] = []
|
|
|
+ const { ingredients, name } = packing;
|
|
|
+ const structures: Structure[] = [];
|
|
|
for (const iName in ingredients) {
|
|
|
- if (ctx.shouldUpdate) await ctx.update(iName)
|
|
|
- const s = await getIngredientStructure(ingredients[iName], baseUrl, ingredientFiles)
|
|
|
- if (s) structures.push(s)
|
|
|
+ if (ctx.shouldUpdate) await ctx.update(iName);
|
|
|
+ const s = await getIngredientStructure(ingredients[iName], baseUrl, ingredientFiles);
|
|
|
+ if (s) structures.push(s);
|
|
|
}
|
|
|
|
|
|
- if (ctx.shouldUpdate) await ctx.update(`${name} - units`)
|
|
|
- const builder = Structure.Builder({ label: name })
|
|
|
- let offsetInvariantId = 0
|
|
|
+ if (ctx.shouldUpdate) await ctx.update(`${name} - units`);
|
|
|
+ const builder = Structure.Builder({ label: name });
|
|
|
+ let offsetInvariantId = 0;
|
|
|
for (const s of structures) {
|
|
|
- if (ctx.shouldUpdate) await ctx.update(`${s.label}`)
|
|
|
- let maxInvariantId = 0
|
|
|
+ if (ctx.shouldUpdate) await ctx.update(`${s.label}`);
|
|
|
+ let maxInvariantId = 0;
|
|
|
for (const u of s.units) {
|
|
|
- const invariantId = u.invariantId + offsetInvariantId
|
|
|
- if (u.invariantId > maxInvariantId) maxInvariantId = u.invariantId
|
|
|
- builder.addUnit(u.kind, u.model, u.conformation.operator, u.elements, Unit.Trait.None, invariantId)
|
|
|
+ const invariantId = u.invariantId + offsetInvariantId;
|
|
|
+ if (u.invariantId > maxInvariantId) maxInvariantId = u.invariantId;
|
|
|
+ builder.addUnit(u.kind, u.model, u.conformation.operator, u.elements, Unit.Trait.None, invariantId);
|
|
|
}
|
|
|
- offsetInvariantId += maxInvariantId + 1
|
|
|
+ offsetInvariantId += maxInvariantId + 1;
|
|
|
}
|
|
|
|
|
|
- if (ctx.shouldUpdate) await ctx.update(`${name} - structure`)
|
|
|
- const s = builder.getStructure()
|
|
|
+ if (ctx.shouldUpdate) await ctx.update(`${name} - structure`);
|
|
|
+ const s = builder.getStructure();
|
|
|
for( let i = 0, il = s.models.length; i < il; ++i) {
|
|
|
- const { trajectoryInfo } = s.models[i]
|
|
|
- trajectoryInfo.size = il
|
|
|
- trajectoryInfo.index = i
|
|
|
+ const { trajectoryInfo } = s.models[i];
|
|
|
+ trajectoryInfo.size = il;
|
|
|
+ trajectoryInfo.index = i;
|
|
|
}
|
|
|
- return s
|
|
|
- })
|
|
|
+ return s;
|
|
|
+ });
|
|
|
}
|
|
|
|
|
|
async function handleHivRna(ctx: { runtime: RuntimeContext, fetch: AjaxTask }, packings: CellPacking[], baseUrl: string) {
|
|
|
for (let i = 0, il = packings.length; i < il; ++i) {
|
|
|
if (packings[i].name === 'HIV1_capsid_3j3q_PackInner_0_1_0') {
|
|
|
- const url = `${baseUrl}/extras/rna_allpoints.json`
|
|
|
+ const url = `${baseUrl}/extras/rna_allpoints.json`;
|
|
|
const data = await ctx.fetch({ url, type: 'string' }).runInContext(ctx.runtime);
|
|
|
- const { points } = await (new Response(data)).json() as { points: number[] }
|
|
|
+ const { points } = await (new Response(data)).json() as { points: number[] };
|
|
|
|
|
|
- const curve0: Vec3[] = []
|
|
|
+ const curve0: Vec3[] = [];
|
|
|
for (let j = 0, jl = points.length; j < jl; j += 3) {
|
|
|
- curve0.push(Vec3.fromArray(Vec3(), points, j))
|
|
|
+ curve0.push(Vec3.fromArray(Vec3(), points, j));
|
|
|
}
|
|
|
packings[i].ingredients['RNA'] = {
|
|
|
source: { pdb: 'RNA_U_Base.pdb', transform: { center: false } },
|
|
@@ -375,106 +375,106 @@ async function handleHivRna(ctx: { runtime: RuntimeContext, fetch: AjaxTask }, p
|
|
|
name: 'RNA',
|
|
|
nbCurve: 1,
|
|
|
curve0
|
|
|
- }
|
|
|
+ };
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
async function loadMembrane(name: string, plugin: PluginContext, runtime: RuntimeContext, state: State, params: LoadCellPackModelParams) {
|
|
|
- const fname: string = `${name}.bcif`
|
|
|
- let ingredientFiles: IngredientFiles = {}
|
|
|
+ const fname: string = `${name}.bcif`;
|
|
|
+ let ingredientFiles: IngredientFiles = {};
|
|
|
if (params.ingredients.files !== null) {
|
|
|
for (let i = 0, il = params.ingredients.files.length; i < il; ++i) {
|
|
|
- const file = params.ingredients.files.item(i)
|
|
|
- if (file) ingredientFiles[file.name] = file
|
|
|
+ const file = params.ingredients.files.item(i);
|
|
|
+ if (file) ingredientFiles[file.name] = file;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- let b = state.build().toRoot()
|
|
|
+ let b = state.build().toRoot();
|
|
|
if (fname in ingredientFiles) {
|
|
|
const file = ingredientFiles[fname];
|
|
|
- b = b.apply(StateTransforms.Data.ReadFile, { file, isBinary: true, label: file.name }, { state: { isGhost: true } })
|
|
|
+ b = b.apply(StateTransforms.Data.ReadFile, { file, isBinary: true, label: file.name }, { state: { isGhost: true } });
|
|
|
} else {
|
|
|
- const url = `${params.baseUrl}/membranes/${name}.bcif`
|
|
|
- b = b.apply(StateTransforms.Data.Download, { url, isBinary: true, label: name }, { state: { isGhost: true } })
|
|
|
+ const url = `${params.baseUrl}/membranes/${name}.bcif`;
|
|
|
+ b = b.apply(StateTransforms.Data.Download, { url, isBinary: true, label: name }, { state: { isGhost: true } });
|
|
|
}
|
|
|
|
|
|
const membrane = await b.apply(StateTransforms.Data.ParseCif, undefined, { state: { isGhost: true } })
|
|
|
.apply(StateTransforms.Model.TrajectoryFromMmCif, undefined, { state: { isGhost: true } })
|
|
|
.apply(StateTransforms.Model.ModelFromTrajectory, undefined, { state: { isGhost: true } })
|
|
|
.apply(StateTransforms.Model.StructureFromModel)
|
|
|
- .commit()
|
|
|
+ .commit();
|
|
|
|
|
|
const membraneParams = {
|
|
|
representation: params.preset.representation,
|
|
|
- }
|
|
|
- await CellpackMembranePreset.apply(membrane, membraneParams, plugin)
|
|
|
+ };
|
|
|
+ await CellpackMembranePreset.apply(membrane, membraneParams, plugin);
|
|
|
}
|
|
|
|
|
|
async function loadHivMembrane(plugin: PluginContext, runtime: RuntimeContext, state: State, params: LoadCellPackModelParams) {
|
|
|
- const url = `${params.baseUrl}/membranes/hiv_lipids.bcif`
|
|
|
+ const url = `${params.baseUrl}/membranes/hiv_lipids.bcif`;
|
|
|
const membrane = await state.build().toRoot()
|
|
|
.apply(StateTransforms.Data.Download, { label: 'hiv_lipids', url, isBinary: true }, { state: { isGhost: true } })
|
|
|
.apply(StateTransforms.Data.ParseCif, undefined, { state: { isGhost: true } })
|
|
|
.apply(StateTransforms.Model.TrajectoryFromMmCif, undefined, { state: { isGhost: true } })
|
|
|
.apply(StateTransforms.Model.ModelFromTrajectory, undefined, { state: { isGhost: true } })
|
|
|
.apply(StateTransforms.Model.StructureFromModel)
|
|
|
- .commit()
|
|
|
+ .commit();
|
|
|
|
|
|
const membraneParams = {
|
|
|
representation: params.preset.representation,
|
|
|
- }
|
|
|
+ };
|
|
|
|
|
|
- await CellpackMembranePreset.apply(membrane, membraneParams, plugin)
|
|
|
+ await CellpackMembranePreset.apply(membrane, membraneParams, plugin);
|
|
|
}
|
|
|
|
|
|
async function loadPackings(plugin: PluginContext, runtime: RuntimeContext, state: State, params: LoadCellPackModelParams) {
|
|
|
- let cellPackJson: StateBuilder.To<PSO.Format.Json, StateTransformer<PSO.Data.String, PSO.Format.Json>>
|
|
|
+ let cellPackJson: StateBuilder.To<PSO.Format.Json, StateTransformer<PSO.Data.String, PSO.Format.Json>>;
|
|
|
if (params.source.name === 'id') {
|
|
|
- const url = getCellPackModelUrl(params.source.params, params.baseUrl)
|
|
|
+ const url = getCellPackModelUrl(params.source.params, params.baseUrl);
|
|
|
cellPackJson = state.build().toRoot()
|
|
|
- .apply(StateTransforms.Data.Download, { url, isBinary: false, label: params.source.params }, { state: { isGhost: true } })
|
|
|
+ .apply(StateTransforms.Data.Download, { url, isBinary: false, label: params.source.params }, { state: { isGhost: true } });
|
|
|
} else {
|
|
|
- const file = params.source.params
|
|
|
+ const file = params.source.params;
|
|
|
if (file === null) {
|
|
|
- plugin.log.error('No file selected')
|
|
|
- return
|
|
|
+ plugin.log.error('No file selected');
|
|
|
+ return;
|
|
|
}
|
|
|
cellPackJson = state.build().toRoot()
|
|
|
- .apply(StateTransforms.Data.ReadFile, { file, isBinary: false, label: file.name }, { state: { isGhost: true } })
|
|
|
+ .apply(StateTransforms.Data.ReadFile, { file, isBinary: false, label: file.name }, { state: { isGhost: true } });
|
|
|
}
|
|
|
|
|
|
const cellPackBuilder = cellPackJson
|
|
|
.apply(StateTransforms.Data.ParseJson, undefined, { state: { isGhost: true } })
|
|
|
- .apply(ParseCellPack)
|
|
|
+ .apply(ParseCellPack);
|
|
|
|
|
|
- const cellPackObject = await state.updateTree(cellPackBuilder).runInContext(runtime)
|
|
|
- const { packings } = cellPackObject.obj!.data
|
|
|
+ const cellPackObject = await state.updateTree(cellPackBuilder).runInContext(runtime);
|
|
|
+ const { packings } = cellPackObject.obj!.data;
|
|
|
|
|
|
- await handleHivRna({ runtime, fetch: plugin.fetch }, packings, params.baseUrl)
|
|
|
+ await handleHivRna({ runtime, fetch: plugin.fetch }, packings, params.baseUrl);
|
|
|
|
|
|
for (let i = 0, il = packings.length; i < il; ++i) {
|
|
|
- const p = { packing: i, baseUrl: params.baseUrl, ingredientFiles: params.ingredients.files }
|
|
|
+ const p = { packing: i, baseUrl: params.baseUrl, ingredientFiles: params.ingredients.files };
|
|
|
|
|
|
const packing = await state.build()
|
|
|
.to(cellPackBuilder.ref)
|
|
|
.apply(StructureFromCellpack, p)
|
|
|
- .commit({ revertOnError: true })
|
|
|
+ .commit({ revertOnError: true });
|
|
|
|
|
|
- const structure = packing.obj?.data
|
|
|
+ const structure = packing.obj?.data;
|
|
|
if (structure) {
|
|
|
await CellPackInfoProvider.attach({ fetch: plugin.fetch, runtime }, structure, {
|
|
|
info: { packingsCount: packings.length, packingIndex: i }
|
|
|
- })
|
|
|
+ });
|
|
|
}
|
|
|
|
|
|
const packingParams = {
|
|
|
traceOnly: params.preset.traceOnly,
|
|
|
representation: params.preset.representation,
|
|
|
- }
|
|
|
- await CellpackPackingPreset.apply(packing, packingParams, plugin)
|
|
|
+ };
|
|
|
+ await CellpackPackingPreset.apply(packing, packingParams, plugin);
|
|
|
if ( packings[i].location === 'surface' ){
|
|
|
- await loadMembrane(packings[i].name, plugin, runtime, state, params)
|
|
|
+ await loadMembrane(packings[i].name, plugin, runtime, state, params);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
@@ -500,7 +500,7 @@ const LoadCellPackModelParams = {
|
|
|
traceOnly: PD.Boolean(false),
|
|
|
representation: PD.Select('gaussian-surface', PD.arrayToOptions(['spacefill', 'gaussian-surface', 'point', 'ellipsoid']))
|
|
|
}, { isExpanded: true })
|
|
|
-}
|
|
|
+};
|
|
|
type LoadCellPackModelParams = PD.Values<typeof LoadCellPackModelParams>
|
|
|
|
|
|
export const LoadCellPackModel = StateAction.build({
|
|
@@ -509,12 +509,12 @@ export const LoadCellPackModel = StateAction.build({
|
|
|
from: PSO.Root
|
|
|
})(({ state, params }, ctx: PluginContext) => Task.create('CellPack Loader', async taskCtx => {
|
|
|
if (!ctx.representation.structure.themes.colorThemeRegistry.has(CellPackColorThemeProvider)) {
|
|
|
- ctx.representation.structure.themes.colorThemeRegistry.add(CellPackColorThemeProvider)
|
|
|
+ ctx.representation.structure.themes.colorThemeRegistry.add(CellPackColorThemeProvider);
|
|
|
}
|
|
|
|
|
|
if (params.source.name === 'id' && params.source.params === 'hiv_lipids.bcif') {
|
|
|
- await loadHivMembrane(ctx, taskCtx, state, params)
|
|
|
+ await loadHivMembrane(ctx, taskCtx, state, params);
|
|
|
} else {
|
|
|
- await loadPackings(ctx, taskCtx, state, params)
|
|
|
+ await loadPackings(ctx, taskCtx, state, params);
|
|
|
}
|
|
|
}));
|