util.ts 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. /**
  2. * Copyright (c) 2018-2023 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author Alexander Rose <alexander.rose@weirdbyte.de>
  5. */
  6. import { Sphere3D } from '../../mol-math/geometry';
  7. import { Vec3, Mat4 } from '../../mol-math/linear-algebra';
  8. import { BoundaryHelper } from '../../mol-math/geometry/boundary-helper';
  9. import { TextureFilter } from '../webgl/texture';
  10. import { arrayMinMax } from '../../mol-util/array';
  11. // avoiding namespace lookup improved performance in Chrome (Aug 2020)
  12. const v3fromArray = Vec3.fromArray;
  13. const v3transformMat4Offset = Vec3.transformMat4Offset;
  14. export function calculateTextureInfo(n: number, itemSize: number) {
  15. n = Math.max(n, 2); // observed issues with 1 pixel textures
  16. const sqN = Math.sqrt(n);
  17. let width = Math.ceil(sqN);
  18. width = width + (itemSize - (width % itemSize)) % itemSize;
  19. const height = width > 0 ? Math.ceil(n / width) : 0;
  20. return { width, height, length: width * height * itemSize };
  21. }
  22. export interface TextureImage<T extends Uint8Array | Float32Array | Int32Array> {
  23. readonly array: T
  24. readonly width: number
  25. readonly height: number
  26. readonly flipY?: boolean
  27. readonly filter?: TextureFilter
  28. }
  29. export interface TextureVolume<T extends Uint8Array | Float32Array> {
  30. readonly array: T
  31. readonly width: number
  32. readonly height: number
  33. readonly depth: number
  34. }
  35. export function createTextureImage<T extends Uint8Array | Float32Array>(n: number, itemSize: number, arrayCtor: new (length: number) => T, array?: T): TextureImage<T> {
  36. const { length, width, height } = calculateTextureInfo(n, itemSize);
  37. array = array && array.length >= length ? array : new arrayCtor(length);
  38. return { array, width, height };
  39. }
  40. const DefaultPrintImageOptions = {
  41. scale: 1,
  42. pixelated: false,
  43. id: 'molstar.debug.image',
  44. normalize: false,
  45. };
  46. export type PrintImageOptions = typeof DefaultPrintImageOptions
  47. export function printTextureImage(textureImage: TextureImage<any>, options: Partial<PrintImageOptions> = {}) {
  48. const { array, width, height } = textureImage;
  49. const itemSize = array.length / (width * height);
  50. const data = new Uint8ClampedArray(width * height * 4);
  51. if (itemSize === 1) {
  52. for (let y = 0; y < height; ++y) {
  53. for (let x = 0; x < width; ++x) {
  54. data[(y * width + x) * 4 + 3] = array[y * width + x];
  55. }
  56. }
  57. } else if (itemSize === 4) {
  58. if (options.normalize) {
  59. const [min, max] = arrayMinMax(array);
  60. for (let i = 0, il = width * height * 4; i < il; i += 4) {
  61. data[i] = ((array[i] - min) / (max - min)) * 255;
  62. data[i + 1] = ((array[i + 1] - min) / (max - min)) * 255;
  63. data[i + 2] = ((array[i + 2] - min) / (max - min)) * 255;
  64. data[i + 3] = 255;
  65. }
  66. } else {
  67. data.set(array);
  68. }
  69. } else {
  70. console.warn(`itemSize '${itemSize}' not supported`);
  71. }
  72. return printImageData(new ImageData(data, width, height), options);
  73. }
  74. let tmpCanvas: HTMLCanvasElement;
  75. let tmpCanvasCtx: CanvasRenderingContext2D;
  76. let tmpContainer: HTMLDivElement;
  77. export function printImageData(imageData: ImageData, options: Partial<PrintImageOptions> = {}) {
  78. const o = { ...DefaultPrintImageOptions, ...options };
  79. const canvas = tmpCanvas || document.createElement('canvas');
  80. tmpCanvas = canvas;
  81. canvas.width = imageData.width;
  82. canvas.height = imageData.height;
  83. const ctx = tmpCanvasCtx || canvas.getContext('2d');
  84. tmpCanvasCtx = ctx;
  85. if (!ctx) throw new Error('Could not create canvas 2d context');
  86. ctx.putImageData(imageData, 0, 0);
  87. if (!tmpContainer) {
  88. tmpContainer = document.createElement('div');
  89. tmpContainer.style.position = 'absolute';
  90. tmpContainer.style.bottom = '0px';
  91. tmpContainer.style.right = '0px';
  92. tmpContainer.style.border = 'solid orange';
  93. tmpContainer.style.pointerEvents = 'none';
  94. document.body.appendChild(tmpContainer);
  95. }
  96. canvas.toBlob(imgBlob => {
  97. const objectURL = URL.createObjectURL(imgBlob!);
  98. const existingImg = document.getElementById(o.id) as HTMLImageElement;
  99. const img = existingImg || document.createElement('img');
  100. img.id = o.id;
  101. img.src = objectURL;
  102. img.style.width = imageData.width * o.scale + 'px';
  103. img.style.height = imageData.height * o.scale + 'px';
  104. if (o.pixelated) {
  105. // not supported in Firefox and IE
  106. img.style.imageRendering = 'pixelated';
  107. }
  108. img.style.position = 'relative';
  109. img.style.border = 'solid grey';
  110. img.style.pointerEvents = 'none';
  111. if (!existingImg) tmpContainer.appendChild(img);
  112. }, 'image/png');
  113. }
  114. //
  115. const v = Vec3();
  116. const boundaryHelperCoarse = new BoundaryHelper('14');
  117. const boundaryHelperFine = new BoundaryHelper('98');
  118. function getHelper(count: number) {
  119. return count > 100_000 ? boundaryHelperCoarse : boundaryHelperFine;
  120. }
  121. export function calculateInvariantBoundingSphere(position: Float32Array, positionCount: number, stepFactor: number): Sphere3D {
  122. const step = stepFactor * 3;
  123. const boundaryHelper = getHelper(positionCount);
  124. boundaryHelper.reset();
  125. for (let i = 0, _i = positionCount * 3; i < _i; i += step) {
  126. v3fromArray(v, position, i);
  127. boundaryHelper.includePosition(v);
  128. }
  129. boundaryHelper.finishedIncludeStep();
  130. for (let i = 0, _i = positionCount * 3; i < _i; i += step) {
  131. v3fromArray(v, position, i);
  132. boundaryHelper.radiusPosition(v);
  133. }
  134. const sphere = boundaryHelper.getSphere();
  135. if (positionCount <= 14) {
  136. const extrema: Vec3[] = [];
  137. for (let i = 0, _i = positionCount * 3; i < _i; i += step) {
  138. extrema.push(v3fromArray(Vec3(), position, i));
  139. }
  140. Sphere3D.setExtrema(sphere, extrema);
  141. }
  142. return sphere;
  143. }
  144. const _mat4 = Mat4();
  145. export function calculateTransformBoundingSphere(invariantBoundingSphere: Sphere3D, transform: Float32Array, transformCount: number, transformOffset: number): Sphere3D {
  146. if (transformCount === 1) {
  147. Mat4.fromArray(_mat4, transform, transformOffset);
  148. const s = Sphere3D.clone(invariantBoundingSphere);
  149. return Mat4.isIdentity(_mat4) ? s : Sphere3D.transform(s, s, _mat4);
  150. }
  151. const boundaryHelper = getHelper(transformCount);
  152. boundaryHelper.reset();
  153. const { center, radius, extrema } = invariantBoundingSphere;
  154. // only use extrema if there are not too many transforms
  155. if (extrema && transformCount <= 14) {
  156. for (let i = 0, _i = transformCount; i < _i; ++i) {
  157. for (const e of extrema) {
  158. v3transformMat4Offset(v, e, transform, 0, 0, i * 16 + transformOffset);
  159. boundaryHelper.includePosition(v);
  160. }
  161. }
  162. boundaryHelper.finishedIncludeStep();
  163. for (let i = 0, _i = transformCount; i < _i; ++i) {
  164. for (const e of extrema) {
  165. v3transformMat4Offset(v, e, transform, 0, 0, i * 16 + transformOffset);
  166. boundaryHelper.radiusPosition(v);
  167. }
  168. }
  169. } else {
  170. for (let i = 0, _i = transformCount; i < _i; ++i) {
  171. v3transformMat4Offset(v, center, transform, 0, 0, i * 16 + transformOffset);
  172. boundaryHelper.includePositionRadius(v, radius);
  173. }
  174. boundaryHelper.finishedIncludeStep();
  175. for (let i = 0, _i = transformCount; i < _i; ++i) {
  176. v3transformMat4Offset(v, center, transform, 0, 0, i * 16 + transformOffset);
  177. boundaryHelper.radiusPositionRadius(v, radius);
  178. }
  179. }
  180. return boundaryHelper.getSphere();
  181. }
  182. export function calculateBoundingSphere(position: Float32Array, positionCount: number, transform: Float32Array, transformCount: number, padding = 0, stepFactor = 1): { boundingSphere: Sphere3D, invariantBoundingSphere: Sphere3D } {
  183. const invariantBoundingSphere = calculateInvariantBoundingSphere(position, positionCount, stepFactor);
  184. const boundingSphere = calculateTransformBoundingSphere(invariantBoundingSphere, transform, transformCount, 0);
  185. Sphere3D.expand(boundingSphere, boundingSphere, padding);
  186. Sphere3D.expand(invariantBoundingSphere, invariantBoundingSphere, padding);
  187. return { boundingSphere, invariantBoundingSphere };
  188. }