UNPKG

molstar

Version:

A comprehensive macromolecular library.

200 lines (199 loc) 9.06 kB
/** * Copyright (c) 2018-2026 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author Alexander Rose <alexander.rose@weirdbyte.de> * @author Gianluca Tomasello <giagitom@gmail.com> */ import { ChunkedArray } from '../../../mol-data/util.js'; import { Lines } from './lines.js'; import { Vec3 } from '../../../mol-math/linear-algebra.js'; // avoiding namespace lookup improved performance in Chrome (Aug 2020) const caAdd = ChunkedArray.add; const caAdd2 = ChunkedArray.add2; const caAdd3 = ChunkedArray.add3; const tmpVecA = Vec3(); const tmpVecB = Vec3(); const tmpDir = Vec3(); export var LinesBuilder; (function (LinesBuilder) { function create(initialCount = 2048, chunkSize = 1024, lines) { const groups = ChunkedArray.create(Float32Array, 1, chunkSize, lines ? lines.groupBuffer.ref.value : initialCount); const starts = ChunkedArray.create(Float32Array, 3, chunkSize, lines ? lines.startBuffer.ref.value : initialCount); const ends = ChunkedArray.create(Float32Array, 3, chunkSize, lines ? lines.endBuffer.ref.value : initialCount); const add = (startX, startY, startZ, endX, endY, endZ, group) => { for (let i = 0; i < 4; ++i) { caAdd3(starts, startX, startY, startZ); caAdd3(ends, endX, endY, endZ); caAdd(groups, group); } }; const addVec = (start, end, group) => { for (let i = 0; i < 4; ++i) { caAdd3(starts, start[0], start[1], start[2]); caAdd3(ends, end[0], end[1], end[2]); caAdd(groups, group); } }; const addFixedCountDashes = (start, end, segmentCount, group) => { const d = Vec3.distance(start, end); const isOdd = segmentCount % 2 !== 0; const s = Math.floor((segmentCount + 1) / 2); const step = d / (segmentCount + 0.5); Vec3.setMagnitude(tmpDir, Vec3.sub(tmpDir, end, start), step); Vec3.copy(tmpVecA, start); for (let j = 0; j < s; ++j) { Vec3.add(tmpVecA, tmpVecA, tmpDir); if (isOdd && j === s - 1) { Vec3.copy(tmpVecB, end); } else { Vec3.add(tmpVecB, tmpVecA, tmpDir); } add(tmpVecA[0], tmpVecA[1], tmpVecA[2], tmpVecB[0], tmpVecB[1], tmpVecB[2], group); Vec3.add(tmpVecA, tmpVecA, tmpDir); } }; return { add, addVec, addFixedCountDashes, addFixedLengthDashes: (start, end, segmentLength, group) => { const d = Vec3.distance(start, end); addFixedCountDashes(start, end, d / segmentLength, group); }, addCage: (t, cage, group) => { const { vertices, edges } = cage; for (let i = 0, il = edges.length; i < il; i += 2) { Vec3.fromArray(tmpVecA, vertices, edges[i] * 3); Vec3.fromArray(tmpVecB, vertices, edges[i + 1] * 3); Vec3.transformMat4(tmpVecA, tmpVecA, t); Vec3.transformMat4(tmpVecB, tmpVecB, t); add(tmpVecA[0], tmpVecA[1], tmpVecA[2], tmpVecB[0], tmpVecB[1], tmpVecB[2], group); } }, getLines: () => { const lineCount = groups.elementCount / 4; const vertexCount = groups.elementCount; const gb = ChunkedArray.compact(groups, true); const sb = ChunkedArray.compact(starts, true); const eb = ChunkedArray.compact(ends, true); const mb = lines && lineCount <= lines.lineCount && lines.stripCount.ref.value === 0 ? lines.mappingBuffer.ref.value : new Float32Array(lineCount * 8); const ib = lines && lineCount <= lines.lineCount && lines.stripCount.ref.value === 0 ? lines.indexBuffer.ref.value : new Uint32Array(lineCount * 6); const ob = lines ? lines.stripBuffer.ref.value : new Uint32Array(0); if (!lines || lineCount > lines.lineCount || lines.stripCount.ref.value > 0) fillMappingAndIndices(lineCount, mb, ib); return Lines.create(mb, ib, gb, sb, eb, ob, lineCount, vertexCount, 0, lines); } }; } LinesBuilder.create = create; })(LinesBuilder || (LinesBuilder = {})); function fillMappingAndIndices(n, mb, ib) { for (let i = 0; i < n; ++i) { const mo = i * 8; mb[mo] = -1; mb[mo + 1] = -1; mb[mo + 2] = 1; mb[mo + 3] = -1; mb[mo + 4] = -1; mb[mo + 5] = 1; mb[mo + 6] = 1; mb[mo + 7] = 1; } for (let i = 0; i < n; ++i) { const o = i * 4; const io = i * 6; ib[io] = o; ib[io + 1] = o + 1; ib[io + 2] = o + 2; ib[io + 3] = o + 1; ib[io + 4] = o + 3; ib[io + 5] = o + 2; } } export var StripLinesBuilder; (function (StripLinesBuilder) { function create(initialCount = 2048, chunkSize = 1024, lines) { const groups = ChunkedArray.create(Float32Array, 1, chunkSize, lines ? lines.groupBuffer.ref.value : initialCount); const starts = ChunkedArray.create(Float32Array, 3, chunkSize, lines ? lines.startBuffer.ref.value : initialCount); const ends = ChunkedArray.create(Float32Array, 3, chunkSize, lines ? lines.endBuffer.ref.value : initialCount); const mapping = ChunkedArray.create(Float32Array, 2, chunkSize, lines ? lines.mappingBuffer.ref.value : initialCount); const indices = ChunkedArray.create(Uint32Array, 3, chunkSize, lines ? lines.indexBuffer.ref.value : initialCount); const strips = ChunkedArray.create(Uint32Array, 1, chunkSize, lines ? lines.stripBuffer.ref.value : initialCount); let stripGroup = 0; let pointCount = 0; let firstVertexOffset = 0; let prevX = 0, prevY = 0, prevZ = 0; const addPoint = (x, y, z) => { if (pointCount === 0) { firstVertexOffset = groups.elementCount; prevX = x; prevY = y; prevZ = z; pointCount = 1; return; } const vertexOffset = groups.elementCount; if (pointCount === 1) { caAdd3(starts, prevX, prevY, prevZ); caAdd3(ends, x, y, z); caAdd(groups, stripGroup); caAdd2(mapping, -1, -1); // left, start caAdd3(starts, prevX, prevY, prevZ); caAdd3(ends, x, y, z); caAdd(groups, stripGroup); caAdd2(mapping, 1, -1); // right, start } caAdd3(starts, prevX, prevY, prevZ); caAdd3(ends, x, y, z); caAdd(groups, stripGroup); caAdd2(mapping, -1, 1); // left, end caAdd3(starts, prevX, prevY, prevZ); caAdd3(ends, x, y, z); caAdd(groups, stripGroup); caAdd2(mapping, 1, 1); // right, end const prevOffset = pointCount === 1 ? firstVertexOffset : vertexOffset - 2; const currOffset = pointCount === 1 ? vertexOffset + 2 : vertexOffset; // Triangle 1: prev-left, prev-right, curr-left caAdd3(indices, prevOffset, prevOffset + 1, currOffset); // Triangle 2: prev-right, curr-right, curr-left caAdd3(indices, prevOffset + 1, currOffset + 1, currOffset); prevX = x; prevY = y; prevZ = z; pointCount++; }; return { start: (group) => { stripGroup = group; pointCount = 0; if (strips.elementCount === 0) { caAdd(strips, 0); } }, add: (x, y, z) => { addPoint(x, y, z); }, addVec: (v) => { addPoint(v[0], v[1], v[2]); }, end: () => { pointCount = 0; caAdd(strips, groups.elementCount); }, getLines: () => { const lineCount = indices.elementCount / 2; const vertexCount = groups.elementCount; const stripCount = strips.elementCount - 1; const gb = ChunkedArray.compact(groups, true); const sb = ChunkedArray.compact(starts, true); const eb = ChunkedArray.compact(ends, true); const mb = ChunkedArray.compact(mapping, true); const ib = ChunkedArray.compact(indices, true); const ob = ChunkedArray.compact(strips, true); return Lines.create(mb, ib, gb, sb, eb, ob, lineCount, vertexCount, stripCount, lines); } }; } StripLinesBuilder.create = create; })(StripLinesBuilder || (StripLinesBuilder = {}));