validation.ts 3.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. /**
  2. * Copyright (c) 2023 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author Sebastian Bittrich <sebastian.bittrich@rcsb.org>
  5. */
  6. import { Structure } from 'molstar/lib/mol-model/structure/structure/structure';
  7. import { Unit } from 'molstar/lib/mol-model/structure/structure/unit';
  8. import { OrderedSet } from 'molstar/lib/mol-data/int';
  9. import { UnitIndex } from 'molstar/lib/mol-model/structure/structure/element/element';
  10. import { StructureElement, StructureProperties } from 'molstar/lib/mol-model/structure/structure';
  11. import { Vec3 } from 'molstar/lib/mol-math/linear-algebra/3d/vec3';
  12. import { StrucmotifCtx } from './helpers';
  13. export const MIN_MOTIF_SIZE = 2;
  14. export const MAX_MOTIF_SIZE = 10;
  15. export const MAX_EXCHANGES = 4;
  16. const MAX_MOTIF_EXTENT = 15;
  17. const MAX_MOTIF_EXTENT_SQUARED = MAX_MOTIF_EXTENT * MAX_MOTIF_EXTENT;
  18. export function determineBackboneAtom(structure: Structure, location: StructureElement.Location, element: { unit: Unit; indices: OrderedSet<UnitIndex> }) {
  19. const { label_atom_id } = StructureProperties.atom;
  20. const { indices } = element;
  21. for (let i = 0, il = OrderedSet.size(indices); i < il; i++) {
  22. StructureElement.Location.set(location, structure, element.unit, element.unit.elements[OrderedSet.getAt(indices, i)]);
  23. if (!Unit.isAtomic(location.unit)) return false;
  24. const atomLabelId = label_atom_id(location);
  25. if ('CA' === atomLabelId || `C4'` === atomLabelId) {
  26. return true;
  27. }
  28. }
  29. return false;
  30. }
  31. export function validate(ctx: StrucmotifCtx) {
  32. if (ctx.residueIds.length < MIN_MOTIF_SIZE) return false;
  33. if (ctx.pdbId.size > 1) {
  34. alert('Motifs can only be extracted from a single model!');
  35. return false;
  36. }
  37. if (ctx.sg.size > 1) {
  38. alert('Motifs can only appear in a single space-group!');
  39. return false;
  40. }
  41. if (ctx.hkl.size > 1) {
  42. alert('All motif residues must have matching hkl operators!');
  43. return false;
  44. }
  45. if (ctx.ncs.size > 1) {
  46. alert('All motif residues must have matching NCS operators!');
  47. return false;
  48. }
  49. if (ctx.residueIds.length > MAX_MOTIF_SIZE) {
  50. alert(`Maximum motif size is ${MAX_MOTIF_SIZE} residues!`);
  51. return false;
  52. }
  53. if (ctx.residueIds.filter(v => v.label_seq_id === 0).length > 0) {
  54. alert('Selections may only contain polymeric entities!');
  55. return false;
  56. }
  57. return validateAtomDistances(ctx);
  58. }
  59. function validateAtomDistances(ctx: StrucmotifCtx) {
  60. const { coordinates } = ctx;
  61. // warn if >15 A
  62. const a = Vec3();
  63. const b = Vec3();
  64. // this is not efficient but is good enough for up to 10 residues
  65. for (let i = 0, il = coordinates.length; i < il; i++) {
  66. Vec3.set(a, coordinates[i].coords[0], coordinates[i].coords[1], coordinates[i].coords[2]);
  67. let contact = false;
  68. for (let j = 0, jl = coordinates.length; j < jl; j++) {
  69. if (i === j) continue;
  70. Vec3.set(b, coordinates[j].coords[0], coordinates[j].coords[1], coordinates[j].coords[2]);
  71. const d = Vec3.squaredDistance(a, b);
  72. if (d < MAX_MOTIF_EXTENT_SQUARED) {
  73. contact = true;
  74. }
  75. }
  76. if (!contact) {
  77. const { residueId } = coordinates[i];
  78. alert(`Residue ${residueId.label_seq_id} | ${residueId.label_asym_id} | ${residueId.struct_oper_id} needs to be less than ${MAX_MOTIF_EXTENT} \u212B from another residue - Consider adding more residues to connect far-apart residues.`);
  79. return false;
  80. }
  81. }
  82. return true;
  83. }