diff --git a/src/mol-model/structure/structure/structure.ts b/src/mol-model/structure/structure/structure.ts index 11069dda8cca280e42bf5a907ca10c7b8d2247da..27c0c467c3840d914236663d3a71e0df15987f10 100644 --- a/src/mol-model/structure/structure/structure.ts +++ b/src/mol-model/structure/structure/structure.ts @@ -397,6 +397,8 @@ namespace Structure { if (isWaterChain(model, c as ChainIndex)) { partitionAtomicUnit(model, elements, builder); + } else if (elements.length > 200000) { + partitionAtomicUnitPerResidue(model, elements, builder); } else { builder.addUnit(Unit.Kind.Atomic, model, SymmetryOperator.Default, elements); } @@ -435,6 +437,36 @@ namespace Structure { } } + function partitionAtomicUnitPerResidue(model: Model, indices: SortedArray, builder: StructureBuilder) { + model.atomicHierarchy.residueAtomSegments.offsets + + const startIndices: number[] = [] + const endIndices: number[] = [] + + const residueIt = Segmentation.transientSegments(model.atomicHierarchy.residueAtomSegments, indices) + while (residueIt.hasNext) { + const residueSegment = residueIt.move(); + startIndices[startIndices.length] = residueSegment.start + endIndices[endIndices.length] = residueSegment.end + } + + const { x, y, z } = model.atomicConformation; + const lookup = GridLookup3D({ x, y, z, indices: SortedArray.ofSortedArray(startIndices) }, Vec3.create(256, 256, 256)); + const { offset, count, array } = lookup.buckets; + + for (let i = 0, _i = offset.length; i < _i; i++) { + const start = offset[i]; + const set: number[] = []; + for (let j = 0, _j = count[i]; j < _j; j++) { + const k = array[start + j] + for (let l = startIndices[k], _l = endIndices[k]; l < _l; l++) { + set[set.length] = l; + } + } + builder.addUnit(Unit.Kind.Atomic, model, SymmetryOperator.Default, SortedArray.ofSortedArray(set)); + } + } + function addCoarseUnits(builder: StructureBuilder, model: Model, elements: CoarseElements, kind: Unit.Kind) { const { chainElementSegments } = elements; for (let cI = 0; cI < chainElementSegments.count; cI++) {