@genome-spy/core 0.14.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (226) hide show
  1. package/dist/index.js +224 -0
  2. package/dist/style.css +1 -0
  3. package/package.json +54 -0
  4. package/src/data/collector.js +178 -0
  5. package/src/data/collector.test.js +82 -0
  6. package/src/data/dataFlow.js +109 -0
  7. package/src/data/dataFlow.test.js +3 -0
  8. package/src/data/facetNode.js +17 -0
  9. package/src/data/flow.test.js +71 -0
  10. package/src/data/flowBatch.d.ts +40 -0
  11. package/src/data/flowNode.js +283 -0
  12. package/src/data/flowNode.test.js +49 -0
  13. package/src/data/flowOptimizer.js +117 -0
  14. package/src/data/flowOptimizer.test.js +192 -0
  15. package/src/data/flowTestUtils.js +63 -0
  16. package/src/data/formats/fasta.js +32 -0
  17. package/src/data/formats/fasta.test.js +26 -0
  18. package/src/data/sources/dataSource.js +22 -0
  19. package/src/data/sources/dataSourceFactory.js +24 -0
  20. package/src/data/sources/dataUtils.js +31 -0
  21. package/src/data/sources/dynamicCallbackSource.js +56 -0
  22. package/src/data/sources/dynamicSource.js +36 -0
  23. package/src/data/sources/inlineSource.js +69 -0
  24. package/src/data/sources/inlineSource.test.js +55 -0
  25. package/src/data/sources/namedSource.js +74 -0
  26. package/src/data/sources/sequenceSource.js +46 -0
  27. package/src/data/sources/sequenceSource.test.js +45 -0
  28. package/src/data/sources/urlSource.js +74 -0
  29. package/src/data/transforms/aggregate.js +69 -0
  30. package/src/data/transforms/clone.js +40 -0
  31. package/src/data/transforms/clone.test.js +10 -0
  32. package/src/data/transforms/coverage.js +187 -0
  33. package/src/data/transforms/coverage.test.js +122 -0
  34. package/src/data/transforms/filter.js +37 -0
  35. package/src/data/transforms/filter.test.js +17 -0
  36. package/src/data/transforms/filterScoredLabels.js +134 -0
  37. package/src/data/transforms/flattenCompressedExons.js +57 -0
  38. package/src/data/transforms/flattenDelimited.js +68 -0
  39. package/src/data/transforms/flattenDelimited.test.js +86 -0
  40. package/src/data/transforms/flattenSequence.js +39 -0
  41. package/src/data/transforms/flattenSequence.test.js +33 -0
  42. package/src/data/transforms/formula.js +39 -0
  43. package/src/data/transforms/formula.test.js +18 -0
  44. package/src/data/transforms/identifier.js +108 -0
  45. package/src/data/transforms/identifier.test.js +82 -0
  46. package/src/data/transforms/linearizeGenomicCoordinate.js +101 -0
  47. package/src/data/transforms/measureText.js +44 -0
  48. package/src/data/transforms/pileup.js +128 -0
  49. package/src/data/transforms/pileup.test.js +69 -0
  50. package/src/data/transforms/project.js +41 -0
  51. package/src/data/transforms/project.test.js +31 -0
  52. package/src/data/transforms/regexExtract.js +61 -0
  53. package/src/data/transforms/regexExtract.test.js +66 -0
  54. package/src/data/transforms/regexFold.js +141 -0
  55. package/src/data/transforms/regexFold.test.js +159 -0
  56. package/src/data/transforms/sample.js +101 -0
  57. package/src/data/transforms/sample.test.js +37 -0
  58. package/src/data/transforms/stack.js +137 -0
  59. package/src/data/transforms/stack.test.js +90 -0
  60. package/src/data/transforms/transformFactory.js +60 -0
  61. package/src/encoder/accessor.js +82 -0
  62. package/src/encoder/accessor.test.js +46 -0
  63. package/src/encoder/encoder.js +369 -0
  64. package/src/encoder/encoder.test.js +97 -0
  65. package/src/fonts/Lato-Regular.json +1267 -0
  66. package/src/fonts/Lato-Regular.png +0 -0
  67. package/src/fonts/OFL.txt +93 -0
  68. package/src/fonts/README.md +3 -0
  69. package/src/fonts/bmFont.d.ts +58 -0
  70. package/src/fonts/bmFontManager.js +357 -0
  71. package/src/fonts/bmFontMetrics.js +108 -0
  72. package/src/genome/genome.js +305 -0
  73. package/src/genome/genome.test.js +152 -0
  74. package/src/genome/genomeStore.js +54 -0
  75. package/src/genome/locusFormat.js +31 -0
  76. package/src/genome/scaleIndex.js +199 -0
  77. package/src/genome/scaleIndex.test.js +61 -0
  78. package/src/genome/scaleLocus.js +112 -0
  79. package/src/genome/scaleLocus.test.js +3 -0
  80. package/src/genomeSpy.js +753 -0
  81. package/src/gl/arrayBuilder.js +199 -0
  82. package/src/gl/dataToVertices.js +621 -0
  83. package/src/gl/includes/common.glsl +63 -0
  84. package/src/gl/includes/fp64-arithmetic.glsl +187 -0
  85. package/src/gl/includes/fp64-utils.js +132 -0
  86. package/src/gl/includes/picking.fragment.glsl +3 -0
  87. package/src/gl/includes/picking.vertex.glsl +29 -0
  88. package/src/gl/includes/sampleFacet.glsl +107 -0
  89. package/src/gl/includes/scales.glsl +79 -0
  90. package/src/gl/includes/scales_fp64.glsl +30 -0
  91. package/src/gl/link.fragment.glsl +18 -0
  92. package/src/gl/link.vertex.glsl +111 -0
  93. package/src/gl/point.fragment.glsl +123 -0
  94. package/src/gl/point.vertex.glsl +128 -0
  95. package/src/gl/rect.fragment.glsl +51 -0
  96. package/src/gl/rect.vertex.glsl +114 -0
  97. package/src/gl/rule.fragment.glsl +52 -0
  98. package/src/gl/rule.vertex.glsl +89 -0
  99. package/src/gl/text.fragment.glsl +31 -0
  100. package/src/gl/text.vertex.glsl +246 -0
  101. package/src/gl/webGLHelper.js +490 -0
  102. package/src/img/bowtie.svg +1 -0
  103. package/src/img/genomespy-favicon.svg +34 -0
  104. package/src/index.html +11 -0
  105. package/src/index.js +151 -0
  106. package/src/marks/link.js +189 -0
  107. package/src/marks/mark.js +867 -0
  108. package/src/marks/markUtils.js +109 -0
  109. package/src/marks/pointMark.js +279 -0
  110. package/src/marks/rectMark.js +236 -0
  111. package/src/marks/rule.js +231 -0
  112. package/src/marks/text.js +274 -0
  113. package/src/options.d.ts +9 -0
  114. package/src/scale/colorUtils.js +184 -0
  115. package/src/scale/glslScaleGenerator.js +462 -0
  116. package/src/scale/scale.js +441 -0
  117. package/src/scale/scale.test.js +323 -0
  118. package/src/scale/ticks.js +198 -0
  119. package/src/scale/ticks.test.js +39 -0
  120. package/src/singlePageApp.js +13 -0
  121. package/src/spec/axis.d.ts +296 -0
  122. package/src/spec/channel.d.ts +127 -0
  123. package/src/spec/data.d.ts +185 -0
  124. package/src/spec/font.d.ts +15 -0
  125. package/src/spec/genome.d.ts +35 -0
  126. package/src/spec/mark.d.ts +432 -0
  127. package/src/spec/root.d.ts +22 -0
  128. package/src/spec/scale.d.ts +265 -0
  129. package/src/spec/tooltip.d.ts +9 -0
  130. package/src/spec/transform.d.ts +479 -0
  131. package/src/spec/view.d.ts +215 -0
  132. package/src/styles/genome-spy.scss +153 -0
  133. package/src/tooltip/dataTooltipHandler.js +59 -0
  134. package/src/tooltip/refseqGeneTooltipHandler.js +77 -0
  135. package/src/tooltip/tooltipHandler.ts +12 -0
  136. package/src/types/filetypes.d.ts +4 -0
  137. package/src/types/flatqueue.d.ts +53 -0
  138. package/src/types/glsl.d.ts +4 -0
  139. package/src/types/object.d.ts +21 -0
  140. package/src/types/vega-scale.d.ts +60 -0
  141. package/src/utils/animator.js +83 -0
  142. package/src/utils/arrayUtils.js +55 -0
  143. package/src/utils/binnedRangeIndex.js +83 -0
  144. package/src/utils/clamp.js +8 -0
  145. package/src/utils/cloner.js +32 -0
  146. package/src/utils/cloner.test.js +23 -0
  147. package/src/utils/coalesce.js +11 -0
  148. package/src/utils/coalesce.test.js +15 -0
  149. package/src/utils/concatIterables.js +26 -0
  150. package/src/utils/concatIterables.test.js +7 -0
  151. package/src/utils/debounce.js +37 -0
  152. package/src/utils/domainArray.js +224 -0
  153. package/src/utils/domainArray.test.js +129 -0
  154. package/src/utils/eerp.js +13 -0
  155. package/src/utils/expression.js +32 -0
  156. package/src/utils/field.js +28 -0
  157. package/src/utils/fisheye.js +60 -0
  158. package/src/utils/formatObject.js +31 -0
  159. package/src/utils/html.js +23 -0
  160. package/src/utils/html.test.js +13 -0
  161. package/src/utils/indexer.js +43 -0
  162. package/src/utils/indexer.test.js +46 -0
  163. package/src/utils/inertia.js +124 -0
  164. package/src/utils/interactionEvent.js +33 -0
  165. package/src/utils/iterateNestedMaps.js +21 -0
  166. package/src/utils/iterateNestedMaps.test.js +32 -0
  167. package/src/utils/kWayMerge.js +42 -0
  168. package/src/utils/kWayMerge.test.js +25 -0
  169. package/src/utils/layout/flexLayout.js +336 -0
  170. package/src/utils/layout/flexLayout.test.js +296 -0
  171. package/src/utils/layout/padding.js +107 -0
  172. package/src/utils/layout/point.js +23 -0
  173. package/src/utils/layout/rectangle.js +282 -0
  174. package/src/utils/layout/rectangle.test.js +171 -0
  175. package/src/utils/mergeObjects.js +99 -0
  176. package/src/utils/mergeObjects.test.js +41 -0
  177. package/src/utils/numberExtractor.js +24 -0
  178. package/src/utils/numberExtractor.test.js +5 -0
  179. package/src/utils/point.js +14 -0
  180. package/src/utils/propertyCacher.js +70 -0
  181. package/src/utils/propertyCacher.test.js +84 -0
  182. package/src/utils/propertyCoalescer.js +37 -0
  183. package/src/utils/propertyCoalescer.test.js +21 -0
  184. package/src/utils/reservationMap.js +103 -0
  185. package/src/utils/reservationMap.test.js +19 -0
  186. package/src/utils/scaleNull.js +19 -0
  187. package/src/utils/setOperations.js +75 -0
  188. package/src/utils/smoothstep.js +10 -0
  189. package/src/utils/throttle.js +34 -0
  190. package/src/utils/topK.js +76 -0
  191. package/src/utils/topK.test.js +63 -0
  192. package/src/utils/transition.js +74 -0
  193. package/src/utils/ui/tooltip.js +189 -0
  194. package/src/utils/url.js +22 -0
  195. package/src/utils/variableTools.js +24 -0
  196. package/src/utils/variableTools.test.js +12 -0
  197. package/src/view/axisResolution.js +135 -0
  198. package/src/view/axisResolution.test.js +200 -0
  199. package/src/view/axisView.js +746 -0
  200. package/src/view/channel.js +5 -0
  201. package/src/view/concatView.js +296 -0
  202. package/src/view/containerView.js +141 -0
  203. package/src/view/decoratorView.js +510 -0
  204. package/src/view/facetView.js +488 -0
  205. package/src/view/flowBuilder.js +362 -0
  206. package/src/view/flowBuilder.test.js +124 -0
  207. package/src/view/importView.js +19 -0
  208. package/src/view/layerView.js +60 -0
  209. package/src/view/rendering.d.ts +44 -0
  210. package/src/view/renderingContext/compositeViewRenderingContext.js +51 -0
  211. package/src/view/renderingContext/deferredViewRenderingContext.js +174 -0
  212. package/src/view/renderingContext/layoutRecorderViewRenderingContext.js +128 -0
  213. package/src/view/renderingContext/simpleViewRenderingContext.js +62 -0
  214. package/src/view/renderingContext/svgViewRenderingContext.js +121 -0
  215. package/src/view/renderingContext/viewRenderingContext.js +41 -0
  216. package/src/view/scaleResolution.js +756 -0
  217. package/src/view/scaleResolution.test.js +571 -0
  218. package/src/view/scaleResolutionApi.d.ts +40 -0
  219. package/src/view/testUtils.js +48 -0
  220. package/src/view/unitView.js +368 -0
  221. package/src/view/view.js +589 -0
  222. package/src/view/view.test.js +213 -0
  223. package/src/view/viewContext.d.ts +57 -0
  224. package/src/view/viewFactory.js +179 -0
  225. package/src/view/viewFactory.test.js +16 -0
  226. package/src/view/viewUtils.js +420 -0
