diff --git a/dist/boostlet.min.js b/dist/boostlet.min.js index 83405661..750a6f15 100644 --- a/dist/boostlet.min.js +++ b/dist/boostlet.min.js @@ -46,13 +46,19 @@ var $lRZZ2 = parcelRequire("lRZZ2"); var $lBIzH = parcelRequire("lBIzH"); var $7Srn3 = parcelRequire("7Srn3"); + +var $jYzCB = parcelRequire("jYzCB"); + +var $3okZG = parcelRequire("3okZG"); class Util { static detect_framework() { let framework11 = null; if (Util.is_defined(window.nv)) framework11 = new (0, $lBIzH.NiiVue)(window.nv); else if (Util.is_defined(window.niivue)) framework11 = new (0, $lBIzH.NiiVue)(window.niivue); else if (Util.is_defined(window.cornerstone)) framework11 = new (0, $lRZZ2.Cornerstone2D)(window.cornerstone); + else if (Util.is_defined(window.r)) framework11 = new (0, $jYzCB.Xtk)(window.r); else if (Util.is_defined(window.OpenSeadragon)) framework11 = new (0, $7Srn3.OpenSeaDragon)(window.OpenSeadragon); + else if (Util.is_defined(window.papayaContainers)) framework11 = new (0, $3okZG.Papaya)(window.papayaContainers); // TODO: fallback to general canvas or webgl framework return framework11; } @@ -561,6 +567,156 @@ class $5bc2ced335fcdbf6$export$ac966f05fb2ad3 extends (0, $bsgU1.Framework) { }); +parcelRegister("jYzCB", function(module, exports) { + +$parcel$export(module.exports, "Xtk", () => $e8af40a51d292379$export$d4ec65ebec44af30); + +var $bsgU1 = parcelRequire("bsgU1"); + +var $gk9zP = parcelRequire("gk9zP"); +class $e8af40a51d292379$export$d4ec65ebec44af30 extends (0, $bsgU1.Framework) { + constructor(instance){ + super(instance); + this.name = "xtk"; + } + get_image(from_canvas) { + let canvas = this.instance.ca; + let ctx = canvas.getContext("2d"); + let image = ctx.getImageData(0, 0, canvas.width, canvas.height); + let rgba_image = (0, $gk9zP.Util).rgba_to_grayscale(image.data); + return { + data: rgba_image, + width: image.width, + height: image.height + }; + // return {'data':pixels, 'width':image.width, 'height':image.height}; + } + set_image(new_pixels) { + let originalcanvas = this.instance.ca; + let newcanvas = window.document.createElement("canvas"); + newcanvas.width = originalcanvas.width; + newcanvas.height = originalcanvas.height; + let ctx = newcanvas.getContext("2d"); + let newPixelsRgba = (0, $gk9zP.Util).grayscale_to_rgba(new_pixels); + let newPixelsClamped = new Uint8ClampedArray(newPixelsRgba); + let newImageData = new ImageData(newPixelsClamped, newcanvas.width, newcanvas.height); + // Draw the new image data onto the canvas + ctx.putImageData(newImageData, 0, 0); + newcanvas.onclick = function() { + // on click, we will restore the nv canvas + newcanvas.parentNode.replaceChild(originalcanvas, newcanvas); + }; + // replace nv canvas with new one + originalcanvas.parentNode.replaceChild(newcanvas, originalcanvas); + } + set_mask(new_mask) { + let image = this.get_image(); + // TODO here we need to flip one more time, this is until + // we use the official niivue infrastructure for adding + // a segmentation layer + let originalcanvas = this.instance.ca; + let newcanvas = window.document.createElement("canvas"); + newcanvas.width = originalcanvas.width; + newcanvas.height = originalcanvas.height; + // put new_pixels down + let ctx = newcanvas.getContext("2d"); + let imageclamped = new Uint8ClampedArray(image.data); + let imagedata = new ImageData(imageclamped, image.width, image.height); + ctx.putImageData(imagedata, 0, 0); + ctx.save(); + ctx.scale(1, -1); + ctx.drawImage(newcanvas, 0, -newcanvas.height); + ctx.restore(); + image = ctx.getImageData(0, 0, newcanvas.width, newcanvas.height); + // end of flip + let masked_image = (0, $gk9zP.Util).harden_mask(image.data, new_mask); + this.set_image(masked_image); // rgba data, no flip + } + select_box(callback) { + // alert("Click on top left and bottom rght coordinated of the desired selection box.") + let isFirstClick = true; + let x1, y1, x2, y2; + // Function to handle the mouse click event + function handleClick(event) { + if (isFirstClick) { + // Capture x1 and y1 on the first click + x1 = event.clientX; + y1 = event.clientY; + console.log(`First click: (X1: ${x1}, Y1: ${y1})`); + isFirstClick = false; + } else { + // Capture x2 and y2 on the second click + x2 = event.clientX; + y2 = event.clientY; + console.log(`Second click: (X2: ${x2}, Y2: ${y2})`); + isFirstClick = true; + let topleft = { + x: x1, + y: y1 + }; + let bottomright = { + x: x2, + y: y2 + }; + callback(topleft, bottomright); + } + // let topleft = {x: 529, y: 480}; + // let bottomright = {x: 667, y: 588}; + // callback(topleft, bottomright); + } + // Add a click event listener to the document + document.addEventListener("click", handleClick); + } +} + +}); + +parcelRegister("3okZG", function(module, exports) { + +$parcel$export(module.exports, "Papaya", () => $2783e5ed66ad5478$export$26cab353fe186c2e); + +var $bsgU1 = parcelRequire("bsgU1"); + +var $gk9zP = parcelRequire("gk9zP"); +class $2783e5ed66ad5478$export$26cab353fe186c2e extends (0, $bsgU1.Framework) { + constructor(instance){ + super(instance); + this.name = "papaya"; + } + get_image(from_canvas) { + let canvas = this.instance[0].viewer.canvas; + let ctx = canvas.getContext("2d"); + let image = ctx.getImageData(0, 0, canvas.width, canvas.height); + let rgba_image = (0, $gk9zP.Util).rgba_to_grayscale(image.data); + return { + data: rgba_image, + width: image.width, + height: image.height + }; + // return {'data':pixels, 'width':image.width, 'height':image.height}; + } + set_image(new_pixels) { + let originalcanvas = this.instance[0].viewer.canvas; + let newcanvas = window.document.createElement("canvas"); + newcanvas.width = originalcanvas.width; + newcanvas.height = originalcanvas.height; + let ctx = newcanvas.getContext("2d"); + let newPixelsRgba = (0, $gk9zP.Util).grayscale_to_rgba(new_pixels); + let newPixelsClamped = new Uint8ClampedArray(newPixelsRgba); + let newImageData = new ImageData(newPixelsClamped, newcanvas.width, newcanvas.height); + // Draw the new image data onto the canvas + ctx.putImageData(newImageData, 0, 0); + newcanvas.onclick = function() { + // on click, we will restore the nv canvas + newcanvas.parentNode.replaceChild(originalcanvas, newcanvas); + }; + // replace nv canvas with new one + originalcanvas.parentNode.replaceChild(newcanvas, originalcanvas); + } +} + +}); + var $gk9zP = parcelRequire("gk9zP"); diff --git a/dist/boostlet.min.js.map b/dist/boostlet.min.js.map index 06262ebf..f98e155b 100644 --- a/dist/boostlet.min.js.map +++ b/dist/boostlet.min.js.map @@ -1 +1 @@ -{"mappings":"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","sources":["src/util.js","src/framework.js","src/frameworks/cornerstone2d.js","src/frameworks/niivue.js","src/frameworks/openseadragon.js","src/index.js","src/boostlet.js"],"sourcesContent":["import {Framework} from './framework.js';\nimport {Cornerstone2D} from './frameworks/cornerstone2d.js';\nimport {NiiVue} from './frameworks/niivue.js';\nimport { OpenSeaDragon } from './frameworks/openseadragon.js';\n\nexport class Util {\n \n static detect_framework() {\n\n let framework = null;\n\n if (Util.is_defined(window.nv)) {\n \n framework = new NiiVue(window.nv);\n \n } else if (Util.is_defined(window.niivue)) {\n \n framework = new NiiVue(window.niivue);\n\n } else if (Util.is_defined(window.cornerstone)) {\n\n framework = new Cornerstone2D(window.cornerstone);\n\n } else if (Util.is_defined(window.OpenSeadragon)) {\n\n framework = new OpenSeaDragon(window.OpenSeadragon);\n \n }\n\n // TODO: fallback to general canvas or webgl framework\n\n return framework;\n\n }\n\n static async load_script(url, callback) {\n\n // introducing hack to make it work for openneuro\n window.Object.defineProperty(window.Object.prototype, 'global', {\n get( ){\n return window;\n },\n set(newGlobal) {\n globalThis = newGlobal;\n }\n });\n\n const script = window.document.createElement(\"script\")\n script.type = \"text/javascript\"\n script.src = url;\n\n if (Util.is_defined(callback)) {\n script.onload = callback;\n }\n\n window.document.head.appendChild(script);\n eval(script);\n\n }\n\n static async send_http_post(url, data, callback) {\n\n let xhr = new XMLHttpRequest();\n xhr.open(\"POST\", url);\n xhr.onreadystatechange = function () {\n if (xhr.readyState === 4) {\n // request done\n callback( xhr.response );\n\n return;\n\n }\n }\n\n xhr.send(data)\n\n }\n\n static convert_to_png(uint8array, width, height, flip) {\n\n // we are using an offscreen canvas for this\n let offscreen = window.document.createElement('canvas');\n offscreen.width = width;\n offscreen.height = height;\n\n let offscreen_ctx = offscreen.getContext('2d');\n\n let imgdata = offscreen_ctx.createImageData(offscreen.width, offscreen.height);\n let pxdata = imgdata.data;\n\n for (var i =0; i c.charCodeAt(0));\n\n return pngpixels;\n\n }\n\n static filter(pixels, width, height, kernel) {\n\n const kernelSize = Math.sqrt(kernel.length);\n const halfKernelSize = Math.floor(kernelSize / 2);\n\n const new_pixels = pixels.slice();\n\n for (let y = 0; y < height; y++) {\n for (let x = 0; x < width; x++) {\n const dstIndex = y * width + x;\n\n let newValue = 0;\n for (let ky = 0; ky < kernelSize; ky++) {\n for (let kx = 0; kx < kernelSize; kx++) {\n const srcX = x + kx - halfKernelSize;\n const srcY = y + ky - halfKernelSize;\n const srcIndex = srcY * width + srcX;\n\n const kernelValue = kernel[ky * kernelSize + kx];\n newValue += pixels[srcIndex] * kernelValue;\n }\n }\n\n new_pixels[dstIndex] = newValue;\n\n }\n }\n\n return new_pixels;\n\n }\n\n static grayscale_to_rgba(grayscale) {\n\n const rgba = new Uint8Array(grayscale.length * 4);\n\n for (let i = 0; i < grayscale.length; i++) {\n const g = grayscale[i];\n const index = i * 4;\n\n rgba[index] = g;\n rgba[index + 1] = g;\n rgba[index + 2] = g;\n rgba[index + 3] = 255; \n }\n\n return rgba;\n\n }\n\n static rgba_to_grayscale(rgba) {\n\n const grayscale = new Uint8Array(rgba.length / 4);\n\n for (let i = 0; i < rgba.length; i += 4) {\n\n grayscale[i / 4] = rgba[i];\n\n }\n\n return grayscale;\n\n }\n\n /**\n * Harden a mask into a grayscale pixel array.\n * \n * pixels needs to be RGBA\n * \n * and mask binary.\n * \n * maskcolor is optional and falls back to blue.\n * \n **/\n static harden_mask(pixels, mask, maskcolor) {\n\n // Modified from: https://github.com/facebookresearch/segment-anything/blob/40df6e4046d8b07ab8c4519e083408289eb43032/demo/src/components/helpers/maskUtils.tsx\n // Copyright (c) Meta Platforms, Inc. and affiliates.\n // All rights reserved.\n\n\n let maskcolor_ = [0, 114, 189, 255];\n\n if (Util.is_defined(maskcolor)) {\n \n maskcolor_ = maskcolor;\n \n } \n\n for (var i = 0; i < mask.length; i++) {\n\n if (mask[i] > 0.0) {\n pixels[4 * i + 0] = maskcolor_[0];\n pixels[4 * i + 1] = maskcolor_[1];\n pixels[4 * i + 2] = maskcolor_[2];\n pixels[4 * i + 3] = maskcolor_[3];\n }\n\n }\n\n return pixels;\n\n }\n\n static is_defined(variable) {\n\n return (typeof variable != 'undefined');\n\n }\n\n}","import {Util} from './util.js';\n\nexport class Framework {\n\n constructor(instance) {\n\n this.name = 'generic';\n this.instance = instance;\n\n this.flip_on_png = false;\n\n }\n\n get_image(from_canvas) {\n\n throw \"Missing Implementation.\";\n\n }\n\n set_image(new_pixels) {\n\n throw \"Missing Implementation.\";\n\n }\n\n set_mask(new_mask) {\n\n throw \"Missing Implementation.\";\n \n }\n\n select_box(callback) {\n\n throw \"Missing Implementation.\";\n\n }\n\n convert_to_png(uint8array, width, height) {\n\n return Util.convert_to_png(uint8array, width, height, this.flip_on_png);\n\n }\n\n}\n","import {Framework} from '../framework.js';\n\nimport {Util} from '../util.js';\n\nexport class Cornerstone2D extends Framework {\n \n constructor(instance) {\n\n super(instance);\n this.name = 'cornerstone2D';\n\n this.cornerstonetools_instance = null;\n\n if (typeof window.cornerstoneTools != 'undefined') {\n\n // TODO probably not too robust\n this.cornerstonetools_instance = window.cornerstoneTools;\n\n }\n\n this.flip_on_png = false;\n\n }\n\n get_image(from_canvas) {\n\n let element = this.instance.getEnabledElements()[0];\n let pixels = null;\n let width = null;\n let height = null;\n\n if (typeof from_canvas != 'undefined') {\n\n // TODO this is hacky going through the canvas\n // later should grab the real volume data\n\n let canvas = element.canvas;\n width = canvas.width;\n height = canvas.height;\n\n let ctx = canvas.getContext('2d');\n\n let imagedata = ctx.getImageData(0, 0, width, height);\n pixels = imagedata.data;\n\n } else {\n\n // this is the real image slice data\n let imagedata = element.image;\n pixels = imagedata.getPixelData();\n width = imagedata.width;\n height = imagedata.height;\n\n }\n\n return {'data':pixels, 'width':width, 'height':height};\n\n }\n\n set_image(new_pixels) {\n\n let element = this.instance.getEnabledElements()[0];\n let pixels = element.image.getPixelData();\n\n // Set the new pixel values\n pixels.set(new_pixels);\n\n // Re-render the current slice\n cornerstone.renderGrayscaleImage(element, true);\n\n }\n\n set_mask(new_mask) {\n\n // TODO this needs to use segmentation/labelmap layers\n // from cornerstone\n // right now, this is modifying the canvas\n\n let element = this.instance.getEnabledElements()[0];\n\n let canvas = element.canvas;\n width = canvas.width;\n height = canvas.height;\n\n let ctx = canvas.getContext('2d');\n\n let imagedata = ctx.getImageData(0, 0, width, height);\n let pixels = imagedata.data;\n\n let masked_image = Util.harden_mask(pixels, new_mask);\n\n let masked_image_as_imagedata = new ImageData(masked_image, width, height);\n\n ctx.putImageData(masked_image_as_imagedata, 0, 0);\n \n }\n\n select_box(callback) {\n\n this.cornerstonetools_instance.setToolActive('RectangleRoi', { mouseButtonMask: 1 })\n\n let element = this.instance.getEnabledElements()[0];\n let canvas = element.canvas;\n\n canvas.onmouseup = function() {\n\n let state = this.cornerstonetools_instance.globalImageIdSpecificToolStateManager.saveToolState();\n\n let topleft = state[Object.keys(state).pop()].RectangleRoi.data[0].handles.start;\n let bottomright = state[Object.keys(state).pop()].RectangleRoi.data[0].handles.end;\n\n let topleft_c = this.instance.pixelToCanvas(element.element, topleft);\n let bottomright_c = this.instance.pixelToCanvas(element.element, bottomright);\n\n this.cornerstonetools_instance.clearToolState(element.element, 'RectangleRoi');\n this.instance.renderGrayscaleImage(element, true);\n\n callback(topleft_c, bottomright_c);\n\n }.bind(this);\n\n }\n\n}\n","import {Framework} from '../framework.js';\n\nimport {Util} from '../util.js';\n\nexport class NiiVue extends Framework {\n \n constructor(instance) {\n\n super(instance);\n this.name = 'niivue';\n\n this.flip_on_png = true;\n\n this.onMouseDown = false;\n this.x1 = null;\n this.y1 = null;\n this.x2 = null;\n this.y2 = null;\n\n }\n\n get_image(from_canvas) {\n\n let element = this.instance.canvas;\n let pixels = null;\n let width = null;\n let height = null;\n\n\n // TODO this is hacky going through the canvas\n // later should grab the real volume data\n\n let old_crosshaircolor = this.instance.opts.crosshairColor;\n let old_crosshairwidth = this.instance.opts.crosshairWidth;\n\n this.instance.setCrosshairColor([0,0,0,0]);\n this.instance.opts.crosshairWidth=0;\n this.instance.updateGLVolume();\n\n\n let ctx = this.instance.gl;\n\n \n width = ctx.drawingBufferWidth;\n height = ctx.drawingBufferHeight;\n\n pixels = new Uint8Array(width * height * 4);\n ctx.readPixels(\n 0, \n 0, \n width, \n height, \n ctx.RGBA, \n ctx.UNSIGNED_BYTE, \n pixels);\n\n // restore crosshairs\n this.instance.setCrosshairColor(old_crosshaircolor);\n this.instance.opts.crosshairWidth = old_crosshairwidth;\n\n if (!Util.is_defined(from_canvas)) {\n\n // convert rgba pixels to grayscale\n pixels = Util.rgba_to_grayscale(pixels);\n\n } else {\n\n // TODO\n // not easily possible yet\n // we could hack it using \n // nv.back.get_value(x,y,z)\n // based on the dimensions\n // nv.back.dims.slice(1);\n // but devs promised easy access in the future\n\n }\n\n\n return {'data':pixels, 'width':width, 'height':height};\n\n }\n\n /**\n * Sets the NiiVue.js image.\n * \n * If is_rgba==true, we do *not* convert to RGBA before setting on canvas.\n **/\n set_image(new_pixels, is_rgba, no_flip) {\n\n // TODO this is hacky since we dont work with the real volume yet\n // create new canvas\n // put pixels on canvas\n // show canvas\n // hide on click\n\n let originalcanvas = this.instance.canvas;\n\n let newcanvas = window.document.createElement('canvas');\n newcanvas.width = originalcanvas.width;\n newcanvas.height = originalcanvas.height;\n\n // put new_pixels down\n let ctx = newcanvas.getContext('2d');\n\n let new_pixels_rgba = null;\n\n if (Util.is_defined(is_rgba)) {\n\n new_pixels_rgba = new_pixels;\n\n } else {\n\n new_pixels_rgba = Util.grayscale_to_rgba(new_pixels);\n\n\n }\n\n let new_pixels_clamped = new Uint8ClampedArray(new_pixels_rgba);\n\n let new_image_data = new ImageData(new_pixels_clamped, newcanvas.width, newcanvas.height);\n \n\n ctx.putImageData(new_image_data, 0, 0);\n\n if (!Util.is_defined(no_flip)) {\n // some flipping action\n ctx.save();\n ctx.scale(1, -1);\n ctx.drawImage(newcanvas, 0, -newcanvas.height);\n ctx.restore();\n }\n\n\n newcanvas.onclick = function() {\n\n // on click, we will restore the nv canvas\n newcanvas.parentNode.replaceChild(originalcanvas, newcanvas);\n\n }\n\n // replace nv canvas with new one\n // originalcanvas.parentNode.replaceChild(newcanvas, originalcanvas);\n newcanvas.style.width = originalcanvas.clientWidth+'px';\n newcanvas.style.height = originalcanvas.clientHeight+'px';\n originalcanvas.parentNode.replaceChild(newcanvas, originalcanvas);\n\n }\n\n set_mask(new_mask) {\n\n // merge image + mask\n // and then call set_image with that information\n\n let image = this.get_image(true);\n\n // TODO here we need to flip one more time, this is until\n // we use the official niivue infrastructure for adding\n // a segmentation layer\n let originalcanvas = this.instance.canvas;\n\n let newcanvas = window.document.createElement('canvas');\n newcanvas.width = originalcanvas.width;\n newcanvas.height = originalcanvas.height;\n // put new_pixels down\n let ctx = newcanvas.getContext('2d');\n let imageclamped = new Uint8ClampedArray(image.data);\n let imagedata = new ImageData(imageclamped, image.width, image.height);\n ctx.putImageData(imagedata, 0, 0);\n ctx.save();\n ctx.scale(1, -1);\n ctx.drawImage(newcanvas, 0, -newcanvas.height);\n ctx.restore();\n image = ctx.getImageData(0, 0, newcanvas.width, newcanvas.height);\n // end of flip\n\n let masked_image = Util.harden_mask(image.data, new_mask);\n\n this.set_image(masked_image, true, true); // rgba data, no flip\n\n\n }\n\n select_box(callback) {\n\n // TODO also hacky until official API supports this\n\n let canvas = this.instance.canvas;\n\n \n canvas.addEventListener('mousedown', function (e) {\n this.isMouseDown = true;\n\n var rect = e.currentTarget.getBoundingClientRect(),\n offsetX = e.clientX * window.devicePixelRatio - rect.left * window.devicePixelRatio,\n offsetY = e.clientY * window.devicePixelRatio - rect.top * window.devicePixelRatio;\n\n this.x1 = offsetX;\n this.y1 = offsetY;\n }.bind(this));\n\n canvas.addEventListener('mousemove', function (e) {\n if (this.isMouseDown) {\n\n var rect = e.currentTarget.getBoundingClientRect(),\n offsetX = e.clientX * window.devicePixelRatio - rect.left * window.devicePixelRatio,\n offsetY = e.clientY * window.devicePixelRatio - rect.top * window.devicePixelRatio;\n\n this.x2 = offsetX;\n this.y2 = offsetY;\n this.instance.drawSelectionBox([this.x1, this.y1, this.x2-this.x1, this.y2-this.y1]);\n }\n }.bind(this));\n\n\n canvas.addEventListener('mouseup', function (e) {\n var rect = e.currentTarget.getBoundingClientRect(),\n offsetX = e.clientX * window.devicePixelRatio - rect.left * window.devicePixelRatio,\n offsetY = e.clientY * window.devicePixelRatio - rect.top * window.devicePixelRatio;\n \n this.x2 = offsetX;\n this.y2 = offsetY;\n this.isMouseDown = false;\n\n let topleft = {x: this.x1, y: this.y1};\n let bottomright = {x: this.x2, y: this.y2};\n\n callback(topleft, bottomright);\n\n }.bind(this));\n\n }\n\n}\n","import {Framework} from '../framework.js';\n\nimport {Util} from '../util.js';\n\nexport class OpenSeaDragon extends Framework {\n constructor(instance) {\n super(instance);\n this.name = 'opensedragon';\n\n }\n\n get_image(from_canvas) {\n \n\n // TODO needs to be generic and executed with Boostlet.init\n\n let viewer = null;\n let vs = this.instance._viewers;\n vs.forEach(function(e) { \n if (e.id == 'viewer') {\n viewer = e;\n }\n });\n\n if (!viewer) {\n throw \"OpenSeaDragon viewer not found.\";\n }\n\n let canvas = viewer.canvas.children[0];\n let ctx = canvas.getContext(\"2d\");\n\n let image = ctx.getImageData(0, 0, canvas.width, canvas.height);\n let rgba_image = Util.rgba_to_grayscale(image.data);\n let pixels = image.data;\n\n if(from_canvas){\n return {'data':pixels, 'width':image.width, 'height':image.height};\n }\n else{\n return {'data':rgba_image, 'width':image.width, 'height':image.height};\n }\n \n }\n\n set_image(new_pixels) {\n\n let viewer = null;\n let vs = this.instance._viewers;\n vs.forEach(function(e) { \n if (e.id == 'viewer') {\n viewer = e;\n }\n });\n\n if (!viewer) {\n throw \"OpenSeaDragon viewer not found.\";\n }\n\n let canvas = viewer.canvas.children[0];\n let ctx = canvas.getContext(\"2d\");\n let new_image = new ImageData(new Uint8ClampedArray(Util.grayscale_to_rgba(new_pixels)), image.width, image.height);\n\n ctx.putImageData(new_image, 0, 0);\n\n }\n\n set_mask(new_mask) {\n\n let viewer = null;\n let vs = this.instance._viewers;\n vs.forEach(function(e) { \n if (e.id == 'viewer') {\n viewer = e;\n }\n });\n\n if (!viewer) {\n throw \"OpenSeaDragon viewer not found.\";\n }\n\n let canvas = viewer.canvas.children[0];\n width = canvas.width;\n height = canvas.height;\n\n let ctx = canvas.getContext('2d');\n let imagedata = ctx.getImageData(0, 0, width, height);\n let pixels = imagedata.data;\n\n let masked_image = Util.harden_mask(pixels, new_mask);\n\n // console.log(masked_image);\n\n let masked_image_as_imagedata = new ImageData(masked_image, width, height);\n\n // console.log(masked_image_as_imagedata);\n\n ctx.putImageData(masked_image_as_imagedata, 0, 0);\n\n\n }\n\n select_box(callback) {\n\n \n // alert(\"Click on top left and bottom rght coordinated of the desired selection box.\")\n let isFirstClick = true;\n let x1, y1, x2, y2;\n \n // Function to handle the mouse click event\n function handleClick(event) {\n if (isFirstClick) {\n // Capture x1 and y1 on the first click\n x1 = event.clientX;\n y1 = event.clientY;\n console.log(`First click: (X1: ${x1}, Y1: ${y1})`);\n isFirstClick = false;\n } else {\n // Capture x2 and y2 on the second click\n x2 = event.clientX;\n y2 = event.clientY;\n console.log(`Second click: (X2: ${x2}, Y2: ${y2})`);\n isFirstClick = true;\n\n let topleft = {x: x1, y: y1};\n let bottomright = {x: x2, y: y2};\n\n callback(topleft, bottomright);\n }\n\n // let topleft = {x: 529, y: 480};\n // let bottomright = {x: 667, y: 588};\n // callback(topleft, bottomright);\n\n }\n \n \n // Add a click event listener to the document\n document.addEventListener(\"click\", handleClick);\n\n\n }\n\n \n\n}\n \n","\nimport {Boostlet} from \"./boostlet.js\"\n\nwindow.console.log('BOOSTLET VERSION 0.1-beta');\n\n// register global namespace with a new boostlet instance\n// later we might want to support multiple active boostlets\nwindow.Boostlet = new Boostlet();\n","import {Util} from './util.js';\n\nimport {Framework} from './framework.js';\n\nexport class Boostlet {\n\n constructor() {\n\n this.framework = null;\n\n }\n\n /**\n * Initializes the Boostlet.\n * \n * This includes several steps such as identifying the \n * visualization/rendering framework that is available. \n * \n * TODO: Later we want to have fallbacks in place if the framework\n * is not detected.\n * \n */\n init(name, instance) {\n\n if (typeof name != 'undefined' && typeof instance != 'undefined') {\n\n console.log('Framework forced by user!');\n throw \"Forced Framework Not Implemented.\";\n // TODO\n\n } else {\n\n this.framework = Util.detect_framework();\n\n }\n\n if (this.framework) {\n\n console.log('Found', this.framework, '!')\n \n } else {\n\n throw \"Framework Not Found.\";\n\n }\n\n }\n\n /**\n * Let's the user select a region of interest box.\n */\n async select_box(callback) {\n\n this.framework.select_box(callback);\n\n }\n\n /**\n * Let's the user select (multiple) seeds.\n */\n async select_seed(howmany) {\n\n throw \"Missing Implementation.\";\n\n }\n\n /**\n * Loads an external javascript file asynchronously. \n */\n async load_script(url, callback) {\n\n Util.load_script(url, callback);\n\n }\n\n /**\n * Sends a HTTP POST request to a url with some data.\n */\n async send_http_post(url, data, callback) {\n\n Util.send_http_post(url, data, callback);\n\n }\n\n /**\n * Gets the current image (2D).\n * \n * TODO: Optional bounding box should be supported.\n * \n */\n get_image(from_canvas) {\n\n return this.framework.get_image(from_canvas);\n\n }\n\n /**\n * Sets the current image (2D).\n * \n */\n set_image(new_pixels) {\n\n return this.framework.set_image(new_pixels);\n\n }\n\n /**\n * Sets the current mask (2D).\n * \n */\n set_mask(new_mask) {\n\n return this.framework.set_mask(new_mask);\n\n }\n\n /**\n * Encode raw image data to PNG.\n */\n convert_to_png(uint8array, width, height) {\n\n return this.framework.convert_to_png(uint8array, width, height);\n\n }\n\n filter(pixels, width, height, kernel) {\n\n return Util.filter(pixels, width, height, kernel);\n\n }\n\n}\n"],"names":[],"version":3,"file":"boostlet.min.js.map"} \ No newline at end of file +{"mappings":"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","sources":["src/util.js","src/framework.js","src/frameworks/cornerstone2d.js","src/frameworks/niivue.js","src/frameworks/openseadragon.js","src/frameworks/xtk.js","src/frameworks/papaya.js","src/index.js","src/boostlet.js"],"sourcesContent":["import {Framework} from './framework.js';\r\nimport {Cornerstone2D} from './frameworks/cornerstone2d.js';\r\nimport {NiiVue} from './frameworks/niivue.js';\r\nimport { OpenSeaDragon } from './frameworks/openseadragon.js';\r\nimport { Xtk } from './frameworks/xtk.js';\r\nimport { Papaya } from './frameworks/papaya.js';\r\n\r\nexport class Util {\r\n \r\n static detect_framework() {\r\n\r\n let framework = null;\r\n\r\n if (Util.is_defined(window.nv)) {\r\n \r\n framework = new NiiVue(window.nv);\r\n \r\n } else if (Util.is_defined(window.niivue)) {\r\n \r\n framework = new NiiVue(window.niivue);\r\n\r\n } else if (Util.is_defined(window.cornerstone)) {\r\n\r\n framework = new Cornerstone2D(window.cornerstone);\r\n\r\n } else if (Util.is_defined(window.r)) {\r\n \r\n framework = new Xtk(window.r);\r\n \r\n } else if (Util.is_defined(window.OpenSeadragon)) {\r\n\r\n framework = new OpenSeaDragon(window.OpenSeadragon);\r\n \r\n } else if (Util.is_defined(window.papayaContainers)) {\r\n framework = new Papaya(window.papayaContainers)\r\n }\r\n\r\n // TODO: fallback to general canvas or webgl framework\r\n\r\n return framework;\r\n\r\n }\r\n\r\n static async load_script(url, callback) {\r\n\r\n // introducing hack to make it work for openneuro\r\n window.Object.defineProperty(window.Object.prototype, 'global', {\r\n get( ){\r\n return window;\r\n },\r\n set(newGlobal) {\r\n globalThis = newGlobal;\r\n }\r\n });\r\n\r\n const script = window.document.createElement(\"script\")\r\n script.type = \"text/javascript\"\r\n script.src = url;\r\n\r\n if (Util.is_defined(callback)) {\r\n script.onload = callback;\r\n }\r\n\r\n window.document.head.appendChild(script);\r\n eval(script);\r\n\r\n }\r\n\r\n static async send_http_post(url, data, callback) {\r\n\r\n let xhr = new XMLHttpRequest();\r\n xhr.open(\"POST\", url);\r\n xhr.onreadystatechange = function () {\r\n if (xhr.readyState === 4) {\r\n // request done\r\n callback( xhr.response );\r\n\r\n return;\r\n\r\n }\r\n }\r\n\r\n xhr.send(data)\r\n\r\n }\r\n\r\n static convert_to_png(uint8array, width, height, flip) {\r\n\r\n // we are using an offscreen canvas for this\r\n let offscreen = window.document.createElement('canvas');\r\n offscreen.width = width;\r\n offscreen.height = height;\r\n\r\n let offscreen_ctx = offscreen.getContext('2d');\r\n\r\n let imgdata = offscreen_ctx.createImageData(offscreen.width, offscreen.height);\r\n let pxdata = imgdata.data;\r\n\r\n for (var i =0; i c.charCodeAt(0));\r\n\r\n return pngpixels;\r\n\r\n }\r\n\r\n static filter(pixels, width, height, kernel) {\r\n\r\n const kernelSize = Math.sqrt(kernel.length);\r\n const halfKernelSize = Math.floor(kernelSize / 2);\r\n\r\n const new_pixels = pixels.slice();\r\n\r\n for (let y = 0; y < height; y++) {\r\n for (let x = 0; x < width; x++) {\r\n const dstIndex = y * width + x;\r\n\r\n let newValue = 0;\r\n for (let ky = 0; ky < kernelSize; ky++) {\r\n for (let kx = 0; kx < kernelSize; kx++) {\r\n const srcX = x + kx - halfKernelSize;\r\n const srcY = y + ky - halfKernelSize;\r\n const srcIndex = srcY * width + srcX;\r\n\r\n const kernelValue = kernel[ky * kernelSize + kx];\r\n newValue += pixels[srcIndex] * kernelValue;\r\n }\r\n }\r\n\r\n new_pixels[dstIndex] = newValue;\r\n\r\n }\r\n }\r\n\r\n return new_pixels;\r\n\r\n }\r\n\r\n static grayscale_to_rgba(grayscale) {\r\n\r\n const rgba = new Uint8Array(grayscale.length * 4);\r\n\r\n for (let i = 0; i < grayscale.length; i++) {\r\n const g = grayscale[i];\r\n const index = i * 4;\r\n\r\n rgba[index] = g;\r\n rgba[index + 1] = g;\r\n rgba[index + 2] = g;\r\n rgba[index + 3] = 255; \r\n }\r\n\r\n return rgba;\r\n\r\n }\r\n\r\n static rgba_to_grayscale(rgba) {\r\n\r\n const grayscale = new Uint8Array(rgba.length / 4);\r\n\r\n for (let i = 0; i < rgba.length; i += 4) {\r\n\r\n grayscale[i / 4] = rgba[i];\r\n\r\n }\r\n\r\n return grayscale;\r\n\r\n }\r\n\r\n /**\r\n * Harden a mask into a grayscale pixel array.\r\n * \r\n * pixels needs to be RGBA\r\n * \r\n * and mask binary.\r\n * \r\n * maskcolor is optional and falls back to blue.\r\n * \r\n **/\r\n static harden_mask(pixels, mask, maskcolor) {\r\n\r\n // Modified from: https://github.com/facebookresearch/segment-anything/blob/40df6e4046d8b07ab8c4519e083408289eb43032/demo/src/components/helpers/maskUtils.tsx\r\n // Copyright (c) Meta Platforms, Inc. and affiliates.\r\n // All rights reserved.\r\n\r\n\r\n let maskcolor_ = [0, 114, 189, 255];\r\n\r\n if (Util.is_defined(maskcolor)) {\r\n \r\n maskcolor_ = maskcolor;\r\n \r\n } \r\n\r\n for (var i = 0; i < mask.length; i++) {\r\n\r\n if (mask[i] > 0.0) {\r\n pixels[4 * i + 0] = maskcolor_[0];\r\n pixels[4 * i + 1] = maskcolor_[1];\r\n pixels[4 * i + 2] = maskcolor_[2];\r\n pixels[4 * i + 3] = maskcolor_[3];\r\n }\r\n\r\n }\r\n\r\n return pixels;\r\n\r\n }\r\n\r\n static is_defined(variable) {\r\n\r\n return (typeof variable != 'undefined');\r\n\r\n }\r\n\r\n}","import {Util} from './util.js';\r\n\r\nexport class Framework {\r\n\r\n constructor(instance) {\r\n\r\n this.name = 'generic';\r\n this.instance = instance;\r\n\r\n this.flip_on_png = false;\r\n\r\n }\r\n\r\n get_image(from_canvas) {\r\n\r\n throw \"Missing Implementation.\";\r\n\r\n }\r\n\r\n set_image(new_pixels) {\r\n\r\n throw \"Missing Implementation.\";\r\n\r\n }\r\n\r\n set_mask(new_mask) {\r\n\r\n throw \"Missing Implementation.\";\r\n \r\n }\r\n\r\n select_box(callback) {\r\n\r\n throw \"Missing Implementation.\";\r\n\r\n }\r\n\r\n convert_to_png(uint8array, width, height) {\r\n\r\n return Util.convert_to_png(uint8array, width, height, this.flip_on_png);\r\n\r\n }\r\n\r\n}\r\n","import {Framework} from '../framework.js';\r\n\r\nimport {Util} from '../util.js';\r\n\r\nexport class Cornerstone2D extends Framework {\r\n \r\n constructor(instance) {\r\n\r\n super(instance);\r\n this.name = 'cornerstone2D';\r\n\r\n this.cornerstonetools_instance = null;\r\n\r\n if (typeof window.cornerstoneTools != 'undefined') {\r\n\r\n // TODO probably not too robust\r\n this.cornerstonetools_instance = window.cornerstoneTools;\r\n\r\n }\r\n\r\n this.flip_on_png = false;\r\n\r\n }\r\n\r\n get_image(from_canvas) {\r\n\r\n let element = this.instance.getEnabledElements()[0];\r\n let pixels = null;\r\n let width = null;\r\n let height = null;\r\n\r\n if (typeof from_canvas != 'undefined') {\r\n\r\n // TODO this is hacky going through the canvas\r\n // later should grab the real volume data\r\n\r\n let canvas = element.canvas;\r\n width = canvas.width;\r\n height = canvas.height;\r\n\r\n let ctx = canvas.getContext('2d');\r\n\r\n let imagedata = ctx.getImageData(0, 0, width, height);\r\n pixels = imagedata.data;\r\n\r\n } else {\r\n\r\n // this is the real image slice data\r\n let imagedata = element.image;\r\n pixels = imagedata.getPixelData();\r\n width = imagedata.width;\r\n height = imagedata.height;\r\n\r\n }\r\n\r\n return {'data':pixels, 'width':width, 'height':height};\r\n\r\n }\r\n\r\n set_image(new_pixels) {\r\n\r\n let element = this.instance.getEnabledElements()[0];\r\n let pixels = element.image.getPixelData();\r\n\r\n // Set the new pixel values\r\n pixels.set(new_pixels);\r\n\r\n // Re-render the current slice\r\n cornerstone.renderGrayscaleImage(element, true);\r\n\r\n }\r\n\r\n set_mask(new_mask) {\r\n\r\n // TODO this needs to use segmentation/labelmap layers\r\n // from cornerstone\r\n // right now, this is modifying the canvas\r\n\r\n let element = this.instance.getEnabledElements()[0];\r\n\r\n let canvas = element.canvas;\r\n width = canvas.width;\r\n height = canvas.height;\r\n\r\n let ctx = canvas.getContext('2d');\r\n\r\n let imagedata = ctx.getImageData(0, 0, width, height);\r\n let pixels = imagedata.data;\r\n\r\n let masked_image = Util.harden_mask(pixels, new_mask);\r\n\r\n let masked_image_as_imagedata = new ImageData(masked_image, width, height);\r\n\r\n ctx.putImageData(masked_image_as_imagedata, 0, 0);\r\n \r\n }\r\n\r\n select_box(callback) {\r\n\r\n this.cornerstonetools_instance.setToolActive('RectangleRoi', { mouseButtonMask: 1 })\r\n\r\n let element = this.instance.getEnabledElements()[0];\r\n let canvas = element.canvas;\r\n\r\n canvas.onmouseup = function() {\r\n\r\n let state = this.cornerstonetools_instance.globalImageIdSpecificToolStateManager.saveToolState();\r\n\r\n let topleft = state[Object.keys(state).pop()].RectangleRoi.data[0].handles.start;\r\n let bottomright = state[Object.keys(state).pop()].RectangleRoi.data[0].handles.end;\r\n\r\n let topleft_c = this.instance.pixelToCanvas(element.element, topleft);\r\n let bottomright_c = this.instance.pixelToCanvas(element.element, bottomright);\r\n\r\n this.cornerstonetools_instance.clearToolState(element.element, 'RectangleRoi');\r\n this.instance.renderGrayscaleImage(element, true);\r\n\r\n callback(topleft_c, bottomright_c);\r\n\r\n }.bind(this);\r\n\r\n }\r\n\r\n}\r\n","import {Framework} from '../framework.js';\r\n\r\nimport {Util} from '../util.js';\r\n\r\nexport class NiiVue extends Framework {\r\n \r\n constructor(instance) {\r\n\r\n super(instance);\r\n this.name = 'niivue';\r\n\r\n this.flip_on_png = true;\r\n\r\n this.onMouseDown = false;\r\n this.x1 = null;\r\n this.y1 = null;\r\n this.x2 = null;\r\n this.y2 = null;\r\n\r\n }\r\n\r\n get_image(from_canvas) {\r\n\r\n let element = this.instance.canvas;\r\n let pixels = null;\r\n let width = null;\r\n let height = null;\r\n\r\n\r\n // TODO this is hacky going through the canvas\r\n // later should grab the real volume data\r\n\r\n let old_crosshaircolor = this.instance.opts.crosshairColor;\r\n let old_crosshairwidth = this.instance.opts.crosshairWidth;\r\n\r\n this.instance.setCrosshairColor([0,0,0,0]);\r\n this.instance.opts.crosshairWidth=0;\r\n this.instance.updateGLVolume();\r\n\r\n\r\n let ctx = this.instance.gl;\r\n\r\n \r\n width = ctx.drawingBufferWidth;\r\n height = ctx.drawingBufferHeight;\r\n\r\n pixels = new Uint8Array(width * height * 4);\r\n ctx.readPixels(\r\n 0, \r\n 0, \r\n width, \r\n height, \r\n ctx.RGBA, \r\n ctx.UNSIGNED_BYTE, \r\n pixels);\r\n\r\n // restore crosshairs\r\n this.instance.setCrosshairColor(old_crosshaircolor);\r\n this.instance.opts.crosshairWidth = old_crosshairwidth;\r\n\r\n if (!Util.is_defined(from_canvas)) {\r\n\r\n // convert rgba pixels to grayscale\r\n pixels = Util.rgba_to_grayscale(pixels);\r\n\r\n } else {\r\n\r\n // TODO\r\n // not easily possible yet\r\n // we could hack it using \r\n // nv.back.get_value(x,y,z)\r\n // based on the dimensions\r\n // nv.back.dims.slice(1);\r\n // but devs promised easy access in the future\r\n\r\n }\r\n\r\n\r\n return {'data':pixels, 'width':width, 'height':height};\r\n\r\n }\r\n\r\n /**\r\n * Sets the NiiVue.js image.\r\n * \r\n * If is_rgba==true, we do *not* convert to RGBA before setting on canvas.\r\n **/\r\n set_image(new_pixels, is_rgba, no_flip) {\r\n\r\n // TODO this is hacky since we dont work with the real volume yet\r\n // create new canvas\r\n // put pixels on canvas\r\n // show canvas\r\n // hide on click\r\n\r\n let originalcanvas = this.instance.canvas;\r\n\r\n let newcanvas = window.document.createElement('canvas');\r\n newcanvas.width = originalcanvas.width;\r\n newcanvas.height = originalcanvas.height;\r\n\r\n // put new_pixels down\r\n let ctx = newcanvas.getContext('2d');\r\n\r\n let new_pixels_rgba = null;\r\n\r\n if (Util.is_defined(is_rgba)) {\r\n\r\n new_pixels_rgba = new_pixels;\r\n\r\n } else {\r\n\r\n new_pixels_rgba = Util.grayscale_to_rgba(new_pixels);\r\n\r\n\r\n }\r\n\r\n let new_pixels_clamped = new Uint8ClampedArray(new_pixels_rgba);\r\n\r\n let new_image_data = new ImageData(new_pixels_clamped, newcanvas.width, newcanvas.height);\r\n \r\n\r\n ctx.putImageData(new_image_data, 0, 0);\r\n\r\n if (!Util.is_defined(no_flip)) {\r\n // some flipping action\r\n ctx.save();\r\n ctx.scale(1, -1);\r\n ctx.drawImage(newcanvas, 0, -newcanvas.height);\r\n ctx.restore();\r\n }\r\n\r\n\r\n newcanvas.onclick = function() {\r\n\r\n // on click, we will restore the nv canvas\r\n newcanvas.parentNode.replaceChild(originalcanvas, newcanvas);\r\n\r\n }\r\n\r\n // replace nv canvas with new one\r\n // originalcanvas.parentNode.replaceChild(newcanvas, originalcanvas);\r\n newcanvas.style.width = originalcanvas.clientWidth+'px';\r\n newcanvas.style.height = originalcanvas.clientHeight+'px';\r\n originalcanvas.parentNode.replaceChild(newcanvas, originalcanvas);\r\n\r\n }\r\n\r\n set_mask(new_mask) {\r\n\r\n // merge image + mask\r\n // and then call set_image with that information\r\n\r\n let image = this.get_image(true);\r\n\r\n // TODO here we need to flip one more time, this is until\r\n // we use the official niivue infrastructure for adding\r\n // a segmentation layer\r\n let originalcanvas = this.instance.canvas;\r\n\r\n let newcanvas = window.document.createElement('canvas');\r\n newcanvas.width = originalcanvas.width;\r\n newcanvas.height = originalcanvas.height;\r\n // put new_pixels down\r\n let ctx = newcanvas.getContext('2d');\r\n let imageclamped = new Uint8ClampedArray(image.data);\r\n let imagedata = new ImageData(imageclamped, image.width, image.height);\r\n ctx.putImageData(imagedata, 0, 0);\r\n ctx.save();\r\n ctx.scale(1, -1);\r\n ctx.drawImage(newcanvas, 0, -newcanvas.height);\r\n ctx.restore();\r\n image = ctx.getImageData(0, 0, newcanvas.width, newcanvas.height);\r\n // end of flip\r\n\r\n let masked_image = Util.harden_mask(image.data, new_mask);\r\n\r\n this.set_image(masked_image, true, true); // rgba data, no flip\r\n\r\n\r\n }\r\n\r\n select_box(callback) {\r\n\r\n // TODO also hacky until official API supports this\r\n\r\n let canvas = this.instance.canvas;\r\n\r\n \r\n canvas.addEventListener('mousedown', function (e) {\r\n this.isMouseDown = true;\r\n\r\n var rect = e.currentTarget.getBoundingClientRect(),\r\n offsetX = e.clientX * window.devicePixelRatio - rect.left * window.devicePixelRatio,\r\n offsetY = e.clientY * window.devicePixelRatio - rect.top * window.devicePixelRatio;\r\n\r\n this.x1 = offsetX;\r\n this.y1 = offsetY;\r\n }.bind(this));\r\n\r\n canvas.addEventListener('mousemove', function (e) {\r\n if (this.isMouseDown) {\r\n\r\n var rect = e.currentTarget.getBoundingClientRect(),\r\n offsetX = e.clientX * window.devicePixelRatio - rect.left * window.devicePixelRatio,\r\n offsetY = e.clientY * window.devicePixelRatio - rect.top * window.devicePixelRatio;\r\n\r\n this.x2 = offsetX;\r\n this.y2 = offsetY;\r\n this.instance.drawSelectionBox([this.x1, this.y1, this.x2-this.x1, this.y2-this.y1]);\r\n }\r\n }.bind(this));\r\n\r\n\r\n canvas.addEventListener('mouseup', function (e) {\r\n var rect = e.currentTarget.getBoundingClientRect(),\r\n offsetX = e.clientX * window.devicePixelRatio - rect.left * window.devicePixelRatio,\r\n offsetY = e.clientY * window.devicePixelRatio - rect.top * window.devicePixelRatio;\r\n \r\n this.x2 = offsetX;\r\n this.y2 = offsetY;\r\n this.isMouseDown = false;\r\n\r\n let topleft = {x: this.x1, y: this.y1};\r\n let bottomright = {x: this.x2, y: this.y2};\r\n\r\n callback(topleft, bottomright);\r\n\r\n }.bind(this));\r\n\r\n }\r\n\r\n}\r\n","import {Framework} from '../framework.js';\r\n\r\nimport {Util} from '../util.js';\r\n\r\nexport class OpenSeaDragon extends Framework {\r\n constructor(instance) {\r\n super(instance);\r\n this.name = 'opensedragon';\r\n\r\n }\r\n\r\n get_image(from_canvas) {\r\n \r\n\r\n // TODO needs to be generic and executed with Boostlet.init\r\n\r\n let viewer = null;\r\n let vs = this.instance._viewers;\r\n vs.forEach(function(e) { \r\n if (e.id == 'viewer') {\r\n viewer = e;\r\n }\r\n });\r\n\r\n if (!viewer) {\r\n throw \"OpenSeaDragon viewer not found.\";\r\n }\r\n\r\n let canvas = viewer.canvas.children[0];\r\n let ctx = canvas.getContext(\"2d\");\r\n\r\n let image = ctx.getImageData(0, 0, canvas.width, canvas.height);\r\n let rgba_image = Util.rgba_to_grayscale(image.data);\r\n let pixels = image.data;\r\n\r\n if(from_canvas){\r\n return {'data':pixels, 'width':image.width, 'height':image.height};\r\n }\r\n else{\r\n return {'data':rgba_image, 'width':image.width, 'height':image.height};\r\n }\r\n \r\n }\r\n\r\n set_image(new_pixels) {\r\n\r\n let viewer = null;\r\n let vs = this.instance._viewers;\r\n vs.forEach(function(e) { \r\n if (e.id == 'viewer') {\r\n viewer = e;\r\n }\r\n });\r\n\r\n if (!viewer) {\r\n throw \"OpenSeaDragon viewer not found.\";\r\n }\r\n\r\n let canvas = viewer.canvas.children[0];\r\n let ctx = canvas.getContext(\"2d\");\r\n let new_image = new ImageData(new Uint8ClampedArray(Util.grayscale_to_rgba(new_pixels)), image.width, image.height);\r\n\r\n ctx.putImageData(new_image, 0, 0);\r\n\r\n }\r\n\r\n set_mask(new_mask) {\r\n\r\n let viewer = null;\r\n let vs = this.instance._viewers;\r\n vs.forEach(function(e) { \r\n if (e.id == 'viewer') {\r\n viewer = e;\r\n }\r\n });\r\n\r\n if (!viewer) {\r\n throw \"OpenSeaDragon viewer not found.\";\r\n }\r\n\r\n let canvas = viewer.canvas.children[0];\r\n width = canvas.width;\r\n height = canvas.height;\r\n\r\n let ctx = canvas.getContext('2d');\r\n let imagedata = ctx.getImageData(0, 0, width, height);\r\n let pixels = imagedata.data;\r\n\r\n let masked_image = Util.harden_mask(pixels, new_mask);\r\n\r\n // console.log(masked_image);\r\n\r\n let masked_image_as_imagedata = new ImageData(masked_image, width, height);\r\n\r\n // console.log(masked_image_as_imagedata);\r\n\r\n ctx.putImageData(masked_image_as_imagedata, 0, 0);\r\n\r\n\r\n }\r\n\r\n select_box(callback) {\r\n\r\n \r\n // alert(\"Click on top left and bottom rght coordinated of the desired selection box.\")\r\n let isFirstClick = true;\r\n let x1, y1, x2, y2;\r\n \r\n // Function to handle the mouse click event\r\n function handleClick(event) {\r\n if (isFirstClick) {\r\n // Capture x1 and y1 on the first click\r\n x1 = event.clientX;\r\n y1 = event.clientY;\r\n console.log(`First click: (X1: ${x1}, Y1: ${y1})`);\r\n isFirstClick = false;\r\n } else {\r\n // Capture x2 and y2 on the second click\r\n x2 = event.clientX;\r\n y2 = event.clientY;\r\n console.log(`Second click: (X2: ${x2}, Y2: ${y2})`);\r\n isFirstClick = true;\r\n\r\n let topleft = {x: x1, y: y1};\r\n let bottomright = {x: x2, y: y2};\r\n\r\n callback(topleft, bottomright);\r\n }\r\n\r\n // let topleft = {x: 529, y: 480};\r\n // let bottomright = {x: 667, y: 588};\r\n // callback(topleft, bottomright);\r\n\r\n }\r\n \r\n \r\n // Add a click event listener to the document\r\n document.addEventListener(\"click\", handleClick);\r\n\r\n\r\n }\r\n\r\n \r\n\r\n}\r\n \r\n","import { Framework } from \"../framework.js\";\r\n\r\nimport { Util } from \"../util.js\";\r\n\r\nexport class Xtk extends Framework {\r\n constructor(instance) {\r\n super(instance);\r\n this.name = \"xtk\";\r\n }\r\n\r\n get_image(from_canvas) {\r\n let canvas = this.instance.ca;\r\n let ctx = canvas.getContext(\"2d\");\r\n\r\n let image = ctx.getImageData(0, 0, canvas.width, canvas.height);\r\n let rgba_image = Util.rgba_to_grayscale(image.data);\r\n\r\n return { data: rgba_image, width: image.width, height: image.height };\r\n // return {'data':pixels, 'width':image.width, 'height':image.height};\r\n }\r\n\r\n set_image(new_pixels) {\r\n let originalcanvas = this.instance.ca;\r\n\r\n let newcanvas = window.document.createElement(\"canvas\");\r\n newcanvas.width = originalcanvas.width;\r\n newcanvas.height = originalcanvas.height;\r\n\r\n let ctx = newcanvas.getContext(\"2d\");\r\n\r\n let newPixelsRgba = Util.grayscale_to_rgba(new_pixels);\r\n\r\n let newPixelsClamped = new Uint8ClampedArray(newPixelsRgba);\r\n\r\n let newImageData = new ImageData(\r\n newPixelsClamped,\r\n newcanvas.width,\r\n newcanvas.height\r\n );\r\n\r\n // Draw the new image data onto the canvas\r\n ctx.putImageData(newImageData, 0, 0);\r\n\r\n newcanvas.onclick = function () {\r\n // on click, we will restore the nv canvas\r\n newcanvas.parentNode.replaceChild(originalcanvas, newcanvas);\r\n };\r\n\r\n // replace nv canvas with new one\r\n originalcanvas.parentNode.replaceChild(newcanvas, originalcanvas);\r\n }\r\n\r\n set_mask(new_mask) {\r\n let image = this.get_image();\r\n\r\n // TODO here we need to flip one more time, this is until\r\n // we use the official niivue infrastructure for adding\r\n // a segmentation layer\r\n let originalcanvas = this.instance.ca;\r\n\r\n let newcanvas = window.document.createElement('canvas');\r\n newcanvas.width = originalcanvas.width;\r\n newcanvas.height = originalcanvas.height;\r\n \r\n // put new_pixels down\r\n let ctx = newcanvas.getContext('2d');\r\n\r\n let imageclamped = new Uint8ClampedArray(image.data);\r\n\r\n let imagedata = new ImageData(imageclamped, image.width, image.height);\r\n\r\n ctx.putImageData(imagedata, 0, 0);\r\n\r\n ctx.save();\r\n ctx.scale(1, -1);\r\n ctx.drawImage(newcanvas, 0, -newcanvas.height);\r\n ctx.restore();\r\n\r\n image = ctx.getImageData(0, 0, newcanvas.width, newcanvas.height);\r\n // end of flip\r\n\r\n let masked_image = Util.harden_mask(image.data, new_mask);\r\n\r\n this.set_image(masked_image); // rgba data, no flip\r\n }\r\n\r\n select_box(callback) {\r\n // alert(\"Click on top left and bottom rght coordinated of the desired selection box.\")\r\n let isFirstClick = true;\r\n let x1, y1, x2, y2;\r\n\r\n // Function to handle the mouse click event\r\n function handleClick(event) {\r\n if (isFirstClick) {\r\n // Capture x1 and y1 on the first click\r\n x1 = event.clientX;\r\n y1 = event.clientY;\r\n console.log(`First click: (X1: ${x1}, Y1: ${y1})`);\r\n isFirstClick = false;\r\n } else {\r\n // Capture x2 and y2 on the second click\r\n x2 = event.clientX;\r\n y2 = event.clientY;\r\n console.log(`Second click: (X2: ${x2}, Y2: ${y2})`);\r\n isFirstClick = true;\r\n\r\n let topleft = { x: x1, y: y1 };\r\n let bottomright = { x: x2, y: y2 };\r\n\r\n callback(topleft, bottomright);\r\n }\r\n\r\n // let topleft = {x: 529, y: 480};\r\n // let bottomright = {x: 667, y: 588};\r\n // callback(topleft, bottomright);\r\n }\r\n\r\n // Add a click event listener to the document\r\n document.addEventListener(\"click\", handleClick);\r\n }\r\n}\r\n","import { Framework } from \"../framework.js\";\r\n\r\nimport { Util } from \"../util.js\";\r\n\r\nexport class Papaya extends Framework {\r\n constructor(instance) {\r\n super(instance);\r\n this.name = \"papaya\";\r\n }\r\n\r\n get_image(from_canvas) {\r\n let canvas = this.instance[0].viewer.canvas;\r\n let ctx = canvas.getContext(\"2d\");\r\n\r\n let image = ctx.getImageData(0, 0, canvas.width, canvas.height);\r\n let rgba_image = Util.rgba_to_grayscale(image.data)\r\n\r\n return { data: rgba_image, width: image.width, height: image.height };\r\n // return {'data':pixels, 'width':image.width, 'height':image.height};\r\n }\r\n\r\n set_image(new_pixels) {\r\n let originalcanvas = this.instance[0].viewer.canvas;\r\n\r\n let newcanvas = window.document.createElement(\"canvas\");\r\n newcanvas.width = originalcanvas.width;\r\n newcanvas.height = originalcanvas.height;\r\n\r\n let ctx = newcanvas.getContext(\"2d\");\r\n\r\n let newPixelsRgba = Util.grayscale_to_rgba(new_pixels);\r\n\r\n let newPixelsClamped = new Uint8ClampedArray(newPixelsRgba);\r\n\r\n let newImageData = new ImageData(\r\n newPixelsClamped,\r\n newcanvas.width,\r\n newcanvas.height\r\n );\r\n\r\n // Draw the new image data onto the canvas\r\n ctx.putImageData(newImageData, 0, 0);\r\n\r\n newcanvas.onclick = function () {\r\n // on click, we will restore the nv canvas\r\n newcanvas.parentNode.replaceChild(originalcanvas, newcanvas);\r\n };\r\n\r\n // replace nv canvas with new one\r\n originalcanvas.parentNode.replaceChild(newcanvas, originalcanvas);\r\n }\r\n}\r\n","\r\nimport {Boostlet} from \"./boostlet.js\"\r\n\r\nwindow.console.log('BOOSTLET VERSION 0.1-beta');\r\n\r\n// register global namespace with a new boostlet instance\r\n// later we might want to support multiple active boostlets\r\nwindow.Boostlet = new Boostlet();\r\n","import {Util} from './util.js';\r\n\r\nimport {Framework} from './framework.js';\r\n\r\nexport class Boostlet {\r\n\r\n constructor() {\r\n\r\n this.framework = null;\r\n\r\n }\r\n\r\n /**\r\n * Initializes the Boostlet.\r\n * \r\n * This includes several steps such as identifying the \r\n * visualization/rendering framework that is available. \r\n * \r\n * TODO: Later we want to have fallbacks in place if the framework\r\n * is not detected.\r\n * \r\n */\r\n init(name, instance) {\r\n\r\n if (typeof name != 'undefined' && typeof instance != 'undefined') {\r\n\r\n console.log('Framework forced by user!');\r\n throw \"Forced Framework Not Implemented.\";\r\n // TODO\r\n\r\n } else {\r\n\r\n this.framework = Util.detect_framework();\r\n\r\n }\r\n\r\n if (this.framework) {\r\n\r\n console.log('Found', this.framework, '!')\r\n \r\n } else {\r\n\r\n throw \"Framework Not Found.\";\r\n\r\n }\r\n\r\n }\r\n\r\n /**\r\n * Let's the user select a region of interest box.\r\n */\r\n async select_box(callback) {\r\n\r\n this.framework.select_box(callback);\r\n\r\n }\r\n\r\n /**\r\n * Let's the user select (multiple) seeds.\r\n */\r\n async select_seed(howmany) {\r\n\r\n throw \"Missing Implementation.\";\r\n\r\n }\r\n\r\n /**\r\n * Loads an external javascript file asynchronously. \r\n */\r\n async load_script(url, callback) {\r\n\r\n Util.load_script(url, callback);\r\n\r\n }\r\n\r\n /**\r\n * Sends a HTTP POST request to a url with some data.\r\n */\r\n async send_http_post(url, data, callback) {\r\n\r\n Util.send_http_post(url, data, callback);\r\n\r\n }\r\n\r\n /**\r\n * Gets the current image (2D).\r\n * \r\n * TODO: Optional bounding box should be supported.\r\n * \r\n */\r\n get_image(from_canvas) {\r\n\r\n return this.framework.get_image(from_canvas);\r\n\r\n }\r\n\r\n /**\r\n * Sets the current image (2D).\r\n * \r\n */\r\n set_image(new_pixels) {\r\n\r\n return this.framework.set_image(new_pixels);\r\n\r\n }\r\n\r\n /**\r\n * Sets the current mask (2D).\r\n * \r\n */\r\n set_mask(new_mask) {\r\n\r\n return this.framework.set_mask(new_mask);\r\n\r\n }\r\n\r\n /**\r\n * Encode raw image data to PNG.\r\n */\r\n convert_to_png(uint8array, width, height) {\r\n\r\n return this.framework.convert_to_png(uint8array, width, height);\r\n\r\n }\r\n\r\n filter(pixels, width, height, kernel) {\r\n\r\n return Util.filter(pixels, width, height, kernel);\r\n\r\n }\r\n\r\n}\r\n"],"names":[],"version":3,"file":"boostlet.min.js.map"} \ No newline at end of file diff --git a/examples/plotly.js b/examples/plotly.js index c6ac7bc7..9dc24a5f 100644 --- a/examples/plotly.js +++ b/examples/plotly.js @@ -1,6 +1,7 @@ script = document.createElement("script"); script.type = "text/javascript"; script.src = "https://boostlet.org/dist/boostlet.min.js"; + script.onload = run; document.head.appendChild(script); eval(script); diff --git a/examples/sobel.js b/examples/sobel.js index aae5df0a..6d144b9a 100644 --- a/examples/sobel.js +++ b/examples/sobel.js @@ -1,6 +1,7 @@ script = document.createElement("script"); script.type = "text/javascript"; script.src = "https://boostlet.org/dist/boostlet.min.js"; + script.onload = run; document.head.appendChild(script); eval(script); diff --git a/gfx/boostlet.png b/gfx/boostlet.png new file mode 100644 index 00000000..b079a49f Binary files /dev/null and b/gfx/boostlet.png differ diff --git a/gfx/favicon.png b/gfx/favicon.png new file mode 100644 index 00000000..051052ab Binary files /dev/null and b/gfx/favicon.png differ diff --git a/gfx/papaya.png b/gfx/papaya.png new file mode 100644 index 00000000..b2ce6bf9 Binary files /dev/null and b/gfx/papaya.png differ diff --git a/gfx/xtk.png b/gfx/xtk.png new file mode 100644 index 00000000..3f3d70cf Binary files /dev/null and b/gfx/xtk.png differ diff --git a/index.html b/index.html index a334715e..2d052aca 100644 --- a/index.html +++ b/index.html @@ -3,6 +3,7 @@ + BOOSTLET.js