|
@@ -1,107 +0,0 @@
|
|
|
-/**
|
|
|
- * Copyright (c) 2017 mol* contributors, licensed under MIT, See LICENSE file for more info.
|
|
|
- *
|
|
|
- * @author David Sehnal <david.sehnal@gmail.com>
|
|
|
- */
|
|
|
-
|
|
|
-import { Structure, Unit } from '../structure'
|
|
|
-import { SortedArray } from 'mol-data/int';
|
|
|
-
|
|
|
-namespace StructureUtils {
|
|
|
- export function union(source: Structure, structures: Structure[]) {
|
|
|
- if (structures.length === 0) return Structure.Empty;
|
|
|
- if (structures.length === 1) return structures[0];
|
|
|
-
|
|
|
- const unitMap = new Map<number, SortedArray>();
|
|
|
- const fullUnits = new Set<number>();
|
|
|
-
|
|
|
- for (const { units } of structures) {
|
|
|
- for (let i = 0, _i = units.length; i < _i; i++) {
|
|
|
- const u = units[i];
|
|
|
- if (unitMap.has(u.id)) {
|
|
|
- // check if there is anything more to union in this particual unit.
|
|
|
- if (fullUnits.has(u.id)) continue;
|
|
|
- const merged = SortedArray.union(unitMap.get(u.id)!, u.elements);
|
|
|
- unitMap.set(u.id, merged);
|
|
|
- if (merged.length === source.unitMap.get(u.id).elements.length) fullUnits.add(u.id);
|
|
|
- } else {
|
|
|
- unitMap.set(u.id, u.elements);
|
|
|
- if (u.elements.length === source.unitMap.get(u.id).elements.length) fullUnits.add(u.id);
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- const builder = source.subsetBuilder(true);
|
|
|
- unitMap.forEach(buildUnion, builder);
|
|
|
- return builder.getStructure();
|
|
|
- }
|
|
|
-
|
|
|
- function buildUnion(this: Structure.SubsetBuilder, elements: SortedArray, id: number) {
|
|
|
- this.setUnit(id, elements);
|
|
|
- }
|
|
|
-
|
|
|
- export function areIntersecting(sA: Structure, sB: Structure): boolean {
|
|
|
- if (sA === sB) return true;
|
|
|
-
|
|
|
- let a, b;
|
|
|
- if (sA.units.length < sB.units.length) { a = sA; b = sB; }
|
|
|
- else { a = sB; b = sA; }
|
|
|
-
|
|
|
- const aU = a.units, bU = b.unitMap;
|
|
|
-
|
|
|
- for (let i = 0, _i = aU.length; i < _i; i++) {
|
|
|
- const u = aU[i];
|
|
|
- if (!bU.has(u.id)) continue;
|
|
|
- const v = bU.get(u.id);
|
|
|
- if (SortedArray.areIntersecting(u.elements, v.elements)) return true;
|
|
|
- }
|
|
|
-
|
|
|
- return false;
|
|
|
- }
|
|
|
-
|
|
|
- export function intersect(sA: Structure, sB: Structure): Structure {
|
|
|
- if (sA === sB) return sA;
|
|
|
- if (!areIntersecting(sA, sB)) return Structure.Empty;
|
|
|
-
|
|
|
- let a, b;
|
|
|
- if (sA.units.length < sB.units.length) { a = sA; b = sB; }
|
|
|
- else { a = sB; b = sA; }
|
|
|
-
|
|
|
- const aU = a.units, bU = b.unitMap;
|
|
|
- const units: Unit[] = [];
|
|
|
-
|
|
|
- for (let i = 0, _i = aU.length; i < _i; i++) {
|
|
|
- const u = aU[i];
|
|
|
- if (!bU.has(u.id)) continue;
|
|
|
- const v = bU.get(u.id);
|
|
|
- if (SortedArray.areIntersecting(u.elements, v.elements)) {
|
|
|
- const int = SortedArray.intersect(u.elements, v.elements);
|
|
|
- units[units.length] = u.getChild(int);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- return Structure.create(units);
|
|
|
- }
|
|
|
-
|
|
|
- export function subtract(a: Structure, b: Structure): Structure {
|
|
|
- if (a === b) return Structure.Empty;
|
|
|
- if (!areIntersecting(a, b)) return a;
|
|
|
-
|
|
|
- const aU = a.units, bU = b.unitMap;
|
|
|
- const units: Unit[] = [];
|
|
|
-
|
|
|
- for (let i = 0, _i = aU.length; i < _i; i++) {
|
|
|
- const u = aU[i];
|
|
|
- if (!bU.has(u.id)) continue;
|
|
|
- const v = bU.get(u.id);
|
|
|
- const sub = SortedArray.intersect(u.elements, v.elements);
|
|
|
- if (sub.length > 0) {
|
|
|
- units[units.length] = u.getChild(sub);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- return Structure.create(units);
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-export { StructureUtils }
|