@@ -0,0 +1,490 @@
1
+ import {
2
+ addExtensionsToContext,
3
+ createFramebufferInfo,
4
+ createTexture,
5
+ getContext,
6
+ isWebGL2,
7
+ resizeFramebufferInfo,
8
+ setTextureFromArray,
9
+ } from "twgl.js";
10
+ import { isArray, isString } from "vega-util";
11
+ import { getPlatformShaderDefines } from "./includes/fp64-utils";
12
+
13
+ import { isDiscrete, isDiscretizing, isInterpolating } from "vega-scale";
14
+ import {
15
+ createDiscreteColorTexture,
16
+ createDiscreteTexture,
17
+ createInterpolatedColorTexture,
18
+ createSchemeTexture,
19
+ } from "../scale/colorUtils";
20
+ import {
21
+ getDiscreteRangeMapper,
22
+ isColorChannel,
23
+ isDiscreteChannel,
24
+ } from "../encoder/encoder";
25
+
26
+ export default class WebGLHelper {
27
+ /**
28
+ *
29
+ * @param {HTMLElement} container
30
+ * @param {() => {width: number, height: number}} [sizeSource]
31
+ * A function that returns the content size. If a dimension is undefined,
32
+ * the canvas fills the container, otherwise the canvas is adjusted to the content size.
33
+ */
34
+ constructor(container, sizeSource) {
35
+ this._container = container;
36
+ this._sizeSource = sizeSource;
37
+
38
+ /** @type {Map<string, WebGLShader>} */
39
+ this._shaderCache = new Map();
40
+
41
+ /** @type {{ type: string, listener: function}[]} */
42
+ this._listeners = [];
43
+
44
+ /** @type {WeakMap<import("../view/scaleResolution").default, WebGLTexture>} */
45
+ this.rangeTextures = new WeakMap();
46
+
47
+ // --------------------------------------------------------
48
+
49
+ const canvas = document.createElement("canvas");
50
+
51
+ container.appendChild(canvas);
52
+
53
+ // TODO: Consider using high-performance powerPreference:
54
+ // https://www.khronos.org/webgl/public-mailing-list/public_webgl/1912/msg00001.php
55
+
56
+ const gl = /** @type {WebGL2RenderingContext} */ (
57
+ getContext(canvas, {
58
+ antialias: true,
59
+ // Disable depth writes. We don't use depth testing.
60
+ depth: false,
61
+ premultipliedAlpha: true,
62
+ })
63
+ );
64
+
65
+ if (!gl) {
66
+ throw new Error(
67
+ "Unable to initialize WebGL. Your browser or machine may not support it."
68
+ );
69
+ }
70
+
71
+ if (!isWebGL2(gl)) {
72
+ throw new Error(
73
+ "Your web browser does not support WebGL 2.0. Chrome, Firefox, and Safari Tech Preview should work."
74
+ );
75
+ }
76
+
77
+ addExtensionsToContext(gl);
78
+
79
+ // TODO: view background: https://vega.github.io/vega-lite/docs/spec.html#view-background
80
+
81
+ // Always use pre-multiplied alpha
82
+ gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA);
83
+
84
+ this._shaderDefines = getPlatformShaderDefines(gl);
85
+
86
+ this.canvas = canvas;
87
+ this.gl = gl;
88
+
89
+ // Setup framebuffer for piccking
90
+ /** @type {import("twgl.js").AttachmentOptions[]} */
91
+ this._pickingAttachmentOptions = [
92
+ {
93
+ format: gl.RGBA,
94
+ type: gl.UNSIGNED_BYTE,
95
+ minMag: gl.LINEAR,
96
+ wrap: gl.CLAMP_TO_EDGE,
97
+ },
98
+ ];
99
+ this._pickingBufferInfo = createFramebufferInfo(
100
+ gl,
101
+ this._pickingAttachmentOptions
102
+ );
103
+ gl.bindFramebuffer(gl.FRAMEBUFFER, null);
104
+
105
+ this.adjustGl();
106
+
107
+ // TODO: Size should be observed only if the content is not absolutely sized
108
+ this._resizeObserver = new ResizeObserver((entries) => {
109
+ this.invalidateSize();
110
+ this._emit("resize");
111
+ });
112
+ this._resizeObserver.observe(this._container);
113
+
114
+ // TODO: Observe devicePixelRatio
115
+ // https://developer.mozilla.org/en-US/docs/Web/API/Window/devicePixelRatio#Monitoring_screen_resolution_or_zoom_level_changes
116
+
117
+ this._updateDpr();
118
+ }
119
+
120
+ invalidateSize() {
121
+ this._logicalCanvasSize = undefined;
122
+ this._updateDpr();
123
+ this.adjustGl();
124
+ }
125
+
126
+ _updateDpr() {
127
+ this.dpr = window.devicePixelRatio;
128
+ }
129
+
130
+ /**
131
+ * Compiles and caches a shader. The shader source is used as a cache key.
132
+ *
133
+ * @param {number} type gl.VERTEX_SHADER or gl.FRAGMENT_SHADER
134
+ * @param {string | string[]} glsl
135
+ */
136
+ compileShader(type, glsl) {
137
+ const VERSION = "#version 300 es";
138
+ const PRECISION = "precision mediump float;";
139
+
140
+ if (isArray(glsl)) {
141
+ glsl = glsl.join("\n\n");
142
+ }
143
+
144
+ const gl = this.gl;
145
+ const cacheKey = glsl.replaceAll(/ {2,}|^\s*\/\/.*$/gm, "");
146
+
147
+ let shader = this._shaderCache.get(cacheKey);
148
+ if (!shader) {
149
+ const stitchedSource = [VERSION, PRECISION, glsl].join("\n\n");
150
+
151
+ shader = gl.createShader(type);
152
+ gl.shaderSource(shader, stitchedSource);
153
+ gl.compileShader(shader);
154
+
155
+ // Don't check errors here. Only check them if linking fails.
156
+
157
+ this._shaderCache.set(cacheKey, shader);
158
+ }
159
+
160
+ return shader;
161
+ }
162
+
163
+ adjustGl() {
164
+ const logicalSize = this.getLogicalCanvasSize();
165
+ this.canvas.style.width = `${logicalSize.width}px`;
166
+ this.canvas.style.height = `${logicalSize.height}px`;
167
+
168
+ const physicalSize = this.getPhysicalCanvasSize(logicalSize);
169
+ this.canvas.width = physicalSize.width;
170
+ this.canvas.height = physicalSize.height;
171
+
172
+ resizeFramebufferInfo(
173
+ this.gl,
174
+ this._pickingBufferInfo,
175
+ this._pickingAttachmentOptions
176
+ );
177
+ }
178
+
179
+ finalize() {
180
+ this._resizeObserver.unobserve(this._container);
181
+ this.canvas.remove();
182
+ }
183
+
184
+ /**
185
+ * Returns the canvas size in true display pixels
186
+ *
187
+ * @param {{ width: number, height: number }} [logicalSize]
188
+ */
189
+ getPhysicalCanvasSize(logicalSize) {
190
+ logicalSize = logicalSize || this.getLogicalCanvasSize();
191
+ return {
192
+ width: logicalSize.width * this.dpr,
193
+ height: logicalSize.height * this.dpr,
194
+ };
195
+ }
196
+
197
+ /**
198
+ * Returns the canvas size in logical pixels (without devicePixelRatio correction)
199
+ */
200
+ getLogicalCanvasSize() {
201
+ if (this._logicalCanvasSize) {
202
+ return this._logicalCanvasSize;
203
+ }
204
+
205
+ // TODO: The size should never be smaller than the minimum content size!
206
+ const contentSize = this._sizeSource?.() ?? {
207
+ width: undefined,
208
+ height: undefined,
209
+ };
210
+
211
+ const cs = window.getComputedStyle(this._container, null);
212
+ const width =
213
+ contentSize.width ??
214
+ this._container.clientWidth -
215
+ parseFloat(cs.paddingLeft) -
216
+ parseFloat(cs.paddingRight);
217
+
218
+ const height =
219
+ contentSize.height ??
220
+ this._container.clientHeight -
221
+ parseFloat(cs.paddingTop) -
222
+ parseFloat(cs.paddingBottom);
223
+
224
+ this._logicalCanvasSize = { width, height };
225
+ return this._logicalCanvasSize;
226
+ }
227
+
228
+ /**
229
+ * @param {"render"|"resize"} eventType
230
+ * @param {function} listener
231
+ */
232
+ addEventListener(eventType, listener) {
233
+ this._listeners.push({ type: eventType, listener });
234
+ }
235
+
236
+ /**
237
+ * @param {string} eventType
238
+ */
239
+ _emit(eventType) {
240
+ for (const entry of this._listeners) {
241
+ if (entry.type === eventType) {
242
+ entry.listener();
243
+ }
244
+ }
245
+ }
246
+
247
+ /**
248
+ *
249
+ * @param {number} x
250
+ * @param {number} y
251
+ */
252
+ readPickingPixel(x, y) {
253
+ const gl = this.gl;
254
+
255
+ x *= this.dpr;
256
+ y *= this.dpr;
257
+
258
+ const height = this.getPhysicalCanvasSize().height;
259
+
260
+ const pixel = new Uint8Array(4);
261
+ gl.bindFramebuffer(gl.FRAMEBUFFER, this._pickingBufferInfo.framebuffer);
262
+ gl.readPixels(
263
+ x,
264
+ height - y - 1,
265
+ 1,
266
+ 1,
267
+ gl.RGBA,
268
+ gl.UNSIGNED_BYTE,
269
+ pixel
270
+ );
271
+ gl.bindFramebuffer(gl.FRAMEBUFFER, null);
272
+
273
+ return pixel;
274
+ }
275
+
276
+ clearAll() {
277
+ const gl = this.gl;
278
+ const { width, height } = this.getPhysicalCanvasSize();
279
+ gl.viewport(0, 0, width, height);
280
+ gl.disable(gl.SCISSOR_TEST);
281
+ gl.clearColor(0, 0, 0, 0);
282
+ gl.clear(gl.COLOR_BUFFER_BIT);
283
+ }
284
+
285
+ /**
286
+ * Creates textures for color schemes and discrete/discretizing ranges.
287
+ * N.B. Discrete range textures need domain. Thus, this cannot be called
288
+ * before the final domains are resolved.
289
+ *
290
+ * TODO: This may be too specific to be included in WebGLHelper. Find a better place.
291
+ *
292
+ * @param {import("../view/scaleResolution").default} resolution
293
+ * @param {boolean} update Update the texture if it exists already.
294
+ */
295
+ createRangeTexture(resolution, update = false) {
296
+ const existingTexture = this.rangeTextures.get(resolution);
297
+ if (!update && existingTexture) {
298
+ return;
299
+ }
300
+
301
+ /**
302
+ * TODO: The count configuration logic etc should be combined
303
+ * with scale.js that configures d3 scales using vega specs
304
+ * @param {number} count
305
+ * @param {any} scale
306
+ * @returns {number}
307
+ */
308
+ function fixCount(count, scale) {
309
+ if (isDiscrete(scale.type)) {
310
+ return scale.domain().length;
311
+ } else if (scale.type == "threshold") {
312
+ return scale.domain().length + 1;
313
+ } else if (scale.type == "quantize") {
314
+ return count ?? 4;
315
+ } else if (scale.type == "quantile") {
316
+ return count ?? 4;
317
+ }
318
+ return count;
319
+ }
320
+
321
+ const channel = resolution.channel;
322
+
323
+ if (isColorChannel(channel)) {
324
+ const props = resolution.getScaleProps();
325
+
326
+ const scale = resolution.getScale();
327
+
328
+ /** @type {WebGLTexture} */
329
+ let texture;
330
+
331
+ if (props.scheme) {
332
+ let count = isString(props.scheme)
333
+ ? undefined
334
+ : props.scheme.count;
335
+
336
+ count = fixCount(count, scale);
337
+
338
+ texture = createSchemeTexture(
339
+ props.scheme,
340
+ this.gl,
341
+ count,
342
+ existingTexture
343
+ );
344
+ } else {
345
+ // No scheme, assume that colors are specified in the range
346
+
347
+ const range = /** @type {any[]} */ (scale.range());
348
+
349
+ if (isInterpolating(scale.type)) {
350
+ texture = createInterpolatedColorTexture(
351
+ range,
352
+ props.interpolate,
353
+ this.gl,
354
+ existingTexture
355
+ );
356
+ } else {
357
+ texture = createDiscreteColorTexture(
358
+ range,
359
+ this.gl,
360
+ scale.domain().length,
361
+ existingTexture
362
+ );
363
+ }
364
+ }
365
+
366
+ this.rangeTextures.set(resolution, texture);
367
+ } else {
368
+ const scale = resolution.getScale();
369
+
370
+ if (scale.type === "ordinal" || isDiscretizing(scale.type)) {
371
+ /** @type {function(any):number} Handle "shape" etc */
372
+ const mapper = isDiscreteChannel(channel)
373
+ ? getDiscreteRangeMapper(channel)
374
+ : (x) => x;
375
+
376
+ const range = /** @type {any[]} */ (
377
+ resolution.getScale().range()
378
+ );
379
+
380
+ this.rangeTextures.set(
381
+ resolution,
382
+ createDiscreteTexture(
383
+ range.map(mapper),
384
+ this.gl,
385
+ scale.domain().length,
386
+ existingTexture
387
+ )
388
+ );
389
+ }
390
+ }
391
+ }
392
+ }
393
+
394
+ /**
395
+ * Copy-pasted from twgl.js:
396
+ * https://github.com/greggman/twgl.js/blob/master/src/programs.js
397
+ * Copyright 2019 Gregg Tavares, MIT license
398
+ */
399
+ function addLineNumbersWithError(src, log = "", lineOffset = 0) {
400
+ const errorRE = /ERROR:\s*\d+:(\d+)/gi;
401
+ // Note: Error message formats are not defined by any spec so this may or may not work.
402
+ const matches = [...log.matchAll(errorRE)];
403
+ const lineNoToErrorMap = new Map(
404
+ matches.map((m, ndx) => {
405
+ const lineNo = parseInt(m[1]);
406
+ const next = matches[ndx + 1];
407
+ const end = next ? next.index : log.length;
408
+ const msg = log.substring(m.index, end);
409
+ return [lineNo - 1, msg];
410
+ })
411
+ );
412
+ return src
413
+ .split("\n")
414
+ .map((line, lineNo) => {
415
+ const err = lineNoToErrorMap.get(lineNo);
416
+ return `${lineNo + 1 + lineOffset}: ${line}${
417
+ err ? `\n\n^^^ ${err}` : ""
418
+ }`;
419
+ })
420
+ .join("\n");
421
+ }
422
+
423
+ /**
424
+ * @param {WebGL2RenderingContext} gl
425
+ * @param {WebGLShader} vertexShader
426
+ * @param {WebGLShader} fragmentShader
427
+ */
428
+ export function createProgram(gl, vertexShader, fragmentShader) {
429
+ var program = gl.createProgram();
430
+ gl.attachShader(program, vertexShader);
431
+ gl.attachShader(program, fragmentShader);
432
+ gl.linkProgram(program);
433
+
434
+ function getProgramErrors() {
435
+ /** @type {string} */
436
+ let errorMsg;
437
+ /** @type {string} */
438
+ let errorDetail;
439
+
440
+ const linked = gl.getProgramParameter(program, gl.LINK_STATUS);
441
+ if (!linked) {
442
+ errorMsg = gl.getProgramInfoLog(program);
443
+
444
+ for (const shader of [vertexShader, fragmentShader]) {
445
+ const compiled = gl.getShaderParameter(
446
+ shader,
447
+ gl.COMPILE_STATUS
448
+ );
449
+ if (!compiled) {
450
+ errorMsg = gl.getShaderInfoLog(shader);
451
+ errorDetail =
452
+ addLineNumbersWithError(
453
+ gl.getShaderSource(shader),
454
+ errorMsg,
455
+ 0
456
+ ) + `\nError compiling: ${errorMsg}`;
457
+ gl.deleteShader(shader);
458
+ }
459
+ }
460
+ gl.deleteProgram(program);
461
+ }
462
+
463
+ if (errorMsg) {
464
+ return { message: errorMsg, detail: errorDetail };
465
+ }
466
+ }
467
+
468
+ return {
469
+ program,
470
+ getProgramErrors,
471
+ };
472
+ }
473
+
474
+ /**
475
+ * @param {WebGLRenderingContext} gl
476
+ * @param {Omit<import("twgl.js").TextureOptions, "src">} options
477
+ * @param {number[] | ArrayBufferView} src
478
+ * @param {WebGLTexture} [texture]
479
+ */
480
+ export function createOrUpdateTexture(gl, options, src, texture) {
481
+ if (texture) {
482
+ setTextureFromArray(gl, texture, src, options);
483
+ } else {
484
+ texture = createTexture(gl, {
485
+ ...options,
486
+ src,
487
+ });
488
+ }
489
+ return texture;
490
+ }
@@ -0,0 +1 @@
1
+ <svg viewBox="0 0 32 32" xmlns="http://www.w3.org/2000/svg" fill-rule="evenodd" clip-rule="evenodd" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="1.5"><path d="M4.7 21.2s.4 2.3 1.3 3.6C7 26 9.8 28 9.8 28s3.4-2.6 6.4-8.5c0 0 1 .1 1.9-.4.9-.6.8-.4 1-1.2 0 0 2.9.5 6.6 0 2.1-.3 4.3-1 6.2-2.5 0 0-1.1-1.7-2.5-5.1-.5-1.3-2-1.8-4.6-4.6l-8.2 8.6-11.9 6.9z" fill-opacity=".1"/><path d="M12.7 14.8s-4-1.8-12 2.7c0 0 1 3.7 2.5 5.3 1.4 1.5 2.3 3.6 4.6 4.6 0 0 4.7-3 6.9-9.2l3-2s4.7 2.8 12.6-1.6c0 0-.6-3.3-3-6-2.6-3-3.8-4.7-3.8-4.7s-4.1 2.3-7.5 9.3l-3.3 1.6z" fill="#7fbbdd"/><path d="M12.4 15.5c-.7-.5-2.4-.8-4.4-.4-2 .4-4 1.3-4.8 1.8-.5.3-1.2 1-1.2 1.4 0 .7.3 1.8.8 2.4.3.3.7.5 1.4.6.8 0 2.5-1.4 3.5-2 1-.6 1.6-.8 2.7-1.2l-2.9 2.4c-1.3 1.2-2.2 1.5-2.4 2.3 0 .5 0 1.4.5 1.8.4.5.6.8 1.6.8.6 0 1 0 2.6-1.5.9-.9 2.3-3 2.7-3.7.6-1.1 1-2.2.7-3-.2-1-.4-1.4-.8-1.7zM17.4 14.2c-.3-.5-.9-1.2-.2-2.5l1.9-3c.5-.8 2-2.3 2.6-2.6.6-.4 1.5-.6 2-.2.6.4 1 1 1.3 1.5.4.6.7 1.3.2 2-.7 1-1.6.9-2.8 1.7-1.2.8-1.9 1.2-2.5 1.9l3.8-1.8c1.3-.6 2.7-1.1 3.4-.7.8.5.8.7 1 1.4.3 1-.2 1.9-.8 2.4-.5.6-1.5.9-2.6 1.2-1.4.4-4.5 1-5.8.5-1.3-.5-1.3-1.4-1.5-1.8z" fill="#fff"/><path d="M12.7 14.8s-4-1.8-12 2.7c0 0 1 3.7 2.5 5.3 1.4 1.5 2.3 3.6 4.6 4.6 0 0 4.7-3 6.9-9.2l3-2s4.7 2.8 12.6-1.6c0 0-.6-3.3-3-6-2.6-3-3.8-4.7-3.8-4.7s-4.1 2.3-7.5 9.3l-3.3 1.6z" fill="none" stroke="#000" stroke-width=".5"/><path d="M12.4 14.9s2.1-2 3-2c1.1 0 2.3 2.7 2.3 3.5 0 .8-2.1 2.4-3.1 2.3 0 0 0-1.2-.7-2.6-.8-1.3-1.5-1.2-1.5-1.2z" fill="#7fbbdd"/><path d="M13.8 15.3c.9.8.6 2 1.4 1.8 1-.2 1.4-.8 1.3-1.5 0-.7 0-.8-.4-1.6-.3-.7-1-1.1-2-.5-.7.5-1.4 1.3-1.4 1.3s.2-.3 1.1.5z" fill="#fff"/><path d="M12.4 14.9s2.1-2 3-2c1.1 0 2.3 2.7 2.3 3.5 0 .8-2.1 2.4-3.1 2.3 0 0 0-1.2-.7-2.6-.8-1.3-1.5-1.2-1.5-1.2z" fill="none" stroke="#000" stroke-width=".5"/></svg>
@@ -0,0 +1,34 @@
1
+ <?xml version="1.0" encoding="UTF-8" standalone="no"?>
2
+ <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
3
+ <svg width="100%" height="100%" viewBox="0 0 64 64" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xml:space="preserve" xmlns:serif="http://www.serif.com/" style="fill-rule:evenodd;clip-rule:evenodd;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:1.5;">
4
+ <g transform="matrix(0.10221,0.00683825,-0.0067136,0.100347,-8.03424,-13.2222)">
5
+ <path d="M208.629,548.17C208.629,548.17 219.768,594.389 239.814,619.222C259.86,644.055 319.119,680.831 319.119,680.831C319.119,680.831 385.557,622.915 437.87,498.346C437.87,498.346 456.021,499.236 473.808,486.446C491.594,473.656 490.239,478.534 494.883,461.62C494.883,461.62 551.848,467.937 625.826,451.863C667.634,442.778 710.238,425.649 747.382,393.142C747.382,393.142 721.992,360.244 689.422,291.45C677.029,265.275 647.518,257.098 590.747,203.309L438.083,391.273L208.629,548.17Z" style="fill-opacity:0.11;"/>
6
+ </g>
7
+ <g transform="matrix(0.081985,-0.0582493,0.0582493,0.081985,-40.4754,34.7668)">
8
+ <path d="M561.588,349.155C561.588,349.155 516.297,271.341 328.841,252.044C328.841,252.044 302.342,325.189 307.686,368.056C313.031,410.922 303.001,456.147 329.132,501.174C329.132,501.174 443.114,506.743 554.649,428.469L629.204,432.112C629.204,432.112 673.755,534.351 856.964,554.375C856.964,554.375 886.717,491.768 879.193,417.478C871.211,338.673 871.716,293.443 871.716,293.443C871.716,293.443 775.485,283.928 635.418,361.795L561.588,349.155Z" style="fill:rgb(127,187,221);"/>
9
+ <clipPath id="_clip1">
10
+ <path d="M561.588,349.155C561.588,349.155 516.297,271.341 328.841,252.044C328.841,252.044 302.342,325.189 307.686,368.056C313.031,410.922 303.001,456.147 329.132,501.174C329.132,501.174 443.114,506.743 554.649,428.469L629.204,432.112C629.204,432.112 673.755,534.351 856.964,554.375C856.964,554.375 886.717,491.768 879.193,417.478C871.211,338.673 871.716,293.443 871.716,293.443C871.716,293.443 775.485,283.928 635.418,361.795L561.588,349.155Z"/>
11
+ </clipPath>
12
+ <g clip-path="url(#_clip1)">
13
+ <g transform="matrix(0.758637,0.45892,-0.45892,0.758637,425.635,-121.213)">
14
+ <path d="M386.873,397.241C371.345,383.628 335.138,374.782 290.644,380.543C246.149,386.304 196.827,402.266 178.802,412.065C168.202,417.828 151.494,429.354 150.682,439.334C149.527,453.528 153.593,477.066 165.625,491.284C171.69,498.451 179.927,503.445 193.838,505.423C211.701,507.963 252.714,480.627 275.399,469.22C299.332,457.187 313.981,454.56 338.386,447.014C338.386,447.014 291.346,478.587 270.205,494.393C239.61,517.268 218.575,522.444 212.859,539.897C209.766,549.34 212.237,567.767 220.354,577.538C229.427,588.461 233.197,595.069 254.591,597.624C267.303,599.143 276.685,598.802 314.37,568.95C336.06,551.767 370.297,509.817 381.412,494.616C397.581,472.506 405.165,451.462 402.656,435.04C399.045,411.391 394.259,403.716 386.873,397.241Z" style="fill:white;"/>
15
+ </g>
16
+ <g transform="matrix(0.752145,0.454993,-0.454993,0.752145,468.36,-108.293)">
17
+ <path d="M484.733,378.934C479.33,367.886 467.092,350.452 483.886,325.205C502.172,297.717 514.898,280.938 528.117,264.632C541.335,248.326 574.793,218.447 587.739,212.53C600.684,206.613 621.026,203.852 631.889,212.841C642.874,221.93 648.416,233.377 654.92,245.478C662.331,259.267 667.318,274.808 656.518,288.278C641.093,307.515 622.242,305.294 594.912,319.844C567.244,334.573 552.644,342.943 539.244,355.703C539.244,355.703 591.431,334.114 621.229,324.474C651.027,314.833 680.952,304.461 695.995,315.975C711.038,327.49 710.631,331.463 714.425,347.187C719.107,366.59 707.231,385.165 694.294,395.838C681.356,406.51 659.958,411.756 636.875,416.852C606.42,423.576 540.278,431.917 512.819,419.366C485.361,406.815 488.488,386.613 484.733,378.934Z" style="fill:white;"/>
18
+ </g>
19
+ </g>
20
+ <path d="M561.588,349.155C561.588,349.155 516.297,271.341 328.841,252.044C328.841,252.044 302.342,325.189 307.686,368.056C313.031,410.922 303.001,456.147 329.132,501.174C329.132,501.174 443.114,506.743 554.649,428.469L629.204,432.112C629.204,432.112 673.755,534.351 856.964,554.375C856.964,554.375 886.717,491.768 879.193,417.478C871.211,338.673 871.716,293.443 871.716,293.443C871.716,293.443 775.485,283.928 635.418,361.795L561.588,349.155Z" style="fill:none;stroke:black;stroke-width:16.57px;"/>
21
+ </g>
22
+ <g transform="matrix(0.100298,-0.00740488,0.00740488,0.100298,-15.7457,-5.17528)">
23
+ <path d="M384.916,385.459C384.916,385.459 430.781,347.868 450.027,350.093C471.685,352.596 490.49,407.747 490.208,424.416C489.927,441.085 442.948,469.866 422.537,466.24C422.537,466.24 425.525,442.129 412.114,413.335C398.702,384.54 384.916,385.459 384.916,385.459Z" style="fill:rgb(127,187,221);"/>
24
+ <clipPath id="_clip2">
25
+ <path d="M384.916,385.459C384.916,385.459 430.781,347.868 450.027,350.093C471.685,352.596 490.49,407.747 490.208,424.416C489.927,441.085 442.948,469.866 422.537,466.24C422.537,466.24 425.525,442.129 412.114,413.335C398.702,384.54 384.916,385.459 384.916,385.459Z"/>
26
+ </clipPath>
27
+ <g clip-path="url(#_clip2)">
28
+ <g transform="matrix(0.909069,3.67641e-17,-3.79818e-17,0.85852,37.051,52.1829)">
29
+ <path d="M413.282,402.697C430.77,421.58 423.307,448.426 441.683,446.137C463.409,443.431 472.604,430.356 473.316,413.402C474.028,396.448 472.245,393.911 466.172,375.321C460.099,356.731 447.953,352.176 422.842,357.179C401.957,361.339 405.039,360.215 398.735,367.228C392.43,374.242 384.315,393.182 384.315,393.182C384.315,393.182 392.715,380.492 413.282,402.697Z" style="fill:white;"/>
30
+ </g>
31
+ </g>
32
+ <path d="M384.916,385.459C384.916,385.459 430.781,347.868 450.027,350.093C471.685,352.596 490.49,407.747 490.208,424.416C489.927,441.085 442.948,469.866 422.537,466.24C422.537,466.24 425.525,442.129 412.114,413.335C398.702,384.54 384.916,385.459 384.916,385.459Z" style="fill:none;stroke:black;stroke-width:16.57px;"/>
33
+ </g>
34
+ </svg>
package/src/index.html ADDED
@@ -0,0 +1,11 @@
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="UTF-8" />
5
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
+ <title>GenomeSpy</title>
7
+ </head>
8
+ <body>
9
+ <script type="module" src="/singlePageApp.js"></script>
10
+ </body>
11
+ </html>
package/src/index.js ADDED
@@ -0,0 +1,151 @@
1
+ import { isObject, isString } from "vega-util";
2
+ import { loader as vegaLoader } from "vega-loader";
3
+ import { html } from "lit-html";
4
+
5
+ import GenomeSpy from "./genomeSpy.js";
6
+ import icon from "./img/bowtie.svg";
7
+
8
+ export { GenomeSpy, html, icon };
9
+
10
+ /**
11
+ * Embeds GenomeSpy into the DOM
12
+ *
13
+ * @param {HTMLElement | string} el HTMLElement or a query selector
14
+ * @param {object | string} spec a spec object or an url to a json spec
15
+ * @param {import("./options.js").EmbedOptions} [options] options
16
+ */
17
+ export async function embed(el, spec, options = {}) {
18
+ /** @type {HTMLElement} */
19
+ let element;
20
+
21
+ if (isString(el)) {
22
+ element = document.querySelector(el);
23
+ if (!element) {
24
+ throw new Error(`No such element: ${el}`);
25
+ }
26
+ } else if (el instanceof HTMLElement) {
27
+ element = el;
28
+ } else {
29
+ throw new Error(`Invalid element: ${el}`);
30
+ }
31
+
32
+ /** @type {GenomeSpy} */
33
+ let genomeSpy;
34
+
35
+ try {
36
+ const specObject = /** @type {import("./spec/root").RootSpec} */ (
37
+ isObject(spec) ? spec : await loadSpec(spec)
38
+ );
39
+
40
+ specObject.baseUrl = specObject.baseUrl || "";
41
+
42
+ if (!("width" in specObject)) {
43
+ specObject.width = "container";
44
+ }
45
+
46
+ if (!("padding" in specObject)) {
47
+ specObject.padding = 10;
48
+ }
49
+
50
+ if (element == document.body) {
51
+ // Need to add a wrapper to make sizing behavior more stable
52
+ const wrapper = document.createElement("div");
53
+ wrapper.style.position = "fixed";
54
+ wrapper.style.inset = "0";
55
+ wrapper.style.overflow = "hidden";
56
+ element.appendChild(wrapper);
57
+ element = wrapper;
58
+ }
59
+
60
+ genomeSpy = new GenomeSpy(element, specObject, options);
61
+ applyOptions(genomeSpy, options);
62
+ await genomeSpy.launch();
63
+ } catch (e) {
64
+ // eslint-disable-next-line require-atomic-updates
65
+ element.innerText = e.toString();
66
+ console.error(e);
67
+ }
68
+
69
+ return {
70
+ finalize() {
71
+ genomeSpy.destroy();
72
+ while (element.firstChild) {
73
+ element.firstChild.remove();
74
+ }
75
+ },
76
+
77
+ /**
78
+ * @param {string} type
79
+ * @param {(event: any) => void} listener
80
+ */
81
+ addEventListener(type, listener) {
82
+ const listenersByType = genomeSpy._eventListeners;
83
+
84
+ let listeners = listenersByType.get(type);
85
+ if (!listeners) {
86
+ listeners = new Set();
87
+ listenersByType.set(type, listeners);
88
+ }
89
+
90
+ listeners.add(listener);
91
+ },
92
+
93
+ /**
94
+ * @param {string} type
95
+ * @param {(event: any) => void} listener
96
+ */
97
+ removeEventListener(type, listener) {
98
+ const listenersByType = genomeSpy._eventListeners;
99
+
100
+ listenersByType.get(type)?.delete(listener);
101
+ },
102
+
103
+ /**
104
+ * @param {string} name
105
+ * @returns {import("./view/scaleResolutionApi").ScaleResolutionApi}
106
+ */
107
+ getScaleResolutionByName(name) {
108
+ return genomeSpy.getNamedScaleResolutions().get(name);
109
+ },
110
+ };
111
+ }
112
+
113
+ /**
114
+ *
115
+ * @param {import("./genomeSpy").default} genomeSpy
116
+ * @param {Record<string, any>} opt
117
+ */
118
+ function applyOptions(genomeSpy, opt) {
119
+ if (opt.namedDataProvider) {
120
+ genomeSpy.registerNamedDataProvider(opt.namedDataProvider);
121
+ }
122
+
123
+ if (opt.beforeLaunchCallback) {
124
+ opt.beforeLaunchCallback(genomeSpy);
125
+ }
126
+ }
127
+
128
+ /**
129
+ * Loads the spec from the given url and sets the baseUrl if it is not
130
+ * defined in the spec.
131
+ *
132
+ * @param {string} url
133
+ */
134
+ export async function loadSpec(url) {
135
+ let spec;
136
+
137
+ try {
138
+ spec = JSON.parse(await vegaLoader().load(url));
139
+ } catch (e) {
140
+ throw new Error(
141
+ `Could not load or parse configuration: ${url}, reason: ${e.message}`
142
+ );
143
+ }
144
+
145
+ if (!spec.baseUrl) {
146
+ const m = url.match(/^[^?#]*\//);
147
+ spec.baseUrl = (m && m[0]) || "./";
148
+ }
149
+
150
+ return spec;
151
+ }