\n
\n \n \n
\n
\n
\n
\n \n
\n ${state.pasteButtonPosition === pasteButtonPositions.RIGHT ? `
` : ''}\n
\n \n
\n
`;\n}\n\nexport default class MyassaysParamsGroupTable extends MyassaysComponent {\n static get uniqueId() {\n if (this._uniqueIdCounter === undefined) {\n this._uniqueIdCounter = 1;\n } else {\n this._uniqueIdCounter++;\n }\n return `--myassays-params-group-table-${this._uniqueIdCounter}-`;\n }\n\n static events = {\n VALUE_CHANGE: 'valuechange',\n VALUE_CHANGE_VALIDATED: 'valuechangevalidated',\n DATA_CHANGE: 'datachange',\n ROW_SELECT: 'rowselect',\n REPLICATES_CHANGE: 'replicateschange',\n ERROR: 'error',\n }\n\n static errorCodes = ParamsTableView.errorCodes;\n\n static get propTypes() {\n return {\n id: PropTypes.string,\n nameTitle: PropTypes.string.required\n .comment('Title of left hand column'),\n dataTitle: PropTypes.string.required\n .comment('Title of right hand column'),\n dataType: PropTypes.string.lookup([dataTypes.NUMERIC, dataTypes.STRING]).required\n .comment('Determines the data type of the right hand column only'),\n typeIcons: PropTypes.bool.default(false)\n .comment('When true, shows coloured circles before names'),\n typeColour: PropTypes.string.default('#ffffff')\n .comment('Colour of the icons'),\n groupsMode: PropTypes.string.lookup(groupsModes).default(groupsModes.NORMAL)\n .comment('\"ids\" is a special use case for editing group name ids'),\n numGroups: PropTypes.number.required.observed\n .comment('Number of rows in the table unless group-merge is true (see below)'),\n groupMerge: PropTypes.bool.default(false)\n .comment('Allows use of replicates. Number of rows in the table = num-groups / replicates'),\n groupMergeName: PropTypes.string.default('Replicates:')\n .comment('The label of the no. of replicates field'),\n seriesEditor: PropTypes.bool.default(false)\n .comment('If true, series editor is included'),\n seriesEditorState: PropTypes.bool.default(false)\n .comment('If true, series editor is open by default'),\n seriesDataBase: PropTypes.number.default(1).observed\n .comment('The start value of the series'),\n seriesDataFactor: PropTypes.number.min(0).default(1).observed\n .comment('The series factor'),\n seriesDataIncrement: PropTypes.number.default(1).observed\n .comment('The series increment (overrides the factor if non-zero)'),\n seriesRepeatEditor: PropTypes.bool.default(false),\n seriesRepeatEditorState: PropTypes.bool.default(false),\n seriesRepeatDefault: PropTypes.number.default(1),\n tipText: PropTypes.string.default(''),\n unitsEditor: PropTypes.bool.default(false),\n unitsValue: PropTypes.string.default(''),\n reset: PropTypes.bool.default(false),\n resetAction: PropTypes.string.default(''),\n decimalSeparator: PropTypes.string.lookup([DECIMAL_SEPARATOR_POINT, DECIMAL_SEPARATOR_COMMA]).default(DECIMAL_SEPARATOR_POINT),\n nameValue: PropTypes.string.default('')\n .comment('A comma separated list, if supplied overrides generated names'),\n nameBase: PropTypes.string.default(NULL_NAME_BASE)\n .comment('If not supplied, the name-title is used, adding a number suffix for each row. A null string can be used to give numbers only'),\n dataValue: PropTypes.string.default('')\n .comment('A comma separated list, if supplied overrides any generated series'),\n dataValidation: PropTypes.string.default(''),\n dataPaddingNumeric: PropTypes.number.default(1)\n .comment('Used when values cannot be derived, e.g. when num-groups is increased and there is no dynamic series'),\n dataPaddingString: PropTypes.string.default('(Unknown)')\n .comment('Used when values cannot be derived, e.g. when num-groups is increased and there is no dynamic series'),\n pasteButtonPosition: PropTypes.string.lookup(pasteButtonPositions).default(pasteButtonPositions.RIGHT)\n .comment('\"right\" puts the button below the controls on the right of the table.\\n\"bottom\" puts it below the data column'),\n }\n }\n constructor() {\n super();\n\n // NOTE: all props are copied to state, these are additional state only attributes.\n this._state = new Model({\n pasteAllowed: !!navigator.clipboard.readText,\n groupMergeReplicates: 1,\n replicatesOptions: [],\n });\n\n this._root = this;\n\n this._hasBeenRendered = false;\n }\n\n static get observedAttributes() {\n return PropTypes.getObserved(this.propTypes);\n }\n\n connectedCallback() {\n const state = PropTypes.attributesToProps(this);\n\n // if it's a dynamic table, and a data-value is provided, override the series-editor-state to false;\n if (state.seriesEditor && state.seriesEditorState && state.dataValue !== '') {\n state.seriesEditorState = false;\n }\n\n if (state.groupMerge && state.dataValue) {\n // analyse the data-value to see if we can infer a groupMergeReplicates value\n const values = state.dataValue.trim().split(/ *, */).map(item => state.dataType === dataTypes.NUMERIC ? Number(item) : item);\n const replicatesFound = [1];\n let replicatesFoundIndex = 0;\n\n let value = values.shift();\n while (values.length > 0) {\n if (values[0] === value) {\n replicatesFound[replicatesFoundIndex]++;\n } else {\n replicatesFoundIndex++;\n replicatesFound.push(1);\n }\n value = values.shift();\n }\n const minReplicates = Math.min(...replicatesFound);\n if (minReplicates > 1 && Math.max(...replicatesFound.map(item => item % minReplicates)) === 0) {\n state.groupMergeReplicates = minReplicates;\n }\n }\n\n this._state.set(state);\n\n this._root.innerHTML = template(MyassaysParamsGroupTable.uniqueId, this._state.attributes);\n this.renderParamsTable();\n this.renderSeriesEditor();\n this.addControlListeners();\n this._hasBeenRendered = true;\n\n this._state.addEventListener('change', evt => {\n this.onStateChange();\n });\n\n this._state.set({replicateOptions: this.getReplicateOptions(state.numGroups)});\n\n this._paramsTableView.items = this.getItems({\n initial: true,\n });\n }\n\n renderParamsTable() {\n const { typeColour } = this._state.attributes;\n this._paramsTableView = ParamsTableView.options\n .rootElement(this._root.querySelector('.params-table'))\n .columns(this.columns)\n .typeColour(typeColour)\n .showPasteButtons(this._state.get('pasteButtonPosition') === pasteButtonPositions.BOTTOM)\n .selectable(true)\n .createInstance();\n }\n\n renderSeriesEditor() {\n const state = this._state.attributes;\n const open = state.seriesEditorState;\n const increment = state.seriesDataIncrement;\n const factor = state.seriesDataFactor;\n const operation = (() => {\n switch (true) {\n case factor === 0 && increment >= 0:\n return operations.ADD;\n case factor >= 1 && (increment === 0 || increment === 1):\n return operations.MULTIPLY;\n case factor < 1 && (increment === 0 || increment === 1):\n return operations.DIVIDE;\n case increment >= 0:\n return operations.ADD;\n case increment < 0:\n return operations.SUBTRACT;\n }\n })();\n const operatorMode = (() => {\n switch (true) {\n case factor === 0 && increment === 0:\n return operationsModes.BOTH;\n case factor === 0:\n return operationsModes.ADD_SUBTRACT;\n case increment === 0:\n return operationsModes.MULTIPLY_DIVIDE;\n default:\n return operationsModes.BOTH;\n }\n })();\n const addSubtractOperand = (() => {\n switch (true) {\n case factor === 0 && increment >= 0:\n return increment;\n case factor >= 1 && (increment === 0 || increment === 1):\n return increment;\n case factor < 1 && (increment === 0 || increment === 1):\n return increment;\n default:\n return Math.abs(increment);\n }\n })();\n const multiplyDivideOperand = (() => {\n switch (true) {\n case factor === 0:\n return 1;\n default:\n return factor >= 1 ? factor : 1/factor;\n }\n })();\n this._seriesEditorView = new SeriesEditorView(this._root.querySelector('.series-editor-root'), open, operation, operatorMode, addSubtractOperand, multiplyDivideOperand);\n\n if (state.groupsMode !== groupsModes.IDS && state.seriesEditor) {\n this._root.querySelector(`.series-editor`).classList.add(`show`);\n }\n }\n\n addControlListeners() {\n const setState = attributes => this._state.set(attributes);\n const toggleState = attrName => {\n setState({[attrName]: !this._state.get(attrName)});\n }\n const addListener = (selector, type, listener) => {\n const element = this._root.querySelector(selector);\n if (element) {\n element.addEventListener(type, listener);\n }\n }\n\n this._seriesEditorView.addEventListener(SeriesEditorView.events.STATE_CHANGE, this.onSeriesEditorStateChange);\n\n this._paramsTableView.addEventListener(ParamsTableView.events.STATE_CHANGE, this.onParamsTableStateChange);\n this._paramsTableView.addEventListener(ParamsTableView.events.ERROR, this.onParamsTableError);\n\n addListener('[name=\"units\"]', 'change', this.onUnitsChange);\n addListener('[name=\"replicates\"]', 'change', this.onReplicatesChange);\n addListener('[name=\"open-series-repeat-editor\"]', 'change', evt => {\n Utils.doAndResizeSmoothly(this._root.querySelector('.series-repeat-editor'), () => {\n setState({seriesRepeatEditorState: evt.target.checked});\n });\n });\n addListener('[name=\"paste\"]', 'focus', this.onPasteFocus);\n addListener('[name=\"paste\"]', 'click', this.onPasteClick);\n addListener('[name=\"reset\"]', 'click', this.onResetClick);\n }\n\n onUnitsChange = evt => {\n\n }\n\n onReplicatesChange = evt => {\n Utils.doAndResizeSmoothly(this._root.querySelector('.params-table'), () => {\n this._state.set({groupMergeReplicates: Number(evt.target.value)});\n });\n }\n\n onSeriesEditorStateChange = evt => {\n const model = evt.detail;\n model.hasChanged('open', newValue => {\n this._state.set({seriesEditorState: newValue});\n newValue && this.updateItems();\n });\n model.hasChanged('addSubtractOperand', newValue => {\n this.updateItems();\n });\n model.hasChanged('multiplyDivideOperand', newValue => {\n this.updateItems();\n });\n model.hasChanged('operation', newValue => {\n this.updateItems();\n });\n }\n\n onParamsTableStateChange = evt => {\n const { seriesEditor, seriesEditorState, seriesDataBase } = this._state.attributes;\n const model = evt.detail;\n model.hasChanged('items', (newValue, previousValue) => {\n delete this._oldFixedData;\n if (seriesEditor && seriesEditorState) {\n if (seriesDataBase !== newValue[0].data) {\n this._state.set({seriesDataBase: newValue[0].data});\n // this triggers a re-calc of the series\n return;\n }\n }\n if (previousValue && previousValue.length > 0) {\n this.dispatchValueChangeEvent();\n } else {\n setTimeout(() => this.dispatchValueChangeEvent(), 1);\n }\n const previousData = (previousValue ? previousValue.map(item => item.data).join(',') : '');\n const newData = newValue.map(item => item.data).join(',');\n if (newData !== previousData) {\n if (previousData === '') {\n setTimeout(() => {\n this.dispatchDataChangeEvent();\n if (this._state.get('groupMergeReplicates') > 1) {\n this.dispatchReplicatesChangeEvent();\n }\n }, 1);\n } else {\n this.dispatchDataChangeEvent();\n }\n }\n });\n model.hasChanged('selectedRowIndex', (newValue) => {\n this.dispatchRowSelectEvent(newValue);\n });\n }\n\n onParamsTableError = evt => {\n evt.preventDefault();\n const error = evt.detail;\n const event = new BlazorCompatibleEvent(MyassaysParamsGroupTable.events.ERROR, {\n detail: error,\n cancelable: true,\n });\n this.dispatchEvent(event) && alert(error.message);\n }\n\n onStateChange(force) {\n // if no event is passed, assume all has changed\n const hasChanged = (name, handler) => {\n force ? handler(this._state.get(name)) : this._state.hasChanged(name, handler);\n }\n const toggleClass= (selector, className, force) => {\n this._root.querySelector(selector).classList.toggle(className, force);\n }\n hasChanged('seriesEditorState', newValue => {\n this._paramsTableView.columns = this.columns;\n if (newValue) {\n const oldFixedData = this._paramsTableView.items.map(item => item.data);\n setTimeout(() => {\n this._oldFixedData = oldFixedData;\n }, 1);\n } else {\n if (this._oldFixedData) {\n this._paramsTableView.items = this._paramsTableView.items.map((item, i) => ({...item, data: this._oldFixedData[i]}))\n }\n }\n });\n hasChanged('seriesDataBase', newValue => {\n this._paramsTableView.items = this.getItems();\n });\n hasChanged('seriesRepeatEditorState', newValue => {\n this.updateItems();\n });\n hasChanged('numGroups', (newValue, previousValue) => {\n const items = this._paramsTableView.items;\n const { seriesEditor, seriesEditorState, groupsMode, groupMerge, groupMergeReplicates, dataType, dataValue, dataPaddingNumeric, dataPaddingString } = this._state.attributes;\n let newItems = [];\n const stateOverride = {groupMergeReplicates};\n if (groupMerge && (newValue % groupMergeReplicates !== 0)) {\n stateOverride.groupMergeReplicates = 1;\n }\n if (previousValue !== undefined) {\n if (newValue < previousValue && groupMergeReplicates === stateOverride.groupMergeReplicates) {\n newItems = items.slice(0, newValue/groupMergeReplicates);\n } else if (seriesEditor && seriesEditorState) {\n newItems = this.getItems({stateOverride});\n } else {\n const dataValues = (dataValue === '') ? [] : dataValue.trim().split(/ *, */).map(item => dataType === dataTypes.NUMERIC ? Number(item) : item);\n\n newItems = this.getItems({stateOverride}).map((item, i) => {\n if (items[i] !== undefined) {\n item.data = items[i].data;\n } else {\n if (groupsMode !== groupsModes.IDS) {\n item.data = dataValues[i * stateOverride.groupMergeReplicates] !== undefined ? dataValues[i * stateOverride.groupMergeReplicates] : (dataType === dataTypes.NUMERIC ? dataPaddingNumeric : dataPaddingString);\n }\n }\n return item;\n });\n }\n this._paramsTableView.items = newItems;\n this._state.set({...stateOverride, replicateOptions: this.getReplicateOptions(newValue)});\n }\n });\n hasChanged('replicateOptions', newValue => {\n const { groupMergeReplicates } = this._state.attributes;\n this._root.querySelector('[name=\"replicates\"]').innerHTML = newValue.map(item => `