{"version":3,"file":"myassays-layout-viewer.iife.js","sources":["../../../src/myassays-global/PropTypes.js","../../../src/myassays-global/FlashTimer.js","../../../src/myassays-global/abstractClasses/MyassaysComponent.js","../../../src/myassays-global/Utils.js","../styles.css?raw","../MyassaysLayoutViewer.js","../index.js"],"sourcesContent":["\nconst types = {\n NUMBER: 'number',\n STRING: 'string',\n JSON: 'json',\n BOOL: 'bool',\n DIMENSION: 'dimension',\n STYLE: 'style',\n};\n\nfunction attributesToObject(attributes) {\n if (attributes instanceof NamedNodeMap) {\n const result = {};\n for (let i = 0; i < attributes.length; i++) {\n const attr = attributes.item(i);\n result[attr.name.toLowerCase()] = {name: attr.name, value: attr.value};\n }\n return result;\n } else {\n return attributes;\n }\n}\n\nclass PropType {\n constructor(type) {\n this._type = type;\n this._required = false;\n this._observed = false;\n this._comment = '';\n this._defaultValue = undefined;\n this._attrName = '';\n\n if (type === types.STRING) {\n this._lookup = undefined;\n\n this.lookup = valueArrayOrObject => {\n this._lookup = Object.values(valueArrayOrObject);\n this._caseSensitive = false;\n Object.defineProperty(this, 'caseSensitive', {\n get() {\n this._caseSensitive = true;\n return this;\n }\n });\n return this;\n }\n this.regExp = regExp => {\n this._regExp = regExp;\n return this;\n }\n }\n\n if (type === types.NUMBER) {\n this._minValue = undefined;\n this._maxValue = undefined;\n\n this.min = value => {\n this._minValue = value;\n return this;\n }\n\n this.max = value => {\n this._maxValue = value;\n return this;\n }\n }\n }\n\n get required() {\n this._required = true;\n return this;\n }\n\n get observed() {\n this._observed = true;\n return this;\n }\n\n comment(value) {\n this._comment = value;\n return this;\n }\n\n default(value) {\n this._defaultValue = value;\n return this;\n }\n\n attrName(value) {\n this._attrName = value;\n return this;\n }\n\n _setAttrNameFromPropName(value, noHyphens) {\n if (! this._attrName) {\n this._attrName = (noHyphens ? value : value.replace(/([A-Z])/g, '-$1')).toLowerCase();\n }\n }\n}\n\nclass PropTypes {\n static ALLOW_HYPHENS_IN_ATTRIBUTE_NAMES = '_allowHyphensInAttributeNames';\n static get number() { return new PropType(types.NUMBER) };\n static get string() { return new PropType(types.STRING) };\n static get bool() { return new PropType(types.BOOL) };\n static get dimension() { return new PropType(types.DIMENSION) };\n static get style() { return new PropType(types.STYLE) };\n static get json() { return new PropType(types.JSON) };\n\n static getDefaults(propTypes) {\n const defaults = {};\n Object.keys(propTypes).map(key => {\n if (key !== this.ALLOW_HYPHENS_IN_ATTRIBUTE_NAMES) {\n const propType = propTypes[key];\n if (propType._defaultValue !== undefined) {\n defaults[key] = propType._defaultValue;\n }\n }\n });\n return defaults;\n }\n\n static getObserved(propTypes) {\n const observed = [];\n const noHyphens = propTypes._allowHyphensInAttributeNames === false;\n Object.keys(propTypes).map(key => {\n if (key !== this.ALLOW_HYPHENS_IN_ATTRIBUTE_NAMES) {\n const propType = propTypes[key];\n propType._setAttrNameFromPropName(key, noHyphens);\n if (propType._observed) {\n observed.push(propType._attrName);\n }\n }\n });\n return observed;\n }\n\n static propTypesToDoc(propTypes) {\n let docLines = [];\n\n let commentsExist = false;\n const noHyphens = propTypes._allowHyphensInAttributeNames === false;\n\n Object.keys(propTypes).forEach(key => {\n if (key !== this.ALLOW_HYPHENS_IN_ATTRIBUTE_NAMES) {\n const propType = propTypes[key];\n propType._setAttrNameFromPropName(key, noHyphens);\n if (propType._comment !== '') {\n commentsExist = true;\n }\n docLines.push(`| ${[\n `
'),\n ].join(' | ')} |`);\n }\n });\n if (!commentsExist) {\n docLines = docLines.map(item => item.replace(/\\| +\\|$/, '|'))\n }\n docLines.unshift('| --- | --- | --- | --- | --- | --- |' + (commentsExist ? ' --- |' : ''));\n docLines.unshift('| Attribute | Type | Valid Values | Default | Reqrd | Obsvd |' + (commentsExist ? ' Comment |' : ''));\n return docLines.join('\\n').replace(/_/g, '\\\\_');\n }\n\n static attributesToProps(element, singleAttrName) {\n const attributes = element.attributes;\n const propTypes = element.constructor.propTypes;\n const hash = {};\n const required = [];\n const noHyphens = propTypes._allowHyphensInAttributeNames === false;\n Object.keys(propTypes).forEach(key => {\n if (key !== this.ALLOW_HYPHENS_IN_ATTRIBUTE_NAMES) {\n const propType = propTypes[key];\n propType._setAttrNameFromPropName(key, noHyphens);\n hash[propType._attrName] = {\n name: key,\n propType: propType,\n }\n if (propType._required) {\n required.push(key);\n }\n }\n });\n const props = {};\n const attrObject = attributesToObject(attributes);\n const elementId = `Location: ${element.id ? `id=${element.id}` : element.outerHTML}`;\n Object.keys(attrObject).filter(key => !singleAttrName || key === singleAttrName).map(key => {\n const { name, value } = attrObject[key];\n let convertedValue;\n const info = hash[key];\n if (info === undefined) {\n throw new Error(`Unknown attribute \"${name}\". ${elementId}`);\n } else {\n const propType = hash[key].propType;\n switch (propType._type) {\n case types.BOOL:\n if (typeof value === 'boolean') {\n convertedValue = value;\n } else {\n const lowerCaseValue = value.toLowerCase();\n if (lowerCaseValue !== 'true' && lowerCaseValue !== 'false') {\n throw new Error(`Attribute \"${name}\" must be either \"true\" or \"false\". ${elementId}`);\n } else {\n convertedValue = lowerCaseValue === 'true';\n }\n }\n break;\n case types.DIMENSION:\n if (! /^(-?\\d+(\\.\\d+)?(px|%)?|content)$/.test(value)) {\n throw new Error(`Attribute \"${name}\" must be a number with optional \"px\" or \"%\" suffix, or \"content\". ${elementId}`);\n } else {\n convertedValue = value.replace(/([-\\d.]+)(px|)$/, '$1px');\n }\n break;\n case types.STRING:\n convertedValue = value;\n if (propType._required && !convertedValue) {\n throw new Error(`Attribute \"${name}\" must be one of ${propType._lookup.join('|')}. ${elementId}`);\n }\n if (propType._lookup !== undefined) {\n let testValue, testLookup;\n if (propType._caseSensitive) {\n testValue = convertedValue;\n testLookup = propType._lookup;\n } else {\n testValue = convertedValue.toLowerCase();\n testLookup = propType._lookup.map(item => item.toLowerCase());\n }\n if (! testLookup.includes(testValue)) {\n throw new Error(`Attribute \"${name}\" must be one of ${propType._lookup.join('|')}. ${elementId}`);\n }\n convertedValue = testValue;\n }\n if (propType._regExp !== undefined) {\n if (! propType._regExp.test(convertedValue)) {\n throw new Error(`Attribute \"${name}\" must conform to reg. exp. \"${propType._regExp.toString()}\". ${elementId}`);\n }\n }\n break;\n case types.NUMBER:\n convertedValue = parseFloat(value);\n if (isNaN(convertedValue)) {\n throw new Error(`Attribute \"${name}\" must be a valid number. ${elementId}`);\n }\n if (propType._minValue !== undefined && convertedValue < propType._minValue) {\n throw new Error(`Attribute \"${name}\" must not be less than ${propType._minValue}. ${elementId}`);\n }\n if (propType._maxValue !== undefined && convertedValue > propType._maxValue) {\n throw new Error(`Attribute \"${name}\" must not be greater than ${propType._maxValue}. ${elementId}`);\n }\n break;\n case types.STYLE:\n convertedValue = value.replace(/^\\s*(.*?);?\\s*$/, '$1;');\n if (!/^([a-zA-Z\\-\\s]+:[a-zA-Z0-9\\-.\\s#%\\(\\)\\+,']*;)+$/.test(convertedValue)) {\n throw new Error(`Attribute \"${name}\" must be a valid inline style declaration. ${elementId}`);\n }\n break;\n case types.JSON:\n try {\n convertedValue = JSON.parse(value);\n } catch (error) {\n throw new Error(`Attribute \"${name}\" must be a valid JSON string. ${elementId}`);\n }\n break;\n }\n props[hash[key].name] = convertedValue;\n }\n });\n\n if (!singleAttrName) {\n required.forEach(name => {\n if (props[name] === undefined) {\n throw new Error(`Attribute \"${name.replace(/([A-Z])/g, '-$1').toLowerCase()}\" is required. ${elementId}`);\n }\n });\n }\n\n return {\n ...(singleAttrName ? {} : PropTypes.getDefaults(propTypes)),\n ...props,\n };\n }\n}\n\nexport default PropTypes;\n","\nexport default class FlashTimer {\n static _startTimer() {\n if (this._timerId === undefined) {\n this._state = false;\n this._timerId = setInterval(() => this._toggleState(), 500);\n }\n }\n\n static _stopTimer() {\n clearInterval(this._timerId);\n delete this._timerId;\n }\n\n static _toggleState() {\n this._state = !this._state;\n this._subscribers.forEach(handler => handler(this._state));\n }\n\n static subscribe(handler) {\n this._subscribers || (this._subscribers = []);\n if (! this._subscribers.includes(handler)) {\n this._subscribers.push(handler);\n }\n this._startTimer();\n }\n\n static unSubscribe(handler) {\n this._subscribers = this._subscribers.filter(item => item !== handler);\n if (this._subscribers.length === 0) {\n delete this._subscribers;\n this._stopTimer();\n }\n }\n\n static get state() {\n return this._state;\n }\n}\n","import { PropTypes } from 'myassays-global';\n\nexport default class MyassaysComponent extends HTMLElement {\n static get documentationMarkdown() {\n const { propTypes } = this;\n if (propTypes) {\n return `\\\n${PropTypes.propTypesToDoc(propTypes)}`;\n } else {\n return 'No documentation available';\n }\n }\n}\n","\nexport default class Utils {\n static repeat (times, iterator) {\n return Array(times).fill('').map((val, index) => iterator(index)).join('');\n }\n\n static setCssVariable(css, name, value) {\n const cssString = String(css);\n if (cssString.indexOf(`${name}:`) > -1) {\n return cssString.replace(new RegExp(`${name}:.*?(;|})`), `${name}: ${value}$1`);\n } else {\n return `* {${name}: ${value};} ${cssString}`;\n }\n }\n\n static classes(...list) {\n let classNames = [];\n let className;\n while (className = list.shift()) {\n if (!!list.shift()) classNames.push(className);\n }\n return classNames.join(' ');\n }\n\n static range(start, stop, step) {\n if (stop === null) {\n stop = start || 0;\n start = 0;\n }\n if (!step) {\n step = stop < start ? -1 : 1;\n }\n\n const length = Math.max(Math.ceil((stop - start) / step), 0);\n const range = Array(length);\n\n for (let idx = 0; idx < length; idx++, start += step) {\n range[idx] = start;\n }\n\n return range;\n }\n\n static isBetween (num1, num2, num3) {\n if (num2 > num3) {\n return num1 <= num2 && num1 >= num3;\n } else {\n return num1 >= num2 && num1 <= num3;\n }\n }\n\n static makeDataUrl(data) {\n return `data:image/svg+xml,${encodeURIComponent(data)}`;\n }\n\n static xmlToObject(xmlString, options = {}) {\n const { arrays = [], exclude = [], booleans = [], numbers = [], preserveCapitals = false } = options;\n\n function formatName(name) {\n return preserveCapitals ? name : name.charAt(0).toLowerCase() + name.slice(1);\n }\n\n function convertType(name, value) {\n if (booleans.includes(name)) {\n return value.toLowerCase() === 'true';\n }\n if (numbers.includes(name)) {\n return Number(value);\n }\n return value;\n }\n\n const parser = new DOMParser();\n let dom;\n try {\n dom = parser.parseFromString(xmlString, 'application/xml');\n } catch (error) {\n console.log('ERROR: XmlToObject failed to parse XML');\n return null;\n }\n\n function parseNode(xmlNode, result) {\n let nodeName = xmlNode.nodeName;\n if (nodeName === \"#text\") {\n let v = xmlNode.nodeValue;\n if (v.trim()) result['#text'] = v;\n return;\n }\n\n if (exclude.indexOf(nodeName) !== -1) {\n return;\n }\n\n name = formatName(nodeName);\n\n let jsonNode = {},\n existing = result[name];\n if (existing) {\n if (!Array.isArray(existing)) result[name] = [existing, jsonNode];\n else result[name].push(jsonNode);\n }\n else {\n if (arrays.indexOf(nodeName) !== -1) result[name] = [jsonNode];\n else result[name] = jsonNode;\n }\n\n if (xmlNode.attributes) for (let attribute of xmlNode.attributes) jsonNode[formatName(attribute.nodeName)] = convertType(attribute.nodeName, attribute.nodeValue);\n\n for (let node of xmlNode.childNodes) parseNode(node, jsonNode);\n }\n\n let result = {};\n for (let node of dom.childNodes) parseNode(node, result);\n\n return result;\n }\n}\n","export default \"\\n:host {\\n display: block;\\n}\\n\\n.layout {\\n}\\n\\ntable {\\n --cell_size: 0;\\n font-size: calc(var(--cell_size) / 1.8);\\n padding: 0;\\n table-layout: fixed;\\n border-spacing: 0;\\n border-collapse: collapse;\\n user-select: none;\\n width: 100%;\\n}\\n\\ntd {\\n text-align: center;\\n padding: 0;\\n width: var(--cell_size);\\n height: var(--cell_size);\\n max-width: var(--cell_size);\\n max-height: var(--cell_size);\\n min-width: var(--cell_size);\\n min-height: var(--cell_size);\\n box-sizing: border-box;\\n}\\n\\ntd.cell {\\n color: black;\\n}\\n\\n.flashing {\\n transition: background-color 1000ms;\\n}\\n\\ntable:not(.flash-on) .flashing {\\n transition: background-color 300ms;\\n}\\n\\n\\n.heading {\\n font-size: calc(var(--cell_size) / 2.5);\\n}\\n\\ntr:first-child .heading:first-child {\\n border: none;\\n}\\n\\ntr:nth-child(2) .heading:first-child {\\n border-top: none;\\n}\\n\\ntr:first-child .heading:nth-child(2) {\\n border-left: none;\\n}\\n\\n\\n\"","import {Utils, PropTypes, FlashTimer, MyassaysComponent} from 'myassays-global';\nimport styles from './styles.css?raw';\n\nconst { repeat } = Utils;\n\nexport default class MyassaysLayoutViewer extends MyassaysComponent {\n static get propTypes() {\n return {\n id: PropTypes.string,\n class: PropTypes.string,\n style: PropTypes.string,\n onclick: PropTypes.string,\n dataConfig: PropTypes.json.default({}).observed,\n flashEnabled: PropTypes.bool.default(true).observed,\n flashWellPositions: PropTypes.string.regExp(/^\\d*( *, *\\d*)*$/).default('').observed,\n flashColour: PropTypes.string.default('black'),\n flashBackgroundColour: PropTypes.string.default('white'),\n }\n }\n constructor(options) {\n super();\n\n if (options) {\n this.options = options;\n this.id = options.id;\n }\n\n this.shadow = this.attachShadow({mode: \"open\"});\n this._hasBeenRendered = false;\n\n if (options) {\n const parentElem = options.parent;\n if (parentElem) {\n parentElem.appendChild(this);\n }\n }\n }\n\n static get observedAttributes() {\n return PropTypes.getObserved(this.propTypes);\n }\n\n static template = data => `\n \n\n
\n ${repeat(data.width, col => ` | ${col + 1} | `)}\n
${data.toRowCode(row + 1)} | ${repeat(data.width, col => {\n const wellIndex = row*data.totalColumns + col;\n const flashingClass = data.flashEnabled && data.flashWellPositions.includes(wellIndex + 1) ? `flashing` : '';\n \n const sampleType = data.default.shift();\n let number = data.default.shift();\n if (sampleType === '1' || number === '0') {\n number = ' ';\n }\n return `${number} | `;\n })}