{"version":3,"file":"myassays-pad.iife.js","sources":["../../../src/myassays-global/Model.js","../model/DataModel.js","../MyassaysPad.css?raw","../templates/padStatusTemplate.js","../templates/endPointTemplate.js","../templates/kineticsTemplate.js","../templates/kineticsIndexedTemplate.js","../templates/spectralTemplate.js","../templates/wellScanTemplate.js","../templates/templateHelpers.js","../templates/gridTemplate.js","../../../src/myassays-global/PropTypes.js","../../../src/myassays-global/FlashTimer.js","../ActivityMonitor.js","../../../src/myassays-global/abstractClasses/MyassaysComponent.js","../../../node_modules/webidl-conversions/lib/index.js","../../../node_modules/jsdom/lib/jsdom/living/generated/utils.js","../../../node_modules/jsdom/lib/jsdom/living/helpers/namespaces.js","../../../node_modules/jsdom/lib/jsdom/living/nodes/HTMLCollection-impl.js","../../../node_modules/jsdom/lib/jsdom/living/generated/HTMLCollection.js","../__vite-browser-external","../../../node_modules/whatwg-url/lib/utils.js","../../../node_modules/punycode/punycode.es6.js","../../../node_modules/tr46/lib/regexes.js","../../../node_modules/tr46/lib/statusMapping.js","../../../node_modules/tr46/index.js","../../../node_modules/whatwg-url/lib/infra.js","../../../node_modules/whatwg-url/lib/encoding.js","../../../node_modules/whatwg-url/lib/percent-encoding.js","../../../node_modules/whatwg-url/lib/url-state-machine.js","../../../node_modules/whatwg-url/lib/urlencoded.js","../../../node_modules/whatwg-url/lib/Function.js","../../../node_modules/whatwg-url/lib/URLSearchParams-impl.js","../../../node_modules/whatwg-url/lib/URLSearchParams.js","../../../node_modules/whatwg-url/lib/URL-impl.js","../../../node_modules/whatwg-url/lib/URL.js","../../../node_modules/whatwg-url/webidl2js-wrapper.js","../../../node_modules/whatwg-url/index.js","../../../node_modules/symbol-tree/lib/SymbolTreeNode.js","../../../node_modules/symbol-tree/lib/TreePosition.js","../../../node_modules/symbol-tree/lib/TreeIterator.js","../../../node_modules/symbol-tree/lib/SymbolTree.js","../../../node_modules/jsdom/lib/jsdom/living/helpers/internal-constants.js","../__vite-optional-peer-dep:canvas:jsdom","../../../node_modules/jsdom/lib/jsdom/utils.js","../MyassaysPad.js","../index.js"],"sourcesContent":["\nfunction cloneValue(value) {\n if (value instanceof Model) {\n return value.attributes;\n } else if (value instanceof Array) {\n return value.map(cloneValue);\n } else if (value instanceof Object) {\n const returnValue = {};\n Object.keys(value).forEach(key => {\n returnValue[key] = cloneValue(value[key]);\n });\n return returnValue;\n } else {\n return value;\n }\n}\n\nexport default class Model extends EventTarget {\n constructor(initialAttributes = {}) {\n super();\n this._previousAttributes = {};\n this._attributes = {...this.constructor.defaults, ...initialAttributes};\n }\n\n set(newAttributes, noTrigger = false) {\n this._previousAttributes = {...this._attributes};\n const previousAttributes = {...this._attributes};\n Object.keys(newAttributes).forEach(key => {\n const newAttribute = newAttributes[key];\n if (this._attributes[key] instanceof Model) {\n if (newAttribute instanceof Model) {\n this._attributes[key] = newAttribute;\n } else {\n this._attributes[key].set(newAttribute);\n }\n } else {\n this._attributes[key] = newAttribute;\n }\n });\n\n if (!noTrigger) {\n const evt = new Event('change');\n const attributes = this._attributes;\n evt.data = {\n attributes,\n previousAttributes,\n hasChanged: (attributeName, handler) => {\n const prevValue = previousAttributes[attributeName];\n const newValue = attributes[attributeName];\n const changed = newValue !== prevValue;\n changed && handler && handler(newValue, prevValue);\n return changed;\n },\n }\n this.dispatchEvent(evt);\n }\n noTrigger || Object.keys(this._attributes).forEach(key => {\n const previousValue = previousAttributes[key];\n const newValue = this._attributes[key];\n if (previousValue !== newValue) {\n const evt = new Event(`change:${key}`);\n evt.data = {\n previousValue: cloneValue(previousValue),\n newValue: cloneValue(newValue),\n }\n this.dispatchEvent(evt);\n }\n });\n }\n\n getExact(attributeName) {\n return this._attributes[attributeName];\n }\n\n get(attributeName) {\n const value = this._attributes[attributeName];\n if (value instanceof Model) {\n return value;\n } else {\n return cloneValue(value);\n }\n }\n\n reset() {\n this._previousAttributes = {};\n this._attributes = {...this.constructor.defaults};\n }\n\n hasChanged(attributeName, handler) {\n const prevValue = this._previousAttributes[attributeName];\n const newValue = this._attributes[attributeName];\n const changed = newValue !== prevValue;\n changed && handler && handler(newValue, prevValue);\n return changed;\n }\n\n set attributes(value) {\n this._attributes = {...value};\n }\n\n get attributes() {\n return cloneValue(this._attributes);\n }\n\n get previousAttributes() {\n return cloneValue(this._previousAttributes);\n }\n\n static defaults = {};\n}\n","import Model from 'myassays-global/Model';\n\nfunction splitLine(raw) {\n if (raw.includes('\\t')) {\n return raw.trim().split(/\\t/).map(item => item.trim());\n } else {\n return raw.trim().split(/ +/);\n }\n}\n\nclass WellScanDataPoint extends Model {\n static defaults = {\n value: '',\n }\n}\n\nclass KineticsDataPoint extends Model {\n static defaults = {\n time: '',\n value: '',\n }\n}\n\nclass KineticsIndexedDataPoint extends Model {\n static defaults = {\n value: '',\n }\n}\n\nclass SpectralDataPoint extends Model {\n static defaults = {\n wavelength: '',\n value: '',\n }\n}\n\nclass EndPointDataPoint extends Model {\n static defaults = {\n value: '',\n }\n}\n\nclass Well extends Model {\n constructor(rawText) {\n super();\n this.fromRaw(rawText);\n }\n\n fromRaw(raw) {\n\n }\n\n toRaw() {\n return '';\n }\n get wellLabelToRowColumn() {\n return Well._wellLabelToRowColumn;\n }\n get rowColumnToWellLabel() {\n return Well._rowColumnToWellLabel;\n }\n static defaults = {\n row: 0,\n column: 0,\n dataPoints: [],\n }\n static _wellLabelToRowColumn(wellLabel) {\n const row = wellLabel.charCodeAt(0) - 'A'.charCodeAt(0);\n const column = Number(wellLabel.slice(1)) - 1;\n return {row, column};\n }\n static _rowColumnToWellLabel(row, column) {\n return String.fromCharCode('A'.charCodeAt(0) + row) + String(column + 1);\n }\n}\n\nclass WellScanWell extends Well {\n fromRaw(raw) {\n // A1 6.738 12.862 21.816 27.748\n const parts = splitLine(raw);\n const { row, column } = this.wellLabelToRowColumn(parts.shift());\n const dataPoints = parts.map(part => {\n return new WellScanDataPoint({value: part});\n });\n this.set({row, column, dataPoints});\n }\n\n toRaw() {\n const { row, column, dataPoints } = this.attributes;\n const parts = [this.rowColumnToWellLabel(row, column)];\n dataPoints.forEach(dataPoint => {\n parts.push(dataPoint.value);\n });\n\n return parts.join(' ');\n }\n}\n\nclass KineticsWell extends Well {\n fromRaw(raw) {\n // A1 0.000;7.769 0.100;17.496 0.200;23.541 0.300;33.412 0.400;39.010 0.500;52.017 0.600;59.519 0.700;60.001 0.800;72.080 0.900;84.380\n const parts = splitLine(raw);\n const { row, column } = this.wellLabelToRowColumn(parts.shift());\n const dataPoints = parts.map(part => {\n const [ time, value ] = part.split(';');\n return new KineticsDataPoint({time, value});\n });\n this.set({row, column, dataPoints});\n }\n\n toRaw() {\n const { row, column, dataPoints } = this.attributes;\n const parts = [this.rowColumnToWellLabel(row, column)];\n dataPoints.forEach(dataPoint => {\n const { time, value } = dataPoint;\n parts.push(`${time};${value}`);\n })\n return parts.join(' ');\n }\n}\n\nclass KineticsIndexedWell extends Well {\n fromRaw(raw) {\n // A1 1.300 2.274 3.869 4.827 5.704 6.985 9.071 8.979 11.448 11.410\n const parts = splitLine(raw);\n const { row, column } = this.wellLabelToRowColumn(parts.shift());\n const dataPoints = parts.map(part => {\n return new KineticsIndexedDataPoint({value: part});\n });\n this.set({row, column, dataPoints});\n }\n\n toRaw() {\n const { row, column, dataPoints } = this.attributes;\n const parts = [this.rowColumnToWellLabel(row, column)];\n dataPoints.forEach(dataPoint => {\n const { value } = dataPoint;\n parts.push(value);\n })\n return parts.join(' ');\n }\n}\n\nclass SpectralWell extends Well {\n fromRaw(raw) {\n // A1 300.000;0.061 350.000;0.187 400.000;0.418 450.000;0.676 500.000;0.794 550.000;0.676 600.000;0.418 650.000;0.187 700.000;0.061\n const parts = splitLine(raw);\n const { row, column } = this.wellLabelToRowColumn(parts.shift());\n const dataPoints = parts.map(part => {\n const [ wavelength, value ] = part.split(';');\n return new SpectralDataPoint({wavelength, value});\n });\n this.set({row, column, dataPoints});\n }\n\n toRaw() {\n const { row, column, dataPoints } = this.attributes;\n const parts = [this.rowColumnToWellLabel(row, column)];\n dataPoints.forEach(dataPoint => {\n const { wavelength, value } = dataPoint;\n parts.push(`${wavelength};${value}`);\n })\n return parts.join(' ');\n }\n}\n\nclass EndPointWell extends Well {\n constructor(rawText, row, column) {\n super(rawText);\n this.set({row, column});\n }\n\n fromRaw(raw) {\n this.set({row: this.row, dataPoints: [new EndPointDataPoint({value: raw})]});\n }\n\n toRaw() {\n return this.get('dataPoints').map(dataPoint => dataPoint.value).join('');\n }\n}\n\nclass Matrix extends Model {\n constructor(matrixTypeId, matrixData, rawWellData, dataSpec) {\n super(matrixData);\n this.matrixTypeId = matrixTypeId;\n this.dataSpec = dataSpec;\n this.fromRaw(rawWellData);\n }\n\n get _WellClass() {\n switch (this.matrixTypeId.toLowerCase()) {\n case 'xy-kinetics': return KineticsWell;\n case 'xy-kinetics-indexed': return KineticsIndexedWell;\n case 'xy-spectral': return SpectralWell;\n case 'xyz-well-scan': return WellScanWell;\n default: return EndPointWell;\n }\n }\n\n static defaults = {\n wells: [],\n label: '',\n indicesLabel: '',\n }\n\n fromRaw(raw) {\n let rawWells;\n if (this.matrixTypeId === 'endpoint') {\n rawWells = raw.split(/[\\n\\t ]+/);\n\n while (rawWells.length < this.dataSpec.containerWidth * this.dataSpec.containerHeight) {\n rawWells.push('-');\n }\n this.set({wells: rawWells.map((rawWell, i) => {\n const row = Math.floor(i / this.dataSpec.containerWidth);\n const column = i - row * this.dataSpec.containerWidth;\n return new EndPointWell(rawWell, row, column);\n })});\n } else {\n rawWells = raw.split(/ ?\\n ?/);\n this.set({wells: rawWells.map(rawWell => new this._WellClass(rawWell))});\n }\n }\n\n toRaw() {\n if (this.matrixTypeId === 'endpoint') {\n const re = new RegExp(`((?:[^ ]* ){${this.dataSpec.containerWidth - 1}}[^ ]*) `, 'g');\n const colWidths = Array(this.dataSpec.containerWidth).fill(0);\n let raw = this.getExact('wells').map(well => {\n const rawWell = well.toRaw();\n const column = well.get('column');\n colWidths[column] = Math.max(colWidths[column], rawWell.length);\n return rawWell;\n }).join(' ').replace(re, '$1\\n').trim();\n\n let col = 0;\n return raw.replace(/[^ \\n]+/mg, match => {\n const width = colWidths[col];\n col++;\n if (col === this.dataSpec.containerWidth) {\n col = 0;\n }\n return match.padEnd(width, ' ');\n });\n } else {\n let raw = '';\n const colWidths = [];\n raw += this.getExact('wells').map(well => {\n const rawWell = well.toRaw()\n rawWell.split(/ +/).forEach((value, col) => {\n const width = colWidths[col] || 0;\n colWidths[col] = Math.max(width, value.length);\n });\n return rawWell;\n }).join('\\n');\n let col = 0;\n raw = raw.replace(/[^ \\n]+/mg, match => {\n if (/^[A-Z][0-9]{1,2}$/.test(match)) {\n col = 0;\n }\n const width = colWidths[col];\n col++;\n return match.padEnd(width, ' ');\n });\n if (this.matrixTypeId === 'xyz-well-scan') {\n raw = `${this.get('indicesLabel')}\\n` + raw;\n }\n return raw;\n }\n }\n}\n\nclass Container extends Model {\n static defaults = {\n label: '',\n matrices: [],\n }\n\n toRaw() {\n const matrices = this.getExact('matrices');\n const showHeadings = matrices.length > 1;\n\n return matrices.map((matrix, i) => {\n let label = '';\n if (showHeadings) {\n label = matrix.get('label');\n if (!label) {\n label = `Raw${i + 1}`;\n }\n }\n return `${label ? `${label}\\n` : ''}${matrix.toRaw()}`;\n }).join('\\n');\n }\n}\n\nexport default class DataModel extends Model {\n constructor(rawText, dataSpec) {\n super();\n this.dataSpec = dataSpec;\n this.fromRaw(rawText);\n }\n\n static defaults = {\n containers: [],\n }\n\n toRaw() {\n const containers = this.getExact('containers');\n return containers.map((container, i) => {\n let label = '';\n if (this.dataSpec.multipleContainers) {\n label = container.get('label');\n if (!label) {\n label = (this.dataSpec.containerType === 'plex' ? 'Analyte' : 'Plate') + ` ${i + 1}`;\n }\n }\n return `${label ? `${label}\\n` : ''}${container.toRaw()}`;\n }).join('\\n');\n }\n\n fromRaw(raw) {\n // consume the raw file line by line\n const whitespacePattern = '[ \\\\n\\\\t]+';\n\n const numberPattern = '-?[0-9]+([.,][0-9]+e-?[0-9]{1,2}|[.,][0-9]+|[.,]?e-?[0-9]{1,2})?';\n const wellLabelPattern = '[A-Z][1-9][0-9]?';\n const xyValuePattern = `${numberPattern};${numberPattern}`;\n\n const getRE = matrixTypeId => {\n switch (matrixTypeId) {\n case 'endpoint': return new RegExp(`^((${numberPattern}|-)${whitespacePattern})*(${numberPattern}|-)$`, 'i');\n case 'xy-kinetics-indexed':\n case 'xyz-well-scan': return new RegExp(`^${wellLabelPattern}(${whitespacePattern}(${numberPattern}|-))*$`, 'i');\n case 'xy-kinetics':\n case 'xy-spectral': return new RegExp(`^${wellLabelPattern}(${whitespacePattern}${xyValuePattern})*$`, 'i');\n default: throw new Error(`matrix type \"${matrixTypeId}\" not recognised.`);\n }\n }\n\n const lines = raw.split('\\n').map(line => line.trim()).filter(line => line !== '');\n\n const containers = [];\n while (lines.length) {\n const containerData = {};\n if (this.dataSpec.multipleContainers) {\n containerData.label = lines.shift();\n }\n containerData.matrices = [];\n this.dataSpec.matrices.forEach(matrixTypeId => {\n const re = getRE(matrixTypeId);\n const matrixData = {};\n if (this.dataSpec.matrices.length > 1) {\n matrixData.label = lines.shift();\n }\n if (matrixTypeId === 'xyz-well-scan') {\n matrixData.indicesLabel = lines.shift();\n }\n const rawWellData = [];\n while (lines.length) {\n const line = lines.shift();\n\n if (re.test(line)) {\n rawWellData.push(line);\n } else {\n const previousLine = rawWellData.pop();\n if (re.test(previousLine + ' ' + line)) {\n rawWellData.push(previousLine + ' ' + line);\n } else {\n rawWellData.push(previousLine);\n lines.unshift(line);\n break;\n }\n }\n }\n if (rawWellData.length > 0) {\n containerData.matrices.push(new Matrix(matrixTypeId, matrixData, rawWellData.join('\\n'), this.dataSpec));\n }\n });\n if (containerData.matrices.length === this.dataSpec.matrices.length) {\n containers.push(new Container(containerData));\n }\n if (!this.dataSpec.multipleContainers) break;\n }\n this.set({containers});\n }\n}\n","export default \":host {\\n position: relative;\\n display: inline-block;\\n width: 100%;\\n}\\n\\n* {\\n box-sizing: border-box;\\n}\\n\\n#grid-container {\\n position: relative;\\n top: 0;\\n left: 0;\\n margin: 0;\\n padding: 0;\\n overflow: hidden;\\n}\\n\\n#grid {\\n position: absolute;\\n top: 0;\\n left: 0;\\n margin: 0;\\n padding: 0;\\n border: 1px solid transparent;\\n}\\n#grid .matrix {\\n display: grid;\\n}\\n#grid .matrix > * {\\n outline: 1px solid black;\\n outline-offset: -0.5px;\\n}\\n#grid .well-label {\\n grid-column-start: 1;\\n}\\n#grid .unknown {\\n background-color: white;\\n color: transparent;\\n}\\n\\n#pad {\\n position: absolute;\\n top: 0;\\n left: 0;\\n width: 100%;\\n border: 1px solid transparent;\\n background: transparent;\\n margin: 0;\\n white-space: pre;\\n overflow: auto;\\n resize: none;\\n}\\n#pad.scrolling-x {\\n border-right-color: black;\\n}\\n#pad.scrolling-y {\\n border-bottom-color: black;\\n}\\n#pad:focus {\\n border: 1px solid black;\\n}\\n#pad.display {\\n color: transparent;\\n}\\n\\n@keyframes spin {\\n 100% {\\n transform: rotate(360deg);\\n }\\n}\\n#pad-status {\\n position: sticky;\\n top: 0;\\n left: 0;\\n bottom: 0;\\n right: 0;\\n z-index: 3;\\n font-family: sans-serif;\\n font-size: 14px;\\n text-align: center;\\n line-height: 22px;\\n background: white;\\n color: black;\\n border: 2px solid black;\\n border-radius: 13px;\\n padding: 0 3px;\\n margin: 4px 3px;\\n display: inline-block;\\n}\\n#pad-status svg {\\n vertical-align: -3px;\\n}\\n#pad-status.ok {\\n color: green;\\n border-color: green;\\n}\\n#pad-status.validating {\\n color: green;\\n border-color: green;\\n}\\n#pad-status.validating svg {\\n animation: spin 1s linear infinite;\\n}\\n#pad-status.failed {\\n color: red;\\n border-color: red;\\n}\\n#pad-status.typing {\\n display: none;\\n}\\n#pad-status.none {\\n display: none;\\n}\\n\\n/*# sourceMappingURL=MyassaysPad.css.map */\\n\"","\nexport default function padStatusTemplate (status, error) {\n if (status === 'ok') {\n return ``;\n } else if (status === 'validating') {\n return ``\n } else if (status === 'failed') {\n return ` ${error}`;\n } else {\n return '';\n }\n}\n","import {wellStyle, wellFlashClass} from './templateHelpers.js';\n\nexport default function endPointTemplate(data, containerIndex, matrixIndex, matrixData) {\n return `\\\n