\n Total expenditures: \n ${formatCurrency(supp.total())}\n \n
`;\n },\n linkEditBtns() {\n let btns = document.querySelectorAll('.edit-supp');\n btns.forEach(btn => {\n btn.addEventListener('click', function () {\n visitPage('new-inits');\n });\n });\n },\n update() {\n const suppDiv = document.querySelector('#supp-stats');\n suppDiv.innerHTML = this.html();\n this.linkEditBtns();\n }\n};\n/* harmony default export */ const supp_section = (SuppSection);\n;// CONCATENATED MODULE: ./src/js/components/sidebar/sidebar.js\n\n\n\n\n// fetch CSS variables saved in :root\nconst root = document.documentElement;\nconst sideBarWidth = getComputedStyle(root).getPropertyValue('--sidebar-width').trim();\nfunction hideSidebar() {\n document.getElementById('sidebar-panel').style.display = 'none';\n document.getElementById('main-panel').style.width = '100%';\n document.querySelector('header').style.width = '100%';\n}\nfunction showSidebar() {\n const sidebar = document.getElementById('sidebar-panel');\n const mainPanel = document.getElementById('main-panel');\n const header = document.querySelector('header');\n\n // update values\n updateTotals();\n sidebar.style.display = 'block'; // Show the sidebar\n\n // Calculate the remaining width for the main panel and header\n var contentWidth = document.documentElement.clientWidth;\n mainPanel.style.width = `${contentWidth - parseInt(sideBarWidth, 10)}px`;\n header.style.width = `${contentWidth - parseInt(sideBarWidth, 10)}px`;\n\n // add event listener to resize content if window is adjusted\n window.addEventListener('resize', showSidebar);\n}\nfunction updateSidebarTitle(new_title) {\n document.getElementById('sidebar-title').textContent = new_title;\n}\nfunction updateTotals() {\n supp_section.update();\n BaselineSection.update();\n}\nfunction resetAll() {\n localStorage.clear();\n // reset all stats to 0\n updateTotals();\n}\nconst Sidebar = {\n SuppSection: supp_section,\n BaselineSection: BaselineSection,\n hide: hideSidebar,\n show: showSidebar,\n updateTitle: updateSidebarTitle,\n updateTotals: updateTotals,\n reset: resetAll\n};\n/* harmony default export */ const components_sidebar_sidebar = (Sidebar);\n// EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js!./src/js/components/nav_buttons/nav_buttons.css\nvar nav_buttons = __webpack_require__(595);\n;// CONCATENATED MODULE: ./src/js/components/nav_buttons/nav_buttons.css\n\n \n \n \n \n \n \n \n \n \n\nvar nav_buttons_options = {};\n\nnav_buttons_options.styleTagTransform = (styleTagTransform_default());\nnav_buttons_options.setAttributes = (setAttributesWithoutAttributes_default());\nnav_buttons_options.insert = insertBySelector_default().bind(null, \"head\");\nnav_buttons_options.domAPI = (styleDomAPI_default());\nnav_buttons_options.insertStyleElement = (insertStyleElement_default());\n\nvar nav_buttons_update = injectStylesIntoStyleTag_default()(nav_buttons/* default */.A, nav_buttons_options);\n\n\n\n\n /* harmony default export */ const nav_buttons_nav_buttons = (nav_buttons/* default */.A && nav_buttons/* default */.A.locals ? nav_buttons/* default */.A.locals : undefined);\n\n;// CONCATENATED MODULE: ./src/js/components/nav_buttons/nav_buttons.js\n// nav_buttons.js\n\n\nfunction initializeNavButtons() {\n // initialize last button\n const last_btn = document.getElementById('btn-last');\n last_btn.addEventListener('click', lastPage);\n // initialize next button\n const next_btn = document.getElementById('btn-next');\n next_btn.addEventListener('click', nextPage);\n}\nfunction nav_buttons_disable(button_id) {\n const button = document.getElementById(button_id);\n button.classList.add('disabled');\n button.disabled = true; // Also disable it at the DOM level\n}\nfunction nav_buttons_enable(button_id) {\n const button = document.getElementById(button_id);\n button.classList.remove('disabled');\n button.disabled = false; // Enable it at the DOM level\n}\nconst Next = {\n disable: function () {\n nav_buttons_disable('btn-next');\n },\n enable: function () {\n nav_buttons_enable('btn-next');\n },\n addAction: function (fn) {\n document.querySelector(`#btn-next`).addEventListener('click', fn);\n },\n removeAction: function (fn) {\n document.querySelector(`#btn-next`).removeEventListener('click', fn);\n }\n};\nconst Last = {\n disable: function () {\n nav_buttons_disable('btn-last');\n },\n enable: function () {\n nav_buttons_enable('btn-last');\n }\n};\nconst NavButtons = {\n hide: function () {\n document.getElementById('nav-btns').style.display = 'none';\n },\n show: function () {\n document.getElementById('nav-btns').style.display = 'block';\n initializeNavButtons();\n },\n Next: Next,\n Last: Last\n};\n/* harmony default export */ const components_nav_buttons_nav_buttons = (NavButtons);\n// EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js!./src/js/components/body/body.css\nvar body = __webpack_require__(9);\n;// CONCATENATED MODULE: ./src/js/components/body/body.css\n\n \n \n \n \n \n \n \n \n \n\nvar body_options = {};\n\nbody_options.styleTagTransform = (styleTagTransform_default());\nbody_options.setAttributes = (setAttributesWithoutAttributes_default());\nbody_options.insert = insertBySelector_default().bind(null, \"head\");\nbody_options.domAPI = (styleDomAPI_default());\nbody_options.insertStyleElement = (insertStyleElement_default());\n\nvar body_update = injectStylesIntoStyleTag_default()(body/* default */.A, body_options);\n\n\n\n\n /* harmony default export */ const body_body = (body/* default */.A && body/* default */.A.locals ? body/* default */.A.locals : undefined);\n\n// EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js!./src/js/components/accordion/accordion.css\nvar accordion = __webpack_require__(995);\n;// CONCATENATED MODULE: ./src/js/components/accordion/accordion.css\n\n \n \n \n \n \n \n \n \n \n\nvar accordion_options = {};\n\naccordion_options.styleTagTransform = (styleTagTransform_default());\naccordion_options.setAttributes = (setAttributesWithoutAttributes_default());\naccordion_options.insert = insertBySelector_default().bind(null, \"head\");\naccordion_options.domAPI = (styleDomAPI_default());\naccordion_options.insertStyleElement = (insertStyleElement_default());\n\nvar accordion_update = injectStylesIntoStyleTag_default()(accordion/* default */.A, accordion_options);\n\n\n\n\n /* harmony default export */ const accordion_accordion = (accordion/* default */.A && accordion/* default */.A.locals ? accordion/* default */.A.locals : undefined);\n\n// EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js!./src/js/components/table/table.css\nvar table = __webpack_require__(279);\n;// CONCATENATED MODULE: ./src/js/components/table/table.css\n\n \n \n \n \n \n \n \n \n \n\nvar table_options = {};\n\ntable_options.styleTagTransform = (styleTagTransform_default());\ntable_options.setAttributes = (setAttributesWithoutAttributes_default());\ntable_options.insert = insertBySelector_default().bind(null, \"head\");\ntable_options.domAPI = (styleDomAPI_default());\ntable_options.insertStyleElement = (insertStyleElement_default());\n\nvar table_update = injectStylesIntoStyleTag_default()(table/* default */.A, table_options);\n\n\n\n\n /* harmony default export */ const table_table = (table/* default */.A && table/* default */.A.locals ? table/* default */.A.locals : undefined);\n\n;// CONCATENATED MODULE: ./src/js/components/table/subcomponents/headers.js\nfunction addTableHeaders(cols) {\n // Get the table element by its ID\n const table = document.getElementById('main-table');\n\n // Create a table header row element\n const headerRow = document.createElement('tr');\n cols.forEach(col => {\n // Create a header cell element\n const headerCell = document.createElement('th');\n headerCell.textContent = col['title'];\n headerCell.classList.add(col['className']);\n\n // Append the header cell to the header row\n headerRow.appendChild(headerCell);\n });\n\n // Append the header row to the table header\n let thead = table.querySelector('thead');\n thead.appendChild(headerRow);\n}\nconst Header = {\n add: function (header_array) {\n addTableHeaders(header_array);\n }\n};\n/* harmony default export */ const headers = (Header);\n;// CONCATENATED MODULE: ./src/js/components/table/subcomponents/rows.js\n\n\nasync function addNewRow(data_dictionary, columns = []) {\n // Get the table element by its ID\n const table = document.getElementById('main-table');\n\n // check if header has already been added\n let header_row = table.querySelector('thead tr');\n if (!header_row) {\n headers.add(columns);\n header_row = table.querySelector('thead tr');\n }\n\n // initialize new row of data\n const new_row = document.createElement('tr');\n\n // go through each header and add the right cell value depending on its class\n let thElements = header_row.querySelectorAll('th');\n thElements.forEach(header_cell => {\n // Create new cell and add it to the row\n const newCell = document.createElement('td');\n new_row.appendChild(newCell);\n // if the data has an appropriate class, add the info to the cell. \n // Otherwise, keep empty cell\n Object.keys(data_dictionary).forEach(className => {\n if (header_cell.classList.contains(className)) {\n newCell.textContent = data_dictionary[className];\n newCell.classList.add(className);\n }\n });\n });\n\n // Append the new row to the table body\n let tbody = table.querySelector('tbody');\n tbody.appendChild(new_row);\n}\nfunction saveRowEdits(row) {\n var cells = row.querySelectorAll('td');\n cells.forEach(cell => {\n // save dropdown values\n if (cell.querySelector('select')) {\n var serviceSelector = cell.querySelector('select');\n cell.textContent = serviceSelector.value;\n } else if (cell.querySelector('input')) {\n // save new entered value in textbox\n var textbox = cell.querySelector('input');\n var enteredValue = textbox.value;\n // update display and format with currency if relevant\n if (cell.classList.contains('cost')) {\n // if cost, remove commas first\n enteredValue = enteredValue.replaceAll(',', '');\n cell.textContent = formatCurrency(enteredValue);\n // set value attribute to the new user input\n cell.setAttribute('value', enteredValue);\n } else {\n cell.textContent = enteredValue;\n }\n } else if (cell.querySelector('textarea')) {\n // save new entered value in textbox\n var enteredValue = cell.querySelector('textarea').value;\n cell.textContent = enteredValue;\n }\n });\n}\nconst Rows = {\n add: function (data_dictionary, cols) {\n addNewRow(data_dictionary, cols);\n },\n saveEdits: function (row) {\n saveRowEdits(row);\n }\n};\n/* harmony default export */ const rows = (Rows);\n;// CONCATENATED MODULE: ./src/js/components/table/subcomponents/buttons.js\n\nfunction hideButton(className) {\n return function () {\n var buttons = document.getElementsByClassName(className);\n for (var i = 0; i < buttons.length; i++) {\n buttons[i].style.display = 'none';\n }\n };\n}\nfunction showButton(className) {\n return function () {\n var buttons = document.getElementsByClassName(className);\n for (var i = 0; i < buttons.length; i++) {\n buttons[i].style.display = 'inline';\n }\n };\n}\nfunction updateButtonText(className, text) {\n document.querySelector(`.${className}`).textContent = text;\n}\nfunction handleRowEdit(actionOnClick, updateCallback = null) {\n // attach an event listener to each edit button in every row\n var editButtons = document.getElementsByClassName('btn-edit');\n for (var i = 0; i < editButtons.length; i++) {\n editButtons[i].addEventListener('click', async function (event) {\n // Determine what was clicked on within the table\n var rowToEdit = event.target.closest('tr');\n // mark row as being edited\n rowToEdit.classList.add('active-editing');\n\n // turn relevant entries into textboxes, usually\n actionOnClick();\n\n // hide edit buttons\n Edit.hide();\n if (updateCallback) {\n initializeConfirmButton(updateCallback);\n }\n });\n }\n ;\n}\nfunction initializeConfirmButton(updateCallback) {\n // get element and add listener for click\n var rowToEdit = document.querySelector('.active-editing');\n const confirm_btn = rowToEdit.querySelector(\".btn-confirm\");\n // show the row's confirm button\n confirm_btn.style.display = 'block';\n confirm_btn.addEventListener('click', function () {\n ;\n // save row edits\n rows.saveEdits(rowToEdit);\n // update values in sidebar\n updateCallback();\n // make row no longer green\n rowToEdit.classList.remove('active-editing');\n // show edit buttons and hide confirm buttons\n Edit.show();\n Confirm.hide();\n });\n}\nconst Edit = {\n html: '',\n hide: hideButton('btn-edit'),\n show: showButton('btn-edit'),\n init: function (actionOnClick, updateCallback) {\n handleRowEdit(actionOnClick, updateCallback);\n }\n};\nconst Delete = {\n html: '',\n hide: hideButton('btn-delete'),\n show: showButton('btn-delete')\n};\nconst Confirm = {\n html: '',\n hide: hideButton('btn-confirm'),\n show: showButton('btn-confirm')\n};\nconst AddRow = {\n hide: hideButton('btn-add'),\n show: showButton('btn-add'),\n updateText: function (text) {\n updateButtonText('btn-add', text);\n }\n};\nconst buttons_Buttons = {\n Delete: Delete,\n Edit: Edit,\n Confirm: Confirm,\n AddRow: AddRow,\n edit_confirm_btns: Edit.html + Confirm.html,\n all_btns: Delete.html + Edit.html + Confirm.html\n};\n/* harmony default export */ const subcomponents_buttons = (buttons_Buttons);\n;// CONCATENATED MODULE: ./src/js/components/form/subcomponents/dropdown.js\nfunction createDropdown(dataArray) {\n // Creating a select element\n const selectElement = document.createElement('select');\n\n // add a default blank option to the dataArray\n dataArray = [''].concat(dataArray);\n\n // Looping through the array and creating an option for each element\n dataArray.forEach(item => {\n const optionElement = document.createElement('option');\n optionElement.value = item;\n optionElement.textContent = item;\n selectElement.appendChild(optionElement); // Appending the option to the select\n });\n\n // Return the select element so it can be appended to the document\n return selectElement;\n}\nconst Dropdown = {\n create: function (dataArray) {\n return createDropdown(dataArray);\n }\n};\n/* harmony default export */ const subcomponents_dropdown = (Dropdown);\n;// CONCATENATED MODULE: ./src/js/components/table/subcomponents/cells.js\n\n\n\n\n// return cell value attribute or 0 if it does not exist\nfunction getCellValue(row, className) {\n var cell = row.querySelector(`.${className}`);\n var cellValue = cell ? cell.getAttribute('value') : null;\n return cellValue ? parseFloat(cellValue) : 0;\n}\n\n// return text in cell\nfunction getCellText(row, className) {\n var cell = row.querySelector(`.${className}`);\n if (cell) {\n return cell.textContent;\n } else {\n //console.log(`Error retrieving cell text for class ${className}`);\n return '';\n }\n}\nfunction updateTableCell(row, col_class, new_value) {\n const cell = row.querySelector(`.${col_class}`);\n cell.setAttribute('value', new_value);\n cell.textContent = formatCurrency(new_value);\n}\nfunction createEditableCell(cellClass, isCost, type = 'input') {\n // get cell\n const cell = document.querySelector(`.active-editing td.${cellClass}`);\n // Create an input element to edit the value\n var textbox = document.createElement(type);\n if (type == 'input') {\n textbox.type = 'text';\n }\n ;\n if (isCost) {\n var value = cell.getAttribute('value');\n textbox.value = displayWithCommas(value);\n } else {\n textbox.value = cell.textContent;\n }\n // Clear the current content and append the textbox to the cell\n cell.innerHTML = '';\n cell.appendChild(textbox);\n}\nfunction cells_createDropdown(cellClass, optionArray) {\n // get cell\n const cell = document.querySelector(`.active-editing td.${cellClass}`);\n // add service dropdown\n const dropdown = subcomponents_dropdown.create(optionArray);\n dropdown.value = cell.textContent;\n // Clear the current content and append the textbox to the cell\n cell.innerHTML = '';\n cell.appendChild(dropdown);\n}\nconst Cell = {\n getValue: function (row, className) {\n return getCellValue(row, className);\n },\n getText: function (row, className) {\n return getCellText(row, className);\n },\n updateValue: function (row, col_class, new_value) {\n updateTableCell(row, col_class, new_value);\n },\n createTextbox: function (className, isCost, type) {\n createEditableCell(className, isCost, type);\n },\n createServiceDropdown: () => {\n cells_createDropdown('service', services.list());\n },\n createDropdown: cells_createDropdown\n};\n/* harmony default export */ const cells = (Cell);\n;// CONCATENATED MODULE: ./src/js/components/table/subcomponents/columns.js\n\n\n// position is index at which new column will be inserted\nfunction addCol(position, htmlContent = '', headerTitle = '') {\n // Get the table element by its ID\n let table = document.getElementById('main-table');\n\n // Validate position\n let maxPosition = table.rows[0].cells.length;\n if (position < 0 || position > maxPosition) {\n console.error(`Position ${position} is out of bounds.`);\n return;\n }\n\n // Insert the header if provided\n let thead = table.tHead;\n if (headerTitle && thead) {\n let th = document.createElement('th');\n th.innerHTML = headerTitle; // Use innerHTML to insert HTML content\n thead.rows[0].insertBefore(th, thead.rows[0].cells[position]);\n }\n\n // Insert new cells into each row of the table body\n let tbody = table.tBodies[0];\n if (tbody) {\n for (let i = 0; i < tbody.rows.length; i++) {\n let row = tbody.rows[i];\n let td = document.createElement('td');\n td.innerHTML = htmlContent; // Use innerHTML to insert HTML content\n row.insertBefore(td, row.cells[position]);\n }\n }\n}\nfunction ncols() {\n const table = document.getElementById('main-table');\n // Ensure that the row exists before counting the columns\n return table.rows[0].cells.length;\n}\nfunction addColToEnd(htmlContents = [], headerTitle = '') {\n // count columns and add new column to the end\n const position = ncols('main-table');\n addCol(position, htmlContents, headerTitle);\n}\nfunction assignClassToColumn(headerName, className) {\n // Get the table element by its ID\n let table = document.getElementById('main-table');\n\n // Find the index of the column by its header name\n const thead = table.tHead;\n let headerCellIndex = -1;\n const headerCells = thead.rows[0].cells; // Assuming the first row contains header cells (
)\n for (let i = 0; i < headerCells.length; i++) {\n if (headerCells[i].textContent.trim() === headerName) {\n // assign the class to the header cell\n headerCells[i].classList.add(className);\n headerCellIndex = i;\n break;\n }\n }\n\n // error check\n if (headerCellIndex === -1) {\n console.error(`No header found with name \"${headerName}\"`);\n return;\n }\n\n // Assign the class to each cell in the specified column index within the tbody\n let tbody = table.tBodies[0];\n if (tbody) {\n let bodyRows = tbody.rows;\n for (let row of bodyRows) {\n if (row.cells[headerCellIndex]) {\n row.cells[headerCellIndex].classList.add(className);\n }\n }\n }\n}\nfunction addCostClass(headerName) {\n assignClassToColumn(headerName, 'cost');\n\n // Get all the cells with the specified class name\n const cells = document.querySelectorAll(`td.cost`);\n cells.forEach(cell => {\n // Get the current text content of the cell and assign it to 'value' attribute\n if (!cell.getAttribute('value')) {\n const cellText = cell.textContent.trim();\n const cellValue = isNaN(cellText) || cellText === '' ? 0 : parseFloat(cellText);\n cell.setAttribute('value', cellValue);\n\n // Now format the text content like currency and replace it in the cell\n const formattedCurrency = formatCurrency(parseFloat(cellValue));\n cell.textContent = formattedCurrency;\n }\n });\n}\nfunction assignColumnClasses(columnDefinitions) {\n columnDefinitions.forEach(column => {\n // Assign class to column\n assignClassToColumn(column.title, column.className);\n\n // If the column is a cost column, add the specific cost class\n if (column.isCost) {\n addCostClass(column.title);\n }\n\n // show the column\n if (!column.hide) {\n showColumnByTitle(column.title);\n }\n });\n}\nfunction hideColumn(index) {\n var table = document.getElementById('main-table');\n var rows = table.rows;\n for (var i = 0; i < rows.length; i++) {\n var cells = rows[i].cells;\n if (cells.length > index) {\n cells[index].style.display = 'none';\n }\n }\n}\n\n// Function to show a specific column\nfunction showColumn(index) {\n var table = document.getElementById('main-table');\n var rows = table.rows;\n for (var i = 0; i < rows.length; i++) {\n var cells = rows[i].cells;\n if (cells.length > index) {\n cells[index].style.display = 'table-cell';\n }\n }\n}\n\n// Function to get the column index by title\nfunction getColumnIndexByTitle(table, title) {\n var headers = table.querySelectorAll('th');\n for (var i = 0; i < headers.length; i++) {\n if (headers[i].innerText === title) {\n return i;\n }\n }\n return -1; // Column not found\n}\n\n// Function to hide a column by title\nfunction hideColumnByTitle(title) {\n var table = document.getElementById('main-table');\n var index = getColumnIndexByTitle(table, title);\n if (index !== -1) {\n hideColumn(index);\n } else {\n console.log(`Column with title \"${title}\" not found.`);\n }\n}\n\n// Function to show a column by title\nfunction showColumnByTitle(title) {\n var table = document.getElementById('main-table');\n var index = getColumnIndexByTitle(table, title);\n if (index !== -1) {\n showColumn(index);\n } else {\n console.log(`Column with title \"${title}\" not found.`);\n }\n}\nconst Column = {\n add: function (position, htmlContent, headerTitle) {\n return addCol(position, htmlContent, headerTitle);\n },\n addAtEnd: function (htmlContent, headerTitle) {\n return addColToEnd(htmlContent, headerTitle);\n },\n assignClasses: function (column_definitions) {\n return assignColumnClasses(column_definitions);\n },\n hide: function (colName) {\n hideColumnByTitle(colName);\n },\n show: function (colName) {\n showColumnByTitle(colName);\n }\n};\n/* harmony default export */ const columns = (Column);\n;// CONCATENATED MODULE: ./src/js/components/table/subcomponents/data.js\n\n\n\n\nfunction fillTable(data) {\n try {\n const table = document.getElementById('main-table');\n const thead = table.querySelector('thead');\n const tbody = table.querySelector('tbody');\n\n // clear existing data\n thead.innerHTML = '';\n tbody.innerHTML = '';\n\n // Create table header row\n const headerRow = document.createElement('tr');\n Object.keys(data[0]).forEach(key => {\n const header = document.createElement('th');\n header.textContent = key;\n headerRow.appendChild(header);\n });\n thead.appendChild(headerRow);\n\n // Create table body rows\n data.forEach(item => {\n const row = document.createElement('tr');\n Object.values(item).forEach(val => {\n const cell = document.createElement('td');\n cell.innerHTML = val;\n row.appendChild(cell);\n });\n tbody.appendChild(row);\n });\n } catch (error) {\n console.error('No table saved in localStorage:', error);\n }\n}\nasync function loadFromStorage() {\n // look up table name in storage\n if (current_fund.number()) {\n var key = `${current_page.load()}_${current_fund.number()}`;\n } else {\n var key = current_page.load();\n }\n // load from local storage\n const data = localStorage.getItem(key);\n // if nothing in storage, return a zero\n if (!data) {\n return 0;\n } else {\n // otherwise, fill table in HTML and return success (1)\n components_table_table.show();\n fillTable(await JSON.parse(data));\n return 1;\n }\n}\nfunction loadFunds() {\n // get list of funds from storage\n const fundDict = fund_lookup_table.retrieve();\n // build out data in correct format\n const ret = [];\n Object.keys(fundDict).forEach(key => {\n // determine if the fund has already been edited\n if (fundDict[key]['viewed']) {\n // todo: add a checkmark here\n ret.push({\n 'Fund': ` \n \n ${fundDict[key]['name']}\n `\n });\n } else {\n ret.push({\n 'Fund': ` \n ${fundDict[key]['name']}\n `\n });\n }\n });\n fillTable(ret);\n}\nfunction getColumnIndexByClass(tbody, className) {\n const firstRow = tbody.rows[0];\n if (!firstRow) return -1; // Return -1 if there's no row to examine\n for (let cellIndex = 0; cellIndex < firstRow.cells.length; cellIndex++) {\n if (firstRow.cells[cellIndex].classList.contains(className)) {\n return cellIndex;\n }\n }\n return -1; // Return -1 if class name not found\n}\n\n// function sort(primaryClass, secondaryClass) {\n// const table = document.getElementById('main-table');\n// const tbody = table.tBodies[0];\n// const rows = Array.from(tbody.rows);\n\n// // Get the column indices by class name\n// const primaryColIndex = getColumnIndexByClass(tbody, primaryClass);\n// const secondaryColIndex = getColumnIndexByClass(tbody, secondaryClass);\n\n// // exit if classes don't exist\n// if (primaryColIndex === -1 ) {\n// console.error(`Column class ${classA} not found in table`);\n// return; \n// } else if (secondaryColIndex === -1) {\n// console.error(`Column class ${classB} not found in table`);\n// return; \n// }\n\n// // Sort the rows based on the text content of the cells\n// rows.sort((rowA, rowB) => {\n// // Primary column comparison\n// const primaryA = unformatCurrency(rowA.cells[primaryColIndex].textContent);\n// const primaryB = unformatCurrency(rowB.cells[primaryColIndex].textContent);\n\n// if (primaryA < primaryB) return -1;\n// if (primaryA > primaryB) return 1;\n\n// // Secondary column comparison (if primary is equal)\n// const secondaryA = unformatCurrency(rowA.cells[secondaryColIndex].textContent);\n// const secondaryB = unformatCurrency(rowB.cells[secondaryColIndex].textContent);\n\n// if (secondaryA < secondaryB) return -1;\n// if (secondaryA > secondaryB) return 1;\n\n// // If both columns are equal\n// return 0;\n// });\n\n// // Reattach sorted rows to the table body\n// rows.forEach((row) => tbody.appendChild(row));\n// }\n\nconst Data = {\n load: loadFromStorage,\n loadFunds: loadFunds,\n sort: function (colA, colB) {\n sort(colA, colB);\n }\n};\n/* harmony default export */ const data = (Data);\n;// CONCATENATED MODULE: ./src/js/components/table/subcomponents/filters.js\n// Helper functions & constants\n\n// object to hold all current filter statuses\nconst filterSettings = {\n 'approp-name': '',\n 'cc-name': '',\n 'object-name': '',\n 'object-category': ''\n};\n\n// helper function to filter data based on all filters\nfunction filterData() {\n // Get all rows in the table\n const rows = document.querySelectorAll('#main-table tbody tr');\n\n // Iterate through each row and determine if it should be hidden or shown\n rows.forEach(row => {\n let isVisible = true;\n\n // Check each filter setting against the row's cells\n for (const [filterId, filterValue] of Object.entries(filterSettings)) {\n const cell = row.querySelector(`.${filterId}`);\n\n // only show row if values pass through all filters \n if (filterValue && cell && cell.textContent.trim() !== filterValue) {\n isVisible = false;\n break;\n }\n }\n\n // Show or hide the row based on visibility\n row.classList.toggle('hidden', !isVisible);\n });\n}\nconst Filter = {\n html(filterLabel, filterClass) {\n // basic html with only 'All' option\n return `\n `;\n },\n addOption(filterClass, option) {\n // Add another option to the dropdown for the filter\n const filterObj = document.querySelector(`#filter-${filterClass}`);\n const optionObj = document.createElement('option');\n optionObj.value = option;\n optionObj.textContent = option;\n filterObj.appendChild(optionObj);\n },\n add(filterLabel, filterClass) {\n // create a div to contain the html and insert inside filter-container\n const filterContainer = document.querySelector('#filter-container');\n const filterDiv = document.createElement('div');\n filterDiv.innerHTML = this.html(filterLabel, filterClass);\n filterContainer.appendChild(filterDiv);\n // add all relevant options from that column in the table\n this.addAllOptions(filterClass);\n // Bind change event to the select element\n filterDiv.querySelector('.filter-dropdown').addEventListener('change', event => {\n // Update filter settings\n filterSettings[filterClass] = event.target.value;\n // Apply all filters\n filterData();\n });\n },\n addAllOptions(filterClass) {\n // get matching column from table\n const column = document.querySelectorAll(`#main-table td.${filterClass}`);\n\n // Use a Set to store unique values in the column of interest\n const uniqueValues = new Set();\n\n // Iterate over the NodeList to get the unique values\n column.forEach(td => {\n // Add each textContent to the Set\n uniqueValues.add(td.textContent.trim());\n });\n // add all values as options to the filter dropdown\n uniqueValues.forEach(option => {\n this.addOption(filterClass, option);\n });\n },\n deleteAll() {\n document.querySelector('#filter-container').innerHTML = '';\n },\n updateOptions(filterClass) {\n const filterObj = document.querySelector(`#filter-${filterClass}`);\n if (filterObj) {\n // Clear all existing options except for the default 'All' option\n filterObj.options.length = 1;\n // Add new options\n this.addAllOptions(filterClass);\n }\n }\n};\n/* harmony default export */ const filters = (Filter);\n;// CONCATENATED MODULE: ./src/js/models/gold_book.js\n// the Gold Book contains a lookup table for each job code, which maps to the BU, the job title, and the fringe rate\n\nconst GoldBook = {\n init(sheet) {\n // Convert sheet to JSON, with no automatic header row parsing\n const rawData = XLSX.utils.sheet_to_json(sheet, {\n header: 1,\n defval: ''\n });\n\n // Assuming header is the second row (index 1), and data starts at the third row (index 2)\n const headers = rawData[1];\n const data = rawData.slice(2);\n\n // Store headers and data in localStorage\n localStorage.setItem('goldbook_headers', JSON.stringify(headers));\n localStorage.setItem('goldbook_data', JSON.stringify(data));\n },\n fetch() {\n const headers = JSON.parse(localStorage.getItem('goldbook_headers'));\n const data = JSON.parse(localStorage.getItem('goldbook_data'));\n return {\n headers,\n data\n };\n },\n fetchByCode(job_code) {\n const {\n headers,\n data\n } = this.fetch();\n\n // Create a map of column names to their respective indices\n const headerMap = headers.reduce((acc, header, index) => {\n acc[header] = index;\n return acc;\n }, {});\n return data.filter(row => row[headerMap['Job Code']] == job_code);\n },\n getHeaderIX(header_name) {\n const {\n headers\n } = this.fetch();\n const headerMap = headers.reduce((acc, header, index) => {\n acc[header] = index;\n return acc;\n }, {});\n return headerMap[header_name];\n },\n lookupByJobCode(job_code, header_name) {\n const codeData = this.fetchByCode(job_code);\n if (codeData.length > 0) {\n return codeData[0][this.getHeaderIX(header_name)];\n }\n return null;\n },\n getTitle(job_code) {\n return this.lookupByJobCode(job_code, 'Job Description');\n },\n getFringeRate(job_code) {\n return parseFloat(this.lookupByJobCode(job_code, 'Fringe Rate'));\n },\n codeExists(job_code) {\n return this.fetchByCode(job_code).length > 0;\n },\n // method to add GoldBook data as a new sheet to the workbook\n xlsx() {\n const {\n headers,\n data\n } = this.fetch();\n\n // Combine headers and data into one array\n const combinedData = [headers].concat(data);\n\n // Create a worksheet\n return XLSX.utils.aoa_to_sheet(combinedData);\n }\n};\n/* harmony default export */ const gold_book = (GoldBook);\n;// CONCATENATED MODULE: ./src/js/utils/XLSX_handlers.js\n\n\n\n\n\n\n\n// Helper functions\n\n/**\r\n * Deletes the top rows until a row containing complete data is found.\r\n * @param {Array} data - The raw data extracted from the sheet.\r\n * @returns {Array} - The cleaned data with incomplete top rows removed.\r\n */\nfunction deleteTopRowsUntilFullData(data) {\n let fullDataRowFound = false;\n while (!fullDataRowFound && data.length > 0) {\n const row = data[0]; // Get the top row\n let hasAllData = true;\n for (const cell of row) {\n if (cell == null || cell === '') {\n hasAllData = false;\n break;\n }\n }\n if (hasAllData && row.length > 1) {\n fullDataRowFound = true;\n } else {\n // delete the top row if it's not the header row\n data.shift();\n }\n }\n return data;\n}\n\n/**\r\n * Reads the workbook from the provided array buffer.\r\n * @param {ArrayBuffer} arrayBuffer - The array buffer containing the workbook data.\r\n * @returns {Object} - The parsed workbook.\r\n */\nfunction readWorkbook(arrayBuffer) {\n return XLSX.read(arrayBuffer, {\n type: 'array'\n });\n}\n\n/**\r\n * Processes sheets to be split by fund and saves the relevant data.\r\n * @param {string} sheetName - The name of the sheet being processed.\r\n * @param {Object} sheet - The sheet object from the workbook.\r\n */\nfunction processSheet(sheetName, sheet) {\n // Read in sheets\n const rawData = XLSX.utils.sheet_to_json(sheet, {\n header: 1,\n defval: ''\n });\n\n // Clean the data by removing top rows with incomplete data\n const dataRows = deleteTopRowsUntilFullData(rawData);\n\n // Get new headers\n const headers = dataRows[0];\n const fundIndex = headers.indexOf('Fund');\n if (fundIndex === -1) {\n console.error(`No 'Fund' column found in sheet ${sheetName}`);\n return;\n }\n\n // Save a dictionary of data for each fund for each sheet\n const fundData = {};\n dataRows.forEach(row => {\n const fund = row[fundIndex];\n if (fund && fund !== \"Fund\") {\n if (!fundData[fund]) {\n fundData[fund] = [];\n }\n const rowData = {};\n headers.forEach((header, index) => {\n rowData[removeNewLines(header)] = row[index];\n });\n fundData[fund].push(rowData);\n }\n });\n\n // Save fund number and name as we go along\n fund_lookup_table.update(fundData);\n console.log('updating fund lookup table');\n Object.keys(fundData).forEach(fund => {\n const key = `${SHEETS[sheetName]}_${fund}`;\n localStorage.setItem(key, JSON.stringify(fundData[fund]));\n });\n}\n\n/**\r\n * Processes the 'Drop-Down Menus' sheet to extract services data.\r\n * @param {Object} sheet - The sheet object from the workbook.\r\n */\nfunction processDropDownMenusSheet(sheet) {\n const sheetData = XLSX.utils.sheet_to_json(sheet, {\n header: 1\n });\n const headerRow = sheetData[0];\n const servicesIndex = headerRow.indexOf('Services');\n\n // save drop down menu for later excel downloads\n localStorage.setItem('dropdowns', JSON.stringify(sheetData));\n if (servicesIndex === -1) {\n console.error('Header \"Services\" not found');\n } else {\n const servicesColumn = sheetData.slice(1).map(row => row[servicesIndex]);\n const cleanedServicesColumn = servicesColumn.filter(value => value != null);\n services.save(cleanedServicesColumn);\n }\n}\n\n/**\r\n * Processes the 'Dept Summary' sheet to get and save the target for the general fund.\r\n * @param {Object} sheet - The sheet object from the workbook.\r\n */\nfunction processDeptSummarySheet(sheet) {\n if (sheet[TARGET_CELL_ADDRESS]) {\n const cellValue = sheet[TARGET_CELL_ADDRESS].v; // Access the cell value\n localStorage.setItem('target', cellValue);\n } else {\n console.error(`Cell ${TARGET_CELL_ADDRESS} not found`);\n }\n\n // save the sheet to add to future excel downloads\n const sheetData = XLSX.utils.sheet_to_json(sheet, {\n header: 1\n });\n // TODO: remove excess empty rows\n const newSheetData = sheetData.map(row => row.slice(0, 3));\n localStorage.setItem('dept-summary', JSON.stringify(newSheetData));\n}\n\n/**\r\n * Processes the 'FY{FISCAL_YEAR} Gold Book' sheet to initialize the Gold Book.\r\n * @param {Object} sheet - The sheet object from the workbook.\r\n */\nfunction processGoldBookSheet(sheet) {\n gold_book.init(sheet);\n}\nfunction processNewInitsSheet(sheet) {\n // Read in sheets\n const rawData = XLSX.utils.sheet_to_json(sheet, {\n header: 1,\n defval: ''\n });\n\n // Clean the data by removing top rows with incomplete data\n const dataRows = deleteTopRowsUntilFullData(rawData);\n\n // Get new headers\n const headers = dataRows[0];\n // final data output\n let fullData = [];\n\n // Convert data to JSON form and filter out rows where first value is missing\n dataRows.slice(1).forEach(row => {\n // Skip headers row\n // skip any empty rows at the end\n if (row[0] != '' && row[0] != '-') {\n const rowData = {};\n headers.forEach((header, index) => {\n rowData[removeNewLines(header)] = row[index];\n });\n // only keep supplemental initiatives \n if (rowData['Baseline or Supplemental'].includes('Supplemental')) {\n fullData.push(rowData);\n }\n }\n });\n // save in local storage\n localStorage.setItem('new-inits', JSON.stringify(fullData));\n}\n\n// Main function to read and process the workbook\nfunction processWorkbook(arrayBuffer) {\n const workbook = readWorkbook(arrayBuffer);\n workbook.SheetNames.forEach(sheetName => {\n // Only convert sheets we need; treat new inits separately because they shouldn't save by fund\n if (sheetName == Object.keys(SHEETS)[4]) {\n const sheet = workbook.Sheets[sheetName];\n processNewInitsSheet(sheet);\n } else if (Object.keys(SHEETS).includes(sheetName)) {\n const sheet = workbook.Sheets[sheetName];\n processSheet(sheetName, sheet);\n } else if (sheetName === 'Drop-Down Menus') {\n const sheet = workbook.Sheets[sheetName];\n processDropDownMenusSheet(sheet);\n } else if (sheetName === 'Dept Summary') {\n const sheet = workbook.Sheets[sheetName];\n processDeptSummarySheet(sheet);\n } else if (sheetName === `FY${FISCAL_YEAR} Gold Book`) {\n const sheet = workbook.Sheets[sheetName];\n processGoldBookSheet(sheet);\n }\n });\n console.log('all excel data saved');\n}\n\n// Utility function to append a sheet to the workbook if data is present\nfunction appendSheetToWorkbook(workbook, data, sheetName) {\n if (data.length > 0) {\n const sheet = XLSX.utils.json_to_sheet(data);\n XLSX.utils.book_append_sheet(workbook, sheet, sheetName);\n }\n}\nfunction downloadXLSX() {\n // grab data from baseline object\n const baseline = new models_baseline();\n const workbook = XLSX.utils.book_new(); // Create a new workbook\n\n // Initialize sheet data based on the names of each tab in the Excel doc\n const sheetData = Object.keys(SHEETS).reduce((acc, key) => {\n acc[key] = [];\n return acc;\n }, {});\n\n // Aggregate all rows across funds and combine for each tab\n baseline.funds.forEach(fund => {\n Object.keys(SHEETS).forEach(sheetName => {\n if (fund[SHEETS[sheetName]] && fund[SHEETS[sheetName]].table) {\n sheetData[sheetName].push(...fund[SHEETS[sheetName]].table);\n }\n });\n });\n\n // Add initiatives data (which isn't stored by fund)\n sheetData[Object.keys(SHEETS)[4]] = JSON.parse(localStorage.getItem('new-inits'));\n\n // Create a tab for each table\n Object.keys(sheetData).forEach(sheetName => {\n appendSheetToWorkbook(workbook, sheetData[sheetName], sheetName);\n });\n\n // Add a tab for the GoldBook\n XLSX.utils.book_append_sheet(workbook, gold_book.xlsx(), `FY${FISCAL_YEAR} Gold Book`);\n\n // add a tab for the drop downs and dept summary (just targets)\n writeJSONtoNewTab('dropdowns', 'Drop-Down Menus', workbook);\n writeJSONtoNewTab('dept-summary', 'Dept Summary', workbook);\n\n // Generate a downloadable file\n const wbout = XLSX.write(workbook, {\n bookType: 'xlsx',\n type: 'array'\n });\n const blob = new Blob([wbout], {\n type: 'application/octet-stream'\n });\n\n // Create a link and trigger the download\n const link = document.createElement(\"a\");\n link.href = URL.createObjectURL(blob);\n link.download = \"Filled_Detail_Sheet.xlsx\";\n document.body.appendChild(link);\n link.click();\n document.body.removeChild(link);\n}\nfunction writeJSONtoNewTab(storage_key, tab_name, workbook) {\n const data = JSON.parse(localStorage.getItem(storage_key));\n console.log(data);\n const worksheet = XLSX.utils.aoa_to_sheet(data);\n XLSX.utils.book_append_sheet(workbook, worksheet, tab_name);\n}\nfunction excelSerialDateToJSDate(serial) {\n if (!serial) {\n return null;\n }\n ;\n // Excel considers 1900-01-01 as day 1, but JavaScript's Date considers\n // 1970-01-01 as day 0. Therefore, we calculate the number of milliseconds\n // between 1900-01-01 and 1970-01-01.\n const excelEpoch = new Date(Date.UTC(1899, 11, 30)); // JavaScript Consider December month as '11'\n\n // Calculate the JS date by adding serial days to the epoch date\n const date = new Date(excelEpoch.getTime() + serial * 24 * 60 * 60 * 1000);\n\n // Set the time part to zero (midnight)\n date.setUTCHours(0, 0, 0, 0);\n\n // Return the date part of the ISO string\n return date.toISOString().split('T')[0];\n}\n// EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js!./src/js/components/tooltip/tooltip.css\nvar tooltip = __webpack_require__(843);\n;// CONCATENATED MODULE: ./src/js/components/tooltip/tooltip.css\n\n \n \n \n \n \n \n \n \n \n\nvar tooltip_options = {};\n\ntooltip_options.styleTagTransform = (styleTagTransform_default());\ntooltip_options.setAttributes = (setAttributesWithoutAttributes_default());\ntooltip_options.insert = insertBySelector_default().bind(null, \"head\");\ntooltip_options.domAPI = (styleDomAPI_default());\ntooltip_options.insertStyleElement = (insertStyleElement_default());\n\nvar tooltip_update = injectStylesIntoStyleTag_default()(tooltip/* default */.A, tooltip_options);\n\n\n\n\n /* harmony default export */ const tooltip_tooltip = (tooltip/* default */.A && tooltip/* default */.A.locals ? tooltip/* default */.A.locals : undefined);\n\n;// CONCATENATED MODULE: ./src/js/components/tooltip/tooltip.js\n\n\n\n\n\n\nfunction hideTooltip() {\n document.getElementById('tooltip').style.visibility = 'hidden';\n}\nfunction showTooltip() {\n document.getElementById('tooltip').style.visibility = 'visible';\n}\nfunction editTooltipText(newText) {\n // edit text to display inside tooltip\n const tooltip = document.getElementById('tooltip');\n tooltip.innerHTML = newText;\n}\nfunction showAccountString(row) {\n const approp = cells.getText(row, 'approp-name');\n const cc = cells.getText(row, 'cc-name');\n const obj = cells.getText(row, 'object-name');\n const fund = cells.getText(row, 'fund-name');\n var message = `Fund: ${fund} \n Appropriation: ${approp} \n Cost Center: ${cc}`;\n if (obj) {\n message += ` Object: ${obj}`;\n }\n message += ` If you believe this account string is incorrect, please leave a comment \n in the comment column.`;\n editTooltipText(message);\n}\nfunction showSalaryProjection(row) {\n const general_increase = cells.getText(row, 'general-increase-rate');\n const merit_increase = cells.getText(row, 'merit-increase-rate');\n const current_salary = cells.getValue(row, 'current-salary');\n const proj_salary = cells.getValue(row, 'avg-salary');\n if (current_salary) {\n var message = `The average salary/wage for this position was \n ${formatCurrency(current_salary)} as of September 20${FISCAL_YEAR - 2}. \n Given a ${general_increase * 100}% general increase rate and a ${merit_increase * 100}% \n merit increase, the FY${FISCAL_YEAR} projection for this position's average \n annual salary/wage is ${formatCurrency(proj_salary)}. This projection takes into \n account the number of vacant vs. full positions.`;\n } else {\n var message = `The average salary/wage for this position was \n unknown as of September 20${FISCAL_YEAR - 2}, or the position\n did not exist. The FY${FISCAL_YEAR} projection for this position's \n average annual salary/wage is ${formatCurrency(proj_salary)}.`;\n }\n editTooltipText(message);\n}\nfunction showFinalPersonnelCost(row) {\n const proj_salary = cells.getValue(row, 'avg-salary');\n const ftes = cells.getText(row, 'baseline-ftes');\n const fringe = parseFloat(cells.getText(row, 'fringe'));\n const avg_benefits = proj_salary * fringe;\n const message = `The total cost captures ${ftes} position(s) at\n an annual salary/wage of ${formatCurrency(proj_salary)}, \n plus fringe benefits that cost ${formatCurrency(avg_benefits)} \n per position per year, on average.`;\n editTooltipText(message);\n}\nfunction showFICA(row) {\n const fica = parseFloat(cells.getText(row, 'fica'));\n const ficaPercentage = (fica * 100).toFixed(2);\n const message = `This total is overtime / holiday / shift premium pay, plus FICA (payroll tax), \n which is ${ficaPercentage}% for this cost center.`;\n editTooltipText(message);\n}\nfunction showCPA(row) {\n const cpa = parseFloat(cells.getText(row, 'cpa'));\n const description = cells.getText(row, 'cpa-description');\n const vendor = cells.getText(row, 'vendor');\n var contract_end = cells.getText(row, 'contract-end');\n // convert to normal date format from excel \n contract_end = excelSerialDateToJSDate(contract_end);\n const remaining = cells.getValue(row, 'remaining');\n if (cpa) {\n var message = `CPA #${cpa}`;\n } else {\n var message = `No CPA`;\n }\n if (vendor) {\n message += ` Vendor: ${vendor}`;\n }\n ;\n if (description) {\n message += ` Description: ${description}`;\n }\n ;\n if (contract_end) {\n message += ` Contract End Date: ${contract_end}`;\n }\n if (remaining) {\n message += ` Amount Remaining on Contract: ${formatCurrency(remaining)}`;\n }\n editTooltipText(message);\n}\nfunction tooltip_link(element, displayFn) {\n // add class to show cell with an underline, etc\n element.classList.add('tooltip-cell');\n\n // Create and append (detail)\n const detail = document.createElement('span');\n detail.classList.add('detail');\n detail.textContent = '(detail)';\n element.appendChild(detail);\n\n // add event listener to show tooltip on mouseover\n element.addEventListener('click', function (event) {\n const row = event.target.closest('tr');\n displayFn(row);\n showTooltip();\n });\n // and hide when mouse moves off\n element.addEventListener('mouseout', function () {\n hideTooltip();\n });\n // Update tooltip position on mouse move\n element.addEventListener('mousemove', function (event) {\n const tooltip = document.getElementById('tooltip');\n tooltip.style.top = event.clientY + 10 + 'px';\n tooltip.style.left = event.clientX + 10 + 'px';\n });\n}\nfunction linkAccountStringCol() {\n // get all relevant cells\n document.querySelectorAll('td.account-string').forEach(cell => {\n tooltip_link(cell, showAccountString);\n });\n}\nfunction linkSalaryCol() {\n // get all relevant cells\n document.querySelectorAll('td.avg-salary').forEach(cell => {\n tooltip_link(cell, showSalaryProjection);\n });\n}\nfunction linkTotalPersonnelCostCol() {\n // get all relevant cells\n document.querySelectorAll('td.total-baseline').forEach(cell => {\n tooltip_link(cell, showFinalPersonnelCost);\n });\n}\nfunction linkTotalOTCol() {\n // get all relevant cells\n document.querySelectorAll('td.total').forEach(cell => {\n tooltip_link(cell, showFICA);\n });\n}\nfunction linkCPACol() {\n // get all relevant cells\n document.querySelectorAll('td.cpa').forEach(cell => {\n tooltip_link(cell, showCPA);\n });\n}\nconst Tooltip = {\n hide: hideTooltip,\n show: showTooltip,\n linkAll: () => {\n linkAccountStringCol();\n switch (current_page.load()) {\n case 'personnel':\n // linkAccountStringCol();\n linkSalaryCol();\n linkTotalPersonnelCostCol();\n break;\n case 'overtime':\n linkTotalOTCol();\n // linkAccountStringCol();\n break;\n case 'nonpersonnel':\n // linkAccountStringCol();\n linkCPACol();\n break;\n // case 'revenue':\n // linkAccountStringCol();\n // break;\n // case 'new-inits':\n // linkAccountStringCol();\n // break;\n default:\n break;\n }\n },\n unlink: function () {\n let details = document.querySelectorAll('.detail');\n details.forEach(span => {\n span.remove();\n });\n }\n};\n/* harmony default export */ const components_tooltip_tooltip = (Tooltip);\n;// CONCATENATED MODULE: ./src/js/utils/JSON_data_handlers.js\nasync function fetchJSON(jsonFilePath) {\n return fetch(jsonFilePath).then(response => {\n if (!response.ok) {\n throw new Error('Network response was not ok');\n }\n return response.json();\n });\n}\nfunction convertToJSON(table, colsToRemove = []) {\n const rows = table.rows;\n // Extract headers from the first row\n const headerRow = rows[0].cells;\n const headers = [];\n for (let j = 0; j < headerRow.length; j++) {\n headers.push(headerRow[j].innerText);\n }\n\n // initialize data\n var tableData = [];\n for (var i = 1; i < rows.length; i++) {\n const cols = rows[i].cells;\n const rowData = {};\n headers.forEach((header, index) => {\n if (colsToRemove.includes(header)) {\n return;\n } else if (cols[index].classList.contains('cost')) {\n rowData[header] = cols[index].getAttribute('value');\n } else {\n rowData[header] = cols[index].innerText;\n }\n });\n tableData.push(rowData);\n }\n return JSON.stringify(tableData);\n}\n;// CONCATENATED MODULE: ./src/js/components/table/table.js\n\n\n\n\n\n\n\n\n\n\n\n\n\nfunction adjustTableWidth(width_pct) {\n const table = document.getElementById('main-table');\n table.style.width = width_pct;\n}\nfunction showTable() {\n const tableContainer = document.querySelector('.table-container');\n tableContainer.innerHTML = Table.html;\n const table = document.getElementById('main-table');\n table.style.display = 'table';\n}\nfunction hideTable() {\n // delete table object from table container\n const tableContainer = document.querySelector('.table-container');\n tableContainer.innerHTML = '';\n subcomponents_buttons.AddRow.hide();\n}\nfunction saveTableData() {\n // remove the detail text\n components_tooltip_tooltip.unlink();\n // get table\n var table = document.getElementById('main-table');\n // determine save_as name\n if (current_fund.number()) {\n var save_as = `${current_page.load()}_${current_fund.number()}`;\n } else {\n var save_as = current_page.load();\n }\n localStorage.setItem(save_as, convertToJSON(table, ['Edit']));\n // update sidebar with new data\n components_sidebar_sidebar.updateTotals();\n // relink, depending on page\n components_tooltip_tooltip.linkAll();\n}\nconst Table = {\n html: `\n
\n \n \n
`,\n Buttons: subcomponents_buttons,\n Cell: cells,\n Columns: columns,\n Header: headers,\n Rows: rows,\n Data: data,\n Filter: filters,\n // functions\n adjustWidth: function (width_pct) {\n adjustTableWidth(width_pct);\n },\n clear: hideTable,\n hide: hideTable,\n show: showTable,\n save: saveTableData\n};\n/* harmony default export */ const components_table_table = (Table);\n;// CONCATENATED MODULE: ./src/js/components/accordion/accordion.js\n\n\n\n\n\nfunction redirectForEdit() {\n const row = document.querySelector(`.active-editing`);\n const table = row.parentElement;\n const section = table.closest('.summary-container');\n // new initiative edits should all redirect to the new-inits page\n if (section.id == 'supp-accordion') {\n visitPage('new-inits');\n } else {\n // Split the string into parts using '-' as the delimiter; retain fund as 1st numeric segment\n const fund = table.id.split('-')[1];\n current_fund.update(fund);\n const lineItem = row.querySelector('.line-item').textContent;\n // visit the correct page for editing\n switch (lineItem) {\n case 'Personnel Expenditures':\n visitPage('personnel');\n break;\n case 'Non-Personnel Expenditures':\n visitPage('nonpersonnel');\n break;\n case 'Revenues':\n visitPage('revenue');\n break;\n case 'Overtime Expenditures':\n visitPage('overtime');\n break;\n default:\n console.error('Name of line item in table does not match a page destination.');\n }\n }\n}\nconst ExpenseTable = {\n table_id: account_string => {\n return `table-${account_string}`;\n },\n init(account_string) {\n // create empty table and put it in the accordion\n var table = document.createElement('table');\n table.id = this.table_id(account_string);\n table.classList.add('accordion-table');\n var parent = document.querySelector(`#string_${account_string}_content .accordion-body`);\n parent.appendChild(table);\n },\n createNewCell(content, row, className) {\n const newCell = document.createElement('td');\n newCell.innerHTML = content;\n newCell.classList.add(className);\n row.appendChild(newCell);\n },\n addRow(account_string, row_name, number) {\n var table = document.getElementById(this.table_id(account_string));\n var new_row = document.createElement('tr');\n table.appendChild(new_row);\n // Create a cell for the line item label\n this.createNewCell(row_name, new_row, 'line-item');\n // create a cell for the amount\n this.createNewCell(formatCurrency(number), new_row, 'cost');\n // create Edit button \n var button = '';\n if (row_name != 'Net Expenditures (Revenues)') {\n button = components_table_table.Buttons.Edit.html;\n }\n this.createNewCell(button, new_row);\n },\n fillFromFund(fund) {\n // use just fund as account string to initialize table inside accordion\n this.init(fund);\n const fundObject = new models_fund(fund);\n\n // Add a row for each appropriation in the fund\n const id = cleanString(fund);\n fundObject.getAppropriations().forEach(appropObj => {\n // if the total for the appropriation is > $0, add an accordion for all the CCs\n if (appropObj.total() != 0) {\n Item.add(appropObj.accountString(), `#string_${id}_content .accordion-body`);\n Item.updateHeader(appropObj.name(), appropObj.accountString(), appropObj.total());\n this.fillFromApprop(appropObj);\n }\n });\n },\n fillFromApprop(appropObj) {\n // initialize the table object\n this.init(appropObj.accountString());\n // add a collapsible row for each cost center\n appropObj.getCostCenters().forEach(ccObj => {\n if (ccObj.getTotal() != 0) {\n Item.add(ccObj.accountString(), `#string_${appropObj.accountString()}_content .accordion-body`);\n Item.updateHeader(ccObj.getName(), ccObj.accountString(), ccObj.getTotal());\n this.fillFromCC(ccObj);\n }\n });\n },\n fillFromCC(ccObj) {\n // initialize a table and summarize the line items\n this.init(ccObj.accountString());\n this.addRow(ccObj.accountString(), 'Personnel Expenditures', ccObj.getPersonnelCost());\n this.addRow(ccObj.accountString(), 'Overtime Expenditures', ccObj.getOvertimeCost());\n this.addRow(ccObj.accountString(), 'Non-Personnel Expenditures', ccObj.getNonPersonnelCost());\n this.addRow(ccObj.accountString(), 'Revenues', ccObj.getRevenue());\n this.addRow(ccObj.accountString(), 'Net Expenditures (Revenues)', ccObj.getTotal());\n },\n fillFromInit(program) {\n // Fill out info for each supplemental init\n this.init(program.id());\n this.addRow(program.id(), 'Estimated Revenue', program.revenue());\n this.addRow(program.id(), 'Personnel Expenditures', program.personnel());\n this.addRow(program.id(), 'Non-Personnel Operating', program.operating());\n this.addRow(program.id(), 'Non-Personnel Capital', program.capital());\n this.addRow(program.id(), 'Total Expenditures', program.total());\n }\n};\nconst Item = {\n accountString(fund, approp = '', cc = '') {\n var account_string = cleanString(fund);\n if (approp) {\n account_string += approp;\n }\n ;\n if (cc) {\n account_string += cc;\n }\n ;\n return account_string;\n },\n html(account_string) {\n return `
\n \n
\n
\n \n
`;\n },\n add: function (account_string, accordion_query) {\n // get accordion and add a new item to it\n const parent = document.querySelector(accordion_query);\n const item_element = document.createElement('div');\n item_element.classList.add('accordion-item');\n item_element.innerHTML = this.html(account_string);\n parent.appendChild(item_element);\n },\n ExpenseTable: ExpenseTable,\n updateHeader: function (title, account_string, new_amount) {\n const header_btn = document.querySelector(`#string_${account_string}_header button`);\n header_btn.querySelector('span.name').textContent = title;\n header_btn.querySelector('span.amount').textContent = formatCurrency(new_amount);\n }\n};\nconst AddInitButton = {\n init() {\n const btn = document.querySelector('.btn-add-init');\n btn.addEventListener('click', function () {\n visitPage('new-inits');\n });\n }\n};\nconst Accordion = {\n Item: Item,\n AddInitButton: AddInitButton,\n hide: function () {\n document.querySelector('#accordion-div').style.display = 'none';\n // reset to delete content\n document.querySelector('#baseline-accordion .summary-accordion').innerHTML = '';\n document.querySelector('#supp-accordion .summary-accordion').innerHTML = '';\n },\n show: function () {\n document.querySelector('#accordion-div').style.display = 'block';\n },\n async createBaseline() {\n var funds = fund_lookup_table.listFunds();\n funds.forEach(fund => {\n Item.add(fund, '#baseline-accordion .summary-accordion');\n Item.ExpenseTable.fillFromFund(fund);\n const fundObject = new models_fund(fund);\n Item.updateHeader(`Fund ${fund_lookup_table.getName(fund)}`, fund, fundObject.getTotal());\n });\n\n // color-code GF baseline\n const GF = new models_fund(1000);\n // text to color code\n let topline = document.querySelector('#string_1000_header .amount');\n if (GF.getTotal() <= models_baseline.target()) {\n topline.style.color = 'green';\n topline.style.weight = 4;\n } else {\n topline.style.color = 'red';\n }\n },\n createSupp() {\n const supp = new supplemental();\n supp.initiatives.forEach(program => {\n Item.add(program.id(), '#supp-accordion .summary-accordion');\n Item.ExpenseTable.fillFromInit(program);\n Item.updateHeader(program.name, program.id(), program.total());\n });\n },\n updateTopLines() {\n // adjuse baseline\n // const baseline = new Baseline;\n const baseline = new models_baseline();\n const baselineAmount = document.querySelector('#baseline-title .top-line-amount');\n baselineAmount.textContent = formatCurrency(baseline.total());\n // adjust supplementals\n const supp = new supplemental();\n const suppAmount = document.querySelector('#supp-title .top-line-amount');\n suppAmount.textContent = formatCurrency(supp.total());\n },\n build() {\n this.createBaseline();\n this.createSupp();\n // initialize edit buttons\n components_table_table.Buttons.Edit.init(redirectForEdit);\n this.AddInitButton.init();\n this.updateTopLines();\n }\n};\n/* harmony default export */ const components_accordion_accordion = (Accordion);\n// EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js!./src/js/components/file_upload/file_upload.css\nvar file_upload = __webpack_require__(39);\n;// CONCATENATED MODULE: ./src/js/components/file_upload/file_upload.css\n\n \n \n \n \n \n \n \n \n \n\nvar file_upload_options = {};\n\nfile_upload_options.styleTagTransform = (styleTagTransform_default());\nfile_upload_options.setAttributes = (setAttributesWithoutAttributes_default());\nfile_upload_options.insert = insertBySelector_default().bind(null, \"head\");\nfile_upload_options.domAPI = (styleDomAPI_default());\nfile_upload_options.insertStyleElement = (insertStyleElement_default());\n\nvar file_upload_update = injectStylesIntoStyleTag_default()(file_upload/* default */.A, file_upload_options);\n\n\n\n\n /* harmony default export */ const file_upload_file_upload = (file_upload/* default */.A && file_upload/* default */.A.locals ? file_upload/* default */.A.locals : undefined);\n\n;// CONCATENATED MODULE: ./src/js/components/file_upload/file_upload.js\n// file_upload.js\n\n\n\nconst FileUpload = {\n init: function () {\n components_nav_buttons_nav_buttons.Next.disable();\n const inputObject = document.getElementById('file-input');\n inputObject.addEventListener('change', function (event) {\n readXL(event);\n });\n },\n show: function () {\n const inputObject = document.getElementById('file-input');\n inputObject.style.display = '';\n },\n hide: function () {\n const inputObject = document.getElementById('file-input');\n inputObject.style.display = 'none';\n }\n};\nfunction readXL(event) {\n const file = event.target.files[0]; // read uploaded file\n const spinner = document.getElementById('upload-spinner'); // get the spinner element\n\n if (file) {\n // Show the spinner\n spinner.style.display = 'block';\n\n // read in new data\n const reader = new FileReader();\n reader.onload = function (e) {\n const arrayBuffer = e.target.result;\n try {\n processWorkbook(arrayBuffer);\n\n // Hide the spinner once processing is done\n spinner.style.display = 'none';\n components_nav_buttons_nav_buttons.Next.enable(); // Enable the next button after processing\n } catch (error) {\n console.error('Error processing workbook:', error);\n\n // Hide the spinner in case of an error\n spinner.style.display = 'none';\n components_nav_buttons_nav_buttons.Next.enable(); // Ensure the button is re-enabled in case of an error\n }\n };\n reader.onerror = function (err) {\n console.error('Error reading file:', err);\n\n // Hide the spinner in case of an error\n spinner.style.display = 'none';\n components_nav_buttons_nav_buttons.Next.enable(); // Ensure the button is re-enabled in case of an error\n };\n reader.readAsArrayBuffer(file); // Read the file as an ArrayBuffer\n }\n}\n/* harmony default export */ const components_file_upload_file_upload = (FileUpload);\n// EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js!./src/js/components/modal/modal.css\nvar modal = __webpack_require__(95);\n;// CONCATENATED MODULE: ./src/js/components/modal/modal.css\n\n \n \n \n \n \n \n \n \n \n\nvar modal_options = {};\n\nmodal_options.styleTagTransform = (styleTagTransform_default());\nmodal_options.setAttributes = (setAttributesWithoutAttributes_default());\nmodal_options.insert = insertBySelector_default().bind(null, \"head\");\nmodal_options.domAPI = (styleDomAPI_default());\nmodal_options.insertStyleElement = (insertStyleElement_default());\n\nvar modal_update = injectStylesIntoStyleTag_default()(modal/* default */.A, modal_options);\n\n\n\n\n /* harmony default export */ const modal_modal = (modal/* default */.A && modal/* default */.A.locals ? modal/* default */.A.locals : undefined);\n\n;// CONCATENATED MODULE: ./src/js/components/modal/modal.js\n\nfunction clearModal() {\n updateModalTitle('');\n document.getElementById('modal-body').innerHTML = '';\n}\nfunction hideModal() {\n $('#main-modal').modal('hide');\n}\nfunction showModal() {\n $('#main-modal').modal('show');\n}\nfunction showModalHandler() {\n showModal('main-modal');\n}\nconst Submit = {\n handler: null,\n init: function (onSubmit) {\n const modal = document.getElementById('main-modal');\n\n // Deinitialize first to avoid multiple bindings\n this.deinit();\n this.handler = function (event) {\n event.preventDefault();\n onSubmit(event);\n };\n\n // Adding the handler as the event listener\n modal.addEventListener('submit', this.handler);\n\n // add event listener to enable close x\n const x = modal.querySelector('#modal-close-x');\n x.addEventListener('click', hideModal);\n },\n deinit: function () {\n const modal = document.getElementById('main-modal');\n if (this.handler !== null) {\n // Removing the event listener and clear the handler\n modal.removeEventListener('submit', this.handler);\n this.handler = null;\n }\n\n // remove event listener to enable close x\n const x = modal.querySelector('#modal-close-x');\n x.removeEventListener('click', hideModal);\n }\n};\nconst Link = {\n add: function (button_id) {\n document.getElementById(button_id).addEventListener('click', showModalHandler);\n },\n remove: function (button_id) {\n document.getElementById(button_id).removeEventListener('click', showModalHandler);\n }\n};\nfunction updateModalTitle(title) {\n document.getElementById('modal-title').textContent = title;\n}\nconst Title = {\n update: function (title) {\n updateModalTitle(title);\n }\n};\nconst Modal = {\n hide: hideModal,\n show: showModal,\n clear: clearModal,\n Title: Title,\n Link: Link,\n Submit: Submit\n};\n/* harmony default export */ const components_modal_modal = (Modal);\n;// CONCATENATED MODULE: ./src/js/components/body/body.js\n\n\n\n\n\n\n\n\n\n\nfunction resetPage() {\n // hide everything in the body\n components_welcome_welcome.hide();\n components_modal_modal.clear();\n components_modal_modal.hide();\n components_nav_buttons_nav_buttons.hide();\n js_components_prompt_prompt.hide();\n components_table_table.hide();\n components_sidebar_sidebar.hide();\n components_accordion_accordion.hide();\n components_file_upload_file_upload.hide();\n components_tooltip_tooltip.hide();\n\n // reset filter dropdowns\n components_table_table.Filter.deleteAll();\n js_components_prompt_prompt.Buttons.reset();\n // disable submit button\n components_modal_modal.Submit.deinit();\n}\nconst Body = {\n reset: resetPage\n};\n/* harmony default export */ const components_body_body = (Body);\n// EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js!./src/js/components/header/header.css\nvar header = __webpack_require__(107);\n;// CONCATENATED MODULE: ./src/js/components/header/header.css\n\n \n \n \n \n \n \n \n \n \n\nvar header_options = {};\n\nheader_options.styleTagTransform = (styleTagTransform_default());\nheader_options.setAttributes = (setAttributesWithoutAttributes_default());\nheader_options.insert = insertBySelector_default().bind(null, \"head\");\nheader_options.domAPI = (styleDomAPI_default());\nheader_options.insertStyleElement = (insertStyleElement_default());\n\nvar header_update = injectStylesIntoStyleTag_default()(header/* default */.A, header_options);\n\n\n\n\n /* harmony default export */ const header_header = (header/* default */.A && header/* default */.A.locals ? header/* default */.A.locals : undefined);\n\n;// CONCATENATED MODULE: ./src/js/components/header/header.js\n\n\n\nconst header_Title = {\n update: function (title) {\n document.getElementById(\"title\").textContent = title;\n },\n default() {\n this.update(`FY${FISCAL_YEAR} Budget Request`);\n }\n};\nconst Subtitle = {\n update: function (subtitle) {\n // get current fund\n var fund = current_fund.name();\n if (fund) {\n var subtitle = `${subtitle}: ${fund}`;\n }\n document.getElementById(\"subtitle\").textContent = subtitle;\n }\n};\n// EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js!./src/js/components/form/form.css\nvar form_form = __webpack_require__(109);\n;// CONCATENATED MODULE: ./src/js/components/form/form.css\n\n \n \n \n \n \n \n \n \n \n\nvar form_options = {};\n\nform_options.styleTagTransform = (styleTagTransform_default());\nform_options.setAttributes = (setAttributesWithoutAttributes_default());\nform_options.insert = insertBySelector_default().bind(null, \"head\");\nform_options.domAPI = (styleDomAPI_default());\nform_options.insertStyleElement = (insertStyleElement_default());\n\nvar form_update = injectStylesIntoStyleTag_default()(form_form/* default */.A, form_options);\n\n\n\n\n /* harmony default export */ const components_form_form = (form_form/* default */.A && form_form/* default */.A.locals ? form_form/* default */.A.locals : undefined);\n\n;// CONCATENATED MODULE: ./src/js/components/form/subcomponents/fields.js\n// function to add questions to forms\n\n\n\n// inputType is for validation ('number' or 'text', etc)\nfunction appendFormElement(label, inputEl, inputId, required) {\n // change if we want forms elsewhere\n const form = document.getElementById('new-form');\n\n // create outer wrapper for element\n const wrapper = document.createElement('div');\n\n // label question\n const labelEl = document.createElement('label');\n labelEl.textContent = label;\n\n // mark as required if applicable\n inputEl.required = required;\n\n // If an ID is provided, set it on the element\n if (inputId) {\n //inputEl.id = `input-${inputId}`;\n inputEl.id = inputId;\n }\n\n // create validation text/ a place to display errors\n const validationText = document.createElement('p');\n validationText.id = `${inputId}-validation`;\n validationText.classList.add('error-message');\n validationText.style.color = 'red';\n\n // add elements\n wrapper.appendChild(labelEl);\n wrapper.appendChild(inputEl);\n wrapper.appendChild(validationText);\n form.appendChild(wrapper);\n}\nconst NewField = {\n shortText: function (label, inputId, required = false) {\n const inputEl = document.createElement('input');\n inputEl.type = 'text';\n appendFormElement(label, inputEl, inputId, required);\n },\n longText: function (label, inputId, required = false) {\n const inputEl = document.createElement('textarea');\n appendFormElement(label, inputEl, inputId, required);\n },\n numericInput: function (label, inputId, required = false) {\n const inputEl = document.createElement('input');\n inputEl.type = 'number';\n appendFormElement(label, inputEl, inputId, required);\n },\n dropdown: function (label, inputId, optionArray, required = false) {\n var inputEl = subcomponents_dropdown.create(optionArray);\n appendFormElement(label, inputEl, inputId, required);\n }\n};\n/* harmony default export */ const fields = (NewField);\n;// CONCATENATED MODULE: ./src/js/components/form/subcomponents/submit.js\nfunction addSubmitButtonToForm(form_id) {\n // Find the form by its ID\n const form = document.getElementById(form_id);\n\n // Create the container `div` for the button\n const buttonContainer = document.createElement('div');\n buttonContainer.id = 'submit-btn-container';\n\n // Create the submit input\n const submitInput = document.createElement('input');\n submitInput.className = 'btn btn-submit'; // Use appropriate class for your design\n submitInput.type = 'submit';\n submitInput.value = 'Submit';\n\n // Append the submit input to the container\n buttonContainer.appendChild(submitInput);\n\n // Append the container to the form\n form.appendChild(buttonContainer);\n}\nconst SubmitButton = {\n add: function () {\n addSubmitButtonToForm('new-form');\n }\n};\n/* harmony default export */ const subcomponents_submit = (SubmitButton);\n;// CONCATENATED MODULE: ./src/js/components/form/form.js\n\n\n\n\nfunction fetchAllResponses(event) {\n // Assuming `event.target` is the form itself\n const form = event.target;\n\n // Initialize an empty array to hold the input values\n let formData = {};\n\n // Loop through each form element\n for (let i = 0; i < form.elements.length; i++) {\n const element = form.elements[i];\n\n // Exclude elements that aren't inputs, textareas, or select\n if (element.tagName === 'INPUT' || element.tagName === 'TEXTAREA' || element.tagName === 'SELECT') {\n // Exclude input types that are not considered for submission (such as `submit`)\n if (element.type !== 'submit' && element.type !== 'button') {\n formData[element.id] = element.value;\n }\n }\n }\n form.reset();\n return formData;\n}\nfunction addForm(element_id = 'modal-body', form_id = 'new-form') {\n const target_elem = document.getElementById(element_id);\n\n // create form\n const form = document.createElement('form');\n form.setAttribute('id', form_id);\n\n // Append the form to the modal body\n target_elem.appendChild(form);\n}\nconst Form = {\n new: function (parent_elem_id) {\n addForm(parent_elem_id, 'new-form');\n },\n fetchAllResponses: function (event) {\n return fetchAllResponses(event);\n },\n NewField: fields,\n Dropdown: subcomponents_dropdown,\n SubmitButton: subcomponents_submit\n};\n/* harmony default export */ const js_components_form_form = (Form);\n;// CONCATENATED MODULE: ./src/js/views/view_class.js\n\n\n\n\n\n\n\n\n\nclass View {\n constructor() {\n // page state in local storage\n this.page_state = '';\n\n // whether to display\n this.navButtons = true;\n this.sidebar = true;\n\n // text to show in the prompt area\n this.prompt = null;\n\n // subtitle text\n this.subtitle = '';\n\n // table object of class ViewTable or null\n this.table = null;\n }\n visit() {\n // update page state\n current_page.update(this.page_state);\n\n // start with a blank page\n components_body_body.reset();\n\n // add default title\n header_Title.default();\n\n // default to showing navbuttons\n if (this.navButtons) {\n components_nav_buttons_nav_buttons.show();\n }\n ;\n\n // default to showing sidebar\n if (this.sidebar) {\n components_sidebar_sidebar.show();\n }\n ;\n\n // initialize prompt text and buttons\n if (this.prompt) {\n js_components_prompt_prompt.Text.update(this.prompt);\n }\n ;\n\n // initialize table\n if (this.table) {\n this.table.build();\n }\n\n // show page subtitle\n if (this.subtitle) {\n Subtitle.update(this.subtitle);\n }\n ;\n }\n cleanup() {\n return;\n }\n}\nclass ViewTable {\n constructor() {\n // Ensure methods retain the correct `this` context\n this.submitNewRow = this.submitNewRow.bind(this);\n this.columns = [{\n title: 'Account String',\n className: 'account-string'\n }, {\n title: 'Appropriation Name',\n className: 'approp-name',\n hide: true\n }, {\n title: 'Appropriation',\n className: 'approp',\n hide: true\n }, {\n title: 'Cost Center Name',\n className: 'cc-name',\n hide: true\n }, {\n title: 'Cost Center',\n className: 'cc',\n hide: true\n }, {\n title: 'Fund Name',\n className: 'fund-name',\n hide: true\n }, {\n title: 'Fund',\n className: 'fund',\n hide: true\n }, {\n title: 'Edit',\n className: 'edit'\n }];\n\n // whether to add an edit column\n this.addEdit = true;\n\n // message to show if there's no saved data\n this.noDataMessage = null;\n\n // text to show for new row button\n this.addButtonText = null;\n\n // whether to show as a datatable\n this.dataTable = true;\n }\n async refreshData() {\n // check for data\n await components_table_table.Data.load();\n\n // create a datatable object\n if (this.dataTable) {\n this.initDataTable();\n }\n\n // add an edit column if needed\n if (this.addEdit) {\n components_table_table.Columns.addAtEnd(components_table_table.Buttons.edit_confirm_btns, 'Edit');\n // activate edit buttons\n components_table_table.Buttons.Edit.init(this.actionOnEdit, this.updateTable);\n }\n\n // assign the correct classes based on the table columns\n components_table_table.Columns.assignClasses(this.columns);\n\n // Apply any update function to make sure sidebar is up to date\n this.updateTable();\n // add any newly created cc or approp to the filters\n this.updateFilters();\n }\n async build() {\n // build table from local storage and initialize edit buttons\n\n // add the add new row button if needed\n if (this.addButtonText) {\n this.setUpForm();\n }\n\n // check for data\n if (await components_table_table.Data.load()) {\n // if there's data, update the table and add filters \n this.addFilters();\n await this.refreshData();\n } else {\n // show a message if there's no saved table data for the selected fund\n if (this.noDataMessage) {\n js_components_prompt_prompt.Text.update(this.noDataMessage);\n }\n }\n }\n addFilters() {\n // Add all relevant filters to table\n if (this.columns.some(column => column.className === 'approp-name')) {\n components_table_table.Filter.add('Appropriation', 'approp-name');\n }\n if (this.columns.some(column => column.className === 'cc-name')) {\n components_table_table.Filter.add('Cost Center', 'cc-name');\n }\n if (this.columns.some(column => column.className === 'object-name')) {\n components_table_table.Filter.add('Object', 'object-name');\n }\n ;\n if (this.columns.some(column => column.className === 'object-category')) {\n components_table_table.Filter.add('Object Category', 'object-category');\n }\n }\n updateFilters() {\n // update filters with any new values\n if (this.columns.some(column => column.className === 'approp-name')) {\n components_table_table.Filter.updateOptions('Appropriation', 'approp-name');\n }\n if (this.columns.some(column => column.className === 'cc-name')) {\n components_table_table.Filter.updateOptions('Cost Center', 'cc-name');\n }\n if (this.columns.some(column => column.className === 'object-name')) {\n components_table_table.Filter.updateOptions('object-name');\n }\n ;\n if (this.columns.some(column => column.className === 'object-category')) {\n components_table_table.Filter.updateOptions('object-category');\n }\n }\n initDataTable() {\n components_table_table.adjustWidth('100%');\n // Initialize Datatables\n if (!$.fn.dataTable.isDataTable('#main-table')) {\n $('#main-table').DataTable({\n paging: false,\n // Disable pagination\n info: false,\n // Disable table information display\n searching: false // Disable the search bar\n });\n }\n ;\n }\n\n // placeholder for action on row edit click\n actionOnEdit() {\n return;\n }\n\n // update function for the sidebar; default to just saving the table\n updateTable() {\n components_table_table.save();\n }\n\n // extra questions of the form to add a new row\n addCustomQuestions() {\n return;\n }\n addValidationListener(inputId, fieldLabel, validationId, length) {\n const inputElement = document.getElementById(inputId);\n inputElement.addEventListener('change', function () {\n if (inputElement.value === 'Add new') {\n // Add a new field after the selected element\n js_components_form_form.NewField.shortText(`Type new ${fieldLabel}:`, inputId.slice(0, -5), true);\n let newInputElement = document.getElementById(inputId.slice(0, -5)); // Remove '-name' suffix\n inputElement.parentElement.insertAdjacentElement('afterend', newInputElement.parentElement);\n\n // Add an event listener for validation on the new input field\n newInputElement.addEventListener('blur', function () {\n let validationText = document.getElementById(validationId);\n validationText.textContent = '';\n if (newInputElement.value.length !== length) {\n validationText.textContent = `${fieldLabel} codes must be exactly ${length} numbers.`;\n } else if (isNaN(Number(newInputElement.value))) {\n validationText.textContent = `${fieldLabel} codes must be numeric.`;\n }\n });\n } else {\n // Remove the new input field if 'Add new' is not selected\n let newInputElement = document.getElementById(inputId.slice(0, -5));\n if (newInputElement) {\n newInputElement.parentElement.innerHTML = '';\n }\n }\n });\n }\n\n // Modified addModalValidation method\n addModalValidation() {\n this.addValidationListener('approp-name', 'Appropriation', 'approp-validation', 5);\n this.addValidationListener('cc-name', 'Cost Center', 'cc-validation', 6);\n }\n setUpForm() {\n // show add button\n components_table_table.Buttons.AddRow.show();\n components_table_table.Buttons.AddRow.updateText(this.addButtonText);\n\n // set up modal for form when add button is pressed\n components_modal_modal.clear();\n components_modal_modal.Link.add('add-btn');\n components_modal_modal.Title.update(this.addButtonText);\n\n // create form\n js_components_form_form.new('modal-body');\n\n // add custom questions\n this.addCustomQuestions();\n // any validation or special functions\n this.addModalValidation();\n // add submit button\n js_components_form_form.SubmitButton.add();\n\n // Initialize form submission to table data\n components_modal_modal.Submit.init(this.submitNewRow);\n }\n editColumns(responses) {\n // if a new appropriation was entered, fix it\n if (responses['approp']) {\n responses['approp-name'] = `${responses['approp']} - New`;\n }\n ;\n // same for cost center\n if (responses['cc']) {\n responses['cc-name'] = `${responses['cc']} - New`;\n }\n ;\n\n // get numbers from account string names\n if (responses['fund-name']) {\n responses['fund'] = account_string.getNumber(responses['fund-name']);\n }\n ;\n if (responses['approp-name']) {\n responses['approp'] = account_string.getNumber(responses['approp-name']);\n }\n ;\n if (responses['cc-name']) {\n responses['cc'] = account_string.getNumber(responses['cc-name']);\n }\n ;\n if (responses['object-name']) {\n responses['object'] = account_string.getNumber(responses['object-name']);\n }\n ;\n responses['account-string'] = account_string.build(responses['approp-name'], responses['cc-name'], responses['object-name'], responses['fund']);\n return responses;\n }\n submitNewRow(event) {\n // get answers from form, hide form, show answers in table\n var responses = js_components_form_form.fetchAllResponses(event);\n\n // edit inputs from modal\n responses = this.editColumns(responses);\n\n // make sure it's not an empty response\n if (Object.values(responses)[0] != '') {\n // change page view\n components_modal_modal.hide();\n\n // add data to table\n components_table_table.Rows.add(responses, this.columns);\n components_table_table.save();\n\n // rebuild table\n this.refreshData();\n }\n }\n}\n;// CONCATENATED MODULE: ./src/js/views/00_welcome.js\n\n\n\n\nclass WelcomeView extends View {\n constructor() {\n super();\n this.page_state = 'welcome';\n this.subtitle = 'Welcome';\n this.sidebar = false;\n this.navButtons = false;\n this.prompt = `This tool will help you prepare your budget request submission for fiscal year \n 20${FISCAL_YEAR}. Start by uploading the detail sheet sent to you by your budget analyst. This\n tool will use the data in that sheet to pre-fill your submission with last year's request.\n
\n Then, you will be guided through each fund to make any adjustments from last year's request. \n As you \n make these adjustments, you will see a sidebar tracking your total expenditures by fund and your target\n baseline for the general fund. \n
\n Once you get through all funds and set your request at or below the \n target, you will be able to download a filled version of the Excel detail sheet. Downloading \n this file is the only permanent way to save your progress.`;\n }\n visit() {\n super.visit();\n\n // show welcome section\n components_welcome_welcome.show();\n\n // initialize links in buttons\n document.getElementById('step-upload').addEventListener('click', () => visitPage('upload'));\n // document.getElementById('step-initiatives').addEventListener('click', () => visitPage('new-inits'));\n // document.getElementById('step-revenue').addEventListener('click', () => visitPage('baseline-landing'));\n // document.getElementById('step-finish').addEventListener('click', () => visitPage('summary'));\n }\n}\n/* harmony default export */ const _00_welcome = (WelcomeView);\n;// CONCATENATED MODULE: ./src/js/views/01_upload.js\n\n\nclass UploadView extends View {\n constructor() {\n super();\n this.page_state = 'upload';\n this.prompt = `Upload the baseline detail sheet given by your budget analyst.`;\n this.subtitle = 'Excel Upload';\n this.sidebar = false;\n }\n visit() {\n super.visit();\n components_file_upload_file_upload.show();\n components_file_upload_file_upload.init();\n }\n}\n/* harmony default export */ const _01_upload = (UploadView);\n;// CONCATENATED MODULE: ./src/js/views/02_baseline_landing.js\n\n\n\n\nclass FundView extends View {\n constructor() {\n super();\n this.page_state = 'baseline-landing';\n this.prompt = `We will now ask you a series of questions about your BASELINE budget request.\n At the end, we will ask you about any new initiatives (ie. supplemental requests).\n Select one of your funds then click continue.`;\n this.subtitle = 'Baseline Budget Request';\n this.table = new FundTable();\n this.sidebar = false;\n this.dataTable = false;\n }\n visit() {\n // remove fund selection\n localStorage.setItem(\"fund\", '');\n super.visit();\n }\n}\nclass FundTable extends ViewTable {\n constructor() {\n super();\n\n // add additional revenue columns to the table\n this.columns = [{\n title: 'Fund',\n className: 'fund-name'\n }];\n this.noDataMessage = 'No funds found.';\n this.addEdit = false;\n }\n build() {\n // load in fund data\n components_table_table.show();\n components_table_table.Data.loadFunds();\n components_table_table.Columns.assignClasses(this.columns);\n components_table_table.adjustWidth('30%');\n\n // disable next button until a fund is selected\n components_nav_buttons_nav_buttons.Next.disable();\n allowRowSelection();\n }\n}\nfunction allowRowSelection() {\n var tableRows = document.querySelectorAll(\"tbody tr\");\n\n // enable highlight on hover and on select\n tableRows.forEach(function (row) {\n row.addEventListener('mouseover', function () {\n this.classList.add('hover-effect');\n });\n row.addEventListener('mouseout', function () {\n this.classList.remove('hover-effect');\n });\n row.addEventListener('click', function () {\n selectFund(tableRows, this);\n });\n });\n}\nfunction selectFund(tableRows, selected_row) {\n // remove selected class from any other rows\n tableRows.forEach(function (tableRow) {\n tableRow.classList = '';\n });\n\n // add selected class to clicked row\n selected_row.classList.add('selected');\n\n // get fund and save selected fund\n var fund = selected_row.querySelector('.fund-name').textContent;\n var fundNumber = parseInt(fund);\n current_fund.update(fundNumber);\n\n // enable next step\n components_nav_buttons_nav_buttons.Next.enable();\n}\n/* harmony default export */ const _02_baseline_landing = (FundView);\n;// CONCATENATED MODULE: ./src/js/views/03_revenue.js\n\n\nclass RevenueView extends View {\n constructor(fiscal_year) {\n super();\n this.page_state = 'revenue';\n this.prompt = `Review and edit revenue line items. If you change the estimate or \n notice an error in an account string, please note it in the notes column. Click edit \n to change values in a row.`;\n this.subtitle = 'Revenues';\n this.table = new RevenueTable(fiscal_year);\n }\n}\nclass RevenueTable extends ViewTable {\n constructor(fiscal_year) {\n super();\n\n // add additional revenue columns to the table\n this.columns = this.columns.concat([{\n title: 'Recurring or One-Time',\n className: 'recurring'\n }, {\n title: 'Object Category',\n className: 'object-category'\n }, {\n title: `FY${fiscal_year} Departmental Estimate`,\n className: 'request',\n isCost: true\n }, {\n title: 'Departmental Estimate Notes',\n className: 'notes'\n },\n // hidden\n {\n title: 'Object Name',\n className: 'object-name',\n hide: true\n }, {\n title: 'Object',\n className: 'object',\n hide: true\n }]);\n this.noDataMessage = 'No revenues for this fund.';\n }\n\n // action on row edit click: make cells editable\n actionOnEdit() {\n // only allow edits in the notes box\n components_table_table.Cell.createTextbox('notes', false, 'textarea');\n }\n}\n/* harmony default export */ const _03_revenue = (RevenueView);\n;// CONCATENATED MODULE: ./src/js/views/04_personnel.js\n\n\n\n\n\n\n\nclass PersonnelView extends View {\n constructor(fiscal_year) {\n super();\n this.page_state = 'personnel';\n this.prompt = `\n This table displays the number of FTEs in each job code for in your department's \n current (amended) FY25 budget. To make edits to the number of positions, click the\n \"Edit\" button on the row you would like to edit. The \"Total Cost\" column and the \n summary sidebar will also update to reflect any edits.`;\n this.subtitle = 'Personnel';\n this.table = new PersonnelTable(fiscal_year);\n }\n}\nclass PersonnelTable extends ViewTable {\n constructor(fiscal_year) {\n super();\n this.fiscal_year = fiscal_year;\n // add additional personnel columns to the table\n this.columns = this.columns.concat([{\n title: 'Job Title',\n className: 'job-name'\n }, {\n title: 'Job Code',\n className: 'job-code'\n }, {\n title: 'Employee Type',\n className: 'employee-type'\n }, {\n title: 'Service',\n className: 'service'\n }, {\n title: `FY${this.fiscal_year} Requested FTE`,\n className: 'baseline-ftes'\n }, {\n title: `FY${this.fiscal_year} Projected Average Salary/Wage`,\n className: 'avg-salary',\n isCost: true\n }, {\n title: 'Total Sal/Wag & Ben Request',\n className: 'total-baseline',\n isCost: true\n }, {\n title: 'Departmental Request Notes',\n className: 'notes'\n },\n // hidden columns\n {\n title: 'Fringe Benefits Rate',\n className: 'fringe',\n hide: true\n }, {\n title: 'General Increase Rate',\n className: 'general-increase-rate',\n hide: true\n }, {\n title: 'Step/Merit Increase Rate',\n className: 'merit-increase-rate',\n hide: true\n }, {\n title: `Average Salary/Wage as of 9/1/20${this.fiscal_year - 2}`,\n className: 'current-salary',\n isCost: true,\n hide: true\n }]);\n this.noDataMessage = 'No personnel expenditures for this fund.';\n this.addButtonText = 'Add new job';\n }\n\n // action on row edit click: make cells editable\n actionOnEdit() {\n components_table_table.Cell.createTextbox('baseline-ftes');\n components_table_table.Cell.createServiceDropdown(services.list());\n components_table_table.Cell.createTextbox('notes', false, 'textarea');\n }\n updateTable() {\n // calculate for each row\n let rows = document.getElementsByTagName('tr');\n for (let i = 1; i < rows.length; i++) {\n // fetch values for calculations\n let avg_salary = components_table_table.Cell.getValue(rows[i], 'avg-salary');\n let fringe = parseFloat(components_table_table.Cell.getText(rows[i], 'fringe'));\n let baseline_ftes = components_table_table.Cell.getText(rows[i], 'baseline-ftes');\n\n // calcuate #FTEs x average salary + COLA adjustments + merit adjustments + fringe\n let total_baseline_cost = avg_salary * baseline_ftes * (1 + fringe);\n\n // update total column\n components_table_table.Cell.updateValue(rows[i], 'total-baseline', total_baseline_cost);\n }\n\n // Save the table after all updates are done\n components_table_table.save();\n }\n addCustomQuestions() {\n // form questions to add a new job\n js_components_form_form.NewField.shortText('Job Code:', 'job-code', true);\n js_components_form_form.NewField.shortText('Job Title:', 'job-name', true);\n js_components_form_form.NewField.dropdown('Employee Type:', 'employee-type', EMPLOYEE_TYPES, true), js_components_form_form.NewField.dropdown('Appropriation:', 'approp-name', fund_lookup_table.getApprops('Add new'), true);\n js_components_form_form.NewField.dropdown('Cost Center:', 'cc-name', fund_lookup_table.getCostCenters('Add new'), true);\n js_components_form_form.NewField.dropdown('Service', 'service', services.list(), true);\n js_components_form_form.NewField.shortText('Number of FTEs requested:', 'baseline-ftes', true);\n js_components_form_form.NewField.shortText(`Projected average salary IN FISCAL YEAR ${this.fiscal_year}:`, 'avg-salary', true);\n //Form.NewField.shortText(`Expected fringe rate (as a percentage)`, 'fringe', true);\n }\n addModalValidation() {\n super.addModalValidation();\n\n // lock the job description \n const jobDescription = document.getElementById('job-name');\n jobDescription.readOnly = true;\n // give a message if attempt to edit\n const nameValidationText = document.getElementById('job-name-validation');\n jobDescription.addEventListener('click', function () {\n nameValidationText.textContent = 'This field is not editable and will auto-fill from the job code above.';\n });\n // hide message on blur\n jobDescription.addEventListener('blur', function () {\n nameValidationText.textContent = '';\n });\n\n // confirm that entered job code is in the gold book\n const jobCodeInput = document.getElementById('job-code');\n jobCodeInput.addEventListener('blur', function () {\n // get entered job code\n const jobCode = jobCodeInput.value;\n // get validation text element\n const validationText = document.getElementById('job-code-validation');\n if (!gold_book.codeExists(jobCode)) {\n // if the job code doesn't exist, show an error message\n validationText.textContent = 'This job code does not exist in the current Gold Book. Please enter another code.';\n // and clear any previous entry in the job title \n jobDescription.value = '';\n } else {\n // if it does exist, change the job title accordingly\n jobDescription.value = gold_book.getTitle(jobCode);\n // and clear any error messages\n validationText.textContent = '';\n }\n });\n }\n editColumns(responses) {\n responses = super.editColumns(responses);\n // edit inputs from modal\n responses['avg-salary'] = unformatCurrency(responses['avg-salary']);\n // use gold book to look up info based on job code\n responses['job-name'] = gold_book.getTitle(responses['job-code']);\n responses['fringe'] = gold_book.getFringeRate(responses['job-code']);\n return responses;\n }\n}\n/* harmony default export */ const _04_personnel = (PersonnelView);\n;// CONCATENATED MODULE: ./src/js/views/05_overtime.js\n\n\n\n\n\n\nclass OvertimeView extends View {\n constructor(fiscal_year) {\n super();\n this.page_state = 'overtime';\n this.prompt = `\n Please see your baseline overtime / holiday pay / shift premiums in the table below.\n Make any edits and continue.`;\n this.subtitle = 'Overtime Estimates';\n this.table = new OvertimeTable(fiscal_year);\n }\n}\nclass OvertimeTable extends ViewTable {\n constructor(fiscal_year) {\n super();\n\n // add additional OT columns to the table\n this.columns = this.columns.concat([{\n title: 'Service',\n className: 'service'\n }, {\n title: 'Recurring or One-Time',\n className: 'recurring'\n }, {\n title: 'Departmental Request OT/SP/Hol',\n className: 'OT-pay',\n isCost: true\n }, {\n title: `FY${fiscal_year} Total OT/SP/Hol + FICA Request`,\n className: 'total',\n isCost: true\n }, {\n title: 'Object Name',\n className: 'object-name'\n }, {\n title: 'Departmental Request Notes',\n className: 'notes'\n },\n // hidden columns\n {\n title: 'FICA Rate',\n className: 'fica',\n hide: true\n }]);\n this.noDataMessage = 'No overtime expenditures for this fund.';\n this.addButtonText = 'Add new cost center';\n }\n\n // action on row edit click: make cells editable\n actionOnEdit() {\n components_table_table.Cell.createTextbox('OT-pay', true);\n components_table_table.Cell.createServiceDropdown(services.list());\n components_table_table.Cell.createDropdown('recurring', ['One-Time', 'Recurring']);\n components_table_table.Cell.createDropdown('object-name', OT_OBJECTS);\n components_table_table.Cell.createTextbox('notes');\n }\n updateTable() {\n // calculate for each row\n let rows = document.getElementsByTagName('tr');\n for (let i = 1; i < rows.length; i++) {\n // fetch values for calculations\n let OT_salary = components_table_table.Cell.getValue(rows[i], 'OT-pay');\n let fica_rate = components_table_table.Cell.getText(rows[i], 'fica');\n\n // add salary and wages and fringe benefits (FICA)\n let row_total = OT_salary * (1 + parseFloat(fica_rate));\n\n // update total\n components_table_table.Cell.updateValue(rows[i], 'total', row_total);\n }\n\n // Save the table after all updates are done\n components_table_table.save();\n }\n addCustomQuestions() {\n // form questions to add a new job\n js_components_form_form.NewField.dropdown('Appropriation:', 'approp-name', fund_lookup_table.getApprops('Add new'), true);\n js_components_form_form.NewField.dropdown('Cost Center:', 'cc-name', fund_lookup_table.getCostCenters('Add new'), true);\n js_components_form_form.NewField.dropdown('Object (salary or wage):', 'object-name', OT_OBJECTS, true);\n js_components_form_form.NewField.dropdown('Service', 'service', services.list(), true);\n js_components_form_form.NewField.dropdown('Recurring or One-Time', 'recurring', ['Recurring', 'One-Time'], true);\n js_components_form_form.NewField.shortText('Overtime amount requested:', 'OT-pay', true);\n }\n editColumns(responses) {\n responses = super.editColumns(responses);\n // edit inputs from modal\n responses['OT-pay'] = unformatCurrency(responses['OT-pay']);\n responses['fica'] = 0.0765;\n return responses;\n }\n}\n/* harmony default export */ const _05_overtime = (OvertimeView);\n;// CONCATENATED MODULE: ./src/js/views/06_nonpersonnel.js\n\n\n\n\n\n\nclass NonPersonnelView extends View {\n constructor(fiscal_year) {\n super();\n this.page_state = 'nonpersonnel';\n this.prompt = `Review and edit non-personnel line items. The CPA numbers are the \n Contract and Procurement Account numbers. Click on the 'detail' link for a CPA \n to see the contract details. Some line items won't have a CPA number.`;\n this.subtitle = 'Non-Personnel';\n this.table = new NonPersonnelTable(fiscal_year);\n }\n}\nclass NonPersonnelTable extends ViewTable {\n constructor(fiscal_year) {\n super();\n\n // add additional personnel columns to the table\n this.columns = this.columns.concat([{\n title: `FY${fiscal_year} Departmental Request Total`,\n className: 'request',\n isCost: true\n }, {\n title: 'Service',\n className: 'service'\n }, {\n title: 'Recurring or One-Time',\n className: 'recurring'\n }, {\n title: 'Vendor Name',\n className: 'vendor'\n }, {\n title: 'CPA #',\n className: 'cpa'\n },\n // hidden columns\n {\n title: 'End Date',\n className: 'contract-end',\n hide: true\n }, {\n title: 'BPA/CPA Amount Remaining',\n className: 'remaining',\n isCost: true,\n hide: true\n }, {\n title: 'Object Name',\n className: 'object-name',\n hide: true\n }, {\n title: 'Object',\n className: 'object',\n hide: true\n }, {\n title: 'Object Category',\n className: 'object-category',\n hide: true\n }, {\n title: 'BPA/CPA Description',\n className: 'cpa-description',\n hide: true\n }, {\n title: 'Departmental Request Notes',\n className: 'notes'\n }]);\n this.noDataMessage = 'No non-personnel expenditures for this fund.';\n this.addButtonText = 'Add new non-personnel item';\n }\n\n // action on row edit click: make cells editable\n actionOnEdit() {\n components_table_table.Cell.createTextbox('request', true);\n components_table_table.Cell.createServiceDropdown();\n components_table_table.Cell.createDropdown('recurring', ['One-Time', 'Recurring']);\n components_table_table.Cell.createTextbox('notes');\n }\n addCustomQuestions() {\n // form questions to add a new row\n js_components_form_form.NewField.dropdown('Appropriation:', 'approp-name', fund_lookup_table.getApprops('Add new'), true);\n js_components_form_form.NewField.dropdown('Cost Center:', 'cc-name', fund_lookup_table.getCostCenters('Add new'), true);\n js_components_form_form.NewField.dropdown('Object Category:', 'object-category', OBJ_CATEGORIES.list, true);\n // TODO: maybe give dropdown based on selected obj category\n js_components_form_form.NewField.shortText('Object Number (if known):', 'object', false);\n js_components_form_form.NewField.dropdown('Service', 'service', services.list(), true);\n js_components_form_form.NewField.longText('Describe your new request:', 'cpa-description', true);\n js_components_form_form.NewField.dropdown('Recurring or One-Time', 'recurring', ['Recurring', 'One-Time'], true);\n js_components_form_form.NewField.shortText('Amount requested:', 'request', true);\n }\n editColumns(responses) {\n responses = super.editColumns(responses);\n responses['avg-salary'] = unformatCurrency(responses['avg-salary']);\n responses['fringe'] = parseFloat(responses['fringe']) / 100;\n return responses;\n }\n}\n/* harmony default export */ const _06_nonpersonnel = (NonPersonnelView);\n;// CONCATENATED MODULE: ./src/js/views/07_new_initiatives.js\n\n\n\n\n\n\nconst dropdownOptions = ['N/A', 'One-Time', 'Recurring'];\n\n// set up page and initialize all buttons\nclass InitiativesView extends View {\n constructor() {\n super();\n this.page_state = 'new-inits';\n this.prompt = `\n This is the place to propose new initiatives for FY${FISCAL_YEAR}.\n New initiative submissions will count as supplemental line items and will be the starting \n point for a conversation with both OB and ODFS, who will help with the details.`;\n this.subtitle = 'New Initiatives';\n this.table = new InitiativesTable();\n }\n visit() {\n // remove fund selection\n localStorage.setItem(\"fund\", '');\n super.visit();\n }\n}\nclass InitiativesTable extends ViewTable {\n constructor() {\n super();\n\n // add additional columns to the table\n this.columns = [{\n title: 'Fund',\n className: 'fund'\n }, {\n title: 'Supplemental Initiative',\n className: 'init-name'\n }, {\n title: 'Total Initiative Request',\n className: 'total',\n isCost: true\n },\n // { title: 'Personnel FTE', className: 'ftes'},\n {\n title: 'Personnel Salary & Benefits',\n className: 'personnel',\n isCost: true\n }, {\n title: 'Non-Personnel Operating',\n className: 'nonpersonnel',\n isCost: true\n }, {\n title: 'Non-Personnel Capital',\n className: 'nonpersonnel-capital',\n isCost: true\n },\n // { title: 'Revenue', className: 'revenue', isCost: true },\n // { title: 'Revenue Type', className: 'rev-type' },\n {\n title: 'Description & Justification',\n className: 'notes'\n }, {\n title: 'Recurring or One-Time',\n className: 'rev-type'\n }, {\n title: 'Edit',\n className: 'edit'\n }];\n this.addButtonText = 'Add new initiative';\n }\n addCustomQuestions() {\n // general questions\n js_components_form_form.NewField.shortText('Initiative Name:', 'init-name', true);\n js_components_form_form.NewField.longText('What is the business case for the Initiative?', 'q1', true);\n js_components_form_form.NewField.longText(`Why is the initiative needed? What is the value-add to residents? \n What is the Department’s plan for implementing the Initiative?`, 'q2', true);\n js_components_form_form.NewField.longText(`Why can’t the Initiative be funded with the Department’s baseline budget?`, 'q3', true);\n js_components_form_form.NewField.dropdown(`Is this initiative one-time or recurring?`, 'rev-type', dropdownOptions);\n\n // Account string info\n js_components_form_form.NewField.dropdown('Fund:', 'fund-name', fund_lookup_table.listFundNames(), true);\n\n // Numbers\n js_components_form_form.NewField.numericInput('What is your ballpark estimate of TOTAL ADDITONAL expenses associated with this initiative?', 'total', false);\n js_components_form_form.NewField.numericInput('Estimate of ADDITONAL personnel cost?', 'personnel', false);\n js_components_form_form.NewField.numericInput('Estimate of ADDITONAL nonpersonnel operating cost?', 'nonpersonnel', false);\n js_components_form_form.NewField.numericInput('Estimate of ADDITONAL nonpersonnel capital costs?', 'nonpersonnel-capital', false);\n js_components_form_form.NewField.numericInput('Estimate of ADDITONAL revenue (if applicable)?', 'revenue', false);\n }\n editColumns(responses) {\n responses['fund'] = account_string.getNumber(responses['fund-name']);\n responses['notes'] = `${responses['q1']} ${responses['q2']} ${responses['q3']}`;\n return responses;\n }\n addModalValidation() {\n return;\n }\n\n // action on row edit click: make cells editable\n actionOnEdit() {\n components_table_table.Cell.createTextbox('total', true);\n // Table.Cell.createTextbox('revenue', true);\n components_table_table.Cell.createTextbox('personnel', true);\n components_table_table.Cell.createTextbox('nonpersonnel', true);\n components_table_table.Cell.createTextbox('init-name');\n components_table_table.Cell.createDropdown('rev-type', dropdownOptions);\n components_table_table.Cell.createTextbox('notes');\n }\n}\n/* harmony default export */ const _07_new_initiatives = (InitiativesView);\n;// CONCATENATED MODULE: ./src/js/views/08_summary.js\n\n\n\n\n\n\n\n\nfunction compareToTarget() {\n const baseline = new models_baseline();\n if (baseline.genFundTotal() <= models_baseline.target()) {\n js_components_prompt_prompt.Text.update(`Congrats! Your General Fund budget is below your target! \n Edit any line items below or download your completed Excel.`);\n } else {\n js_components_prompt_prompt.Text.update(`Your General Fund budget is above your GF target of ${formatCurrency(models_baseline.target())}. \n Please expand the summary table below and edit line items until you meet your target. \n When you meet the target, you will be able to download the Excel sheet.`);\n js_components_prompt_prompt.Buttons.Right.disable();\n }\n js_components_prompt_prompt.show();\n}\nfunction returnToWelcome() {\n const welcome = new _00_welcome();\n const left = document.getElementById('option1');\n welcome.visit();\n}\n;\nclass SummaryView extends View {\n constructor() {\n super();\n this.page_state = 'summary';\n this.subtitle = 'Summary';\n this.sidebar = false;\n }\n visit() {\n super.visit();\n\n // reset fund\n current_fund.reset();\n\n // show summary accordion\n Accordion.build();\n Accordion.show();\n\n // add prompt buttons\n js_components_prompt_prompt.Buttons.Right.updateText('Download Excel');\n js_components_prompt_prompt.Buttons.Left.updateText('Start over with new Excel upload');\n // add button links\n js_components_prompt_prompt.Buttons.Left.addAction(returnToWelcome);\n js_components_prompt_prompt.Buttons.Right.addAction(downloadXLSX);\n\n // update prompt text depending on target matching\n compareToTarget();\n }\n cleanup() {\n // delete event listeners\n js_components_prompt_prompt.Buttons.Left.removeAction(returnToWelcome);\n js_components_prompt_prompt.Buttons.Right.removeAction(downloadXLSX);\n js_components_prompt_prompt.Buttons.Right.enable();\n }\n}\n/* harmony default export */ const _08_summary = (SummaryView);\n;// CONCATENATED MODULE: ./src/js/views/view_logic.js\n\n\n\n\n\n\n\n\n\n\n\n\n// Initialize pages globally once\nconst PAGES = {\n 'welcome': new _00_welcome(),\n 'upload': new _01_upload(),\n 'baseline-landing': new _02_baseline_landing(),\n 'revenue': new _03_revenue(FISCAL_YEAR),\n 'personnel': new _04_personnel(FISCAL_YEAR),\n 'overtime': new _05_overtime(FISCAL_YEAR),\n 'nonpersonnel': new _06_nonpersonnel(FISCAL_YEAR),\n 'new-inits': new _07_new_initiatives(),\n 'summary': new _08_summary()\n};\nfunction visitPage(new_page_key) {\n var page_state = current_page.load();\n\n // Perform cleanup from the current page\n if (PAGES[page_state]) {\n PAGES[page_state].cleanup();\n }\n if (PAGES[new_page_key]) {\n PAGES[new_page_key].visit();\n } else {\n console.error(`No page initializer found for state: ${new_page_key}`);\n }\n}\nfunction nextPage() {\n var page_state = current_page.load();\n const keys = Object.keys(PAGES);\n const currentIndex = keys.indexOf(page_state);\n const returnPages = ['revenue', 'nonpersonnel', 'new-inits', 'overtime', 'personnel'];\n if (!fund_lookup_table.fundsLeft() && returnPages.includes(current_page.load())) {\n visitPage('summary');\n return;\n }\n if (current_page.load() == 'nonpersonnel') {\n fund_lookup_table.editFund(current_fund.number());\n if (fund_lookup_table.fundsLeft()) {\n visitPage('baseline-landing');\n return;\n }\n }\n if (currentIndex >= 0 && currentIndex < keys.length - 1) {\n const nextKey = keys[currentIndex + 1];\n visitPage(nextKey);\n }\n}\nfunction lastPage() {\n var page_state = current_page.load();\n const keys = Object.keys(PAGES);\n const currentIndex = keys.indexOf(page_state);\n if (current_page.load() == 'new-inits') {\n visitPage('baseline-landing');\n return;\n }\n if (currentIndex >= 1) {\n const lastKey = keys[currentIndex - 1];\n visitPage(lastKey);\n }\n}\n;// CONCATENATED MODULE: ./src/js/models/current_page.js\n\nconst CurrentPage = {\n update: function (page) {\n localStorage.setItem('page_state', page);\n },\n load: function () {\n const pageState = localStorage.getItem('page_state');\n return pageState !== null ? pageState : 'welcome';\n },\n visit: function () {\n visitPage(this.load());\n }\n};\n/* harmony default export */ const current_page = (CurrentPage);\n;// CONCATENATED MODULE: ./src/js/utils/storage_utils.js\nfunction getLocalStorageSize() {\n let total = 0;\n for (let key in localStorage) {\n if (localStorage.hasOwnProperty(key)) {\n let keySize = key.length * 2; // each character is 2 bytes\n let valueSize = localStorage[key].length * 2; // each character is 2 bytes\n total += keySize + valueSize;\n }\n }\n return formatBytes(total);\n}\nfunction formatBytes(bytes, decimals = 2) {\n if (bytes === 0) return '0 Bytes';\n const k = 1024;\n const dm = decimals < 0 ? 0 : decimals;\n const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB'];\n const i = Math.floor(Math.log(bytes) / Math.log(k));\n return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i];\n}\n;// CONCATENATED MODULE: ./src/js/init.js\n// Import styles\n\n\n// Import functions\n\n\n\n// Initialize only once when the document is fully loaded\ndocument.addEventListener('DOMContentLoaded', function () {\n current_page.visit();\n console.log('Local Storage Used:', getLocalStorageSize());\n});//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"863.js","mappings":";;;;;;;;;;;;;;;;;;;;;;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxBA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;;ACfA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AAQA;;AAEA;AACA;AAWA;;;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxBA;AACA;AACA;AAEA;AACA;AACA;AAGA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAAA;AAAA;AACA;AAEA;;ACpBA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAEA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;AC9EA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxBA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7BA;;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;;ACzEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;AC/HA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;AClBA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;;ACzCA;AACA;;AAEA;AACA;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;;AClNA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAAA;AAAA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;;AC1DA;;AAEA;AACA;AAEA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;;AC/DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;ACTA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AAAA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AAEA;;AC3CA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACTA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1GA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;;ACnDA;AACA;AAEA;;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;;;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxBA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;;;ACrDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxBA;AAEA;AACA;;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;;AC7BA;AACA;AAEA;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;AC7EA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;ACvGA;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AAGA;AACA;AAAA;AAAA;AACA;AAEA;;ACzBA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AACA;AACA;AAEA;;AC3EA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AAEA;AAEA;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;ACrMA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAGA;AACA;AACA;AACA;AAAA;AAAA;AACA;AAEA;;AC/IA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;ACvGA;;AAEA;AACA;AACA;AACA;AAAA;AAAA;AAAA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAAA;AAAA;AAAA;AACA;AAEA;AACA;AAAA;AAAA;AAAA;;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AAGA;AACA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAGA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAAA;AAAA;AAAA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AAEA;;AC5EA;AACA;AACA;AACA;AACA;AACA;;AAGA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AAAA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AACA;AACA;;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AAAA;AAAA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAAA;AAAA;AAAA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;;;AC1RA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxBA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAAA;AAAA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AAEA;AACA;AAEA;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;AC7MA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtCA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;AC7EA;AAGA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAGA;;;;ACrOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxBA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;;;;AC5DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxBA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AAEA;AACA;;AAEA;AACA;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;ACnFA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;;;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxBA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxBA;;AAEA;;AAEA;AACA;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;AC5DA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AAEA;AACA;AAAA;AAAA;AACA;AAEA;;ACzBA;AAEA;AACA;AACA;AAEA;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AAEA;AAEA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AACA;AACA;AACA;AAEA;;ACtDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAGA;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AAAA;AAAA;AAAA;;AAEA;AACA;AAAA;AAAA;AAAA;;AAEA;AACA;AAAA;AAAA;AAAA;;AAEA;AACA;AAAA;AAAA;;AAEA;AACA;AAAA;AAAA;AAAA;AACA;AAEA;AAAA;AAAA;AAEA;AAEA;AAEA;AACA;AACA;AAEA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;;AAGA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AAEA;AAEA;AACA;;AAEA;AACA;AAAA;AAAA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AAEA;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;;AAEA;AACA;AAAA;AAAA;;AAEA;AACA;AAAA;AAAA;;AAEA;AACA;AAAA;AAAA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;;AAEA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AAKA;AACA;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AAEA;;AC9SA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AAEA;AAEA;AAEA;;AC5CA;AAEA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;;ACrBA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;;AAEA;AACA;AACA;AAAA;AAAA;AAGA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AAGA;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AAEA;;AC3FA;AAEA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;;AAEA;AACA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AAGA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;;AC3CA;AAEA;AACA;AAEA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAGA;AAEA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AAAA;AAGA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;AC5IA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;;AAEA;AACA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AAGA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;AC3FA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;;AAEA;AACA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAGA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;AC1EA;AACA;AACA;AACA;AACA;AACA;AAGA;;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AAEA;AAEA;AACA;;AAEA;AACA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AACA;AAAA;AAAA;AAGA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;;AAGA;AACA;;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AAAA;AAAA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAAA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;;AChFA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;;ACfA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrBA;AACA;;AAEA;AACA;AAEA;;AAEA;AACA;AACA;AACA;AACA","sources":["webpack://budget-request-form/./src/css/common.css?8ff6","webpack://budget-request-form/./src/js/components/welcome/welcome.css?dee1","webpack://budget-request-form/./src/js/components/welcome/welcome.js?535a","webpack://budget-request-form/./src/js/constants/budget_constants.js?1502","webpack://budget-request-form/./src/js/components/prompt/prompt.css?328b","webpack://budget-request-form/./src/js/components/prompt/subcomponents/text.js?f6df","webpack://budget-request-form/./src/js/components/prompt/subcomponents/buttons.js?8d3b","webpack://budget-request-form/./src/js/components/prompt/prompt.js?a771","webpack://budget-request-form/./src/js/components/sidebar/sidebar.css?3bc7","webpack://budget-request-form/./src/js/constants/excel_constants.js?f687","webpack://budget-request-form/./src/js/constants/index.js?25c4","webpack://budget-request-form/./src/js/utils/common_utils.js?44f2","webpack://budget-request-form/./src/js/models/fund_lookup_table.js?dc66","webpack://budget-request-form/./src/js/models/current_fund.js?e6db","webpack://budget-request-form/./src/js/models/account_string.js?6c8d","webpack://budget-request-form/./src/js/models/fund.js?6cfa","webpack://budget-request-form/./src/js/models/baseline.js?d494","webpack://budget-request-form/./src/js/models/initiative.js?f96a","webpack://budget-request-form/./src/js/models/services.js?139f","webpack://budget-request-form/./src/js/models/supplemental.js?99ce","webpack://budget-request-form/./src/js/models/index.js?05e1","webpack://budget-request-form/./src/js/components/sidebar/subcomponents/baseline_section.js?13bd","webpack://budget-request-form/./src/js/components/sidebar/subcomponents/supp_section.js?0c4e","webpack://budget-request-form/./src/js/components/sidebar/sidebar.js?80f0","webpack://budget-request-form/./src/js/components/nav_buttons/nav_buttons.css?707f","webpack://budget-request-form/./src/js/components/nav_buttons/nav_buttons.js?abd1","webpack://budget-request-form/./src/js/components/body/body.css?d1c7","webpack://budget-request-form/./src/js/components/accordion/accordion.css?5a6c","webpack://budget-request-form/./src/js/components/table/table.css?784c","webpack://budget-request-form/./src/js/components/table/subcomponents/headers.js?1623","webpack://budget-request-form/./src/js/components/table/subcomponents/rows.js?fb79","webpack://budget-request-form/./src/js/components/table/subcomponents/buttons.js?c992","webpack://budget-request-form/./src/js/components/form/subcomponents/dropdown.js?ba6b","webpack://budget-request-form/./src/js/components/table/subcomponents/cells.js?73cb","webpack://budget-request-form/./src/js/components/table/subcomponents/columns.js?674a","webpack://budget-request-form/./src/js/components/table/subcomponents/data.js?d9b6","webpack://budget-request-form/./src/js/components/table/subcomponents/filters.js?aecc","webpack://budget-request-form/./src/js/models/gold_book.js?66c5","webpack://budget-request-form/./src/js/utils/XLSX_handlers.js?ba8a","webpack://budget-request-form/./src/js/components/tooltip/tooltip.css?bc8c","webpack://budget-request-form/./src/js/components/tooltip/tooltip.js?d23c","webpack://budget-request-form/./src/js/utils/JSON_data_handlers.js?c9d2","webpack://budget-request-form/./src/js/components/table/table.js?3071","webpack://budget-request-form/./src/js/components/accordion/accordion.js?1f37","webpack://budget-request-form/./src/js/components/file_upload/file_upload.css?db09","webpack://budget-request-form/./src/js/components/file_upload/file_upload.js?5188","webpack://budget-request-form/./src/js/components/modal/modal.css?40db","webpack://budget-request-form/./src/js/components/modal/modal.js?5533","webpack://budget-request-form/./src/js/components/body/body.js?f9ba","webpack://budget-request-form/./src/js/components/header/header.css?72d7","webpack://budget-request-form/./src/js/components/header/header.js?a0a5","webpack://budget-request-form/./src/js/components/form/form.css?5415","webpack://budget-request-form/./src/js/components/form/subcomponents/fields.js?459f","webpack://budget-request-form/./src/js/components/form/subcomponents/submit.js?c596","webpack://budget-request-form/./src/js/components/form/form.js?d942","webpack://budget-request-form/./src/js/views/view_class.js?a79e","webpack://budget-request-form/./src/js/views/00_welcome.js?08bb","webpack://budget-request-form/./src/js/views/01_upload.js?c448","webpack://budget-request-form/./src/js/views/02_baseline_landing.js?7b8e","webpack://budget-request-form/./src/js/views/03_revenue.js?b878","webpack://budget-request-form/./src/js/views/04_personnel.js?fbff","webpack://budget-request-form/./src/js/views/05_overtime.js?bf70","webpack://budget-request-form/./src/js/views/06_nonpersonnel.js?4d3d","webpack://budget-request-form/./src/js/views/07_new_initiatives.js?41a7","webpack://budget-request-form/./src/js/views/08_summary.js?8b32","webpack://budget-request-form/./src/js/views/view_logic.js?46c6","webpack://budget-request-form/./src/js/models/current_page.js?ffb0","webpack://budget-request-form/./src/js/utils/storage_utils.js?fdf4","webpack://budget-request-form/./src/js/init.js?1b2a"],"sourcesContent":["\n      import API from \"!../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../node_modules/css-loader/dist/cjs.js!./common.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../node_modules/css-loader/dist/cjs.js!./common.css\";\n       export default content && content.locals ? content.locals : undefined;\n","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./welcome.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./welcome.css\";\n       export default content && content.locals ? content.locals : undefined;\n","import './welcome.css';\r\n\r\n// Hide and unhide welcome buttons\r\nfunction unhideWelcomeButtons(){\r\n    document.getElementById(\"welcome-page\").style.display = \"block\";\r\n}\r\nfunction hideWelcomeButtons(){\r\n    document.getElementById(\"welcome-page\").style.display = \"none\";\r\n}\r\n\r\nexport const Welcome = {\r\n    show: unhideWelcomeButtons,\r\n    hide : hideWelcomeButtons\r\n}\r\n\r\nexport default Welcome;","// Set to equal current fiscal year\r\nexport var FISCAL_YEAR = '26';\r\n\r\n// object categories (from obj part of account string)\r\nexport const OBJ_CATEGORIES = {\r\n    list : [\r\n        // 'Salaries & Wages',\r\n        // 'Employee Benefits',\r\n        'Professional & Contractual Services',\r\n        'Operating Supplies',\r\n        'Operating Services',\r\n        'Equipment Acquisition',\r\n        'Capital Outlays',\r\n        'Fixed Charges',\r\n        'Other Expenses'\r\n    ]\r\n}\r\n\r\n// from the drop-down menu\r\nexport const EMPLOYEE_TYPES = [\r\n    'Regular',\r\n    'TASS',\r\n    'Seasonal',\r\n    'Uniform Fire',\r\n    'Uniform Police',\r\n    'Appointed',\r\n    'Elected',\r\n    'Long Term Disability'\r\n]\r\n\r\nexport const OT_OBJECTS = [\r\n    '601300 - Salar-Overtime-Gen City', \r\n    '601305 - Salaries-Overtime-Police Unif',\r\n    '601310 - Salaries-Overtime-Fire Unif',\r\n    '602300 - Wages-Overtime-Gen City'\r\n]","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./prompt.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./prompt.css\";\n       export default content && content.locals ? content.locals : undefined;\n","function showPrompt(){\r\n    document.getElementById(\"prompt-div\").style.display = \"block\";\r\n}\r\n\r\nfunction hidePrompt(){\r\n    document.getElementById('prompt-div').style.display = 'none';\r\n}\r\n\r\n\r\nfunction updatePrompt(prompt){\r\n    document.getElementById('prompt').innerHTML = prompt;\r\n    showPrompt();\r\n}\r\n\r\nexport const Text = {\r\n    show : showPrompt,\r\n    hide : hidePrompt,\r\n    update : function(text) { updatePrompt(text) }\r\n}\r\n\r\nexport default Text;","function showPromptButton(id){\r\n    // make buttons visible\r\n    document.getElementById(id).style.display = 'inline';\r\n}\r\n\r\nfunction updatePromptButton(id, text){\r\n    document.getElementById(id).textContent = text;\r\n    showPromptButton(id);\r\n}\r\n\r\nfunction hidePromptButton(id){\r\n    document.getElementById(id).style.display = 'none';\r\n}\r\n\r\nfunction unclickAll(){\r\n    document.getElementById('option1').classList.remove('clicked');\r\n    document.getElementById('option2').classList.remove('clicked');\r\n}\r\n\r\nfunction applyClickedStyle(button){\r\n    unclickAll();\r\n    button.classList.add('clicked');\r\n}\r\n\r\nfunction addPromptButtonAction(button_id, action_fn){\r\n    const buttonElement = document.getElementById(button_id);\r\n    buttonElement.addEventListener('click', action_fn);\r\n    buttonElement.addEventListener('click', function(){\r\n        applyClickedStyle(this);\r\n    });\r\n}\r\n\r\nfunction removePromptButtonAction(button_id, action_fn){\r\n    document.getElementById(button_id).removeEventListener('click', action_fn);\r\n}\r\n\r\nfunction disable(button_id){\r\n    document.querySelector(`#${button_id}`).classList.add('disabled');\r\n}\r\n\r\nfunction enable(button_id){\r\n    document.querySelector(`#${button_id}`).classList.remove('disabled');\r\n}\r\n\r\nexport const Left = {\r\n    show : function() { showPromptButton('option1') },\r\n    hide : function() { hidePromptButton('option1') },\r\n    updateText : function(text) { updatePromptButton('option1', text) },\r\n    addAction : function(action_fn) { addPromptButtonAction('option1', action_fn) },\r\n    removeAction : function(action_fn) { removePromptButtonAction('option1', action_fn) },\r\n    disable : function() { disable('option1') },\r\n    enable : function() { enable('option1') }\r\n}\r\n\r\nexport const Right = {\r\n    show : function() { showPromptButton('option2') },\r\n    hide : function() { hidePromptButton('option2') },\r\n    updateText : function(text) { updatePromptButton('option2', text) },\r\n    addAction : function(action_fn) { addPromptButtonAction('option2', action_fn) },\r\n    removeAction : function(action_fn) { removePromptButtonAction('option2', action_fn) },\r\n    disable : function() { disable('option2') },\r\n    enable : function() { enable('option2') }\r\n}\r\n\r\nexport const Buttons = {\r\n    Left : Left,\r\n    Right : Right,\r\n    show : function() {\r\n        showPromptButton('option1');\r\n        showPromptButton('option2');\r\n    },\r\n    hide : function() {\r\n        hidePromptButton('option1');\r\n        hidePromptButton('option2');\r\n    },\r\n    reset : unclickAll\r\n}\r\n\r\nexport default Buttons;","import './prompt.css';\r\n\r\nimport Text from \"./subcomponents/text.js\";\r\nimport Buttons from \"./subcomponents/buttons.js\";\r\n\r\nexport const Prompt = {\r\n    Text : Text,\r\n    Buttons : Buttons,\r\n    hide : function(){\r\n        Text.hide();\r\n        Buttons.hide();\r\n    },\r\n    show : function(){\r\n        Text.show();\r\n        Buttons.show();\r\n    }\r\n}\r\n\r\nexport default Prompt","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./sidebar.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./sidebar.css\";\n       export default content && content.locals ? content.locals : undefined;\n","import { FISCAL_YEAR } from \"./budget_constants\";\r\n\r\n// sheets to expect on detail sheet\r\nexport const SHEETS = {\r\n    'FTE, Salary-Wage, & Benefits' : 'personnel' ,\r\n    'Overtime & Other Personnel' : 'overtime',\r\n    'Non-Personnel' : 'nonpersonnel',\r\n    'Revenue' : 'revenue',\r\n    'Initiatives Summary': 'new-inits'\r\n}\r\n\r\n// where to find the general fund target\r\nexport const TARGET_CELL_ADDRESS = 'C14'\r\n\r\nexport const TOTAL_COLUMNS = {\r\n    'personnel': 'Total Sal/Wag & Ben Request',\r\n    'overtime':`FY${FISCAL_YEAR} Total OT/SP/Hol + FICA Request`,\r\n    'nonpersonnel': `FY${FISCAL_YEAR} Departmental Request Total`,\r\n    'revenue': `FY${FISCAL_YEAR} Departmental Estimate`,\r\n    'new-inits': `Total Initiative Request`\r\n};\r\n\r\nexport const NEW_INIT_COLS = {\r\n    'personnel' : 'Personnel Salary & Benefits',\r\n    'operating' : 'Non-Personnel Operating',\r\n    'capital' : 'Non-Personnel Capital',\r\n    'revenue' : 'Revenue',\r\n    'name' : 'Supplemental Initiative',\r\n    'total' : TOTAL_COLUMNS['new-inits']\r\n}","export * from './budget_constants';\r\nexport * from './excel_constants';","// Function to format number as currency\r\nexport const formatCurrency = (amount, return_zero = false) => {\r\n    var amount = Math.round(parseFloat(amount));\r\n    if (amount == NaN){\r\n        return \"$ -\"\r\n    }\r\n    if (amount < 0){\r\n        return '($' + amount.toString().replace(/\\B(?=(\\d{3})+(?!\\d))/g, ',') + ')';\r\n    } else if (amount == 0) {\r\n        if (return_zero){\r\n            return '$0';    \r\n        }\r\n        return \"$ -\"\r\n    }\r\n    return '$' + amount.toString().replace(/\\B(?=(\\d{3})+(?!\\d))/g, ',');\r\n} ;\r\n\r\n// function to convert formatted number to a float\r\nexport const unformatCurrency = (formattedAmount) => {\r\n    if (!formattedAmount) { return 0 };\r\n    // Remove any currency symbols and commas\r\n    let numericalPart = formattedAmount.replace(/[^0-9.-]+/g, \"\");\r\n    if (numericalPart == '-'){\r\n        return 0;\r\n    }\r\n    return parseFloat(numericalPart);\r\n};\r\n\r\nexport function displayWithCommas(value) {\r\n    if (value == 0){\r\n        return 0;\r\n    }\r\n    return formatCurrency(value).replace('$', '');\r\n}\r\n\r\nexport function cleanString(str){\r\n    return str.toLowerCase().replaceAll(' ', '-');\r\n}\r\n\r\nexport function removeNewLines(str){\r\n    // replace all new lines with spaces\r\n    str = str.replaceAll(/[\\r\\n]+/g, \" \");\r\n    // remove any extra spaces or trailing/leading whitespace\r\n    str = str.replaceAll('  ', ' ');\r\n    str = str.replaceAll('  ', ' ');\r\n    str = str.replace(/^\\s+|\\s+$/g, '');\r\n    return str;\r\n}\r\n\r\nexport function colSum(table, colName) {\r\n    // fill with zero until there is something saved in storage\r\n    if(!table || table == ''){ \r\n        return 0; \r\n    }\r\n    const headers = Object.keys(table[0]);\r\n    if (headers.includes(colName)) {\r\n        let sum = 0;\r\n        for (let i = 0; i < table.length; i++){\r\n            var value = Math.round(parseFloat(table[i][colName]));\r\n            // treat NaN (non-numerics) as zeroes\r\n            if (value) { sum += value; }\r\n        }\r\n        return sum;\r\n    } else {\r\n        // console.error(`Could not find expected total column in saved data for ${name}. Returning 0. See StoredTable.totalCol() switch.`);\r\n        return 0;\r\n    }\r\n\r\n}\r\n\r\nexport function getUniqueValues(data, key) {\r\n    const values = data.map(obj => obj[key]);\r\n    return Array.from(new Set(values));\r\n}\r\n","import CurrentFund from \"./current_fund\";\r\nimport { getUniqueValues } from \"../utils/common_utils\";\r\n\r\nexport const FundLookupTable = {\r\n    retrieve : function() {\r\n        return JSON.parse(localStorage.getItem('fund-lookup-table')) || {};\r\n    },\r\n    save : function(fundDict){\r\n        localStorage.setItem('fund-lookup-table', JSON.stringify(fundDict));\r\n    },\r\n\r\n    update : function(fundData){\r\n        // reset\r\n        this.reset();\r\n        // fill in with data stored in fundDate\r\n        const table = this.retrieve();\r\n        for (let fund of Object.keys(fundData)){\r\n\r\n            // add to lookup table if not in there already\r\n            if (!table[fund]){\r\n                // get fund name\r\n                const fundName = fundData[fund][0]['Fund Name'];\r\n                // add fund to dictionary\r\n                table[fund] = {};\r\n                table[fund]['name'] = fundName;\r\n                table[fund]['viewed'] = false;\r\n                // build lists of unique cost centers and appropriations\r\n                table[fund]['approp'] = getUniqueValues(fundData[fund], 'Appropriation Name');\r\n                table[fund]['cc'] = getUniqueValues(fundData[fund], 'Cost Center Name');\r\n            }\r\n        }\r\n        // save any updates\r\n        this.save(table);\r\n    },\r\n\r\n    getAll: function(key) {\r\n        // function to aggregate all approps or CCs for every fund in one array\r\n        const funds = this.retrieve();\r\n        const ret = [];\r\n        for (const fund in funds) {\r\n            if (funds.hasOwnProperty(fund)) {\r\n                for (let i in funds[fund][key]){\r\n                    ret.push(funds[fund][key][i]);\r\n                }\r\n            }\r\n        }\r\n        return ret;\r\n    },\r\n\r\n    getCostCenters : function( extraOption ) {\r\n        // get current fund\r\n        const fund = CurrentFund.number()\r\n        let ret = [];\r\n        if (this.retrieve()[fund]){\r\n            ret = this.retrieve()[fund]['cc'];\r\n        } else {\r\n            // if no fund (ie. we're on the new initiative page), return all options\r\n            ret = this.getAll('cc');\r\n        };\r\n        // add extra option if given\r\n        if (extraOption) {\r\n            ret.push(extraOption);\r\n        }\r\n        return ret;\r\n    },\r\n\r\n    getApprops : function( extraOption ) {\r\n        // get current fund\r\n        const fund = CurrentFund.number()\r\n        let ret = [];\r\n        if (this.retrieve()[fund]){\r\n            ret = this.retrieve()[fund]['approp'];\r\n        } else {\r\n            // if no fund (ie. we're on the new initiative page), return all options\r\n            ret = this.getAll('approp');\r\n        };\r\n        // add extra option if given\r\n        if (extraOption) {\r\n            ret.push(extraOption);\r\n        }\r\n        return ret;\r\n    },\r\n\r\n    reset : function() {\r\n        this.save({});\r\n    },\r\n    getName : function(number){\r\n        if(!number || !this.retrieve()) { return '' };\r\n        return this.retrieve()[number]['name'];\r\n    },\r\n    listFunds : function(){\r\n        return Object.keys(this.retrieve());\r\n    },\r\n    listFundNames : function(){\r\n        const funds = this.retrieve();\r\n        // initialize array\r\n        var ret = [];\r\n        Object.keys(funds).forEach( (fund_number) => {\r\n            var fund_name = funds[fund_number]['name'];\r\n            ret.push(fund_name);\r\n        });\r\n        return ret;\r\n    },\r\n    editFund : function(fund){\r\n        const table = this.retrieve();\r\n        if (table[fund]){\r\n            table[fund]['viewed'] = true;\r\n            this.save(table);\r\n        } else {\r\n            console.error('No fund selected.');\r\n        }\r\n        \r\n    },\r\n    listUneditedFunds : function(){\r\n        const table = this.retrieve();\r\n        const ret = [];\r\n        this.listFunds().forEach(key => {\r\n            if (!table[key]['viewed']){\r\n                ret.push(key);\r\n            }\r\n        });\r\n        return ret;\r\n    },\r\n    fundsLeft : function(){\r\n        return (this.listUneditedFunds().length > 0);\r\n    }\r\n}\r\n\r\nexport default FundLookupTable","\r\nimport FundLookupTable from \"./fund_lookup_table\";\r\n\r\nexport const CurrentFund = {\r\n    update : function(fund){\r\n        localStorage.setItem('fund', fund);\r\n    },\r\n    number : function(){\r\n        return localStorage.getItem(\"fund\");\r\n    },\r\n    name : function(){\r\n        return FundLookupTable.getName( this.number());\r\n    },\r\n    reset : function() {\r\n        this.update('');\r\n    }\r\n}\r\n\r\nexport default CurrentFund;","import CurrentFund from \"./current_fund\";\r\n\r\nexport const AccountString = {\r\n    getNumber: function(input) {\r\n        // isolate the numerical part of a appropriation/cost center/object\r\n        const match = input.match(/^\\d+/);\r\n        return match ? match[0] : null;\r\n    },\r\n\r\n    build : function(approp, cc, obj = null, fund = null) {\r\n        // put together account string fund-approp-costcenter[-obj] (w optional object)\r\n        if (!fund) { fund = CurrentFund.number() };\r\n        // hits error here\r\n        approp = this.getNumber(approp);\r\n        cc = this.getNumber(cc);\r\n        var string = `${fund}-${approp}-${cc}`;\r\n        string = obj ? `${string}-${this.getNumber(obj)}` : string;\r\n        return string;\r\n    },\r\n\r\n    getAccountStringSection : function(account_string, section) {\r\n        const sections = account_string.split(\"-\");\r\n        return sections.length > section ? sections[section] : null;\r\n    },\r\n\r\n    fund : function(account_string) { \r\n        return this.getAccountStringSection(account_string, 0) \r\n    },\r\n\r\n    approp : function(account_string) { \r\n        return this.getAccountStringSection(account_string, 1) \r\n    },\r\n\r\n    costCenter : function(account_string) { \r\n        return this.getAccountStringSection(account_string, 2) \r\n    },\r\n\r\n    object : function(account_string) {\r\n        return this.getAccountStringSection(account_string, 3) \r\n    },\r\n}\r\n\r\nexport default AccountString;","\r\nimport { colSum } from \"../utils/common_utils\";\r\nimport { TOTAL_COLUMNS } from '../constants/';\r\n\r\n// Helper function to create a unique list of appropriations represented in a table\r\nfunction fetchValueSet(table, key) {\r\n\r\n    // IF there's no data, return an empty list\r\n    if(!table){\r\n        return [];\r\n    }\r\n\r\n    // Use a Set to store unique values\r\n    const uniqValues = new Set();\r\n  \r\n    // Iterate through each object in the table\r\n    table.forEach(row => {\r\n      if (row[key]) {\r\n        uniqValues.add(row[key]);\r\n      }\r\n    });\r\n  \r\n    // Convert the Set to an array if needed\r\n    return uniqValues;\r\n  }\r\n\r\n// Class to hold information on a specific fund and table\r\nclass StoredTable {\r\n    constructor(page, fund){\r\n        this.name = `${page}_${fund}`;\r\n        this.page = page;\r\n        this.table = JSON.parse(localStorage.getItem(this.name));\r\n        this.approps = fetchValueSet(this.table, 'Appropriation');\r\n        this.cc = fetchValueSet(this.table, 'Cost Center');\r\n\r\n    }\r\n\r\n    totalCol() {\r\n        return TOTAL_COLUMNS[this.page];\r\n    }\r\n    getSum() {\r\n        // fill with zero until there is something saved in storage\r\n        return colSum(this.table, this.totalCol(), this.name);\r\n    }\r\n\r\n    // key is the column to filter on (ie. Cost Center)\r\n    filter(key, value) {\r\n        if (this.table){\r\n            // reassign underlying data in object if there's data to filter\r\n            this.table = this.table.filter(row => {   \r\n                return (row[key]) && (row[key] == value);\r\n            });\r\n        }\r\n        return this;\r\n    }\r\n\r\n}\r\n\r\nexport class CostCenter{\r\n    constructor(fund, approp, cc){\r\n        const AppropObj = new Appropriation(fund, approp);\r\n        this.personnel = AppropObj.personnel.filter('Cost Center', cc);\r\n        this.nonpersonnel = AppropObj.nonpersonnel.filter('Cost Center', cc);\r\n        this.overtime = AppropObj.overtime.filter('Cost Center', cc);\r\n        this.revenue = AppropObj.revenue.filter('Cost Center', cc);\r\n        this.cc = cc;\r\n        this.account_string = `${AppropObj.accountString()}-${cc}`;\r\n    }\r\n\r\n    getPersonnelCost() {\r\n        return this.personnel.getSum(); \r\n    }\r\n\r\n    getOvertimeCost() {\r\n        return this.overtime.getSum();\r\n    }\r\n\r\n    getNonPersonnelCost() {\r\n        return this.nonpersonnel.getSum();\r\n    }\r\n\r\n    getRevenue() {\r\n        return this.revenue.getSum();\r\n    }\r\n\r\n    getTotal() { \r\n        // only sum expenditures, not net of revenue\r\n        return this.getNonPersonnelCost() + this.getOvertimeCost() + this.getPersonnelCost(); \r\n    }\r\n\r\n    getName(){\r\n        // just grab the value in the approp name column for the first row of the first non-empty table\r\n        const tables = [this.nonpersonnel.table, this.personnel.table, this.overtime.table, this.revenue.table];\r\n\r\n        for (let table of tables) {\r\n          if (table.length > 0 && table[0]['Cost Center Name']) {\r\n            return 'Cost Center ' + table[0]['Cost Center Name'];\r\n          }\r\n        }\r\n    \r\n        // If all tables are empty, return ''\r\n        return '';\r\n    }  \r\n\r\n    accountString() {\r\n        return this.account_string;\r\n    }\r\n}\r\n\r\nexport class Appropriation {\r\n    constructor(fund, approp){\r\n        const fundObj = new Fund(fund);\r\n        this.personnel = fundObj.personnel.filter('Appropriation', approp);\r\n        this.nonpersonnel = fundObj.nonpersonnel.filter('Appropriation', approp);\r\n        this.overtime = fundObj.overtime.filter('Appropriation', approp);\r\n        this.revenue = fundObj.revenue.filter('Appropriation', approp);\r\n        // own data\r\n        this.approp = approp;\r\n        this.fund = fundObj.fund;\r\n\r\n    }\r\n\r\n    getCostCenters(){\r\n        // build a set of unique appropriations across all line items for the fund\r\n        const cc = new Set([...this.personnel.cc,\r\n            ...this.overtime.cc, \r\n            ...this.nonpersonnel.cc, \r\n            ...this.revenue.cc]);\r\n\r\n        // initialize a list placeholder for the appropriations objects\r\n        const ccList = [];\r\n        // build out list\r\n        cc.forEach(num => {\r\n            ccList.push(new CostCenter(this.fund, this.approp, num));\r\n        });\r\n        return ccList;\r\n    }\r\n\r\n    total(){\r\n        return this.personnel.getSum() + this.overtime.getSum() + this.nonpersonnel.getSum();\r\n    }\r\n\r\n    name(){\r\n        // just grab the value in the approp name column for the first row of the first non-empty table\r\n        const tables = [this.nonpersonnel.table, this.personnel.table, this.overtime.table, this.revenue.table];\r\n\r\n        for (let table of tables) {\r\n          if (table.length > 0 && table[0]['Appropriation Name']) {\r\n            return 'Appropriation ' + table[0]['Appropriation Name'];\r\n          }\r\n        }\r\n    \r\n        // If all tables are empty, return ''\r\n        return '';\r\n    }  \r\n\r\n    accountString(){\r\n        return `${this.fund}-${this.approp}`;\r\n    }\r\n\r\n}\r\n\r\n// Holds all the detailed data for one fund's budget\r\nexport class Fund {\r\n    constructor(fund){\r\n        this.fund = fund;\r\n        this.personnel = new StoredTable('personnel', fund);\r\n        this.overtime = new StoredTable('overtime', fund);\r\n        this.nonpersonnel = new StoredTable('nonpersonnel', fund);\r\n        this.revenue = new StoredTable('revenue', fund);\r\n    }\r\n\r\n    getAppropriations(){\r\n        // build a set of unique appropriations across all line items for the fund\r\n        const approps = new Set([...this.personnel.approps,\r\n            ...this.overtime.approps, \r\n            ...this.nonpersonnel.approps, \r\n            ...this.revenue.approps]);\r\n\r\n        // initialize a list placeholder for the appropriations objects\r\n        const appropsList = [];\r\n        // build out list\r\n        approps.forEach(approp => {\r\n            appropsList.push( new Appropriation(this.fund, approp) );\r\n        });\r\n        return appropsList;\r\n    }\r\n\r\n    getPersonnelCost() {\r\n        return this.personnel.getSum(); \r\n    }\r\n\r\n    getOvertimeCost() {\r\n        return this.overtime.getSum();\r\n    }\r\n\r\n    getNonPersonnelCost() {\r\n        return this.nonpersonnel.getSum();\r\n    }\r\n\r\n    getRevenue() {\r\n        return this.revenue.getSum();\r\n    }\r\n\r\n    getTotal() { \r\n        // only sum expenditures, not net of revenue\r\n        return this.getNonPersonnelCost() + this.getOvertimeCost() + this.getPersonnelCost(); \r\n    }\r\n}\r\n\r\nexport default Fund;","import Fund from \"./fund.js\";\r\nimport FundLookupTable from \"./fund_lookup_table.js\";\r\n\r\n\r\nexport class Baseline {\r\n    // baseline will just contain a list of funds, each with\r\n    // running tallies for their budgets\r\n    constructor() {\r\n        const allFunds = FundLookupTable.listFunds();\r\n        this.funds = [];\r\n        allFunds.forEach((fund) => { \r\n            this.funds.push(new Fund(fund));\r\n        });\r\n    } \r\n\r\n    static target() { return localStorage.getItem('target') };\r\n\r\n    personnel() {\r\n        let total = 0;\r\n        this.funds.forEach(fund => {\r\n            total += fund.getPersonnelCost();\r\n        });\r\n        return total;\r\n    }\r\n\r\n    overtime() {\r\n        let total = 0;\r\n        this.funds.forEach(fund => {\r\n            total += fund.getOvertimeCost();\r\n        });\r\n        return total;\r\n    }\r\n\r\n    nonpersonnel() {\r\n        let total = 0;\r\n        this.funds.forEach(fund => {\r\n            total += fund.getNonPersonnelCost();\r\n        });\r\n        return total;\r\n    }\r\n\r\n    revenue() {\r\n        let total = 0;\r\n        this.funds.forEach(fund => {\r\n            total += fund.getRevenue();\r\n        });\r\n        return total;\r\n    }\r\n\r\n    total() {\r\n        return this.nonpersonnel() + this.personnel() + this.overtime();\r\n    }\r\n\r\n    genFundTotal() {\r\n        const GF = new Fund(1000);\r\n        return GF.getTotal();\r\n    }\r\n}\r\n\r\nexport default Baseline;","\r\n// data structure to hold information on new initiatives\r\n\r\nimport { NEW_INIT_COLS } from \"../constants\";\r\nimport { cleanString } from \"../utils/common_utils\";\r\n\r\nexport class Initiative {\r\n    \r\n    constructor(row) {\r\n        this.data = row;\r\n        this.name = row[NEW_INIT_COLS['name']];\r\n    }\r\n\r\n    id() {\r\n        let hash = 5381;\r\n        for (let i = 0; i < this.name.length; i++) {\r\n            hash = (hash << 5) + hash + this.name.charCodeAt(i);  // hash * 33 + c\r\n        }\r\n        return hash >>> 0;  // Ensure positive unsigned 32-bit integer\r\n    }\r\n\r\n    personnel() { \r\n        if (this.data[NEW_INIT_COLS['personnel']]) {\r\n            return this.data[NEW_INIT_COLS['personnel']];\r\n        } else {\r\n            return 0;\r\n        }\r\n    }\r\n\r\n    operating() { \r\n        if (this.data[NEW_INIT_COLS['operating']]) {\r\n            return this.data[NEW_INIT_COLS['operating']];\r\n        } else {\r\n            return 0;\r\n        }\r\n    }\r\n\r\n    capital() { \r\n        if (this.data[NEW_INIT_COLS['capital']]) {\r\n            return this.data[NEW_INIT_COLS['capital']];\r\n        } else {\r\n            return 0;\r\n        }\r\n    }\r\n\r\n    revenue() { \r\n        if (this.data[NEW_INIT_COLS['revenue']]) {\r\n            return this.data[NEW_INIT_COLS['revenue']];\r\n        } else {\r\n            return 0;\r\n        }\r\n    }\r\n\r\n    total() { \r\n        if (this.data[NEW_INIT_COLS['total']]) {\r\n            return this.data[NEW_INIT_COLS['total']];\r\n        } else {\r\n            return 0;\r\n        }\r\n    }\r\n\r\n}\r\n\r\nexport default Initiative;","// data structure to save the possible service options for the department\r\nexport const Services = {\r\n    save : function(services){\r\n        localStorage.setItem('services-list', JSON.stringify(services));\r\n    },\r\n    list : function(){\r\n        return JSON.parse(localStorage.getItem('services-list')) || {};\r\n    }\r\n}\r\n\r\nexport default Services;","\r\nimport Initiative from \"./initiative.js\";\r\nimport { colSum } from \"../utils/common_utils.js\";\r\nimport { NEW_INIT_COLS, TOTAL_COLUMNS } from \"../constants/excel_constants.js\";\r\n\r\n// data structure to hold supplemental requests\r\nexport class Supplemental {\r\n    constructor() {\r\n        this.table = JSON.parse(localStorage.getItem('new-inits'));\r\n        this.initiatives = [];\r\n        if(this.table){\r\n            this.table.forEach((row) => { \r\n                this.initiatives.push(new Initiative(row));\r\n            });\r\n        }\r\n    }\r\n\r\n    getInits() {\r\n        return this.table.map((item) => { return item[NEW_INIT_COLS['name']] });\r\n    }\r\n\r\n    personnel() {\r\n        return colSum(this.table, NEW_INIT_COLS['personnel']);\r\n    }\r\n\r\n    operating() {\r\n        return colSum(this.table, NEW_INIT_COLS['operating']);\r\n    }\r\n\r\n    capital() {\r\n        return colSum(this.table, NEW_INIT_COLS['capital']);\r\n    }\r\n\r\n    revenue() {\r\n        return colSum(this.table, NEW_INIT_COLS['revenue']);\r\n    }\r\n\r\n    total(){\r\n        return colSum(this.table, TOTAL_COLUMNS['new-inits']);\r\n    }\r\n\r\n}\r\n\r\nexport default Supplemental;","// models/index.js\r\n\r\nexport { default as AccountString } from './account_string.js';\r\nexport { default as Baseline } from './baseline.js';\r\nexport { default as CurrentFund } from './current_fund.js';\r\nexport { default as CurrentPage } from './current_page.js';\r\nexport { default as Fund } from './fund.js';\r\nexport { default as FundLookupTable } from './fund_lookup_table.js';\r\nexport { default as Initiative } from './initiative.js';\r\nexport { default as Services } from './services.js';\r\nexport { default as Supplemental } from './supplemental.js';\r\n","import { FISCAL_YEAR } from \"../../../constants\";\r\nimport { Baseline, FundLookupTable, Fund, CurrentFund } from \"../../../models\";\r\nimport { formatCurrency } from \"../../../utils/common_utils\";\r\nimport { visitPage } from \"../../../views/view_logic\";\r\n\r\nexport const BaselineSection = {\r\n    _data: new Baseline(),\r\n    _genFund : new Fund(1000),\r\n\r\n    get data() {\r\n        this._data = new Baseline();\r\n        return this._data;\r\n    },\r\n\r\n    set data(newData) {\r\n        this._data = newData;\r\n    },\r\n\r\n    get genFund() {\r\n        this._genFund = new Fund(1000);\r\n        return this._genFund;\r\n    },\r\n\r\n    set genFund(newFund) {\r\n        this._genFund = newFund;\r\n    },\r\n\r\n    target_html() {\r\n        return `\r\n            <div class='sidebar-stat-line' id=\"target\">\r\n                <span class=\"stat-label\">FY${FISCAL_YEAR} General Fund baseline target:</span> \r\n                <span class=\"stat\">${formatCurrency(Baseline.target())}</span>\r\n            </div>\r\n            <div class='sidebar-stat-line' id=\"GF-total\">\r\n                <span class=\"stat-label\">Current GF baseline:</span> \r\n                <span class=\"stat\">${formatCurrency(this.genFund.getTotal())}</span>\r\n            </div>\r\n            <br>`;\r\n    },\r\n    \r\n    fund_html(fund) {\r\n        return `\r\n            <h6>${FundLookupTable.getName(fund.fund)}</h6>\r\n            <hr>\r\n            <div class='sidebar-stat-line revenue'>\r\n                <span class=\"stat-label\">Projected revenues:</span> \r\n                <span class=\"stat\">${formatCurrency(fund.getRevenue())}</span>\r\n                <i class=\"fas fa-edit edit-icon\" title=\"Edit\"></i>\r\n            </div>\r\n            <div class='sidebar-stat-line personnel'>\r\n                <span class=\"stat-label\">Personnel cost:</span> \r\n                <span class=\"stat\">${formatCurrency(fund.getPersonnelCost())}</span>\r\n                <i class=\"fas fa-edit edit-icon\" title=\"Edit\"></i>\r\n            </div>\r\n            <div class='sidebar-stat-line overtime'>\r\n                <span class=\"stat-label\">Overtime cost:</span> \r\n                <span class=\"stat\">${formatCurrency(fund.getOvertimeCost())}</span>\r\n                <i class=\"fas fa-edit edit-icon\" title=\"Edit\"></i>\r\n            </div>\r\n            <div class='sidebar-stat-line nonpersonnel'>\r\n                <span class=\"stat-label\">Non-personnel cost:</span> \r\n                <span class=\"stat\">${formatCurrency(fund.getNonPersonnelCost())}</span>\r\n                <i class=\"fas fa-edit edit-icon\" title=\"Edit\"></i>\r\n            </div>\r\n            <div class='sidebar-stat-line fund-total'>\r\n                <span class=\"stat-label\">Total expenditures:</span> \r\n                <span class=\"stat\">${formatCurrency(fund.getTotal())}</span>\r\n            </div>\r\n            <br>`;\r\n    },\r\n\r\n    linkEditBtns() {\r\n        let btns = document.querySelectorAll('.edit-icon');\r\n        btns.forEach((btn) => {\r\n            // Get the fund from the div the button is in\r\n            let fund = btn.closest('.fund-div').id.replace('fund_', '');\r\n            let page = btn.closest('.sidebar-stat-line').classList[1];\r\n    \r\n            btn.addEventListener('click', function() {\r\n                CurrentFund.update(fund);\r\n                visitPage(page);\r\n            });\r\n        });\r\n    },\r\n\r\n    update() {\r\n        const baselineDiv = document.querySelector('#baseline-stats');\r\n        baselineDiv.innerHTML = this.target_html();\r\n\r\n        this.data.funds.forEach((fund) => {\r\n            var fundDiv = document.createElement('div');\r\n            fundDiv.id = `fund_${fund.fund}`;\r\n            fundDiv.classList.add('fund-div');\r\n            fundDiv.innerHTML = this.fund_html(fund);\r\n            baselineDiv.appendChild(fundDiv);\r\n        });\r\n\r\n        if(this.genFund.getTotal() <= Baseline.target()){\r\n            document.querySelector('#GF-total .stat').style.color = \"green\";\r\n            // document.querySelector('#fund_1000 .sidebar-stat-line:last-of-type .stat').style.color = \"green\";\r\n        } else {\r\n            document.querySelector('#GF-total .stat').style.color = \"red\";\r\n            // document.querySelector('#fund_1000 .sidebar-stat-line:last-of-type .stat').style.color = \"red\";\r\n        }\r\n        this.linkEditBtns();\r\n    }\r\n};","import { Supplemental } from \"../../../models\";\r\nimport { formatCurrency } from \"../../../utils/common_utils\";\r\nimport { visitPage } from \"../../../views/view_logic\";\r\n\r\nexport const SuppSection = {\r\n    html() {\r\n        var supp = new Supplemental;\r\n        return `\r\n            <div class='sidebar-stat-line' id=\"supp-revenue\">\r\n                <span class=\"stat-label\">Estimated Revenues:</span> \r\n                <span class=\"stat\">${formatCurrency(supp.revenue())}</span>\r\n                <i class=\"fas fa-edit edit-supp\" title=\"Edit\"></i>\r\n            </div>\r\n            <div class='sidebar-stat-line' id=\"supp-expenses\">\r\n                <span class=\"stat-label\">Personnel cost:</span> \r\n                <span class=\"stat\">${formatCurrency(supp.personnel())}</span>\r\n                <i class=\"fas fa-edit edit-supp\" title=\"Edit\"></i>\r\n            </div>\r\n            <div class='sidebar-stat-line' id=\"supp-expenses\">\r\n                <span class=\"stat-label\">Non-Personnel Operating:</span> \r\n                <span class=\"stat\">${formatCurrency(supp.operating())}</span>\r\n                <i class=\"fas fa-edit edit-supp\" title=\"Edit\"></i>\r\n            </div>\r\n            <div class='sidebar-stat-line' id=\"supp-expenses\">\r\n                <span class=\"stat-label\">Non-Personnel Capital:</span> \r\n                <span class=\"stat\">${formatCurrency(supp.capital())}</span>\r\n                <i class=\"fas fa-edit edit-supp\" title=\"Edit\"></i>\r\n            </div>\r\n            <div class='sidebar-stat-line' id=\"supp-expenses\">\r\n                <span class=\"stat-label\">Total expenditures:</span> \r\n                <span class=\"stat\">${formatCurrency(supp.total())}</span>\r\n                <i class=\"fas fa-edit edit-supp\" title=\"Edit\"></i>\r\n            </div>`\r\n    },\r\n\r\n    linkEditBtns() {\r\n        let btns = document.querySelectorAll('.edit-supp');\r\n        btns.forEach((btn) => {\r\n            btn.addEventListener('click', function() {\r\n                visitPage('new-inits');\r\n            });\r\n        });\r\n    },\r\n\r\n    update() {\r\n        const suppDiv = document.querySelector('#supp-stats');\r\n        suppDiv.innerHTML = this.html();\r\n        this.linkEditBtns();\r\n    }\r\n}\r\n\r\nexport default SuppSection;","import './sidebar.css'\r\nimport { BaselineSection } from './subcomponents/baseline_section';\r\n\r\nimport SuppSection from './subcomponents/supp_section'\r\n\r\n// fetch CSS variables saved in :root\r\nconst root = document.documentElement;\r\nconst sideBarWidth = getComputedStyle(root).getPropertyValue('--sidebar-width').trim();\r\n\r\nfunction hideSidebar() {\r\n    document.getElementById('sidebar-panel').style.display = 'none';\r\n    document.getElementById('main-panel').style.width = '100%'; \r\n    document.querySelector('header').style.width = '100%'\r\n}\r\n\r\nfunction showSidebar() {\r\n    const sidebar = document.getElementById('sidebar-panel');\r\n    const mainPanel = document.getElementById('main-panel');\r\n    const header = document.querySelector('header');\r\n\r\n    // update values\r\n    updateTotals();\r\n    sidebar.style.display = 'block'; // Show the sidebar\r\n    \r\n    // Calculate the remaining width for the main panel and header\r\n    var contentWidth = document.documentElement.clientWidth;\r\n    mainPanel.style.width = `${contentWidth - parseInt(sideBarWidth, 10)}px`; \r\n    header.style.width = `${contentWidth - parseInt(sideBarWidth, 10)}px`; \r\n\r\n    // add event listener to resize content if window is adjusted\r\n    window.addEventListener('resize', showSidebar);\r\n}\r\n\r\nfunction updateSidebarTitle(new_title){\r\n    document.getElementById('sidebar-title').textContent = new_title;\r\n}\r\n\r\nfunction updateTotals(){\r\n    SuppSection.update();\r\n    BaselineSection.update();\r\n}\r\n\r\nfunction resetAll(){\r\n    localStorage.clear();\r\n    // reset all stats to 0\r\n    updateTotals();\r\n}\r\n\r\nconst Sidebar = {\r\n    SuppSection : SuppSection,\r\n    BaselineSection : BaselineSection,\r\n    hide: hideSidebar,\r\n    show: showSidebar,\r\n    updateTitle: updateSidebarTitle,\r\n    updateTotals: updateTotals,\r\n    reset: resetAll\r\n};\r\n\r\nexport default Sidebar;","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./nav_buttons.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./nav_buttons.css\";\n       export default content && content.locals ? content.locals : undefined;\n","// nav_buttons.js\r\nimport './nav_buttons.css';\r\nimport { nextPage, lastPage } from '../../views/view_logic.js';\r\n\r\nfunction initializeNavButtons() {\r\n    // initialize last button\r\n    const last_btn = document.getElementById('btn-last');\r\n    last_btn.addEventListener('click', lastPage); \r\n    // initialize next button\r\n    const next_btn = document.getElementById('btn-next');\r\n    next_btn.addEventListener('click', nextPage);\r\n}\r\n\r\nfunction disable(button_id) {\r\n    const button = document.getElementById(button_id);\r\n    button.classList.add('disabled');\r\n    button.disabled = true; // Also disable it at the DOM level\r\n}\r\n\r\nfunction enable(button_id) {\r\n    const button = document.getElementById(button_id);\r\n    button.classList.remove('disabled');\r\n    button.disabled = false; // Enable it at the DOM level\r\n}\r\n\r\nconst Next = {\r\n    disable: function() { disable('btn-next'); },\r\n    enable: function() { enable('btn-next'); },\r\n    addAction: function(fn) {\r\n        document.querySelector(`#btn-next`).addEventListener('click', fn);\r\n    },\r\n    removeAction: function(fn) {\r\n        document.querySelector(`#btn-next`).removeEventListener('click', fn);\r\n    },\r\n};\r\n\r\nconst Last = {\r\n    disable: function() { disable('btn-last'); },\r\n    enable: function() { enable('btn-last'); },\r\n};\r\n\r\nexport const NavButtons = {\r\n    hide: function() {\r\n        document.getElementById('nav-btns').style.display = 'none';\r\n    },\r\n    show: function() {\r\n        document.getElementById('nav-btns').style.display = 'block';\r\n        initializeNavButtons();\r\n    },\r\n    Next: Next,\r\n    Last: Last,\r\n};\r\n\r\nexport default NavButtons;","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./body.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./body.css\";\n       export default content && content.locals ? content.locals : undefined;\n","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./accordion.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./accordion.css\";\n       export default content && content.locals ? content.locals : undefined;\n","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./table.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./table.css\";\n       export default content && content.locals ? content.locals : undefined;\n","function addTableHeaders(cols){\r\n\r\n    // Get the table element by its ID\r\n    const table = document.getElementById('main-table');\r\n    \r\n    // Create a table header row element\r\n    const headerRow = document.createElement('tr');\r\n\r\n    cols.forEach(col => {\r\n        // Create a header cell element\r\n        const headerCell = document.createElement('th');\r\n        headerCell.textContent = col['title'];\r\n        headerCell.classList.add(col['className']);\r\n        \r\n        // Append the header cell to the header row\r\n        headerRow.appendChild(headerCell);\r\n    });\r\n\r\n    // Append the header row to the table header\r\n    let thead = table.querySelector('thead');\r\n    thead.appendChild(headerRow);\r\n}\r\n\r\nconst Header = {\r\n    add: function(header_array){\r\n        addTableHeaders(header_array)\r\n    }\r\n};\r\n\r\nexport default Header;","import Header from \"./headers.js\";\r\nimport { formatCurrency } from \"../../../utils/common_utils.js\";\r\n\r\nasync function addNewRow(data_dictionary, columns = []){\r\n    \r\n    // Get the table element by its ID\r\n    const table = document.getElementById('main-table');\r\n\r\n    // check if header has already been added\r\n    let header_row = table.querySelector('thead tr');\r\n    if (!header_row) {\r\n        Header.add(columns);\r\n        header_row = table.querySelector('thead tr');\r\n    }\r\n\r\n    // initialize new row of data\r\n    const new_row = document.createElement('tr');\r\n\r\n    // go through each header and add the right cell value depending on its class\r\n    let thElements = header_row.querySelectorAll('th');\r\n    thElements.forEach( (header_cell) => {\r\n        // Create new cell and add it to the row\r\n        const newCell = document.createElement('td');\r\n        new_row.appendChild(newCell);\r\n        // if the data has an appropriate class, add the info to the cell. \r\n        // Otherwise, keep empty cell\r\n        Object.keys(data_dictionary).forEach( (className) => {\r\n            if (header_cell.classList.contains(className) ){\r\n                newCell.textContent = data_dictionary[className];\r\n                newCell.classList.add(className);\r\n            }\r\n        });\r\n    });\r\n\r\n    // Append the new row to the table body\r\n    let tbody = table.querySelector('tbody');\r\n    tbody.appendChild(new_row);\r\n}\r\n\r\nfunction saveRowEdits(row){\r\n    var cells = row.querySelectorAll('td')\r\n    cells.forEach( cell => {\r\n        // save dropdown values\r\n        if (cell.querySelector('select')){\r\n            var serviceSelector = cell.querySelector('select');\r\n            cell.textContent = serviceSelector.value;\r\n        } else if (cell.querySelector('input')) {\r\n            // save new entered value in textbox\r\n            var textbox = cell.querySelector('input');\r\n            var enteredValue = textbox.value;\r\n            // update display and format with currency if relevant\r\n            if ( cell.classList.contains('cost') ){\r\n                // if cost, remove commas first\r\n                enteredValue = enteredValue.replaceAll(',', '');\r\n                cell.textContent = formatCurrency(enteredValue);\r\n                // set value attribute to the new user input\r\n                cell.setAttribute('value', enteredValue);\r\n            } else {\r\n                cell.textContent = enteredValue;\r\n            }\r\n        } else if (cell.querySelector('textarea')){\r\n            // save new entered value in textbox\r\n            var enteredValue = cell.querySelector('textarea').value;\r\n            cell.textContent = enteredValue;\r\n        }\r\n    })\r\n}\r\n\r\nconst Rows = {\r\n    add : function(data_dictionary, cols){\r\n        addNewRow(data_dictionary, cols)\r\n    },\r\n    saveEdits : function(row){\r\n        saveRowEdits(row)\r\n    }\r\n}\r\n\r\nexport default Rows;","import Rows from './rows.js'\r\n\r\nfunction hideButton(className){\r\n    return function() {\r\n        var buttons = document.getElementsByClassName(className);\r\n        for (var i = 0; i < buttons.length; i++) {\r\n            buttons[i].style.display = 'none';\r\n        }\r\n    }\r\n}\r\n\r\nfunction showButton(className){\r\n    return function() {\r\n        var buttons = document.getElementsByClassName(className);\r\n        for (var i = 0; i < buttons.length; i++) {\r\n            buttons[i].style.display = 'inline';\r\n        }\r\n    }\r\n}\r\n\r\nfunction updateButtonText(className, text){\r\n    document.querySelector(`.${className}`).textContent = text;\r\n}\r\n\r\nfunction handleRowEdit(actionOnClick, updateCallback = null){\r\n    // attach an event listener to each edit button in every row\r\n    var editButtons = document.getElementsByClassName('btn-edit');\r\n    for (var i = 0; i < editButtons.length; i++) {\r\n        editButtons[i].addEventListener('click', async function(event) {\r\n            // Determine what was clicked on within the table\r\n            var rowToEdit = event.target.closest('tr');\r\n            // mark row as being edited\r\n            rowToEdit.classList.add('active-editing');\r\n            \r\n            // turn relevant entries into textboxes, usually\r\n            actionOnClick();\r\n\r\n            // hide edit buttons\r\n            Edit.hide();\r\n            if(updateCallback){\r\n                initializeConfirmButton(updateCallback);\r\n            }\r\n        });\r\n    };\r\n}\r\n\r\nfunction initializeConfirmButton(updateCallback){\r\n    // get element and add listener for click\r\n    var rowToEdit = document.querySelector('.active-editing');\r\n    const confirm_btn = rowToEdit.querySelector(\".btn-confirm\");\r\n    // show the row's confirm button\r\n    confirm_btn.style.display = 'block';\r\n    confirm_btn.addEventListener('click', function(){;\r\n        // save row edits\r\n        Rows.saveEdits(rowToEdit);\r\n        // update values in sidebar\r\n        updateCallback();\r\n        // make row no longer green\r\n        rowToEdit.classList.remove('active-editing');\r\n        // show edit buttons and hide confirm buttons\r\n        Edit.show();\r\n        Confirm.hide();\r\n    });\r\n}\r\n\r\nconst Edit = {\r\n    html: '<button class=\"btn btn-edit\">Edit row</button>',\r\n    hide: hideButton('btn-edit'),\r\n    show: showButton('btn-edit'),\r\n    init : function(actionOnClick, updateCallback){\r\n        handleRowEdit(actionOnClick, updateCallback)\r\n    }\r\n};\r\n\r\nconst Delete = {\r\n    html: '<button class=\"btn btn-delete\">Delete</button>',\r\n    hide: hideButton('btn-delete'),\r\n    show: showButton('btn-delete')\r\n};\r\n\r\nconst Confirm = {\r\n    html: '<button class=\"btn btn-confirm\">Confirm</button>',\r\n    hide: hideButton('btn-confirm'),\r\n    show: showButton('btn-confirm')\r\n};\r\n\r\nconst AddRow = {\r\n    hide: hideButton('btn-add'),\r\n    show: showButton('btn-add'),\r\n    updateText: function(text){\r\n        updateButtonText('btn-add', text);\r\n    }\r\n};\r\n\r\nexport const Buttons = {\r\n    Delete: Delete,\r\n    Edit : Edit,\r\n    Confirm : Confirm,\r\n    AddRow : AddRow,\r\n    edit_confirm_btns : Edit.html + Confirm.html ,\r\n    all_btns : Delete.html + Edit.html + Confirm.html\r\n}\r\n\r\nexport default Buttons;","function createDropdown(dataArray) {\r\n\r\n  // Creating a select element\r\n  const selectElement = document.createElement('select');\r\n\r\n  // add a default blank option to the dataArray\r\n  dataArray = [''].concat(dataArray);\r\n\r\n  // Looping through the array and creating an option for each element\r\n  dataArray.forEach(item => {\r\n    const optionElement = document.createElement('option');\r\n    optionElement.value = item;\r\n    optionElement.textContent = item;\r\n    selectElement.appendChild(optionElement); // Appending the option to the select\r\n  });\r\n\r\n  // Return the select element so it can be appended to the document\r\n  return selectElement;\r\n}\r\n\r\n\r\nexport const Dropdown = {\r\n    create : function(dataArray) { return createDropdown(dataArray) },\r\n}\r\n\r\nexport default Dropdown;","import { formatCurrency, displayWithCommas } from \"../../../utils/common_utils.js\";\r\nimport Services from \"../../../models/services.js\";\r\nimport Dropdown from \"../../form/subcomponents/dropdown.js\";\r\n\r\n// return cell value attribute or 0 if it does not exist\r\nfunction getCellValue(row, className) {\r\n    var cell = row.querySelector(`.${className}`);\r\n    var cellValue = cell ? cell.getAttribute('value') : null;\r\n    return cellValue ? parseFloat(cellValue) : 0;\r\n}\r\n\r\n// return text in cell\r\nfunction getCellText(row, className) {\r\n    var cell = row.querySelector(`.${className}`);\r\n    if (cell) {\r\n        return cell.textContent;\r\n    } else {\r\n        //console.log(`Error retrieving cell text for class ${className}`);\r\n        return '';\r\n    }\r\n}\r\n\r\nfunction updateTableCell(row, col_class, new_value){\r\n    const cell = row.querySelector(`.${col_class}`);\r\n    cell.setAttribute('value', new_value);\r\n    cell.textContent = formatCurrency(new_value);\r\n}\r\n\r\nfunction createEditableCell(cellClass, isCost, type = 'input'){\r\n    // get cell\r\n    const cell = document.querySelector(`.active-editing td.${cellClass}`);\r\n    // Create an input element to edit the value\r\n    var textbox = document.createElement(type);\r\n    if (type == 'input'){\r\n        textbox.type = 'text';\r\n    };\r\n    if (isCost){\r\n        var value = cell.getAttribute('value');\r\n        textbox.value = displayWithCommas(value);\r\n    } else {\r\n        textbox.value = cell.textContent;\r\n    }\r\n    // Clear the current content and append the textbox to the cell\r\n    cell.innerHTML = '';\r\n    cell.appendChild(textbox);\r\n}\r\n\r\nfunction createDropdown(cellClass, optionArray){\r\n    // get cell\r\n    const cell = document.querySelector(`.active-editing td.${cellClass}`);\r\n    // add service dropdown\r\n    const dropdown = Dropdown.create(optionArray);\r\n    dropdown.value = cell.textContent;\r\n    // Clear the current content and append the textbox to the cell\r\n    cell.innerHTML = '';\r\n    cell.appendChild(dropdown);\r\n}\r\n\r\nconst Cell = {\r\n    getValue: function(row, className) {\r\n        return getCellValue(row, className); \r\n    },\r\n    getText: function(row, className) {\r\n        return getCellText(row, className); \r\n    },\r\n    updateValue: function(row, col_class, new_value) {\r\n        updateTableCell(row, col_class, new_value); \r\n    },\r\n    createTextbox : function(className, isCost, type) {\r\n        createEditableCell(className, isCost, type)\r\n    },\r\n    createServiceDropdown :  () => { createDropdown('service', Services.list()) },\r\n    createDropdown : createDropdown\r\n};\r\n\r\nexport default Cell;","import { formatCurrency } from \"../../../utils/common_utils.js\";\r\n\r\n// position is index at which new column will be inserted\r\nfunction addCol(position, htmlContent = '', headerTitle = '') {\r\n    // Get the table element by its ID\r\n    let table = document.getElementById('main-table');\r\n\r\n    // Validate position\r\n    let maxPosition = table.rows[0].cells.length;\r\n    if (position < 0 || position > maxPosition) {\r\n      console.error(`Position ${position} is out of bounds.`);\r\n      return;\r\n    }\r\n  \r\n    // Insert the header if provided\r\n    let thead = table.tHead;\r\n    if (headerTitle && thead) {\r\n      let th = document.createElement('th');\r\n      th.innerHTML = headerTitle; // Use innerHTML to insert HTML content\r\n      thead.rows[0].insertBefore(th, thead.rows[0].cells[position]);\r\n    }\r\n  \r\n    // Insert new cells into each row of the table body\r\n    let tbody = table.tBodies[0];\r\n    if (tbody) {\r\n      for (let i = 0; i < tbody.rows.length; i++) {\r\n        let row = tbody.rows[i];\r\n        let td = document.createElement('td');\r\n        td.innerHTML = htmlContent; // Use innerHTML to insert HTML content\r\n        row.insertBefore(td, row.cells[position]);\r\n      }\r\n    }\r\n}\r\n\r\nfunction ncols(){\r\n    const table = document.getElementById('main-table');\r\n    // Ensure that the row exists before counting the columns\r\n    return table.rows[0].cells.length;\r\n}\r\n  \r\nfunction addColToEnd(htmlContents = [], headerTitle = ''){\r\n    // count columns and add new column to the end\r\n    const position = ncols('main-table');\r\n    addCol(position, htmlContents, headerTitle);\r\n}\r\n\r\nfunction assignClassToColumn(headerName, className) {\r\n    // Get the table element by its ID\r\n    let table = document.getElementById('main-table');\r\n\r\n    // Find the index of the column by its header name\r\n    const thead = table.tHead;\r\n    let headerCellIndex = -1;\r\n    const headerCells = thead.rows[0].cells; // Assuming the first row contains header cells (<th>)\r\n    for (let i = 0; i < headerCells.length; i++) {\r\n        if (headerCells[i].textContent.trim() === headerName) {\r\n            // assign the class to the header cell\r\n            headerCells[i].classList.add(className);\r\n            headerCellIndex = i;\r\n            break;\r\n        }\r\n    }\r\n\r\n    // error check\r\n    if (headerCellIndex === -1) {\r\n        console.error(`No header found with name \"${headerName}\"`);\r\n        return;\r\n    }\r\n  \r\n    // Assign the class to each cell in the specified column index within the tbody\r\n    let tbody = table.tBodies[0];\r\n    if (tbody) {\r\n      let bodyRows = tbody.rows;\r\n      for (let row of bodyRows) {\r\n        if (row.cells[headerCellIndex]) {\r\n          row.cells[headerCellIndex].classList.add(className);\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\nfunction addCostClass(headerName){\r\n    assignClassToColumn( headerName, 'cost');\r\n\r\n    // Get all the cells with the specified class name\r\n    const cells = document.querySelectorAll(`td.cost`);\r\n      \r\n    cells.forEach(cell => {\r\n        // Get the current text content of the cell and assign it to 'value' attribute\r\n        if (!cell.getAttribute('value')){\r\n            const cellText = cell.textContent.trim();\r\n            const cellValue = isNaN(cellText) || cellText === '' ? 0 : parseFloat(cellText);\r\n            cell.setAttribute('value', cellValue);\r\n\r\n            // Now format the text content like currency and replace it in the cell\r\n            const formattedCurrency = formatCurrency(parseFloat(cellValue));\r\n            cell.textContent = formattedCurrency;\r\n        }\r\n      \r\n    });\r\n\r\n}\r\n\r\nfunction assignColumnClasses(columnDefinitions) {\r\n    columnDefinitions.forEach(column => {\r\n        // Assign class to column\r\n        assignClassToColumn(column.title, column.className);\r\n\r\n        // If the column is a cost column, add the specific cost class\r\n        if (column.isCost) {\r\n            addCostClass(column.title);\r\n        }\r\n\r\n        // show the column\r\n        if (!column.hide){\r\n            showColumnByTitle(column.title);\r\n        }\r\n    });\r\n}\r\n\r\nfunction hideColumn(index) {\r\n  var table = document.getElementById('main-table');\r\n  var rows = table.rows;\r\n\r\n  for (var i = 0; i < rows.length; i++) {\r\n      var cells = rows[i].cells;\r\n      if (cells.length > index) {\r\n          cells[index].style.display = 'none';\r\n      }\r\n  }\r\n}\r\n\r\n// Function to show a specific column\r\nfunction showColumn(index) {\r\n  var table = document.getElementById('main-table');\r\n  var rows = table.rows;\r\n\r\n  for (var i = 0; i < rows.length; i++) {\r\n      var cells = rows[i].cells;\r\n      if (cells.length > index) {\r\n          cells[index].style.display = 'table-cell';\r\n      }\r\n  }\r\n}\r\n\r\n// Function to get the column index by title\r\nfunction getColumnIndexByTitle(table, title) {\r\n  var headers = table.querySelectorAll('th');\r\n  for (var i = 0; i < headers.length; i++) {\r\n      if (headers[i].innerText === title) {\r\n          return i;\r\n      }\r\n  }\r\n  return -1; // Column not found\r\n}\r\n\r\n// Function to hide a column by title\r\nfunction hideColumnByTitle(title) {\r\n  var table = document.getElementById('main-table');\r\n  var index = getColumnIndexByTitle(table, title);\r\n  if (index !== -1) {\r\n      hideColumn(index);\r\n  } else {\r\n      console.log(`Column with title \"${title}\" not found.`);\r\n  }\r\n}\r\n\r\n// Function to show a column by title\r\nfunction showColumnByTitle(title) {\r\n  var table = document.getElementById('main-table');\r\n  var index = getColumnIndexByTitle(table, title);\r\n  if (index !== -1) {\r\n      showColumn(index);\r\n  } else {\r\n      console.log(`Column with title \"${title}\" not found.`);\r\n  }\r\n}\r\n\r\n\r\nconst Column = {\r\n  add: function(position, htmlContent, headerTitle) {\r\n      return addCol(position, htmlContent, headerTitle);\r\n  },\r\n  addAtEnd: function(htmlContent, headerTitle) {\r\n      return addColToEnd(htmlContent, headerTitle);\r\n  },\r\n  assignClasses: function(column_definitions) {\r\n      return assignColumnClasses(column_definitions);\r\n  },\r\n  hide : function(colName) {\r\n      hideColumnByTitle(colName);\r\n  },\r\n  show : function(colName) {\r\n      showColumnByTitle(colName);\r\n  }\r\n};\r\n\r\nexport default Column;","import FundLookupTable from '../../../models/fund_lookup_table.js';\r\nimport CurrentFund from '../../../models/current_fund.js'\r\nimport CurrentPage from '../../../models/current_page.js'\r\nimport Table from '../table.js';\r\n\r\nfunction fillTable(data) {\r\n    try {\r\n        const table = document.getElementById('main-table');\r\n        const thead = table.querySelector('thead');\r\n        const tbody = table.querySelector('tbody');\r\n\r\n        // clear existing data\r\n        thead.innerHTML = '';\r\n        tbody.innerHTML = '';\r\n\r\n        // Create table header row\r\n        const headerRow = document.createElement('tr');\r\n        Object.keys(data[0]).forEach(key => {\r\n            const header = document.createElement('th');\r\n            header.textContent = key;\r\n            headerRow.appendChild(header);\r\n        });\r\n        thead.appendChild(headerRow);\r\n\r\n        // Create table body rows\r\n        data.forEach(item => {\r\n            const row = document.createElement('tr');\r\n            Object.values(item).forEach(val => {\r\n                const cell = document.createElement('td');\r\n                cell.innerHTML = val;\r\n                row.appendChild(cell);\r\n            });\r\n            tbody.appendChild(row);\r\n        });\r\n    } catch(error) {\r\n        console.error('No table saved in localStorage:', error);\r\n    }\r\n}\r\n\r\nasync function loadFromStorage(){\r\n    // look up table name in storage\r\n    if (CurrentFund.number()){\r\n        var key = `${CurrentPage.load()}_${CurrentFund.number()}`;\r\n    } else {\r\n        var key = CurrentPage.load();\r\n    }\r\n    // load from local storage\r\n    const data = localStorage.getItem(key);\r\n    // if nothing in storage, return a zero\r\n    if ( !data ) {\r\n        return 0;\r\n    } else {\r\n        // otherwise, fill table in HTML and return success (1)\r\n        Table.show();\r\n        fillTable(await JSON.parse(data));\r\n        return 1;\r\n    }\r\n}\r\n\r\n\r\nfunction loadFunds(){\r\n    // get list of funds from storage\r\n    const fundDict = FundLookupTable.retrieve();\r\n    // build out data in correct format\r\n    const ret = [];\r\n    Object.keys(fundDict).forEach(key => {\r\n        // determine if the fund has already been edited\r\n        if (fundDict[key]['viewed']){\r\n            // todo: add a checkmark here\r\n            ret.push({'Fund' :  `<span class = 'viewed-fund'> \r\n                                    <i class=\"fas fa-check\"></i>\r\n                                    ${fundDict[key]['name']}\r\n                                </span>`});\r\n        } else {\r\n            ret.push({'Fund' : `<span class = 'unviewed-fund'> \r\n                                    ${fundDict[key]['name']}\r\n                                </span>`});   \r\n        }\r\n    });\r\n    fillTable(ret);\r\n}\r\n\r\n\r\nfunction getColumnIndexByClass(tbody, className) {\r\n  const firstRow = tbody.rows[0];\r\n  if (!firstRow) return -1; // Return -1 if there's no row to examine\r\n  for (let cellIndex = 0; cellIndex < firstRow.cells.length; cellIndex++) {\r\n      if (firstRow.cells[cellIndex].classList.contains(className)) {\r\n          return cellIndex;\r\n      }\r\n  }\r\n  return -1; // Return -1 if class name not found\r\n}\r\n\r\n// function sort(primaryClass, secondaryClass) {\r\n//   const table = document.getElementById('main-table');\r\n//   const tbody = table.tBodies[0];\r\n//   const rows = Array.from(tbody.rows);\r\n\r\n//   // Get the column indices by class name\r\n//   const primaryColIndex = getColumnIndexByClass(tbody, primaryClass);\r\n//   const secondaryColIndex = getColumnIndexByClass(tbody, secondaryClass);\r\n\r\n//   // exit if classes don't exist\r\n//   if (primaryColIndex === -1 ) {\r\n//       console.error(`Column class ${classA} not found in table`);\r\n//       return; \r\n//   } else if (secondaryColIndex === -1) {\r\n//       console.error(`Column class ${classB} not found in table`);\r\n//       return; \r\n//   }\r\n\r\n//   // Sort the rows based on the text content of the cells\r\n//   rows.sort((rowA, rowB) => {\r\n//       // Primary column comparison\r\n//       const primaryA = unformatCurrency(rowA.cells[primaryColIndex].textContent);\r\n//       const primaryB = unformatCurrency(rowB.cells[primaryColIndex].textContent);\r\n\r\n//       if (primaryA < primaryB) return -1;\r\n//       if (primaryA > primaryB) return 1;\r\n\r\n//       // Secondary column comparison (if primary is equal)\r\n//       const secondaryA = unformatCurrency(rowA.cells[secondaryColIndex].textContent);\r\n//       const secondaryB = unformatCurrency(rowB.cells[secondaryColIndex].textContent);\r\n\r\n//       if (secondaryA < secondaryB) return -1;\r\n//       if (secondaryA > secondaryB) return 1;\r\n\r\n//       // If both columns are equal\r\n//       return 0;\r\n//   });\r\n\r\n//   // Reattach sorted rows to the table body\r\n//   rows.forEach((row) => tbody.appendChild(row));\r\n// }\r\n\r\n\r\nexport const Data = {\r\n    load : loadFromStorage,\r\n    loadFunds : loadFunds,\r\n    sort : function(colA, colB) { sort(colA, colB) }\r\n}\r\n\r\nexport default Data;","// Helper functions & constants\r\n\r\n// object to hold all current filter statuses\r\nconst filterSettings = {\r\n    'approp-name': '',\r\n    'cc-name': '',\r\n    'object-name': '',\r\n    'object-category': ''\r\n};\r\n\r\n// helper function to filter data based on all filters\r\nfunction filterData() {\r\n    // Get all rows in the table\r\n    const rows = document.querySelectorAll('#main-table tbody tr');\r\n    \r\n    // Iterate through each row and determine if it should be hidden or shown\r\n    rows.forEach(row => {\r\n        let isVisible = true;\r\n        \r\n        // Check each filter setting against the row's cells\r\n        for (const [filterId, filterValue] of Object.entries(filterSettings)) {\r\n            const cell = row.querySelector(`.${filterId}`);\r\n\r\n            // only show row if values pass through all filters \r\n            if (filterValue && cell && (cell.textContent.trim() !== filterValue)) {\r\n                isVisible = false;\r\n                break;\r\n            } \r\n        }\r\n\r\n        // Show or hide the row based on visibility\r\n        row.classList.toggle('hidden', !isVisible);\r\n    });\r\n}\r\n\r\nconst Filter = {\r\n    html(filterLabel, filterClass) {\r\n        // basic html with only 'All' option\r\n        return `<label for=\"filter-${filterClass}\">Filter by ${filterLabel}:</label>\r\n                <select id=\"filter-${filterClass}\" class=\"filter-dropdown\">\r\n                    <option value=\"\">All</option>\r\n                </select>`\r\n    },\r\n\r\n    addOption(filterClass, option) {\r\n        // Add another option to the dropdown for the filter\r\n        const filterObj = document.querySelector(`#filter-${filterClass}`);\r\n        const optionObj = document.createElement('option');\r\n        optionObj.value = option;\r\n        optionObj.textContent = option;\r\n        filterObj.appendChild(optionObj);\r\n    },\r\n\r\n    add(filterLabel, filterClass) {\r\n        // create a div to contain the html and insert inside filter-container\r\n        const filterContainer = document.querySelector('#filter-container');\r\n        const filterDiv = document.createElement('div');\r\n        filterDiv.innerHTML = this.html(filterLabel, filterClass);\r\n        filterContainer.appendChild(filterDiv);\r\n        // add all relevant options from that column in the table\r\n        this.addAllOptions(filterClass);\r\n        // Bind change event to the select element\r\n        filterDiv.querySelector('.filter-dropdown').addEventListener('change', event => {\r\n            // Update filter settings\r\n            filterSettings[filterClass] = event.target.value;\r\n            // Apply all filters\r\n            filterData();\r\n        });\r\n    },\r\n\r\n    addAllOptions(filterClass) {\r\n        // get matching column from table\r\n        const column = document.querySelectorAll(`#main-table td.${filterClass}`);\r\n\r\n        // Use a Set to store unique values in the column of interest\r\n        const uniqueValues = new Set();\r\n        \r\n        // Iterate over the NodeList to get the unique values\r\n        column.forEach(td => {\r\n            // Add each textContent to the Set\r\n            uniqueValues.add(td.textContent.trim());\r\n        });\r\n        // add all values as options to the filter dropdown\r\n        uniqueValues.forEach(option => {\r\n            this.addOption(filterClass, option);\r\n        });\r\n    },\r\n\r\n    deleteAll(){\r\n        document.querySelector('#filter-container').innerHTML = '';\r\n    },\r\n\r\n    updateOptions(filterClass) {\r\n        const filterObj = document.querySelector(`#filter-${filterClass}`);\r\n        if (filterObj) {\r\n            // Clear all existing options except for the default 'All' option\r\n            filterObj.options.length = 1;\r\n            // Add new options\r\n            this.addAllOptions(filterClass);\r\n        }\r\n    }\r\n}\r\n\r\nexport default Filter;","// the Gold Book contains a lookup table for each job code, which maps to the BU, the job title, and the fringe rate\r\n\r\nconst GoldBook = {\r\n    init(sheet) {\r\n        // Convert sheet to JSON, with no automatic header row parsing\r\n        const rawData = XLSX.utils.sheet_to_json(sheet, { header: 1, defval: '' });\r\n\r\n        // Assuming header is the second row (index 1), and data starts at the third row (index 2)\r\n        const headers = rawData[1];\r\n        const data = rawData.slice(2); \r\n\r\n        // Store headers and data in localStorage\r\n        localStorage.setItem('goldbook_headers', JSON.stringify(headers));\r\n        localStorage.setItem('goldbook_data', JSON.stringify(data));\r\n    },\r\n\r\n    fetch() {\r\n        const headers = JSON.parse(localStorage.getItem('goldbook_headers'));\r\n        const data = JSON.parse(localStorage.getItem('goldbook_data'));\r\n        return { headers, data };\r\n    },\r\n\r\n    fetchByCode(job_code) {\r\n        const { headers, data } = this.fetch();\r\n\r\n        // Create a map of column names to their respective indices\r\n        const headerMap = headers.reduce((acc, header, index) => {\r\n            acc[header] = index;\r\n            return acc;\r\n        }, {});\r\n\r\n        return data.filter(row => row[headerMap['Job Code']] == job_code);\r\n    },\r\n\r\n    \r\n    getHeaderIX(header_name){\r\n        const { headers } = this.fetch();\r\n            const headerMap = headers.reduce((acc, header, index) => {\r\n              acc[header] = index;\r\n              return acc;\r\n            }, {});\r\n        return headerMap[header_name];\r\n    },\r\n\r\n    lookupByJobCode(job_code, header_name){\r\n        const codeData = this.fetchByCode(job_code);\r\n\r\n        if (codeData.length > 0) {\r\n            return codeData[0][this.getHeaderIX(header_name)];\r\n        }\r\n        return null;\r\n    },\r\n\r\n    getTitle(job_code) {\r\n        return this.lookupByJobCode(job_code, 'Job Description');\r\n    },\r\n\r\n\r\n    getFringeRate(job_code) {\r\n        return parseFloat(this.lookupByJobCode(job_code, 'Fringe Rate'));\r\n    },\r\n\r\n    codeExists(job_code){\r\n        return this.fetchByCode(job_code).length > 0;\r\n    },\r\n\r\n    // method to add GoldBook data as a new sheet to the workbook\r\n    xlsx() {\r\n        const { headers, data } = this.fetch();\r\n\r\n        // Combine headers and data into one array\r\n        const combinedData = [headers].concat(data);\r\n\r\n        // Create a worksheet\r\n        return XLSX.utils.aoa_to_sheet(combinedData);\r\n    }\r\n};\r\n\r\nexport default GoldBook;","\r\n\r\nimport { SHEETS, TARGET_CELL_ADDRESS, FISCAL_YEAR } from '../constants/';\r\nimport FundLookupTable from '../models/fund_lookup_table.js';\r\nimport { removeNewLines } from './common_utils.js';\r\nimport Baseline from '../models/baseline.js';\r\nimport Services from '../models/services.js';\r\nimport GoldBook from '../models/gold_book.js';\r\n\r\n\r\n// Helper functions\r\n\r\n/**\r\n * Deletes the top rows until a row containing complete data is found.\r\n * @param {Array} data - The raw data extracted from the sheet.\r\n * @returns {Array} - The cleaned data with incomplete top rows removed.\r\n */\r\nfunction deleteTopRowsUntilFullData(data) {\r\n    let fullDataRowFound = false;\r\n\r\n    while (!fullDataRowFound && data.length > 0) {\r\n        const row = data[0]; // Get the top row\r\n        let hasAllData = true;\r\n\r\n        for (const cell of row) {\r\n            if (cell == null || cell === '') {\r\n                hasAllData = false;\r\n                break;\r\n            }\r\n        }\r\n\r\n        if (hasAllData && row.length > 1) {\r\n            fullDataRowFound = true;\r\n        } else {\r\n            // delete the top row if it's not the header row\r\n            data.shift();\r\n        }\r\n    }\r\n\r\n    return data;\r\n}\r\n\r\n/**\r\n * Reads the workbook from the provided array buffer.\r\n * @param {ArrayBuffer} arrayBuffer - The array buffer containing the workbook data.\r\n * @returns {Object} - The parsed workbook.\r\n */\r\nfunction readWorkbook(arrayBuffer) {\r\n    return XLSX.read(arrayBuffer, { type: 'array' });\r\n}\r\n\r\n/**\r\n * Processes sheets to be split by fund and saves the relevant data.\r\n * @param {string} sheetName - The name of the sheet being processed.\r\n * @param {Object} sheet - The sheet object from the workbook.\r\n */\r\nfunction processSheet(sheetName, sheet) {\r\n    // Read in sheets\r\n    const rawData = XLSX.utils.sheet_to_json(sheet, { header: 1, defval: '' });\r\n    \r\n    // Clean the data by removing top rows with incomplete data\r\n    const dataRows = deleteTopRowsUntilFullData(rawData);\r\n\r\n    // Get new headers\r\n    const headers = dataRows[0];\r\n    const fundIndex = headers.indexOf('Fund');\r\n    if (fundIndex === -1) {\r\n        console.error(`No 'Fund' column found in sheet ${sheetName}`);\r\n        return;\r\n    }\r\n\r\n    // Save a dictionary of data for each fund for each sheet\r\n    const fundData = {};\r\n\r\n    dataRows.forEach(row => {\r\n        const fund = row[fundIndex];\r\n        if(fund && fund !== \"Fund\"){\r\n            if (!fundData[fund]) {\r\n                fundData[fund] = [];\r\n            }\r\n            const rowData = {};\r\n            headers.forEach((header, index) => {\r\n                rowData[removeNewLines(header)] = row[index];\r\n            });\r\n            fundData[fund].push(rowData);\r\n        }\r\n    });\r\n\r\n    // Save fund number and name as we go along\r\n    FundLookupTable.update(fundData);   \r\n    console.log('updating fund lookup table');       \r\n\r\n    Object.keys(fundData).forEach(fund => {\r\n        const key = `${SHEETS[sheetName]}_${fund}`;\r\n        localStorage.setItem(key, JSON.stringify(fundData[fund]));\r\n    });\r\n}\r\n\r\n/**\r\n * Processes the 'Drop-Down Menus' sheet to extract services data.\r\n * @param {Object} sheet - The sheet object from the workbook.\r\n */\r\nfunction processDropDownMenusSheet(sheet) {\r\n    const sheetData = XLSX.utils.sheet_to_json(sheet, { header: 1 });\r\n    const headerRow = sheetData[0];\r\n    const servicesIndex = headerRow.indexOf('Services');\r\n\r\n    // save drop down menu for later excel downloads\r\n    localStorage.setItem('dropdowns', JSON.stringify(sheetData));\r\n\r\n    if (servicesIndex === -1) {\r\n        console.error('Header \"Services\" not found');\r\n    } else {\r\n        const servicesColumn = sheetData.slice(1).map(row => row[servicesIndex]);\r\n        const cleanedServicesColumn = servicesColumn.filter(value => value != null);\r\n        Services.save(cleanedServicesColumn);\r\n    }\r\n}\r\n\r\n/**\r\n * Processes the 'Dept Summary' sheet to get and save the target for the general fund.\r\n * @param {Object} sheet - The sheet object from the workbook.\r\n */\r\nfunction processDeptSummarySheet(sheet) {\r\n    if(sheet[TARGET_CELL_ADDRESS]) {\r\n        const cellValue = sheet[TARGET_CELL_ADDRESS].v; // Access the cell value\r\n        localStorage.setItem('target', cellValue);\r\n    } else {\r\n        console.error(`Cell ${TARGET_CELL_ADDRESS} not found`);\r\n    }\r\n\r\n    // save the sheet to add to future excel downloads\r\n    const sheetData = XLSX.utils.sheet_to_json(sheet, { header: 1 });\r\n    // TODO: remove excess empty rows\r\n    const newSheetData = sheetData.map(row => row.slice(0, 3));\r\n    localStorage.setItem('dept-summary', JSON.stringify(newSheetData));\r\n}\r\n\r\n/**\r\n * Processes the 'FY{FISCAL_YEAR} Gold Book' sheet to initialize the Gold Book.\r\n * @param {Object} sheet - The sheet object from the workbook.\r\n */\r\nfunction processGoldBookSheet(sheet) {\r\n    GoldBook.init(sheet);\r\n}\r\n\r\nfunction processNewInitsSheet(sheet){\r\n    // Read in sheets\r\n    const rawData = XLSX.utils.sheet_to_json(sheet, { header: 1, defval: '' });\r\n    \r\n    // Clean the data by removing top rows with incomplete data\r\n    const dataRows = deleteTopRowsUntilFullData(rawData);\r\n\r\n    // Get new headers\r\n    const headers = dataRows[0];\r\n    // final data output\r\n    let fullData = []\r\n    \r\n    // Convert data to JSON form and filter out rows where first value is missing\r\n    dataRows.slice(1).forEach(row => { // Skip headers row\r\n        // skip any empty rows at the end\r\n        if ((row[0] != '') && (row[0] != '-')) {\r\n            const rowData = {};\r\n            headers.forEach((header, index) => {\r\n                rowData[removeNewLines(header)] = row[index];\r\n            });\r\n            // only keep supplemental initiatives \r\n            if (rowData['Baseline or Supplemental'].includes('Supplemental')){\r\n                fullData.push(rowData);\r\n            }\r\n        }\r\n    });\r\n    // save in local storage\r\n    localStorage.setItem('new-inits', JSON.stringify(fullData));\r\n}\r\n\r\n// Main function to read and process the workbook\r\nexport function processWorkbook(arrayBuffer) {\r\n    const workbook = readWorkbook(arrayBuffer);\r\n\r\n    workbook.SheetNames.forEach(sheetName => {\r\n        // Only convert sheets we need; treat new inits separately because they shouldn't save by fund\r\n        if (sheetName == Object.keys(SHEETS)[4]) {\r\n            const sheet = workbook.Sheets[sheetName];\r\n            processNewInitsSheet(sheet);\r\n        } else if (Object.keys(SHEETS).includes(sheetName)) {\r\n            const sheet = workbook.Sheets[sheetName];\r\n            processSheet(sheetName, sheet);\r\n        } else if (sheetName === 'Drop-Down Menus') {\r\n            const sheet = workbook.Sheets[sheetName];\r\n            processDropDownMenusSheet(sheet);\r\n        } else if (sheetName === 'Dept Summary') {\r\n            const sheet = workbook.Sheets[sheetName];\r\n            processDeptSummarySheet(sheet);\r\n        } else if (sheetName === `FY${FISCAL_YEAR} Gold Book`) {\r\n            const sheet = workbook.Sheets[sheetName];\r\n            processGoldBookSheet(sheet);\r\n        }\r\n    });\r\n\r\n    console.log('all excel data saved');\r\n}\r\n\r\n// Utility function to append a sheet to the workbook if data is present\r\nfunction appendSheetToWorkbook(workbook, data, sheetName) {\r\n    if (data.length > 0) {\r\n        const sheet = XLSX.utils.json_to_sheet(data);\r\n        XLSX.utils.book_append_sheet(workbook, sheet, sheetName);\r\n    }\r\n}\r\n\r\nexport function downloadXLSX() {\r\n    // grab data from baseline object\r\n    const baseline = new Baseline();\r\n    const workbook = XLSX.utils.book_new(); // Create a new workbook\r\n\r\n    // Initialize sheet data based on the names of each tab in the Excel doc\r\n    const sheetData = Object.keys(SHEETS).reduce((acc, key) => {\r\n        acc[key] = [];\r\n        return acc;\r\n    }, {});\r\n\r\n    // Aggregate all rows across funds and combine for each tab\r\n    baseline.funds.forEach(fund => {\r\n        Object.keys(SHEETS).forEach(sheetName => {\r\n            if (fund[SHEETS[sheetName]] && fund[SHEETS[sheetName]].table) {\r\n                sheetData[sheetName].push(...fund[SHEETS[sheetName]].table);\r\n            }\r\n        });\r\n    });\r\n\r\n    // Add initiatives data (which isn't stored by fund)\r\n    sheetData[Object.keys(SHEETS)[4]] = JSON.parse(localStorage.getItem('new-inits'));\r\n\r\n    // Create a tab for each table\r\n    Object.keys(sheetData).forEach(sheetName => {\r\n        appendSheetToWorkbook(workbook, sheetData[sheetName], sheetName);\r\n    });\r\n\r\n    // Add a tab for the GoldBook\r\n    XLSX.utils.book_append_sheet(workbook, GoldBook.xlsx(), `FY${FISCAL_YEAR} Gold Book`);\r\n\r\n    // add a tab for the drop downs and dept summary (just targets)\r\n    writeJSONtoNewTab('dropdowns', 'Drop-Down Menus', workbook); \r\n    writeJSONtoNewTab('dept-summary', 'Dept Summary', workbook);\r\n\r\n    // Generate a downloadable file\r\n    const wbout = XLSX.write(workbook, { bookType: 'xlsx', type: 'array' });\r\n    const blob = new Blob([wbout], { type: 'application/octet-stream' });\r\n\r\n    // Create a link and trigger the download\r\n    const link = document.createElement(\"a\");\r\n    link.href = URL.createObjectURL(blob);\r\n    link.download = \"Filled_Detail_Sheet.xlsx\";\r\n    document.body.appendChild(link);\r\n    link.click();\r\n    document.body.removeChild(link);\r\n}\r\n\r\nfunction writeJSONtoNewTab(storage_key, tab_name, workbook){\r\n    const data = JSON.parse(localStorage.getItem(storage_key));\r\n    console.log(data);\r\n    const worksheet = XLSX.utils.aoa_to_sheet(data);\r\n    XLSX.utils.book_append_sheet(workbook, worksheet, tab_name); \r\n}\r\n\r\nexport function excelSerialDateToJSDate(serial) {\r\n\r\n    if (!serial) { return null };\r\n    // Excel considers 1900-01-01 as day 1, but JavaScript's Date considers\r\n    // 1970-01-01 as day 0. Therefore, we calculate the number of milliseconds\r\n    // between 1900-01-01 and 1970-01-01.\r\n    const excelEpoch = new Date(Date.UTC(1899, 11, 30)); // JavaScript Consider December month as '11'\r\n    \r\n    // Calculate the JS date by adding serial days to the epoch date\r\n    const date = new Date(excelEpoch.getTime() + (serial * 24 * 60 * 60 * 1000));\r\n    \r\n    // Set the time part to zero (midnight)\r\n    date.setUTCHours(0, 0, 0, 0);\r\n    \r\n    // Return the date part of the ISO string\r\n    return date.toISOString().split('T')[0];\r\n}\r\n","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./tooltip.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./tooltip.css\";\n       export default content && content.locals ? content.locals : undefined;\n","import { FISCAL_YEAR } from '../../constants/';\r\nimport Cell from '../table/subcomponents/cells';\r\nimport { formatCurrency } from '../../utils/common_utils';\r\nimport CurrentPage from '../../models/current_page';\r\nimport { excelSerialDateToJSDate } from '../../utils/XLSX_handlers';\r\n\r\nimport './tooltip.css'\r\n\r\nfunction hideTooltip() {\r\n    document.getElementById('tooltip').style.visibility = 'hidden';\r\n}\r\n\r\nfunction showTooltip() {\r\n    document.getElementById('tooltip').style.visibility = 'visible';\r\n}\r\n\r\nfunction editTooltipText(newText){\r\n    // edit text to display inside tooltip\r\n    const tooltip = document.getElementById('tooltip');\r\n    tooltip.innerHTML = newText;\r\n}\r\n\r\nfunction showAccountString(row){\r\n    const approp = Cell.getText(row, 'approp-name');\r\n    const cc =  Cell.getText(row, 'cc-name');\r\n    const obj =  Cell.getText(row, 'object-name');\r\n    const fund = Cell.getText(row, 'fund-name');\r\n    var message = \r\n        `<strong>Fund</strong>: ${fund}<br>\r\n        <strong>Appropriation</strong>: ${approp}<br>\r\n        <strong>Cost Center</strong>: ${cc}`;\r\n    if (obj) { message += `<br><strong>Object</strong>: ${obj}`}\r\n    message += `<br> If you believe this account string is incorrect, please leave a comment \r\n        in the comment column.`\r\n    editTooltipText(message);\r\n}\r\n\r\nfunction showSalaryProjection(row){\r\n    const general_increase = Cell.getText(row, 'general-increase-rate');\r\n    const merit_increase =  Cell.getText(row, 'merit-increase-rate');\r\n    const current_salary = Cell.getValue(row, 'current-salary');\r\n    const proj_salary = Cell.getValue(row, 'avg-salary');\r\n    if (current_salary){\r\n        var message = `The average salary/wage for this position was \r\n            ${formatCurrency(current_salary)} as of September 20${FISCAL_YEAR-2}. \r\n            Given a ${general_increase*100}% general increase rate and a ${merit_increase*100}% \r\n            merit increase, the FY${FISCAL_YEAR} projection for this position's average \r\n            annual salary/wage is ${formatCurrency(proj_salary)}. This projection takes into \r\n            account the number of vacant vs. full positions.`;\r\n    } else {\r\n        var message = `The average salary/wage for this position was \r\n            unknown as of September 20${FISCAL_YEAR-2}, or the position\r\n            did not exist. The FY${FISCAL_YEAR} projection for this position's \r\n            average annual salary/wage is ${formatCurrency(proj_salary)}.`\r\n    }\r\n\r\n    editTooltipText(message);\r\n}\r\n\r\nfunction showFinalPersonnelCost(row){\r\n    const proj_salary = Cell.getValue(row, 'avg-salary');\r\n    const ftes = Cell.getText(row, 'baseline-ftes');\r\n    const fringe = parseFloat(Cell.getText(row, 'fringe'));\r\n    const avg_benefits = proj_salary * fringe;\r\n    const message = `The total cost captures ${ftes} position(s) at\r\n                    an annual salary/wage of ${formatCurrency(proj_salary)}, \r\n                    plus fringe benefits that cost ${formatCurrency(avg_benefits)} \r\n                    per position per year, on average.`\r\n    editTooltipText(message);\r\n}\r\n\r\nfunction showFICA(row){\r\n    const fica = parseFloat(Cell.getText(row, 'fica'));\r\n    const ficaPercentage = (fica * 100).toFixed(2);\r\n    const message = `This total is overtime / holiday / shift premium pay, plus FICA (payroll tax), \r\n                     which is ${ficaPercentage}% for this cost center.`\r\n    editTooltipText(message);\r\n}\r\n\r\nfunction showCPA(row){\r\n    const cpa = parseFloat(Cell.getText(row, 'cpa'));\r\n    const description = Cell.getText(row, 'cpa-description');\r\n    const vendor = Cell.getText(row, 'vendor');\r\n    var contract_end = Cell.getText(row, 'contract-end');\r\n    // convert to normal date format from excel \r\n    contract_end = excelSerialDateToJSDate(contract_end);\r\n    const remaining = Cell.getValue(row, 'remaining');\r\n    if (cpa) {\r\n        var message = `<strong>CPA #${cpa}</strong>`;\r\n    } else {\r\n        var message = `No CPA`;\r\n    }\r\n    if (vendor) {message += `<br><strong>Vendor</strong>: ${vendor}`};\r\n    if (description) {message += `<br><strong>Description</strong>: ${description}`};\r\n    if (contract_end) {message += `<br><strong>Contract End Date</strong>: ${contract_end}`}\r\n    if (remaining) {message += `<br><strong>Amount Remaining on Contract</strong>: ${formatCurrency(remaining)}`}\r\n\r\n    editTooltipText(message);\r\n}\r\n\r\nfunction link(element, displayFn) {\r\n\r\n    // add class to show cell with an underline, etc\r\n    element.classList.add('tooltip-cell');\r\n\r\n    // Create and append (detail)\r\n    const detail = document.createElement('span');\r\n    detail.classList.add('detail');\r\n    detail.textContent = '(detail)';\r\n    element.appendChild(detail);\r\n\r\n    // add event listener to show tooltip on mouseover\r\n    element.addEventListener('click', function (event) {\r\n        const row = event.target.closest('tr');\r\n        displayFn(row);\r\n        showTooltip();\r\n    });\r\n    // and hide when mouse moves off\r\n    element.addEventListener('mouseout', function () {\r\n        hideTooltip();\r\n    });\r\n    // Update tooltip position on mouse move\r\n    element.addEventListener('mousemove', function (event) {\r\n        const tooltip = document.getElementById('tooltip');\r\n        tooltip.style.top = (event.clientY + 10) + 'px';\r\n        tooltip.style.left = (event.clientX + 10) + 'px';\r\n    });\r\n}\r\n\r\nfunction linkAccountStringCol() {\r\n    // get all relevant cells\r\n    document.querySelectorAll('td.account-string').forEach( (cell) => {\r\n        link(cell, showAccountString);\r\n    })\r\n}\r\n\r\nfunction linkSalaryCol() {\r\n    // get all relevant cells\r\n    document.querySelectorAll('td.avg-salary').forEach( (cell) => {\r\n        link(cell, showSalaryProjection);\r\n    })\r\n}\r\n\r\nfunction linkTotalPersonnelCostCol() {\r\n    // get all relevant cells\r\n    document.querySelectorAll('td.total-baseline').forEach( (cell) => {\r\n        link(cell, showFinalPersonnelCost);\r\n    })\r\n}\r\n\r\nfunction linkTotalOTCol() {\r\n    // get all relevant cells\r\n    document.querySelectorAll('td.total').forEach( (cell) => {\r\n        link(cell, showFICA);\r\n    })\r\n}\r\n\r\nfunction linkCPACol() {\r\n    // get all relevant cells\r\n    document.querySelectorAll('td.cpa').forEach( (cell) => {\r\n        link(cell, showCPA);\r\n    })\r\n}\r\n\r\nexport const Tooltip = {\r\n\r\n    hide : hideTooltip,\r\n    show : showTooltip,\r\n\r\n    linkAll : () => {\r\n        linkAccountStringCol();\r\n        switch(CurrentPage.load()){\r\n            case 'personnel' :\r\n                // linkAccountStringCol();\r\n                linkSalaryCol();\r\n                linkTotalPersonnelCostCol();\r\n                break;\r\n            case 'overtime':\r\n                linkTotalOTCol();\r\n                // linkAccountStringCol();\r\n                break;\r\n            case 'nonpersonnel':\r\n                // linkAccountStringCol();\r\n                linkCPACol();  \r\n                break;\r\n            // case 'revenue':\r\n            //     linkAccountStringCol();\r\n            //     break;\r\n            // case 'new-inits':\r\n            //     linkAccountStringCol();\r\n            //     break;\r\n            default:\r\n                break;\r\n\r\n        }\r\n    },\r\n\r\n    unlink : function() {\r\n        let details = document.querySelectorAll('.detail');\r\n        details.forEach( (span) => {\r\n            span.remove();\r\n        })\r\n    }\r\n}\r\n\r\nexport default Tooltip","export async function fetchJSON(jsonFilePath) {\r\n  return fetch(jsonFilePath)\r\n    .then(response => {\r\n      if (!response.ok) {\r\n        throw new Error('Network response was not ok');\r\n      }\r\n      return response.json();\r\n    });\r\n}\r\n\r\nexport function convertToJSON(table, colsToRemove = []){\r\n    const rows = table.rows;\r\n    // Extract headers from the first row\r\n    const headerRow = rows[0].cells;\r\n    const headers = [];\r\n    for (let j = 0; j < headerRow.length; j++) {\r\n        headers.push(headerRow[j].innerText);\r\n    }\r\n\r\n    // initialize data\r\n    var tableData = [];\r\n\r\n    for (var i = 1; i < rows.length; i++) {\r\n        const cols = rows[i].cells;\r\n        const rowData = {};\r\n            headers.forEach((header, index) => {\r\n                if (colsToRemove.includes(header)){\r\n                    return;\r\n                }\r\n                else if (cols[index].classList.contains('cost')) {\r\n                    rowData[header] = cols[index].getAttribute('value');\r\n                } else {\r\n                    rowData[header] = cols[index].innerText;\r\n                }\r\n            });\r\n        tableData.push(rowData);\r\n    }\r\n    return JSON.stringify(tableData);\r\n}\r\n\r\n\r\n  ","import './table.css';\r\n\r\nimport Buttons from './subcomponents/buttons.js'\r\nimport Cell from './subcomponents/cells.js'\r\nimport Columns from './subcomponents/columns.js'\r\nimport Header from './subcomponents/headers.js'\r\nimport Rows from './subcomponents/rows.js'\r\nimport Data from './subcomponents/data.js'\r\nimport Filter from './subcomponents/filters.js';\r\n\r\nimport Tooltip from '../tooltip/tooltip.js';\r\nimport { convertToJSON } from \"../../utils/JSON_data_handlers.js\";\r\nimport Sidebar from '../sidebar/sidebar.js';\r\nimport CurrentFund from '../../models/current_fund.js';\r\nimport CurrentPage from '../../models/current_page.js';\r\n\r\nfunction adjustTableWidth(width_pct){\r\n    const table = document.getElementById('main-table');\r\n    table.style.width = width_pct;\r\n}\r\n\r\nfunction showTable(){\r\n    const tableContainer = document.querySelector('.table-container');\r\n    tableContainer.innerHTML = Table.html;\r\n    const table = document.getElementById('main-table');\r\n    table.style.display = 'table';\r\n}\r\n\r\nfunction hideTable(){\r\n\r\n    // delete table object from table container\r\n    const tableContainer = document.querySelector('.table-container');\r\n    tableContainer.innerHTML = '';\r\n    Buttons.AddRow.hide();\r\n}\r\n\r\nfunction saveTableData() {\r\n    // remove the detail text\r\n    Tooltip.unlink();\r\n    // get table\r\n    var table = document.getElementById('main-table');\r\n    // determine save_as name\r\n    if (CurrentFund.number()) {\r\n        var save_as = `${CurrentPage.load()}_${CurrentFund.number()}`;\r\n    } else {\r\n        var save_as = CurrentPage.load();\r\n    }\r\n    localStorage.setItem(save_as, convertToJSON(table, ['Edit']));\r\n    // update sidebar with new data\r\n    Sidebar.updateTotals();\r\n    // relink, depending on page\r\n    Tooltip.linkAll();\r\n}\r\n\r\nconst Table = {\r\n    html : `\r\n        <table class=\"table table-bordered mt-3 display\" id=\"main-table\">\r\n            <thead class=\"thead-dark\"></thead>\r\n            <tbody></tbody>\r\n        </table>`,\r\n    Buttons : Buttons,\r\n    Cell : Cell,\r\n    Columns : Columns,\r\n    Header : Header,\r\n    Rows : Rows,\r\n    Data : Data,\r\n    Filter : Filter,\r\n    // functions\r\n    adjustWidth : function(width_pct){\r\n        adjustTableWidth(width_pct)\r\n    },\r\n    clear : hideTable,\r\n    hide : hideTable,\r\n    show : showTable,\r\n    save : saveTableData\r\n}\r\n\r\nexport default Table;","import './accordion.css'\r\n\r\n\r\nimport {Baseline, CurrentFund, Fund, Supplemental, FundLookupTable} from '../../models'\r\nimport { formatCurrency, cleanString } from \"../../utils/common_utils.js\";\r\nimport Table from \"../table/table.js\";\r\nimport { visitPage } from '../../views/view_logic.js';\r\n\r\nfunction redirectForEdit(){\r\n    const row = document.querySelector(`.active-editing`);\r\n    const table = row.parentElement;\r\n    const section = table.closest('.summary-container');\r\n    // new initiative edits should all redirect to the new-inits page\r\n    if (section.id == 'supp-accordion'){\r\n        visitPage('new-inits');\r\n    }\r\n    else {\r\n        // Split the string into parts using '-' as the delimiter; retain fund as 1st numeric segment\r\n        const fund = table.id.split('-')[1]\r\n        \r\n        CurrentFund.update(fund);\r\n        const lineItem = row.querySelector('.line-item').textContent;\r\n        // visit the correct page for editing\r\n        switch(lineItem){\r\n            case 'Personnel Expenditures':\r\n                visitPage('personnel');\r\n                break;\r\n            case 'Non-Personnel Expenditures':\r\n                visitPage('nonpersonnel');\r\n                break;\r\n            case 'Revenues':\r\n                visitPage('revenue');\r\n                break;\r\n            case 'Overtime Expenditures':\r\n                visitPage('overtime');\r\n                break;\r\n            default:\r\n                console.error('Name of line item in table does not match a page destination.')\r\n        }\r\n    }\r\n}\r\n\r\nconst ExpenseTable = {\r\n    table_id : (account_string) => { return `table-${account_string}` },\r\n    init(account_string) {\r\n        // create empty table and put it in the accordion\r\n        var table = document.createElement('table');\r\n        table.id = this.table_id(account_string);\r\n        table.classList.add('accordion-table');\r\n        var parent = document.querySelector(`#string_${account_string}_content .accordion-body`);\r\n        parent.appendChild(table);\r\n    },\r\n    createNewCell(content, row, className) {\r\n        const newCell = document.createElement('td');\r\n        newCell.innerHTML = content;\r\n        newCell.classList.add(className);\r\n        row.appendChild(newCell);\r\n    },\r\n    addRow(account_string, row_name, number){\r\n        var table = document.getElementById(this.table_id(account_string));\r\n        var new_row = document.createElement('tr');\r\n        table.appendChild(new_row);\r\n        // Create a cell for the line item label\r\n        this.createNewCell(row_name, new_row, 'line-item');\r\n        // create a cell for the amount\r\n        this.createNewCell(formatCurrency(number), new_row, 'cost');\r\n        // create Edit button \r\n        var button = '';\r\n        if (row_name != 'Net Expenditures (Revenues)'){\r\n            button = Table.Buttons.Edit.html;\r\n        }\r\n        this.createNewCell(button, new_row);\r\n    },\r\n    fillFromFund(fund) {\r\n        // use just fund as account string to initialize table inside accordion\r\n        this.init(fund);\r\n        const fundObject = new Fund(fund);\r\n\r\n        // Add a row for each appropriation in the fund\r\n        const id = cleanString(fund);\r\n\r\n        fundObject.getAppropriations().forEach( appropObj => {\r\n            // if the total for the appropriation is > $0, add an accordion for all the CCs\r\n            if ( appropObj.total() != 0 ){\r\n                Item.add(appropObj.accountString(), `#string_${id}_content .accordion-body`);\r\n                Item.updateHeader(appropObj.name(), appropObj.accountString(), appropObj.total());\r\n                this.fillFromApprop(appropObj);\r\n            }\r\n        })\r\n    },\r\n    fillFromApprop(appropObj){\r\n        // initialize the table object\r\n        this.init(appropObj.accountString());\r\n        // add a collapsible row for each cost center\r\n        appropObj.getCostCenters().forEach( ccObj => {\r\n            if (ccObj.getTotal() != 0 ){\r\n                Item.add(ccObj.accountString(), `#string_${appropObj.accountString()}_content .accordion-body`);\r\n                Item.updateHeader(ccObj.getName(), ccObj.accountString(), ccObj.getTotal());\r\n                this.fillFromCC(ccObj);\r\n            }\r\n        })\r\n        \r\n    },\r\n    fillFromCC(ccObj){\r\n        // initialize a table and summarize the line items\r\n        this.init(ccObj.accountString());\r\n        this.addRow(ccObj.accountString(), 'Personnel Expenditures', ccObj.getPersonnelCost());\r\n        this.addRow(ccObj.accountString(), 'Overtime Expenditures', ccObj.getOvertimeCost());\r\n        this.addRow(ccObj.accountString(), 'Non-Personnel Expenditures', ccObj.getNonPersonnelCost());\r\n        this.addRow(ccObj.accountString(), 'Revenues', ccObj.getRevenue());\r\n        this.addRow(ccObj.accountString(), 'Net Expenditures (Revenues)', ccObj.getTotal());\r\n    },\r\n    fillFromInit(program) {\r\n        // Fill out info for each supplemental init\r\n        this.init(program.id());\r\n        this.addRow(program.id(), 'Estimated Revenue', program.revenue());\r\n        this.addRow(program.id(), 'Personnel Expenditures', program.personnel());\r\n        this.addRow(program.id(), 'Non-Personnel Operating', program.operating());\r\n        this.addRow(program.id(), 'Non-Personnel Capital', program.capital());\r\n        this.addRow(program.id(), 'Total Expenditures', program.total());\r\n    }\r\n}\r\n\r\nconst Item = {\r\n    accountString(fund, approp = '', cc = '') {\r\n        var account_string = cleanString(fund);\r\n        if (approp) { account_string += approp };\r\n        if (cc) { account_string += cc };\r\n        return account_string;\r\n    },\r\n    html(account_string){\r\n        return `<h2 class=\"accordion-header\" id=\"string_${account_string}_header\">\r\n                    <button class=\"accordion-button collapsed\" type=\"button\" data-bs-toggle=\"collapse\" data-bs-target=\"#string_${account_string}_content\" aria-expanded=\"false\" aria-controls=\"string_${account_string}\">\r\n                        <span class=\"name\"></span>\r\n                        <span class=\"amount\"></span>\r\n                    </button>\r\n                </h2>\r\n                <div id=\"string_${account_string}_content\" class=\"accordion-collapse collapse\" aria-labelledby=\"string_${account_string}_header\">\r\n                    <div class=\"accordion-body\"></div>\r\n                </div>`\r\n    },\r\n    add : function(account_string, accordion_query) {\r\n        // get accordion and add a new item to it\r\n        const parent = document.querySelector(accordion_query);\r\n        const item_element = document.createElement('div');\r\n        item_element.classList.add('accordion-item');\r\n        item_element.innerHTML = this.html(account_string);\r\n        parent.appendChild(item_element);\r\n    },\r\n    ExpenseTable : ExpenseTable,\r\n    updateHeader : function(title, account_string, new_amount) {\r\n        const header_btn = document.querySelector(`#string_${account_string}_header button`);\r\n        header_btn.querySelector('span.name').textContent = title;\r\n        header_btn.querySelector('span.amount').textContent = formatCurrency(new_amount);\r\n    }\r\n}\r\n\r\nconst AddInitButton = {\r\n    init() {\r\n        const btn = document.querySelector('.btn-add-init');\r\n        btn.addEventListener('click', function(){\r\n            visitPage('new-inits');\r\n        })\r\n    }\r\n}\r\n\r\nexport const Accordion = {\r\n    Item : Item,\r\n    AddInitButton: AddInitButton,\r\n    hide : function(){\r\n        document.querySelector('#accordion-div').style.display = 'none';\r\n        // reset to delete content\r\n        document.querySelector('#baseline-accordion .summary-accordion').innerHTML = '';\r\n        document.querySelector('#supp-accordion .summary-accordion').innerHTML = '';\r\n    },\r\n    show : function(){\r\n        document.querySelector('#accordion-div').style.display = 'block';\r\n    },\r\n    async createBaseline(){\r\n        var funds = FundLookupTable.listFunds();\r\n        funds.forEach(fund => {\r\n            Item.add(fund, '#baseline-accordion .summary-accordion');\r\n            Item.ExpenseTable.fillFromFund(fund);\r\n            const fundObject = new Fund(fund);\r\n            Item.updateHeader(`Fund ${FundLookupTable.getName(fund)}`, fund, fundObject.getTotal());\r\n        });\r\n\r\n        // color-code GF baseline\r\n        const GF = new Fund(1000);\r\n        // text to color code\r\n        let topline = document.querySelector('#string_1000_header .amount');\r\n        if (GF.getTotal() <= Baseline.target()){\r\n            topline.style.color = 'green';\r\n            topline.style.weight = 4;\r\n        } else {\r\n           topline.style.color = 'red';\r\n        }\r\n    },\r\n    createSupp() {\r\n        const supp = new Supplemental;\r\n        supp.initiatives.forEach(program => {\r\n            Item.add(program.id(), '#supp-accordion .summary-accordion');\r\n            Item.ExpenseTable.fillFromInit(program);\r\n            Item.updateHeader(program.name, program.id(), program.total());\r\n        });\r\n    },\r\n    updateTopLines() {\r\n        // adjuse baseline\r\n        // const baseline = new Baseline;\r\n        const baseline = new Baseline;\r\n        const baselineAmount = document.querySelector('#baseline-title .top-line-amount')\r\n        baselineAmount.textContent = formatCurrency(baseline.total());\r\n        // adjust supplementals\r\n        const supp = new Supplemental;\r\n        const suppAmount = document.querySelector('#supp-title .top-line-amount')\r\n        suppAmount.textContent = formatCurrency(supp.total());\r\n\r\n    },\r\n    build() {\r\n        this.createBaseline();\r\n        this.createSupp();\r\n        // initialize edit buttons\r\n        Table.Buttons.Edit.init(redirectForEdit);\r\n        this.AddInitButton.init();\r\n        this.updateTopLines();\r\n    }\r\n}\r\n\r\n\r\nexport default Accordion;\r\n","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./file_upload.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./file_upload.css\";\n       export default content && content.locals ? content.locals : undefined;\n","// file_upload.js\r\nimport './file_upload.css';\r\nimport { processWorkbook } from \"../../utils/XLSX_handlers.js\";\r\nimport NavButtons from '../nav_buttons/nav_buttons.js';\r\n\r\nexport const FileUpload = {\r\n    init : function() {\r\n        NavButtons.Next.disable();\r\n        const inputObject = document.getElementById('file-input');\r\n        inputObject.addEventListener('change', function(event) { readXL(event) });\r\n    },\r\n    show : function(){\r\n        const inputObject = document.getElementById('file-input');\r\n        inputObject.style.display = '';\r\n    },\r\n    hide : function(){\r\n        const inputObject = document.getElementById('file-input');\r\n        inputObject.style.display = 'none';\r\n    }\r\n};\r\n\r\nfunction readXL(event) {\r\n    const file = event.target.files[0]; // read uploaded file\r\n    const spinner = document.getElementById('upload-spinner'); // get the spinner element\r\n\r\n    if (file) {\r\n        // Show the spinner\r\n        spinner.style.display = 'block';\r\n\r\n        // read in new data\r\n        const reader = new FileReader();\r\n        reader.onload = function(e) {\r\n            const arrayBuffer = e.target.result;\r\n\r\n            try {\r\n                processWorkbook(arrayBuffer);\r\n\r\n                // Hide the spinner once processing is done\r\n                spinner.style.display = 'none';\r\n                NavButtons.Next.enable(); // Enable the next button after processing\r\n            } catch (error) {\r\n                console.error('Error processing workbook:', error);\r\n\r\n                // Hide the spinner in case of an error\r\n                spinner.style.display = 'none';\r\n                NavButtons.Next.enable(); // Ensure the button is re-enabled in case of an error\r\n            }\r\n        };\r\n        reader.onerror = function(err) {\r\n            console.error('Error reading file:', err);\r\n\r\n            // Hide the spinner in case of an error\r\n            spinner.style.display = 'none';\r\n            NavButtons.Next.enable(); // Ensure the button is re-enabled in case of an error\r\n        };\r\n\r\n        reader.readAsArrayBuffer(file); // Read the file as an ArrayBuffer\r\n    }\r\n}\r\n\r\nexport default FileUpload;","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./modal.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./modal.css\";\n       export default content && content.locals ? content.locals : undefined;\n","import './modal.css';\r\n\r\nfunction clearModal() {\r\n    updateModalTitle('');\r\n    document.getElementById('modal-body').innerHTML = '';\r\n}\r\n\r\nfunction hideModal() {\r\n    $('#main-modal').modal('hide');\r\n}\r\n\r\nfunction showModal() {\r\n    $('#main-modal').modal('show');\r\n}\r\n\r\nfunction showModalHandler() {\r\n    showModal('main-modal');\r\n}\r\n\r\nconst Submit = {\r\n    handler: null,\r\n\r\n    init: function(onSubmit) {\r\n        const modal = document.getElementById('main-modal');\r\n\r\n        // Deinitialize first to avoid multiple bindings\r\n        this.deinit();\r\n\r\n        this.handler = function(event) {\r\n            event.preventDefault();\r\n            onSubmit(event);\r\n        };\r\n\r\n        // Adding the handler as the event listener\r\n        modal.addEventListener('submit', this.handler);\r\n\r\n        // add event listener to enable close x\r\n        const x = modal.querySelector('#modal-close-x');\r\n        x.addEventListener('click', hideModal);\r\n    },\r\n\r\n    deinit: function() {\r\n        const modal = document.getElementById('main-modal');\r\n        if (this.handler !== null) {\r\n            // Removing the event listener and clear the handler\r\n            modal.removeEventListener('submit', this.handler);\r\n            this.handler = null;\r\n        }\r\n\r\n        // remove event listener to enable close x\r\n        const x = modal.querySelector('#modal-close-x');\r\n        x.removeEventListener('click', hideModal);\r\n    }\r\n};\r\n\r\nconst Link = {\r\n    add : function(button_id){\r\n        document.getElementById(button_id).addEventListener('click', showModalHandler);\r\n    },\r\n    remove : function(button_id){\r\n        document.getElementById(button_id).removeEventListener('click', showModalHandler);\r\n    }\r\n};\r\n\r\nfunction updateModalTitle(title) {\r\n    document.getElementById('modal-title').textContent = title;\r\n}\r\n\r\nconst Title = {\r\n    update : function(title) {\r\n        updateModalTitle(title);\r\n    }\r\n};\r\n\r\nexport const Modal = {\r\n    hide: hideModal,\r\n    show: showModal,\r\n    clear: clearModal,\r\n    Title: Title,\r\n    Link: Link,\r\n    Submit: Submit\r\n};\r\n\r\nexport default Modal;","import './body.css';\r\n\r\nimport Welcome from '../../components/welcome/welcome.js'\r\nimport Accordion from '../accordion/accordion.js';\r\nimport FileUpload from '../file_upload/file_upload.js';\r\nimport Modal from '../modal/modal.js';\r\nimport NavButtons from '../nav_buttons/nav_buttons.js';\r\nimport Prompt from '../prompt/prompt.js';\r\nimport Sidebar from '../sidebar/sidebar.js';\r\nimport Table from '../table/table.js';\r\nimport Tooltip from '../tooltip/tooltip.js';  \r\n\r\nfunction resetPage() {\r\n    // hide everything in the body\r\n    Welcome.hide();\r\n    Modal.clear();\r\n    Modal.hide();\r\n    NavButtons.hide();\r\n    Prompt.hide();\r\n    Table.hide();\r\n    Sidebar.hide();\r\n    Accordion.hide();\r\n    FileUpload.hide();\r\n    Tooltip.hide();\r\n\r\n    // reset filter dropdowns\r\n    Table.Filter.deleteAll();\r\n\r\n    Prompt.Buttons.reset();\r\n    // disable submit button\r\n    Modal.Submit.deinit();\r\n}\r\n\r\nexport const Body = {\r\n    reset : resetPage\r\n}\r\n\r\nexport default Body;","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./header.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./header.css\";\n       export default content && content.locals ? content.locals : undefined;\n","import './header.css';\r\n\r\nimport CurrentFund from '../../models/current_fund';\r\nimport { FISCAL_YEAR } from '../../constants';\r\n\r\nexport const Title = {\r\n    update : function(title){\r\n        document.getElementById(\"title\").textContent = title;\r\n    },\r\n    default() {\r\n        this.update(`FY${FISCAL_YEAR} Budget Request`)\r\n    }\r\n}\r\n\r\nexport const Subtitle = {\r\n    update : function(subtitle){\r\n        // get current fund\r\n        var fund = CurrentFund.name();\r\n        if (fund){\r\n            var subtitle = `${subtitle}: ${fund}`;\r\n        }\r\n        document.getElementById(\"subtitle\").textContent = subtitle;\r\n    }\r\n}","\n      import API from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../../node_modules/css-loader/dist/cjs.js!./form.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\noptions.insert = insertFn.bind(null, \"head\");\noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../../node_modules/css-loader/dist/cjs.js!./form.css\";\n       export default content && content.locals ? content.locals : undefined;\n","// function to add questions to forms\r\n\r\nimport Dropdown from \"./dropdown\";\r\n\r\n// inputType is for validation ('number' or 'text', etc)\r\nfunction appendFormElement(label, inputEl, inputId, required) {\r\n\r\n    // change if we want forms elsewhere\r\n    const form = document.getElementById('new-form');\r\n\r\n    // create outer wrapper for element\r\n    const wrapper = document.createElement('div');  \r\n\r\n    // label question\r\n    const labelEl = document.createElement('label');\r\n    labelEl.textContent = label;\r\n\r\n    // mark as required if applicable\r\n    inputEl.required = required;\r\n\r\n    // If an ID is provided, set it on the element\r\n    if (inputId) {\r\n        //inputEl.id = `input-${inputId}`;\r\n        inputEl.id = inputId;\r\n    }\r\n\r\n    // create validation text/ a place to display errors\r\n    const validationText = document.createElement('p');\r\n    validationText.id = `${inputId}-validation`;\r\n    validationText.classList.add('error-message');\r\n    validationText.style.color = 'red';\r\n  \r\n    // add elements\r\n    wrapper.appendChild(labelEl);\r\n    wrapper.appendChild(inputEl);\r\n    wrapper.appendChild(validationText);\r\n    form.appendChild(wrapper);\r\n} \r\n\r\nexport const NewField = {\r\n    shortText : function(label, inputId, required = false) {\r\n        const inputEl = document.createElement('input');\r\n        inputEl.type = 'text';\r\n        appendFormElement(label, inputEl, inputId, required);\r\n    },\r\n    longText : function(label, inputId, required = false) {\r\n        const inputEl = document.createElement('textarea');\r\n        appendFormElement(label, inputEl, inputId, required);\r\n    },\r\n    numericInput : function(label, inputId, required = false) {\r\n        const inputEl = document.createElement('input');\r\n        inputEl.type = 'number';\r\n        appendFormElement(label, inputEl, inputId,required);\r\n    },\r\n    dropdown : function(label, inputId, optionArray, required = false){\r\n        var inputEl = Dropdown.create(optionArray);\r\n        appendFormElement(label, inputEl, inputId, required);\r\n    }\r\n}\r\n\r\nexport default NewField;","function addSubmitButtonToForm(form_id) {\r\n    // Find the form by its ID\r\n    const form = document.getElementById(form_id);\r\n  \r\n    // Create the container `div` for the button\r\n    const buttonContainer = document.createElement('div');\r\n    buttonContainer.id = 'submit-btn-container';\r\n    \r\n    // Create the submit input\r\n    const submitInput = document.createElement('input');\r\n    submitInput.className = 'btn btn-submit'; // Use appropriate class for your design\r\n    submitInput.type = 'submit';\r\n    submitInput.value = 'Submit';\r\n\r\n    // Append the submit input to the container\r\n    buttonContainer.appendChild(submitInput);\r\n    \r\n    // Append the container to the form\r\n    form.appendChild(buttonContainer);\r\n}\r\n\r\nexport const SubmitButton = {\r\n    add : function() { addSubmitButtonToForm('new-form') }\r\n}\r\n\r\nexport default SubmitButton;","import './form.css';\r\n\r\nimport Dropdown from \"./subcomponents/dropdown.js\";\r\nimport NewField from \"./subcomponents/fields.js\";\r\nimport SubmitButton from \"./subcomponents/submit.js\";\r\n\r\nfunction fetchAllResponses(event) {\r\n\r\n    // Assuming `event.target` is the form itself\r\n    const form = event.target;\r\n    \r\n    // Initialize an empty array to hold the input values\r\n    let formData = {};\r\n    \r\n    // Loop through each form element\r\n    for (let i = 0; i < form.elements.length; i++) {\r\n      const element = form.elements[i];\r\n      \r\n      // Exclude elements that aren't inputs, textareas, or select\r\n      if (element.tagName === 'INPUT' ||\r\n          element.tagName === 'TEXTAREA' ||\r\n          element.tagName === 'SELECT') {\r\n        // Exclude input types that are not considered for submission (such as `submit`)\r\n        if (element.type !== 'submit' && element.type !== 'button') {\r\n          formData[element.id] = element.value;\r\n        }\r\n      }\r\n    }\r\n    \r\n    form.reset();\r\n    return formData;\r\n}\r\n\r\nfunction addForm(element_id = 'modal-body', form_id = 'new-form') {\r\n  \r\n  const target_elem = document.getElementById(element_id);\r\n\r\n  // create form\r\n  const form = document.createElement('form');\r\n  form.setAttribute('id', form_id);\r\n\r\n  // Append the form to the modal body\r\n  target_elem.appendChild(form);\r\n\r\n}\r\n\r\nexport const Form = {\r\n  new : function(parent_elem_id) { addForm(parent_elem_id, 'new-form') },\r\n  fetchAllResponses : function(event) { return fetchAllResponses(event) },\r\n  NewField : NewField,\r\n  Dropdown : Dropdown,\r\n  SubmitButton : SubmitButton\r\n}\r\n\r\nexport default Form;","import Prompt from \"../components/prompt/prompt.js\";\r\nimport Sidebar from \"../components/sidebar/sidebar.js\";\r\nimport NavButtons from \"../components/nav_buttons/nav_buttons.js\";\r\nimport Body from \"../components/body/body.js\";\r\nimport { Subtitle, Title } from \"../components/header/header.js\";\r\nimport Table from \"../components/table/table.js\";\r\nimport Form from \"../components/form/form.js\";\r\nimport Modal from \"../components/modal/modal.js\";\r\n\r\nimport { CurrentPage, AccountString } from '../models/'\r\n\r\n\r\nexport class View {\r\n\r\n    constructor() {\r\n        // page state in local storage\r\n        this.page_state = '';\r\n\r\n        // whether to display\r\n        this.navButtons = true; \r\n        this.sidebar = true;\r\n\r\n        // text to show in the prompt area\r\n        this.prompt = null;\r\n\r\n        // subtitle text\r\n        this.subtitle = '';\r\n\r\n        // table object of class ViewTable or null\r\n        this.table = null;\r\n    }\r\n\r\n    visit() {\r\n        // update page state\r\n        CurrentPage.update(this.page_state);\r\n\r\n        // start with a blank page\r\n        Body.reset();\r\n\r\n        // add default title\r\n        Title.default();\r\n        \r\n        // default to showing navbuttons\r\n        if (this.navButtons) { NavButtons.show(); };\r\n        \r\n        // default to showing sidebar\r\n        if (this.sidebar) { Sidebar.show() };\r\n\r\n        // initialize prompt text and buttons\r\n        if (this.prompt) { Prompt.Text.update(this.prompt) };\r\n        \r\n        // initialize table\r\n        if (this.table) { this.table.build(); }\r\n\r\n        // show page subtitle\r\n        if (this.subtitle) { Subtitle.update(this.subtitle) };\r\n    }\r\n\r\n    cleanup() { return; }\r\n\r\n}\r\n\r\nexport class ViewTable {\r\n\r\n    constructor(){\r\n        // Ensure methods retain the correct `this` context\r\n        this.submitNewRow = this.submitNewRow.bind(this);\r\n\r\n        this.columns = [\r\n            { title: 'Account String', className: 'account-string' },\r\n            { title: 'Appropriation Name', className: 'approp-name', hide: true },\r\n            { title: 'Appropriation', className: 'approp', hide: true },\r\n            { title: 'Cost Center Name', className: 'cc-name',  hide: true },\r\n            { title: 'Cost Center', className: 'cc',  hide: true },\r\n            { title: 'Fund Name', className: 'fund-name',  hide: true },\r\n            { title: 'Fund', className: 'fund',  hide: true },\r\n            { title: 'Edit', className: 'edit' },\r\n        ];\r\n        \r\n        // whether to add an edit column\r\n        this.addEdit = true ;\r\n\r\n        // message to show if there's no saved data\r\n        this.noDataMessage = null;\r\n   \r\n        // text to show for new row button\r\n        this.addButtonText = null ;\r\n\r\n        // whether to show as a datatable\r\n        this.dataTable = true;\r\n    }\r\n\r\n    async refreshData() {\r\n        \r\n        // check for data\r\n        await Table.Data.load();\r\n\r\n        // create a datatable object\r\n        if(this.dataTable){this.initDataTable()}\r\n\r\n        // add an edit column if needed\r\n        if (this.addEdit) { \r\n            Table.Columns.addAtEnd(Table.Buttons.edit_confirm_btns, 'Edit'); \r\n            // activate edit buttons\r\n            Table.Buttons.Edit.init(this.actionOnEdit, this.updateTable);\r\n        }\r\n        \r\n        // assign the correct classes based on the table columns\r\n        Table.Columns.assignClasses(this.columns);\r\n\r\n        // Apply any update function to make sure sidebar is up to date\r\n        this.updateTable();\r\n        // add any newly created cc or approp to the filters\r\n        this.updateFilters();\r\n\r\n    }\r\n\r\n    async build() {\r\n        // build table from local storage and initialize edit buttons\r\n\r\n        // add the add new row button if needed\r\n        if (this.addButtonText) { \r\n            this.setUpForm();\r\n        }\r\n\r\n        // check for data\r\n        if(await Table.Data.load()) {  \r\n            // if there's data, update the table and add filters  \r\n            this.addFilters();  \r\n            await this.refreshData();\r\n        } else {\r\n            // show a message if there's no saved table data for the selected fund\r\n            if (this.noDataMessage) {\r\n                Prompt.Text.update(this.noDataMessage);\r\n            }\r\n        }\r\n    }\r\n\r\n    addFilters() {\r\n        // Add all relevant filters to table\r\n        if (this.columns.some(column => column.className === 'approp-name')){\r\n            Table.Filter.add('Appropriation', 'approp-name');\r\n        }\r\n        if (this.columns.some(column => column.className === 'cc-name')){\r\n            Table.Filter.add('Cost Center', 'cc-name');\r\n        }\r\n        if (this.columns.some(column => column.className === 'object-name')){\r\n            Table.Filter.add('Object', 'object-name');\r\n        };\r\n        if (this.columns.some(column => column.className === 'object-category')){\r\n            Table.Filter.add('Object Category', 'object-category');\r\n        }\r\n    }\r\n\r\n    updateFilters() {\r\n        // update filters with any new values\r\n        if (this.columns.some(column => column.className === 'approp-name')){\r\n            Table.Filter.updateOptions('Appropriation', 'approp-name');\r\n        }\r\n        if (this.columns.some(column => column.className === 'cc-name')){\r\n            Table.Filter.updateOptions('Cost Center', 'cc-name');\r\n        }\r\n        if (this.columns.some(column => column.className === 'object-name')){\r\n            Table.Filter.updateOptions('object-name');\r\n        };\r\n        if (this.columns.some(column => column.className === 'object-category')){\r\n            Table.Filter.updateOptions('object-category');\r\n        }\r\n    }\r\n\r\n    initDataTable() {\r\n        Table.adjustWidth('100%');\r\n        // Initialize Datatables\r\n        if ( !$.fn.dataTable.isDataTable('#main-table') ) {\r\n            $('#main-table').DataTable({\r\n                paging: false, // Disable pagination\r\n                info: false, // Disable table information display\r\n                searching: false // Disable the search bar\r\n            });\r\n        };\r\n    }\r\n\r\n    // placeholder for action on row edit click\r\n    actionOnEdit() { return }\r\n\r\n    // update function for the sidebar; default to just saving the table\r\n    updateTable() { Table.save() }\r\n\r\n    // extra questions of the form to add a new row\r\n    addCustomQuestions() { return };\r\n\r\n    addValidationListener(inputId, fieldLabel, validationId, length) {\r\n        const inputElement = document.getElementById(inputId);\r\n        inputElement.addEventListener('change', function () {\r\n            if (inputElement.value === 'Add new') {\r\n                // Add a new field after the selected element\r\n                Form.NewField.shortText(`Type new ${fieldLabel}:`, inputId.slice(0, -5), true);\r\n                let newInputElement = document.getElementById(inputId.slice(0, -5));  // Remove '-name' suffix\r\n                inputElement.parentElement.insertAdjacentElement('afterend', newInputElement.parentElement);\r\n    \r\n                // Add an event listener for validation on the new input field\r\n                newInputElement.addEventListener('blur', function () {\r\n                    let validationText = document.getElementById(validationId);\r\n                    validationText.textContent = '';\r\n                    if (newInputElement.value.length !== length) {\r\n                        validationText.textContent = `${fieldLabel} codes must be exactly ${length} numbers.`;\r\n                    } else if (isNaN(Number(newInputElement.value))) {\r\n                        validationText.textContent = `${fieldLabel} codes must be numeric.`;\r\n                    }\r\n                });\r\n            } else {\r\n                // Remove the new input field if 'Add new' is not selected\r\n                let newInputElement = document.getElementById(inputId.slice(0, -5));\r\n                if (newInputElement) {\r\n                    newInputElement.parentElement.innerHTML = '';\r\n                }\r\n            }\r\n        });\r\n    }\r\n    \r\n    // Modified addModalValidation method\r\n    addModalValidation() {\r\n        this.addValidationListener('approp-name', 'Appropriation', 'approp-validation', 5);\r\n        this.addValidationListener('cc-name', 'Cost Center', 'cc-validation', 6);\r\n    }\r\n\r\n    setUpForm() {\r\n        // show add button\r\n        Table.Buttons.AddRow.show();\r\n        Table.Buttons.AddRow.updateText(this.addButtonText);\r\n\r\n        // set up modal for form when add button is pressed\r\n        Modal.clear();\r\n        Modal.Link.add('add-btn');\r\n        Modal.Title.update(this.addButtonText);\r\n\r\n        // create form\r\n        Form.new('modal-body');\r\n\r\n        // add custom questions\r\n        this.addCustomQuestions();\r\n        // any validation or special functions\r\n        this.addModalValidation();\r\n        // add submit button\r\n        Form.SubmitButton.add();\r\n\r\n        // Initialize form submission to table data\r\n        Modal.Submit.init(this.submitNewRow);\r\n    }\r\n\r\n    editColumns(responses) { \r\n        // if a new appropriation was entered, fix it\r\n        if (responses['approp']){\r\n            responses['approp-name'] = `${responses['approp']} - New`;\r\n        };\r\n        // same for cost center\r\n        if (responses['cc']){\r\n            responses['cc-name'] = `${responses['cc']} - New`;\r\n        };\r\n\r\n        // get numbers from account string names\r\n        if(responses['fund-name']){\r\n            responses['fund'] = AccountString.getNumber(responses['fund-name']);\r\n        };\r\n        if(responses['approp-name']){\r\n            responses['approp'] = AccountString.getNumber(responses['approp-name']);\r\n        };\r\n        if(responses['cc-name']){\r\n            responses['cc'] = AccountString.getNumber(responses['cc-name']);\r\n        };\r\n        if(responses['object-name']){\r\n            responses['object'] = AccountString.getNumber(responses['object-name']);\r\n        };\r\n        responses['account-string'] = \r\n            AccountString.build(responses['approp-name'], \r\n                                responses['cc-name'], \r\n                                responses['object-name'], \r\n                                responses['fund']);\r\n        return responses;\r\n    }\r\n\r\n    submitNewRow(event) {\r\n        // get answers from form, hide form, show answers in table\r\n        var responses = Form.fetchAllResponses(event);\r\n        \r\n        // edit inputs from modal\r\n        responses = this.editColumns(responses);\r\n        \r\n        // make sure it's not an empty response\r\n        if (Object.values(responses)[0] != ''){\r\n\r\n            // change page view\r\n            Modal.hide();\r\n            \r\n            // add data to table\r\n            Table.Rows.add(responses, this.columns);\r\n            Table.save();\r\n            \r\n            // rebuild table\r\n            this.refreshData();\r\n        }\r\n    }\r\n\r\n}","\r\nimport Welcome from '../components/welcome/welcome.js';\r\nimport { FISCAL_YEAR } from '../constants/budget_constants.js';\r\nimport { View } from './view_class.js';\r\nimport { visitPage } from './view_logic.js';\r\n\r\nexport class WelcomeView extends View {\r\n\r\n    constructor() {\r\n        super();\r\n        this.page_state = 'welcome';\r\n        this.subtitle = 'Welcome';\r\n        this.sidebar = false;\r\n        this.navButtons = false;\r\n        this.prompt = `This tool will help you prepare your budget request submission for fiscal year \r\n            20${FISCAL_YEAR}. Start by uploading the detail sheet sent to you by your budget analyst. This\r\n            tool will use the data in that sheet to pre-fill your submission with last year's request.\r\n            <br><br>\r\n            Then, you will be guided through each fund to make any adjustments from last year's request. \r\n            As you \r\n            make these adjustments, you will see a sidebar tracking your total expenditures by fund and your target\r\n            baseline for the general fund. \r\n            <br><br>\r\n            Once you get through all funds and set your request at or below the \r\n            target, you will be able to download a filled version of the Excel detail sheet. Downloading \r\n            this file is the only permanent way to save your progress.`\r\n    }\r\n\r\n    visit() {\r\n        super.visit();\r\n        \r\n        // show welcome section\r\n        Welcome.show();\r\n\r\n        // initialize links in buttons\r\n        document.getElementById('step-upload').addEventListener('click', () => visitPage('upload'));\r\n        // document.getElementById('step-initiatives').addEventListener('click', () => visitPage('new-inits'));\r\n        // document.getElementById('step-revenue').addEventListener('click', () => visitPage('baseline-landing'));\r\n        // document.getElementById('step-finish').addEventListener('click', () => visitPage('summary'));\r\n\r\n    }\r\n\r\n}\r\n\r\nexport default WelcomeView;\r\n","import { View } from './view_class.js'\r\n\r\nimport FileUpload from \"../components/file_upload/file_upload.js\";\r\n\r\nexport class UploadView extends View {\r\n\r\n    constructor() {\r\n        super();\r\n        this.page_state = 'upload';\r\n        this.prompt = `Upload the baseline detail sheet given by your budget analyst.`;\r\n        this.subtitle = 'Excel Upload';\r\n        this.sidebar = false;\r\n    }\r\n\r\n    visit() {\r\n        super.visit();\r\n        FileUpload.show();\r\n        FileUpload.init();\r\n    }\r\n}\r\n\r\nexport default UploadView;","import NavButtons from \"../components/nav_buttons/nav_buttons.js\";\r\nimport Table from \"../components/table/table.js\";\r\nimport { View, ViewTable } from './view_class.js'\r\nimport CurrentFund from \"../models/current_fund.js\";\r\n\r\nexport class FundView extends View {\r\n\r\n    constructor() {\r\n        super();\r\n        this.page_state = 'baseline-landing';\r\n        this.prompt = `We will now ask you a series of questions about your BASELINE budget request.\r\n            At the end, we will ask you about any new initiatives (ie. supplemental requests).\r\n            Select one of your funds then click continue.`;\r\n        this.subtitle = 'Baseline Budget Request';\r\n        this.table = new FundTable();\r\n        this.sidebar = false;\r\n        this.dataTable = false;\r\n    }\r\n\r\n    visit() {\r\n        // remove fund selection\r\n        localStorage.setItem(\"fund\", '');\r\n        super.visit();\r\n    }\r\n}\r\n\r\nclass FundTable extends ViewTable {\r\n\r\n    constructor() {\r\n        super();\r\n\r\n        // add additional revenue columns to the table\r\n        this.columns = [\r\n            { title: 'Fund', className: 'fund-name' }\r\n        ];\r\n\r\n        this.noDataMessage = 'No funds found.'\r\n        this.addEdit = false;\r\n    }\r\n\r\n    build(){\r\n        // load in fund data\r\n        Table.show();\r\n        Table.Data.loadFunds();\r\n        Table.Columns.assignClasses(this.columns);\r\n        Table.adjustWidth('30%');\r\n\r\n        // disable next button until a fund is selected\r\n        NavButtons.Next.disable();\r\n        allowRowSelection();\r\n    }\r\n}\r\n\r\n\r\nfunction allowRowSelection(){\r\n    \r\n    var tableRows = document.querySelectorAll(\"tbody tr\");\r\n    \r\n    // enable highlight on hover and on select\r\n    tableRows.forEach(function(row) {\r\n        row.addEventListener('mouseover', function() {\r\n            this.classList.add('hover-effect');\r\n        });\r\n        row.addEventListener('mouseout', function() {\r\n            this.classList.remove('hover-effect');\r\n        });\r\n        row.addEventListener('click', function() {\r\n            selectFund(tableRows, this);\r\n        });\r\n    });\r\n}\r\n\r\nfunction selectFund(tableRows, selected_row){\r\n\r\n    // remove selected class from any other rows\r\n    tableRows.forEach(function(tableRow) {\r\n        tableRow.classList = '';\r\n    });\r\n\r\n    // add selected class to clicked row\r\n    selected_row.classList.add('selected');\r\n\r\n    // get fund and save selected fund\r\n    var fund = selected_row.querySelector('.fund-name').textContent;\r\n    var fundNumber = parseInt(fund);\r\n    CurrentFund.update(fundNumber);\r\n\r\n    // enable next step\r\n    NavButtons.Next.enable();\r\n}\r\n\r\nexport default FundView;","import { View, ViewTable } from './view_class.js'\r\n\r\nimport Table from '../components/table/table.js';\r\n\r\nexport class RevenueView extends View {\r\n\r\n    constructor(fiscal_year) {\r\n        super();\r\n        this.page_state = 'revenue';\r\n        this.prompt = `Review and edit revenue line items. If you change the estimate or \r\n            notice an error in an account string, please note it in the notes column. Click edit \r\n            to change values in a row.`;\r\n        this.subtitle = 'Revenues';\r\n        this.table = new RevenueTable(fiscal_year);\r\n    }\r\n}\r\n\r\nclass RevenueTable extends ViewTable {\r\n\r\n    constructor(fiscal_year) {\r\n        super();\r\n\r\n        // add additional revenue columns to the table\r\n        this.columns = this.columns.concat([\r\n            { title: 'Recurring or One-Time', className: 'recurring'},\r\n            { title: 'Object Category', className: 'object-category'},\r\n            { title: `FY${fiscal_year} Departmental Estimate`, className: 'request', isCost: true},\r\n            { title: 'Departmental Estimate Notes', className: 'notes'},\r\n            // hidden\r\n            { title: 'Object Name', className: 'object-name', hide: true},\r\n            { title: 'Object', className: 'object', hide: true},\r\n        ]);\r\n\r\n        this.noDataMessage = 'No revenues for this fund.'\r\n    }\r\n\r\n    // action on row edit click: make cells editable\r\n    actionOnEdit() { \r\n        // only allow edits in the notes box\r\n        Table.Cell.createTextbox('notes', false, 'textarea');\r\n    }\r\n}\r\n\r\nexport default RevenueView;","import { View, ViewTable } from './view_class.js'\r\n\r\nimport Table from \"../components/table/table.js\";\r\nimport Form from \"../components/form/form.js\";\r\n\r\nimport { Services, FundLookupTable } from '../models/';\r\nimport { unformatCurrency } from \"../utils/common_utils.js\";\r\nimport { EMPLOYEE_TYPES } from '../constants/budget_constants.js';\r\nimport GoldBook from '../models/gold_book.js';\r\n\r\nexport class PersonnelView extends View {\r\n\r\n    constructor(fiscal_year) {\r\n        super();\r\n        this.page_state = 'personnel';\r\n        this.prompt = `\r\n            This table displays the number of FTEs in each job code for in your department's \r\n            current (amended) FY25 budget. To make edits to the number of positions, click the\r\n            \"Edit\" button on the row you would like to edit. The \"Total Cost\" column and the \r\n            summary sidebar will also update to reflect any edits.`;\r\n        this.subtitle = 'Personnel';\r\n        this.table = new PersonnelTable(fiscal_year);\r\n    }\r\n}\r\n\r\n\r\nclass PersonnelTable extends ViewTable {\r\n\r\n    constructor(fiscal_year) {\r\n        super();\r\n        this.fiscal_year = fiscal_year;\r\n        // add additional personnel columns to the table\r\n        this.columns = this.columns.concat([\r\n            { title: 'Job Title', className: 'job-name' },\r\n            { title: 'Job Code', className: 'job-code' },\r\n            { title: 'Employee Type', className: 'employee-type'},\r\n            { title: 'Service', className: 'service' },\r\n            { title: `FY${this.fiscal_year} Requested FTE`, className: 'baseline-ftes' },\r\n            { title: `FY${this.fiscal_year} Projected Average Salary/Wage`, className: 'avg-salary', isCost: true },\r\n            { title: 'Total Sal/Wag & Ben Request', className: 'total-baseline', isCost: true },\r\n            { title: 'Departmental Request Notes', className: 'notes'},\r\n            // hidden columns\r\n            { title: 'Fringe Benefits Rate', className: 'fringe', hide: true },\r\n            { title: 'General Increase Rate', className: 'general-increase-rate', hide: true},\r\n            { title: 'Step/Merit Increase Rate', className: 'merit-increase-rate', hide: true},\r\n            { title: `Average Salary/Wage as of 9/1/20${this.fiscal_year-2}`, className: 'current-salary', isCost: true, hide: true}\r\n        ]);\r\n\r\n        this.noDataMessage = 'No personnel expenditures for this fund.'\r\n        this.addButtonText = 'Add new job' ;\r\n    }\r\n\r\n    // action on row edit click: make cells editable\r\n    actionOnEdit() { \r\n        Table.Cell.createTextbox('baseline-ftes');\r\n        Table.Cell.createServiceDropdown(Services.list());\r\n        Table.Cell.createTextbox('notes', false, 'textarea');\r\n    }\r\n\r\n    updateTable(){\r\n        // calculate for each row\r\n        let rows = document.getElementsByTagName('tr');\r\n        for (let i = 1; i < rows.length; i++){\r\n            // fetch values for calculations\r\n            let avg_salary = Table.Cell.getValue(rows[i], 'avg-salary');\r\n            let fringe = parseFloat(Table.Cell.getText(rows[i], 'fringe'));\r\n            let baseline_ftes = Table.Cell.getText(rows[i], 'baseline-ftes');\r\n\r\n            // calcuate #FTEs x average salary + COLA adjustments + merit adjustments + fringe\r\n            let total_baseline_cost = avg_salary * baseline_ftes * (1 + fringe);\r\n\r\n            // update total column\r\n            Table.Cell.updateValue(rows[i], 'total-baseline', total_baseline_cost);\r\n        }\r\n\r\n        // Save the table after all updates are done\r\n        Table.save();\r\n    }\r\n\r\n    addCustomQuestions(){\r\n        // form questions to add a new job\r\n        Form.NewField.shortText('Job Code:', 'job-code', true); \r\n        Form.NewField.shortText('Job Title:', 'job-name', true); \r\n        Form.NewField.dropdown('Employee Type:', 'employee-type', EMPLOYEE_TYPES, true),\r\n        Form.NewField.dropdown('Appropriation:', 'approp-name', FundLookupTable.getApprops('Add new'), true);\r\n        Form.NewField.dropdown('Cost Center:', 'cc-name', FundLookupTable.getCostCenters('Add new'), true);\r\n        Form.NewField.dropdown('Service', 'service', Services.list(), true);\r\n        Form.NewField.shortText('Number of FTEs requested:', 'baseline-ftes', true);\r\n        Form.NewField.shortText(`Projected average salary IN FISCAL YEAR ${this.fiscal_year}:`, 'avg-salary', true);\r\n        //Form.NewField.shortText(`Expected fringe rate (as a percentage)`, 'fringe', true);\r\n    }\r\n\r\n    addModalValidation(){\r\n\r\n        super.addModalValidation();\r\n\r\n        // lock the job description \r\n        const jobDescription = document.getElementById('job-name');\r\n        jobDescription.readOnly = true;\r\n        // give a message if attempt to edit\r\n        const nameValidationText = document.getElementById('job-name-validation');\r\n        jobDescription.addEventListener('click', function() {\r\n            nameValidationText.textContent = 'This field is not editable and will auto-fill from the job code above.';\r\n        });\r\n        // hide message on blur\r\n        jobDescription.addEventListener('blur', function() {\r\n            nameValidationText.textContent = '';\r\n        });\r\n\r\n        // confirm that entered job code is in the gold book\r\n        const jobCodeInput = document.getElementById('job-code');\r\n        jobCodeInput.addEventListener('blur', function () {\r\n            // get entered job code\r\n            const jobCode = jobCodeInput.value;\r\n            // get validation text element\r\n            const validationText = document.getElementById('job-code-validation');\r\n            if (!GoldBook.codeExists(jobCode)){\r\n                // if the job code doesn't exist, show an error message\r\n                validationText.textContent = 'This job code does not exist in the current Gold Book. Please enter another code.';\r\n                // and clear any previous entry in the job title \r\n                jobDescription.value = '';\r\n            } else {\r\n                // if it does exist, change the job title accordingly\r\n                jobDescription.value = GoldBook.getTitle(jobCode);\r\n                // and clear any error messages\r\n                validationText.textContent = '';\r\n            }\r\n        });\r\n        \r\n    }\r\n\r\n    editColumns(responses){\r\n        responses = super.editColumns(responses);\r\n        // edit inputs from modal\r\n        responses['avg-salary'] = unformatCurrency(responses['avg-salary']);\r\n        // use gold book to look up info based on job code\r\n        responses['job-name'] = GoldBook.getTitle(responses['job-code']);\r\n        responses['fringe'] = GoldBook.getFringeRate(responses['job-code']);\r\n        return responses;\r\n    }\r\n}\r\n\r\nexport default PersonnelView;","\r\n\r\nimport { View, ViewTable } from './view_class.js'\r\nimport Table from '../components/table/table.js';\r\nimport Form from '../components/form/form.js';\r\nimport { FundLookupTable, Services } from '../models/';\r\nimport { unformatCurrency } from '../utils/common_utils.js';\r\nimport { OT_OBJECTS } from '../constants/';\r\n\r\nexport class OvertimeView extends View {\r\n\r\n    constructor(fiscal_year) {\r\n        super();\r\n        this.page_state = 'overtime';\r\n        this.prompt = `\r\n            Please see your baseline overtime / holiday pay / shift premiums in the table below.\r\n            Make any edits and continue.`;\r\n        this.subtitle = 'Overtime Estimates';\r\n        this.table = new OvertimeTable(fiscal_year);\r\n    }\r\n}\r\n\r\nclass OvertimeTable extends ViewTable {\r\n\r\n    constructor(fiscal_year) {\r\n        super();\r\n\r\n        // add additional OT columns to the table\r\n        this.columns = this.columns.concat([\r\n            { title: 'Service', className: 'service' },\r\n            { title: 'Recurring or One-Time', className: 'recurring'},\r\n            { title: 'Departmental Request OT/SP/Hol', className: 'OT-pay', isCost: true },\r\n            { title: `FY${fiscal_year} Total OT/SP/Hol + FICA Request`, className : 'total', isCost: true},\r\n            { title: 'Object Name', className: 'object-name'},\r\n            { title: 'Departmental Request Notes', className: 'notes'},\r\n            // hidden columns\r\n            { title: 'FICA Rate', className: 'fica', hide: true},\r\n        ]);\r\n\r\n        this.noDataMessage = 'No overtime expenditures for this fund.'\r\n        this.addButtonText = 'Add new cost center' ;\r\n    }\r\n\r\n    // action on row edit click: make cells editable\r\n    actionOnEdit() { \r\n        Table.Cell.createTextbox('OT-pay', true);\r\n        Table.Cell.createServiceDropdown(Services.list());\r\n        Table.Cell.createDropdown('recurring', ['One-Time', 'Recurring']);\r\n        Table.Cell.createDropdown('object-name', OT_OBJECTS);\r\n        Table.Cell.createTextbox('notes');\r\n    }\r\n\r\n    updateTable(){\r\n\r\n        // calculate for each row\r\n        let rows = document.getElementsByTagName('tr');\r\n        for (let i = 1; i < rows.length; i++){\r\n            // fetch values for calculations\r\n            let OT_salary = Table.Cell.getValue(rows[i], 'OT-pay');\r\n            let fica_rate = Table.Cell.getText(rows[i], 'fica');\r\n\r\n            // add salary and wages and fringe benefits (FICA)\r\n            let row_total = OT_salary * (1 + parseFloat(fica_rate));\r\n\r\n            // update total\r\n            Table.Cell.updateValue(rows[i], 'total', row_total);\r\n        }\r\n\r\n        // Save the table after all updates are done\r\n        Table.save();\r\n    }\r\n\r\n    addCustomQuestions(){\r\n        // form questions to add a new job\r\n        Form.NewField.dropdown('Appropriation:', 'approp-name', FundLookupTable.getApprops('Add new'), true);\r\n        Form.NewField.dropdown('Cost Center:', 'cc-name', FundLookupTable.getCostCenters('Add new'), true);\r\n        Form.NewField.dropdown('Object (salary or wage):', 'object-name', OT_OBJECTS, true); \r\n        Form.NewField.dropdown('Service', 'service', Services.list(), true);\r\n        Form.NewField.dropdown('Recurring or One-Time', 'recurring', ['Recurring', 'One-Time'], true);\r\n        Form.NewField.shortText('Overtime amount requested:', 'OT-pay', true);\r\n    }\r\n\r\n    editColumns(responses){\r\n        responses = super.editColumns(responses);\r\n        // edit inputs from modal\r\n        responses['OT-pay'] = unformatCurrency(responses['OT-pay']);\r\n        responses['fica'] = 0.0765;\r\n        return responses;\r\n    }\r\n}\r\n\r\nexport default OvertimeView;","import { View, ViewTable } from './view_class.js'\r\nimport Form from '../components/form/form.js';\r\nimport Table from '../components/table/table.js';\r\nimport { FundLookupTable, Services } from '../models/';\r\nimport { unformatCurrency } from '../utils/common_utils.js';\r\nimport { OBJ_CATEGORIES } from '../constants/';\r\n\r\nexport class NonPersonnelView extends View {\r\n\r\n    constructor(fiscal_year) {\r\n        super();\r\n        this.page_state = 'nonpersonnel';\r\n        this.prompt = `Review and edit non-personnel line items. The CPA numbers are the \r\n            Contract and Procurement Account numbers. Click on the 'detail' link for a CPA \r\n            to see the contract details. Some line items won't have a CPA number.`;\r\n        this.subtitle = 'Non-Personnel';\r\n        this.table = new NonPersonnelTable(fiscal_year);\r\n    }\r\n}\r\n\r\nclass NonPersonnelTable extends ViewTable {\r\n\r\n    constructor(fiscal_year) {\r\n        super();\r\n\r\n        // add additional personnel columns to the table\r\n        this.columns = this.columns.concat([\r\n            { title: `FY${fiscal_year} Departmental Request Total`, className: 'request', isCost: true },\r\n            { title: 'Service', className : 'service' },\r\n            { title: 'Recurring or One-Time', className: 'recurring'},\r\n            { title: 'Vendor Name', className: 'vendor'},\r\n            { title : 'CPA #', className : 'cpa'},\r\n            // hidden columns\r\n            { title: 'End Date', className: 'contract-end', hide: true},\r\n            { title: 'BPA/CPA Amount Remaining', className: 'remaining', isCost: true , hide: true},\r\n            { title: 'Object Name', className: 'object-name', hide: true},\r\n            { title: 'Object', className: 'object', hide: true},\r\n            { title: 'Object Category', className: 'object-category', hide: true},\r\n            { title: 'BPA/CPA Description', className: 'cpa-description', hide: true},\r\n            { title: 'Departmental Request Notes', className: 'notes'}, \r\n        ]);\r\n\r\n        this.noDataMessage = 'No non-personnel expenditures for this fund.'\r\n        this.addButtonText = 'Add new non-personnel item' ;\r\n    }\r\n\r\n    // action on row edit click: make cells editable\r\n    actionOnEdit() { \r\n        Table.Cell.createTextbox('request', true);\r\n        Table.Cell.createServiceDropdown();\r\n        Table.Cell.createDropdown('recurring', ['One-Time', 'Recurring']);\r\n        Table.Cell.createTextbox('notes');\r\n    }\r\n\r\n    addCustomQuestions(){\r\n        // form questions to add a new row\r\n        Form.NewField.dropdown('Appropriation:', 'approp-name', FundLookupTable.getApprops('Add new'), true);\r\n        Form.NewField.dropdown('Cost Center:', 'cc-name', FundLookupTable.getCostCenters('Add new'), true);\r\n        Form.NewField.dropdown('Object Category:', 'object-category', OBJ_CATEGORIES.list, true);\r\n        // TODO: maybe give dropdown based on selected obj category\r\n        Form.NewField.shortText('Object Number (if known):', 'object', false);\r\n        Form.NewField.dropdown('Service', 'service', Services.list(), true);\r\n        Form.NewField.longText('Describe your new request:', 'cpa-description', true);\r\n        Form.NewField.dropdown('Recurring or One-Time', 'recurring', ['Recurring', 'One-Time'], true); \r\n        Form.NewField.shortText('Amount requested:', 'request', true);\r\n    }\r\n\r\n    editColumns(responses){\r\n        responses = super.editColumns(responses);\r\n        responses['avg-salary'] = unformatCurrency(responses['avg-salary']);\r\n        responses['fringe'] = parseFloat(responses['fringe']) / 100;\r\n        return responses;\r\n    }\r\n}\r\n\r\nexport default NonPersonnelView;","\r\nimport { View, ViewTable } from './view_class.js'\r\nimport Table from \"../components/table/table.js\";\r\nimport Form from \"../components/form/form.js\";\r\nimport FundLookupTable from '../models/fund_lookup_table.js';\r\nimport { FISCAL_YEAR } from '../constants/';\r\nimport AccountString from '../models/account_string.js';\r\n\r\n\r\nconst dropdownOptions = ['N/A', 'One-Time', 'Recurring']\r\n\r\n// set up page and initialize all buttons\r\nexport class InitiativesView extends View {\r\n\r\n    constructor() {\r\n        super();\r\n        this.page_state = 'new-inits';\r\n        this.prompt = `\r\n            This is the place to propose new initiatives for FY${FISCAL_YEAR}.\r\n            New initiative submissions will count as supplemental line items and will be the starting \r\n            point for a conversation with both OB and ODFS, who will help with the details.`;\r\n        this.subtitle = 'New Initiatives';\r\n        this.table = new InitiativesTable();\r\n    }\r\n\r\n    visit() {\r\n        // remove fund selection\r\n        localStorage.setItem(\"fund\", '');\r\n        super.visit();\r\n    }\r\n\r\n}\r\n\r\nclass InitiativesTable extends ViewTable {\r\n\r\n    constructor() {\r\n        super();\r\n\r\n        // add additional columns to the table\r\n        this.columns = [\r\n            { title: 'Fund', className: 'fund'},\r\n            { title: 'Supplemental Initiative', className: 'init-name' },\r\n            { title: 'Total Initiative Request', className: 'total', isCost: true },\r\n            // { title: 'Personnel FTE', className: 'ftes'},\r\n            { title: 'Personnel Salary & Benefits', className: 'personnel', isCost: true },\r\n            { title: 'Non-Personnel Operating', className: 'nonpersonnel', isCost: true },\r\n            { title: 'Non-Personnel Capital', className: 'nonpersonnel-capital', isCost: true },\r\n            // { title: 'Revenue', className: 'revenue', isCost: true },\r\n            // { title: 'Revenue Type', className: 'rev-type' },\r\n            { title: 'Description & Justification', className: 'notes' },\r\n            { title: 'Recurring or One-Time', className: 'rev-type'},\r\n            { title: 'Edit', className: 'edit' }\r\n        ];\r\n\r\n        this.addButtonText = 'Add new initiative' ;\r\n    }\r\n\r\n    addCustomQuestions(){\r\n\r\n        // general questions\r\n        Form.NewField.shortText('Initiative Name:', 'init-name', true); \r\n        Form.NewField.longText('What is the business case for the Initiative?', 'q1', true);\r\n        Form.NewField.longText(`Why is the initiative needed? What is the value-add to residents? \r\n            What is the Department’s plan for implementing the Initiative?`, 'q2', true);\r\n        Form.NewField.longText(`Why can’t the Initiative be funded with the Department’s baseline budget?`, 'q3', true);\r\n\r\n        Form.NewField.dropdown(`Is this initiative one-time or recurring?`, \r\n            'rev-type', dropdownOptions);\r\n\r\n        // Account string info\r\n        Form.NewField.dropdown('Fund:', 'fund-name', FundLookupTable.listFundNames(), true);\r\n\r\n        // Numbers\r\n        Form.NewField.numericInput('What is your ballpark estimate of TOTAL ADDITONAL expenses associated with this initiative?', \r\n            'total', false);\r\n        Form.NewField.numericInput('Estimate of ADDITONAL personnel cost?', 'personnel', false);\r\n        Form.NewField.numericInput('Estimate of ADDITONAL nonpersonnel operating cost?', 'nonpersonnel', false);\r\n        Form.NewField.numericInput('Estimate of ADDITONAL nonpersonnel capital costs?', 'nonpersonnel-capital', false);\r\n        Form.NewField.numericInput('Estimate of ADDITONAL revenue (if applicable)?', 'revenue', false);\r\n    }\r\n\r\n    editColumns(responses) {\r\n        responses['fund'] = AccountString.getNumber(responses['fund-name']);\r\n        responses['notes'] = `${responses['q1']} ${responses['q2']} ${responses['q3']}`;\r\n        return responses;\r\n    }\r\n\r\n    addModalValidation() { return }\r\n\r\n    // action on row edit click: make cells editable\r\n    actionOnEdit() { \r\n        Table.Cell.createTextbox('total', true);\r\n        // Table.Cell.createTextbox('revenue', true);\r\n        Table.Cell.createTextbox('personnel', true);\r\n        Table.Cell.createTextbox('nonpersonnel', true);\r\n        Table.Cell.createTextbox('init-name');\r\n        Table.Cell.createDropdown('rev-type', dropdownOptions);\r\n        Table.Cell.createTextbox('notes');\r\n    }\r\n\r\n}\r\n\r\nexport default InitiativesView;","import CurrentFund from '../models/current_fund.js';\r\nimport Baseline from '../models/baseline.js';\r\nimport { Accordion } from \"../components/accordion/accordion.js\";\r\nimport { formatCurrency } from '../utils/common_utils.js';\r\nimport { View } from \"./view_class.js\";\r\nimport Prompt from \"../components/prompt/prompt.js\";\r\nimport { downloadXLSX } from \"../utils/XLSX_handlers.js\";\r\nimport WelcomeView from './00_welcome.js';\r\n\r\nexport function compareToTarget(){\r\n    const baseline = new Baseline;\r\n    if (baseline.genFundTotal() <= Baseline.target()){\r\n        Prompt.Text.update(`Congrats! Your General Fund budget is below your target! \r\n            Edit any line items below or download your completed Excel.`);\r\n    } else {\r\n        Prompt.Text.update(`Your General Fund budget is above your GF target of ${formatCurrency(Baseline.target())}. \r\n            Please expand the summary table below and edit line items until you meet your target. \r\n            When you meet the target, you will be able to download the Excel sheet.`);\r\n        Prompt.Buttons.Right.disable();\r\n    }\r\n    Prompt.show();\r\n}\r\n\r\nfunction returnToWelcome() {\r\n    const welcome = new WelcomeView();\r\n    const left = document.getElementById('option1');\r\n    welcome.visit();\r\n};\r\n\r\nexport class SummaryView extends View {\r\n\r\n    constructor() {\r\n        super();\r\n        this.page_state = 'summary';\r\n        this.subtitle = 'Summary';\r\n        this.sidebar = false;\r\n    }\r\n\r\n    visit() {\r\n        super.visit();\r\n\r\n        // reset fund\r\n        CurrentFund.reset();\r\n\r\n        // show summary accordion\r\n        Accordion.build();\r\n        Accordion.show();\r\n\r\n        // add prompt buttons\r\n        Prompt.Buttons.Right.updateText('Download Excel');\r\n        Prompt.Buttons.Left.updateText('Start over with new Excel upload');\r\n        // add button links\r\n        Prompt.Buttons.Left.addAction(returnToWelcome);\r\n        Prompt.Buttons.Right.addAction(downloadXLSX);\r\n\r\n        // update prompt text depending on target matching\r\n        compareToTarget();\r\n    }\r\n\r\n    cleanup() {\r\n        // delete event listeners\r\n        Prompt.Buttons.Left.removeAction(returnToWelcome);\r\n        Prompt.Buttons.Right.removeAction(downloadXLSX);\r\n        Prompt.Buttons.Right.enable();\r\n    }\r\n}\r\n\r\nexport default SummaryView;","import WelcomeView from './00_welcome.js';\r\nimport UploadView from './01_upload.js';\r\nimport FundView from './02_baseline_landing.js';\r\nimport RevenueView from './03_revenue.js';\r\nimport PersonnelView from './04_personnel.js';\r\nimport OvertimeView from './05_overtime.js';\r\nimport NonPersonnelView from './06_nonpersonnel.js';\r\nimport InitiativesView from './07_new_initiatives.js';\r\nimport SummaryView from './08_summary.js';\r\n\r\nimport { FundLookupTable, CurrentFund, CurrentPage } from '../models/';\r\nimport { FISCAL_YEAR } from '../constants/';\r\n\r\n// Initialize pages globally once\r\nconst PAGES = {\r\n    'welcome': new WelcomeView(),\r\n    'upload': new UploadView(),\r\n    'baseline-landing': new FundView(),\r\n    'revenue': new RevenueView(FISCAL_YEAR),\r\n    'personnel': new PersonnelView(FISCAL_YEAR),\r\n    'overtime': new OvertimeView(FISCAL_YEAR),\r\n    'nonpersonnel': new NonPersonnelView(FISCAL_YEAR),\r\n    'new-inits': new InitiativesView(),\r\n    'summary': new SummaryView()\r\n};\r\n\r\nexport function visitPage(new_page_key) {\r\n    var page_state = CurrentPage.load();\r\n\r\n    // Perform cleanup from the current page\r\n    if (PAGES[page_state]) {\r\n        PAGES[page_state].cleanup();\r\n    }\r\n\r\n    if (PAGES[new_page_key]) {\r\n        PAGES[new_page_key].visit();\r\n    } else {\r\n        console.error(`No page initializer found for state: ${new_page_key}`);\r\n    }\r\n}\r\n\r\nexport function nextPage() {\r\n    var page_state = CurrentPage.load();\r\n    const keys = Object.keys(PAGES);\r\n    const currentIndex = keys.indexOf(page_state);\r\n\r\n    const returnPages = ['revenue', 'nonpersonnel', 'new-inits', 'overtime', 'personnel'];\r\n    if (!FundLookupTable.fundsLeft() && returnPages.includes(CurrentPage.load())) {\r\n        visitPage('summary');\r\n        return;\r\n    }\r\n\r\n    if (CurrentPage.load() == 'nonpersonnel') {\r\n        FundLookupTable.editFund(CurrentFund.number());\r\n        if (FundLookupTable.fundsLeft()) {\r\n            visitPage('baseline-landing');\r\n            return;\r\n        }\r\n    }\r\n\r\n    if (currentIndex >= 0 && currentIndex < keys.length - 1) {\r\n        const nextKey = keys[currentIndex + 1];\r\n        visitPage(nextKey);\r\n    }\r\n}\r\n\r\nexport function lastPage() {\r\n    var page_state = CurrentPage.load();\r\n    const keys = Object.keys(PAGES);\r\n    const currentIndex = keys.indexOf(page_state);\r\n\r\n    if (CurrentPage.load() == 'new-inits') {\r\n        visitPage('baseline-landing');\r\n        return;\r\n    }\r\n    \r\n    if (currentIndex >= 1) {\r\n        const lastKey = keys[currentIndex - 1];\r\n        visitPage(lastKey);\r\n    }\r\n}","import { visitPage } from \"../views/view_logic\";\r\n\r\nexport const CurrentPage = {\r\n    update : function(page){\r\n        localStorage.setItem('page_state', page);\r\n    },\r\n    load : function(){\r\n        const pageState = localStorage.getItem('page_state');\r\n        return pageState !== null ? pageState : 'welcome';\r\n    },\r\n    visit : function(){\r\n        visitPage(this.load());\r\n    }\r\n}\r\n\r\nexport default CurrentPage;","export function getLocalStorageSize() {\r\n    let total = 0;\r\n\r\n    for (let key in localStorage) {\r\n        if (localStorage.hasOwnProperty(key)) {\r\n            let keySize = key.length * 2; // each character is 2 bytes\r\n            let valueSize = localStorage[key].length * 2; // each character is 2 bytes\r\n            total += keySize + valueSize;\r\n        }\r\n    }\r\n\r\n    return formatBytes(total);\r\n}\r\n\r\nfunction formatBytes(bytes, decimals = 2) {\r\n    if (bytes === 0) return '0 Bytes';\r\n    const k = 1024;\r\n    const dm = decimals < 0 ? 0 : decimals;\r\n    const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB'];\r\n    const i = Math.floor(Math.log(bytes) / Math.log(k));\r\n    return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i];\r\n}\r\n","// Import styles\r\nimport '../css/common.css';\r\n\r\n// Import functions\r\nimport CurrentPage from './models/current_page.js';\r\n\r\nimport { getLocalStorageSize } from './utils/storage_utils.js';\r\n\r\n// Initialize only once when the document is fully loaded\r\ndocument.addEventListener('DOMContentLoaded', function () {\r\n    CurrentPage.visit();\r\n    console.log('Local Storage Used:', getLocalStorageSize());\r\n});"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///863\n")},404:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `:root {\r\n /* colors */\r\n --orange: #FB6523;\r\n --yellow: #D99733;\r\n --blue: #43748E;\r\n --lightBlue: #b7c9e2;\r\n --darkGray: #373A37;\r\n --mediumGray: #D0D0D0;\r\n --lightGray: #F2F2F2;\r\n --white: #FBFDFB;\r\n --black: #212121;\r\n --green: #71BC78;\r\n --citygreen: #004445;\r\n --spiritgreen: #279989;\r\n --palegreen: #9FD5B3;\r\n --verypalegreen: #DFF5E3;\r\n --accentyellow: #feb70d;\r\n --nearblack: #18252a;\r\n /* fonts */\r\n font-family: \'Nunito\', sans-serif;\r\n font-size: 14px;\r\n /* spacing */\r\n --header-height : 105px;\r\n --sidebar-width: 300px;\r\n}\r\n\r\n/* Button styling */\r\n\r\n.btn {\r\n cursor: pointer;\r\n padding: 6px;\r\n margin-top: 5px;\r\n margin-right : 2px;\r\n margin-left: 2px;\r\n border-radius: 10px;\r\n background-color: gray;\r\n color: white; \r\n}\r\n\r\n/* Font application for all text */\r\nbody, button, input, textarea, select, .sidebar, table {\r\n font-family: var(--font-family), sans-serif;\r\n }\r\n\r\n#main-panel {\r\n /* display: flex; */\r\n /* flex-direction: column; */\r\n justify-content: center; /* This will center content vertically */\r\n min-height: calc(100vh - var(--header-height)) ; \r\n /* max-width: calc(100vw - var(--sidebar-width)); */\r\n padding-top: 25px;\r\n padding-bottom: 25px;\r\n margin: 0;\r\n}\r\n\r\n/* Font awesome */\r\n\r\ni.fas.fa-check { \r\n font-size: 1.5em;\r\n color: var(--spiritgreen);\r\n margin-right: 10px;\r\n}\r\n`, "",{"version":3,"sources":["webpack://./src/css/common.css"],"names":[],"mappings":"AAAA;IACI,WAAW;IACX,iBAAiB;IACjB,iBAAiB;IACjB,eAAe;IACf,oBAAoB;IACpB,mBAAmB;IACnB,qBAAqB;IACrB,oBAAoB;IACpB,gBAAgB;IAChB,gBAAgB;IAChB,gBAAgB;IAChB,oBAAoB;IACpB,sBAAsB;IACtB,oBAAoB;IACpB,wBAAwB;IACxB,uBAAuB;IACvB,oBAAoB;IACpB,UAAU;IACV,iCAAiC;IACjC,eAAe;IACf,YAAY;IACZ,uBAAuB;IACvB,sBAAsB;AAC1B;;AAEA,mBAAmB;;AAEnB;IACI,eAAe;IACf,YAAY;IACZ,eAAe;IACf,kBAAkB;IAClB,gBAAgB;IAChB,mBAAmB;IACnB,sBAAsB;IACtB,YAAY;AAChB;;AAEA,mCAAmC;AACnC;IACI,2CAA2C;EAC7C;;AAEF;IACI,mBAAmB;IACnB,4BAA4B;IAC5B,uBAAuB,EAAE,wCAAwC;IACjE,+CAA+C;IAC/C,oDAAoD;IACpD,iBAAiB;IACjB,oBAAoB;IACpB,SAAS;AACb;;AAEA,iBAAiB;;AAEjB;IACI,gBAAgB;IAChB,yBAAyB;IACzB,kBAAkB;AACtB","sourcesContent":[":root {\\r\\n /* colors */\\r\\n --orange: #FB6523;\\r\\n --yellow: #D99733;\\r\\n --blue: #43748E;\\r\\n --lightBlue: #b7c9e2;\\r\\n --darkGray: #373A37;\\r\\n --mediumGray: #D0D0D0;\\r\\n --lightGray: #F2F2F2;\\r\\n --white: #FBFDFB;\\r\\n --black: #212121;\\r\\n --green: #71BC78;\\r\\n --citygreen: #004445;\\r\\n --spiritgreen: #279989;\\r\\n --palegreen: #9FD5B3;\\r\\n --verypalegreen: #DFF5E3;\\r\\n --accentyellow: #feb70d;\\r\\n --nearblack: #18252a;\\r\\n /* fonts */\\r\\n font-family: \'Nunito\', sans-serif;\\r\\n font-size: 14px;\\r\\n /* spacing */\\r\\n --header-height : 105px;\\r\\n --sidebar-width: 300px;\\r\\n}\\r\\n\\r\\n/* Button styling */\\r\\n\\r\\n.btn {\\r\\n cursor: pointer;\\r\\n padding: 6px;\\r\\n margin-top: 5px;\\r\\n margin-right : 2px;\\r\\n margin-left: 2px;\\r\\n border-radius: 10px;\\r\\n background-color: gray;\\r\\n color: white; \\r\\n}\\r\\n\\r\\n/* Font application for all text */\\r\\nbody, button, input, textarea, select, .sidebar, table {\\r\\n font-family: var(--font-family), sans-serif;\\r\\n }\\r\\n\\r\\n#main-panel {\\r\\n /* display: flex; */\\r\\n /* flex-direction: column; */\\r\\n justify-content: center; /* This will center content vertically */\\r\\n min-height: calc(100vh - var(--header-height)) ; \\r\\n /* max-width: calc(100vw - var(--sidebar-width)); */\\r\\n padding-top: 25px;\\r\\n padding-bottom: 25px;\\r\\n margin: 0;\\r\\n}\\r\\n\\r\\n/* Font awesome */\\r\\n\\r\\ni.fas.fa-check { \\r\\n font-size: 1.5em;\\r\\n color: var(--spiritgreen);\\r\\n margin-right: 10px;\\r\\n}\\r\\n"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiNDA0LmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSIsInNvdXJjZXMiOlsid2VicGFjazovL2J1ZGdldC1yZXF1ZXN0LWZvcm0vLi9zcmMvY3NzL2NvbW1vbi5jc3M/ZjdjYSJdLCJzb3VyY2VzQ29udGVudCI6WyIvLyBJbXBvcnRzXG5pbXBvcnQgX19fQ1NTX0xPQURFUl9BUElfU09VUkNFTUFQX0lNUE9SVF9fXyBmcm9tIFwiLi4vLi4vbm9kZV9tb2R1bGVzL2Nzcy1sb2FkZXIvZGlzdC9ydW50aW1lL3NvdXJjZU1hcHMuanNcIjtcbmltcG9ydCBfX19DU1NfTE9BREVSX0FQSV9JTVBPUlRfX18gZnJvbSBcIi4uLy4uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvcnVudGltZS9hcGkuanNcIjtcbnZhciBfX19DU1NfTE9BREVSX0VYUE9SVF9fXyA9IF9fX0NTU19MT0FERVJfQVBJX0lNUE9SVF9fXyhfX19DU1NfTE9BREVSX0FQSV9TT1VSQ0VNQVBfSU1QT1JUX19fKTtcbi8vIE1vZHVsZVxuX19fQ1NTX0xPQURFUl9FWFBPUlRfX18ucHVzaChbbW9kdWxlLmlkLCBgOnJvb3Qge1xyXG4gICAgLyogY29sb3JzICovXHJcbiAgICAtLW9yYW5nZTogI0ZCNjUyMztcclxuICAgIC0teWVsbG93OiAjRDk5NzMzO1xyXG4gICAgLS1ibHVlOiAjNDM3NDhFO1xyXG4gICAgLS1saWdodEJsdWU6ICNiN2M5ZTI7XHJcbiAgICAtLWRhcmtHcmF5OiAjMzczQTM3O1xyXG4gICAgLS1tZWRpdW1HcmF5OiAjRDBEMEQwO1xyXG4gICAgLS1saWdodEdyYXk6ICNGMkYyRjI7XHJcbiAgICAtLXdoaXRlOiAjRkJGREZCO1xyXG4gICAgLS1ibGFjazogIzIxMjEyMTtcclxuICAgIC0tZ3JlZW46ICM3MUJDNzg7XHJcbiAgICAtLWNpdHlncmVlbjogIzAwNDQ0NTtcclxuICAgIC0tc3Bpcml0Z3JlZW46ICMyNzk5ODk7XHJcbiAgICAtLXBhbGVncmVlbjogIzlGRDVCMztcclxuICAgIC0tdmVyeXBhbGVncmVlbjogI0RGRjVFMztcclxuICAgIC0tYWNjZW50eWVsbG93OiAjZmViNzBkO1xyXG4gICAgLS1uZWFyYmxhY2s6ICMxODI1MmE7XHJcbiAgICAvKiBmb250cyAqL1xyXG4gICAgZm9udC1mYW1pbHk6ICdOdW5pdG8nLCBzYW5zLXNlcmlmO1xyXG4gICAgZm9udC1zaXplOiAxNHB4O1xyXG4gICAgLyogc3BhY2luZyAqL1xyXG4gICAgLS1oZWFkZXItaGVpZ2h0IDogMTA1cHg7XHJcbiAgICAtLXNpZGViYXItd2lkdGg6IDMwMHB4O1xyXG59XHJcblxyXG4vKiBCdXR0b24gc3R5bGluZyAqL1xyXG5cclxuLmJ0biB7XHJcbiAgICBjdXJzb3I6IHBvaW50ZXI7XHJcbiAgICBwYWRkaW5nOiA2cHg7XHJcbiAgICBtYXJnaW4tdG9wOiA1cHg7XHJcbiAgICBtYXJnaW4tcmlnaHQgOiAycHg7XHJcbiAgICBtYXJnaW4tbGVmdDogMnB4O1xyXG4gICAgYm9yZGVyLXJhZGl1czogMTBweDtcclxuICAgIGJhY2tncm91bmQtY29sb3I6IGdyYXk7XHJcbiAgICBjb2xvcjogd2hpdGU7IFxyXG59XHJcblxyXG4vKiBGb250IGFwcGxpY2F0aW9uIGZvciBhbGwgdGV4dCAgKi9cclxuYm9keSwgYnV0dG9uLCBpbnB1dCwgdGV4dGFyZWEsIHNlbGVjdCwgLnNpZGViYXIsIHRhYmxlIHtcclxuICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1mb250LWZhbWlseSksIHNhbnMtc2VyaWY7XHJcbiAgfVxyXG5cclxuI21haW4tcGFuZWwge1xyXG4gICAgLyogZGlzcGxheTogZmxleDsgKi9cclxuICAgIC8qIGZsZXgtZGlyZWN0aW9uOiBjb2x1bW47ICovXHJcbiAgICBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjsgLyogVGhpcyB3aWxsIGNlbnRlciBjb250ZW50IHZlcnRpY2FsbHkgKi9cclxuICAgIG1pbi1oZWlnaHQ6IGNhbGMoMTAwdmggLSB2YXIoLS1oZWFkZXItaGVpZ2h0KSkgOyBcclxuICAgIC8qIG1heC13aWR0aDogY2FsYygxMDB2dyAtIHZhcigtLXNpZGViYXItd2lkdGgpKTsgICovXHJcbiAgICBwYWRkaW5nLXRvcDogMjVweDtcclxuICAgIHBhZGRpbmctYm90dG9tOiAyNXB4O1xyXG4gICAgbWFyZ2luOiAwO1xyXG59XHJcblxyXG4vKiBGb250IGF3ZXNvbWUgKi9cclxuXHJcbmkuZmFzLmZhLWNoZWNrIHsgXHJcbiAgICBmb250LXNpemU6IDEuNWVtO1xyXG4gICAgY29sb3I6IHZhcigtLXNwaXJpdGdyZWVuKTtcclxuICAgIG1hcmdpbi1yaWdodDogMTBweDtcclxufVxyXG5gLCBcIlwiLHtcInZlcnNpb25cIjozLFwic291cmNlc1wiOltcIndlYnBhY2s6Ly8uL3NyYy9jc3MvY29tbW9uLmNzc1wiXSxcIm5hbWVzXCI6W10sXCJtYXBwaW5nc1wiOlwiQUFBQTtJQUNJLFdBQVc7SUFDWCxpQkFBaUI7SUFDakIsaUJBQWlCO0lBQ2pCLGVBQWU7SUFDZixvQkFBb0I7SUFDcEIsbUJBQW1CO0lBQ25CLHFCQUFxQjtJQUNyQixvQkFBb0I7SUFDcEIsZ0JBQWdCO0lBQ2hCLGdCQUFnQjtJQUNoQixnQkFBZ0I7SUFDaEIsb0JBQW9CO0lBQ3BCLHNCQUFzQjtJQUN0QixvQkFBb0I7SUFDcEIsd0JBQXdCO0lBQ3hCLHVCQUF1QjtJQUN2QixvQkFBb0I7SUFDcEIsVUFBVTtJQUNWLGlDQUFpQztJQUNqQyxlQUFlO0lBQ2YsWUFBWTtJQUNaLHVCQUF1QjtJQUN2QixzQkFBc0I7QUFDMUI7O0FBRUEsbUJBQW1COztBQUVuQjtJQUNJLGVBQWU7SUFDZixZQUFZO0lBQ1osZUFBZTtJQUNmLGtCQUFrQjtJQUNsQixnQkFBZ0I7SUFDaEIsbUJBQW1CO0lBQ25CLHNCQUFzQjtJQUN0QixZQUFZO0FBQ2hCOztBQUVBLG1DQUFtQztBQUNuQztJQUNJLDJDQUEyQztFQUM3Qzs7QUFFRjtJQUNJLG1CQUFtQjtJQUNuQiw0QkFBNEI7SUFDNUIsdUJBQXVCLEVBQUUsd0NBQXdDO0lBQ2pFLCtDQUErQztJQUMvQyxvREFBb0Q7SUFDcEQsaUJBQWlCO0lBQ2pCLG9CQUFvQjtJQUNwQixTQUFTO0FBQ2I7O0FBRUEsaUJBQWlCOztBQUVqQjtJQUNJLGdCQUFnQjtJQUNoQix5QkFBeUI7SUFDekIsa0JBQWtCO0FBQ3RCXCIsXCJzb3VyY2VzQ29udGVudFwiOltcIjpyb290IHtcXHJcXG4gICAgLyogY29sb3JzICovXFxyXFxuICAgIC0tb3JhbmdlOiAjRkI2NTIzO1xcclxcbiAgICAtLXllbGxvdzogI0Q5OTczMztcXHJcXG4gICAgLS1ibHVlOiAjNDM3NDhFO1xcclxcbiAgICAtLWxpZ2h0Qmx1ZTogI2I3YzllMjtcXHJcXG4gICAgLS1kYXJrR3JheTogIzM3M0EzNztcXHJcXG4gICAgLS1tZWRpdW1HcmF5OiAjRDBEMEQwO1xcclxcbiAgICAtLWxpZ2h0R3JheTogI0YyRjJGMjtcXHJcXG4gICAgLS13aGl0ZTogI0ZCRkRGQjtcXHJcXG4gICAgLS1ibGFjazogIzIxMjEyMTtcXHJcXG4gICAgLS1ncmVlbjogIzcxQkM3ODtcXHJcXG4gICAgLS1jaXR5Z3JlZW46ICMwMDQ0NDU7XFxyXFxuICAgIC0tc3Bpcml0Z3JlZW46ICMyNzk5ODk7XFxyXFxuICAgIC0tcGFsZWdyZWVuOiAjOUZENUIzO1xcclxcbiAgICAtLXZlcnlwYWxlZ3JlZW46ICNERkY1RTM7XFxyXFxuICAgIC0tYWNjZW50eWVsbG93OiAjZmViNzBkO1xcclxcbiAgICAtLW5lYXJibGFjazogIzE4MjUyYTtcXHJcXG4gICAgLyogZm9udHMgKi9cXHJcXG4gICAgZm9udC1mYW1pbHk6ICdOdW5pdG8nLCBzYW5zLXNlcmlmO1xcclxcbiAgICBmb250LXNpemU6IDE0cHg7XFxyXFxuICAgIC8qIHNwYWNpbmcgKi9cXHJcXG4gICAgLS1oZWFkZXItaGVpZ2h0IDogMTA1cHg7XFxyXFxuICAgIC0tc2lkZWJhci13aWR0aDogMzAwcHg7XFxyXFxufVxcclxcblxcclxcbi8qIEJ1dHRvbiBzdHlsaW5nICovXFxyXFxuXFxyXFxuLmJ0biB7XFxyXFxuICAgIGN1cnNvcjogcG9pbnRlcjtcXHJcXG4gICAgcGFkZGluZzogNnB4O1xcclxcbiAgICBtYXJnaW4tdG9wOiA1cHg7XFxyXFxuICAgIG1hcmdpbi1yaWdodCA6IDJweDtcXHJcXG4gICAgbWFyZ2luLWxlZnQ6IDJweDtcXHJcXG4gICAgYm9yZGVyLXJhZGl1czogMTBweDtcXHJcXG4gICAgYmFja2dyb3VuZC1jb2xvcjogZ3JheTtcXHJcXG4gICAgY29sb3I6IHdoaXRlOyBcXHJcXG59XFxyXFxuXFxyXFxuLyogRm9udCBhcHBsaWNhdGlvbiBmb3IgYWxsIHRleHQgICovXFxyXFxuYm9keSwgYnV0dG9uLCBpbnB1dCwgdGV4dGFyZWEsIHNlbGVjdCwgLnNpZGViYXIsIHRhYmxlIHtcXHJcXG4gICAgZm9udC1mYW1pbHk6IHZhcigtLWZvbnQtZmFtaWx5KSwgc2Fucy1zZXJpZjtcXHJcXG4gIH1cXHJcXG5cXHJcXG4jbWFpbi1wYW5lbCB7XFxyXFxuICAgIC8qIGRpc3BsYXk6IGZsZXg7ICovXFxyXFxuICAgIC8qIGZsZXgtZGlyZWN0aW9uOiBjb2x1bW47ICovXFxyXFxuICAgIGp1c3RpZnktY29udGVudDogY2VudGVyOyAvKiBUaGlzIHdpbGwgY2VudGVyIGNvbnRlbnQgdmVydGljYWxseSAqL1xcclxcbiAgICBtaW4taGVpZ2h0OiBjYWxjKDEwMHZoIC0gdmFyKC0taGVhZGVyLWhlaWdodCkpIDsgXFxyXFxuICAgIC8qIG1heC13aWR0aDogY2FsYygxMDB2dyAtIHZhcigtLXNpZGViYXItd2lkdGgpKTsgICovXFxyXFxuICAgIHBhZGRpbmctdG9wOiAyNXB4O1xcclxcbiAgICBwYWRkaW5nLWJvdHRvbTogMjVweDtcXHJcXG4gICAgbWFyZ2luOiAwO1xcclxcbn1cXHJcXG5cXHJcXG4vKiBGb250IGF3ZXNvbWUgKi9cXHJcXG5cXHJcXG5pLmZhcy5mYS1jaGVjayB7IFxcclxcbiAgICBmb250LXNpemU6IDEuNWVtO1xcclxcbiAgICBjb2xvcjogdmFyKC0tc3Bpcml0Z3JlZW4pO1xcclxcbiAgICBtYXJnaW4tcmlnaHQ6IDEwcHg7XFxyXFxufVxcclxcblwiXSxcInNvdXJjZVJvb3RcIjpcIlwifV0pO1xuLy8gRXhwb3J0c1xuZXhwb3J0IGRlZmF1bHQgX19fQ1NTX0xPQURFUl9FWFBPUlRfX187XG4iXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///404\n')},995:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `.summary-accordion {\r\n width: 60%;\r\n margin: auto;\r\n}\r\n\r\n.accordion-title {\r\n margin-left: 20%;\r\n font-weight: bold;\r\n /* font-size: 20px; */\r\n padding-bottom: 10px;\r\n padding-left: 15px;\r\n padding-top: 30px;\r\n}\r\n\r\n.accordion-button:active, .accordion-button:not(.collapsed) {\r\n background-color: var(--palegreen);\r\n color: var(--citygreen);\r\n /* font-size: 20px; */\r\n}\r\n\r\n.accordion-table { \r\n width: 100%;\r\n font-size: 1.2em;\r\n /* border-collapse: separate; This is required for rounded corners */\r\n}\r\n\r\n.accordion-table td:last-child {\r\n text-align: right;\r\n padding-right: 15px;\r\n padding-bottom: 5px\r\n}\r\n\r\n.accordion-table tr {\r\n border-bottom: 1px solid black;\r\n border-top: none;\r\n border-right: none;\r\n border-left: none;\r\n}\r\n\r\n.accordion-table tr:last-child {\r\n border-bottom: none;\r\n background-color: var(--mediumGray);\r\n height : 50px;\r\n}\r\n\r\nspan.amount {\r\n font-weight: bold;\r\n position: absolute;\r\n right: 0;\r\n margin-right: 60px;\r\n}\r\n\r\n.accordion-header button {\r\n font-size: 0.8em;\r\n}\r\n\r\n.btn-add-init {\r\n background-color: var(--spiritgreen);\r\n border-color: var(--citygreen);\r\n /* position: absolute; */\r\n right: 0;\r\n /* margin-left: 20px; */\r\n /* margin-right: calc(20vw + 15px); */\r\n margin-top: 15px;\r\n}\r\n\r\n#supp-title {\r\n margin-top: 20px;\r\n}\r\n\r\n.btn-add-init:hover {\r\n background-color: var(--palegreen);\r\n}\r\n.add-init-btn-div{\r\n text-align: center;\r\n}\r\n\r\nspan.top-line-amount {\r\n font-weight: bold;\r\n position: absolute;\r\n right: 0;\r\n margin-right: calc(20vw + 20px);\r\n}`, "",{"version":3,"sources":["webpack://./src/js/components/accordion/accordion.css"],"names":[],"mappings":"AAAA;IACI,UAAU;IACV,YAAY;AAChB;;AAEA;IACI,gBAAgB;IAChB,iBAAiB;IACjB,qBAAqB;IACrB,oBAAoB;IACpB,kBAAkB;IAClB,iBAAiB;AACrB;;AAEA;IACI,kCAAkC;IAClC,uBAAuB;IACvB,qBAAqB;AACzB;;AAEA;IACI,WAAW;IACX,gBAAgB;IAChB,oEAAoE;AACxE;;AAEA;IACI,iBAAiB;IACjB,mBAAmB;IACnB;AACJ;;AAEA;IACI,8BAA8B;IAC9B,gBAAgB;IAChB,kBAAkB;IAClB,iBAAiB;AACrB;;AAEA;IACI,mBAAmB;IACnB,mCAAmC;IACnC,aAAa;AACjB;;AAEA;IACI,iBAAiB;IACjB,kBAAkB;IAClB,QAAQ;IACR,kBAAkB;AACtB;;AAEA;IACI,gBAAgB;AACpB;;AAEA;IACI,oCAAoC;IACpC,8BAA8B;IAC9B,wBAAwB;IACxB,QAAQ;IACR,uBAAuB;IACvB,qCAAqC;IACrC,gBAAgB;AACpB;;AAEA;GACG,gBAAgB;AACnB;;AAEA;IACI,kCAAkC;AACtC;AACA;IACI,kBAAkB;AACtB;;AAEA;IACI,iBAAiB;IACjB,kBAAkB;IAClB,QAAQ;IACR,+BAA+B;AACnC","sourcesContent":[".summary-accordion {\\r\\n width: 60%;\\r\\n margin: auto;\\r\\n}\\r\\n\\r\\n.accordion-title {\\r\\n margin-left: 20%;\\r\\n font-weight: bold;\\r\\n /* font-size: 20px; */\\r\\n padding-bottom: 10px;\\r\\n padding-left: 15px;\\r\\n padding-top: 30px;\\r\\n}\\r\\n\\r\\n.accordion-button:active, .accordion-button:not(.collapsed) {\\r\\n background-color: var(--palegreen);\\r\\n color: var(--citygreen);\\r\\n /* font-size: 20px; */\\r\\n}\\r\\n\\r\\n.accordion-table { \\r\\n width: 100%;\\r\\n font-size: 1.2em;\\r\\n /* border-collapse: separate; This is required for rounded corners */\\r\\n}\\r\\n\\r\\n.accordion-table td:last-child {\\r\\n text-align: right;\\r\\n padding-right: 15px;\\r\\n padding-bottom: 5px\\r\\n}\\r\\n\\r\\n.accordion-table tr {\\r\\n border-bottom: 1px solid black;\\r\\n border-top: none;\\r\\n border-right: none;\\r\\n border-left: none;\\r\\n}\\r\\n\\r\\n.accordion-table tr:last-child {\\r\\n border-bottom: none;\\r\\n background-color: var(--mediumGray);\\r\\n height : 50px;\\r\\n}\\r\\n\\r\\nspan.amount {\\r\\n font-weight: bold;\\r\\n position: absolute;\\r\\n right: 0;\\r\\n margin-right: 60px;\\r\\n}\\r\\n\\r\\n.accordion-header button {\\r\\n font-size: 0.8em;\\r\\n}\\r\\n\\r\\n.btn-add-init {\\r\\n background-color: var(--spiritgreen);\\r\\n border-color: var(--citygreen);\\r\\n /* position: absolute; */\\r\\n right: 0;\\r\\n /* margin-left: 20px; */\\r\\n /* margin-right: calc(20vw + 15px); */\\r\\n margin-top: 15px;\\r\\n}\\r\\n\\r\\n#supp-title {\\r\\n margin-top: 20px;\\r\\n}\\r\\n\\r\\n.btn-add-init:hover {\\r\\n background-color: var(--palegreen);\\r\\n}\\r\\n.add-init-btn-div{\\r\\n text-align: center;\\r\\n}\\r\\n\\r\\nspan.top-line-amount {\\r\\n font-weight: bold;\\r\\n position: absolute;\\r\\n right: 0;\\r\\n margin-right: calc(20vw + 20px);\\r\\n}"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiOTk1LmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9idWRnZXQtcmVxdWVzdC1mb3JtLy4vc3JjL2pzL2NvbXBvbmVudHMvYWNjb3JkaW9uL2FjY29yZGlvbi5jc3M/ZjBmOCJdLCJzb3VyY2VzQ29udGVudCI6WyIvLyBJbXBvcnRzXG5pbXBvcnQgX19fQ1NTX0xPQURFUl9BUElfU09VUkNFTUFQX0lNUE9SVF9fXyBmcm9tIFwiLi4vLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2Nzcy1sb2FkZXIvZGlzdC9ydW50aW1lL3NvdXJjZU1hcHMuanNcIjtcbmltcG9ydCBfX19DU1NfTE9BREVSX0FQSV9JTVBPUlRfX18gZnJvbSBcIi4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvcnVudGltZS9hcGkuanNcIjtcbnZhciBfX19DU1NfTE9BREVSX0VYUE9SVF9fXyA9IF9fX0NTU19MT0FERVJfQVBJX0lNUE9SVF9fXyhfX19DU1NfTE9BREVSX0FQSV9TT1VSQ0VNQVBfSU1QT1JUX19fKTtcbi8vIE1vZHVsZVxuX19fQ1NTX0xPQURFUl9FWFBPUlRfX18ucHVzaChbbW9kdWxlLmlkLCBgLnN1bW1hcnktYWNjb3JkaW9uIHtcclxuICAgIHdpZHRoOiA2MCU7XHJcbiAgICBtYXJnaW46IGF1dG87XHJcbn1cclxuXHJcbi5hY2NvcmRpb24tdGl0bGUge1xyXG4gICAgbWFyZ2luLWxlZnQ6IDIwJTtcclxuICAgIGZvbnQtd2VpZ2h0OiBib2xkO1xyXG4gICAgLyogZm9udC1zaXplOiAyMHB4OyAqL1xyXG4gICAgcGFkZGluZy1ib3R0b206IDEwcHg7XHJcbiAgICBwYWRkaW5nLWxlZnQ6IDE1cHg7XHJcbiAgICBwYWRkaW5nLXRvcDogMzBweDtcclxufVxyXG5cclxuLmFjY29yZGlvbi1idXR0b246YWN0aXZlLCAuYWNjb3JkaW9uLWJ1dHRvbjpub3QoLmNvbGxhcHNlZCkge1xyXG4gICAgYmFja2dyb3VuZC1jb2xvcjogdmFyKC0tcGFsZWdyZWVuKTtcclxuICAgIGNvbG9yOiB2YXIoLS1jaXR5Z3JlZW4pO1xyXG4gICAgLyogZm9udC1zaXplOiAyMHB4OyAqL1xyXG59XHJcblxyXG4uYWNjb3JkaW9uLXRhYmxlIHsgXHJcbiAgICB3aWR0aDogMTAwJTtcclxuICAgIGZvbnQtc2l6ZTogMS4yZW07XHJcbiAgICAvKiBib3JkZXItY29sbGFwc2U6IHNlcGFyYXRlOyBUaGlzIGlzIHJlcXVpcmVkIGZvciByb3VuZGVkIGNvcm5lcnMgKi9cclxufVxyXG5cclxuLmFjY29yZGlvbi10YWJsZSB0ZDpsYXN0LWNoaWxkIHtcclxuICAgIHRleHQtYWxpZ246IHJpZ2h0O1xyXG4gICAgcGFkZGluZy1yaWdodDogMTVweDtcclxuICAgIHBhZGRpbmctYm90dG9tOiA1cHhcclxufVxyXG5cclxuLmFjY29yZGlvbi10YWJsZSB0ciB7XHJcbiAgICBib3JkZXItYm90dG9tOiAxcHggc29saWQgYmxhY2s7XHJcbiAgICBib3JkZXItdG9wOiBub25lO1xyXG4gICAgYm9yZGVyLXJpZ2h0OiBub25lO1xyXG4gICAgYm9yZGVyLWxlZnQ6IG5vbmU7XHJcbn1cclxuXHJcbi5hY2NvcmRpb24tdGFibGUgdHI6bGFzdC1jaGlsZCB7XHJcbiAgICBib3JkZXItYm90dG9tOiBub25lO1xyXG4gICAgYmFja2dyb3VuZC1jb2xvcjogdmFyKC0tbWVkaXVtR3JheSk7XHJcbiAgICBoZWlnaHQgOiA1MHB4O1xyXG59XHJcblxyXG5zcGFuLmFtb3VudCB7XHJcbiAgICBmb250LXdlaWdodDogYm9sZDtcclxuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcclxuICAgIHJpZ2h0OiAwO1xyXG4gICAgbWFyZ2luLXJpZ2h0OiA2MHB4O1xyXG59XHJcblxyXG4uYWNjb3JkaW9uLWhlYWRlciBidXR0b24ge1xyXG4gICAgZm9udC1zaXplOiAwLjhlbTtcclxufVxyXG5cclxuLmJ0bi1hZGQtaW5pdCB7XHJcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiB2YXIoLS1zcGlyaXRncmVlbik7XHJcbiAgICBib3JkZXItY29sb3I6IHZhcigtLWNpdHlncmVlbik7XHJcbiAgICAvKiBwb3NpdGlvbjogYWJzb2x1dGU7ICovXHJcbiAgICByaWdodDogMDtcclxuICAgIC8qIG1hcmdpbi1sZWZ0OiAyMHB4OyAqL1xyXG4gICAgLyogbWFyZ2luLXJpZ2h0OiBjYWxjKDIwdncgKyAxNXB4KTsgKi9cclxuICAgIG1hcmdpbi10b3A6IDE1cHg7XHJcbn1cclxuXHJcbiNzdXBwLXRpdGxlIHtcclxuICAgbWFyZ2luLXRvcDogMjBweDtcclxufVxyXG5cclxuLmJ0bi1hZGQtaW5pdDpob3ZlciB7XHJcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiB2YXIoLS1wYWxlZ3JlZW4pO1xyXG59XHJcbi5hZGQtaW5pdC1idG4tZGl2e1xyXG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xyXG59XHJcblxyXG5zcGFuLnRvcC1saW5lLWFtb3VudCB7XHJcbiAgICBmb250LXdlaWdodDogYm9sZDtcclxuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcclxuICAgIHJpZ2h0OiAwO1xyXG4gICAgbWFyZ2luLXJpZ2h0OiBjYWxjKDIwdncgKyAyMHB4KTtcclxufWAsIFwiXCIse1widmVyc2lvblwiOjMsXCJzb3VyY2VzXCI6W1wid2VicGFjazovLy4vc3JjL2pzL2NvbXBvbmVudHMvYWNjb3JkaW9uL2FjY29yZGlvbi5jc3NcIl0sXCJuYW1lc1wiOltdLFwibWFwcGluZ3NcIjpcIkFBQUE7SUFDSSxVQUFVO0lBQ1YsWUFBWTtBQUNoQjs7QUFFQTtJQUNJLGdCQUFnQjtJQUNoQixpQkFBaUI7SUFDakIscUJBQXFCO0lBQ3JCLG9CQUFvQjtJQUNwQixrQkFBa0I7SUFDbEIsaUJBQWlCO0FBQ3JCOztBQUVBO0lBQ0ksa0NBQWtDO0lBQ2xDLHVCQUF1QjtJQUN2QixxQkFBcUI7QUFDekI7O0FBRUE7SUFDSSxXQUFXO0lBQ1gsZ0JBQWdCO0lBQ2hCLG9FQUFvRTtBQUN4RTs7QUFFQTtJQUNJLGlCQUFpQjtJQUNqQixtQkFBbUI7SUFDbkI7QUFDSjs7QUFFQTtJQUNJLDhCQUE4QjtJQUM5QixnQkFBZ0I7SUFDaEIsa0JBQWtCO0lBQ2xCLGlCQUFpQjtBQUNyQjs7QUFFQTtJQUNJLG1CQUFtQjtJQUNuQixtQ0FBbUM7SUFDbkMsYUFBYTtBQUNqQjs7QUFFQTtJQUNJLGlCQUFpQjtJQUNqQixrQkFBa0I7SUFDbEIsUUFBUTtJQUNSLGtCQUFrQjtBQUN0Qjs7QUFFQTtJQUNJLGdCQUFnQjtBQUNwQjs7QUFFQTtJQUNJLG9DQUFvQztJQUNwQyw4QkFBOEI7SUFDOUIsd0JBQXdCO0lBQ3hCLFFBQVE7SUFDUix1QkFBdUI7SUFDdkIscUNBQXFDO0lBQ3JDLGdCQUFnQjtBQUNwQjs7QUFFQTtHQUNHLGdCQUFnQjtBQUNuQjs7QUFFQTtJQUNJLGtDQUFrQztBQUN0QztBQUNBO0lBQ0ksa0JBQWtCO0FBQ3RCOztBQUVBO0lBQ0ksaUJBQWlCO0lBQ2pCLGtCQUFrQjtJQUNsQixRQUFRO0lBQ1IsK0JBQStCO0FBQ25DXCIsXCJzb3VyY2VzQ29udGVudFwiOltcIi5zdW1tYXJ5LWFjY29yZGlvbiB7XFxyXFxuICAgIHdpZHRoOiA2MCU7XFxyXFxuICAgIG1hcmdpbjogYXV0bztcXHJcXG59XFxyXFxuXFxyXFxuLmFjY29yZGlvbi10aXRsZSB7XFxyXFxuICAgIG1hcmdpbi1sZWZ0OiAyMCU7XFxyXFxuICAgIGZvbnQtd2VpZ2h0OiBib2xkO1xcclxcbiAgICAvKiBmb250LXNpemU6IDIwcHg7ICovXFxyXFxuICAgIHBhZGRpbmctYm90dG9tOiAxMHB4O1xcclxcbiAgICBwYWRkaW5nLWxlZnQ6IDE1cHg7XFxyXFxuICAgIHBhZGRpbmctdG9wOiAzMHB4O1xcclxcbn1cXHJcXG5cXHJcXG4uYWNjb3JkaW9uLWJ1dHRvbjphY3RpdmUsIC5hY2NvcmRpb24tYnV0dG9uOm5vdCguY29sbGFwc2VkKSB7XFxyXFxuICAgIGJhY2tncm91bmQtY29sb3I6IHZhcigtLXBhbGVncmVlbik7XFxyXFxuICAgIGNvbG9yOiB2YXIoLS1jaXR5Z3JlZW4pO1xcclxcbiAgICAvKiBmb250LXNpemU6IDIwcHg7ICovXFxyXFxufVxcclxcblxcclxcbi5hY2NvcmRpb24tdGFibGUgeyBcXHJcXG4gICAgd2lkdGg6IDEwMCU7XFxyXFxuICAgIGZvbnQtc2l6ZTogMS4yZW07XFxyXFxuICAgIC8qIGJvcmRlci1jb2xsYXBzZTogc2VwYXJhdGU7IFRoaXMgaXMgcmVxdWlyZWQgZm9yIHJvdW5kZWQgY29ybmVycyAqL1xcclxcbn1cXHJcXG5cXHJcXG4uYWNjb3JkaW9uLXRhYmxlIHRkOmxhc3QtY2hpbGQge1xcclxcbiAgICB0ZXh0LWFsaWduOiByaWdodDtcXHJcXG4gICAgcGFkZGluZy1yaWdodDogMTVweDtcXHJcXG4gICAgcGFkZGluZy1ib3R0b206IDVweFxcclxcbn1cXHJcXG5cXHJcXG4uYWNjb3JkaW9uLXRhYmxlIHRyIHtcXHJcXG4gICAgYm9yZGVyLWJvdHRvbTogMXB4IHNvbGlkIGJsYWNrO1xcclxcbiAgICBib3JkZXItdG9wOiBub25lO1xcclxcbiAgICBib3JkZXItcmlnaHQ6IG5vbmU7XFxyXFxuICAgIGJvcmRlci1sZWZ0OiBub25lO1xcclxcbn1cXHJcXG5cXHJcXG4uYWNjb3JkaW9uLXRhYmxlIHRyOmxhc3QtY2hpbGQge1xcclxcbiAgICBib3JkZXItYm90dG9tOiBub25lO1xcclxcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiB2YXIoLS1tZWRpdW1HcmF5KTtcXHJcXG4gICAgaGVpZ2h0IDogNTBweDtcXHJcXG59XFxyXFxuXFxyXFxuc3Bhbi5hbW91bnQge1xcclxcbiAgICBmb250LXdlaWdodDogYm9sZDtcXHJcXG4gICAgcG9zaXRpb246IGFic29sdXRlO1xcclxcbiAgICByaWdodDogMDtcXHJcXG4gICAgbWFyZ2luLXJpZ2h0OiA2MHB4O1xcclxcbn1cXHJcXG5cXHJcXG4uYWNjb3JkaW9uLWhlYWRlciBidXR0b24ge1xcclxcbiAgICBmb250LXNpemU6IDAuOGVtO1xcclxcbn1cXHJcXG5cXHJcXG4uYnRuLWFkZC1pbml0IHtcXHJcXG4gICAgYmFja2dyb3VuZC1jb2xvcjogdmFyKC0tc3Bpcml0Z3JlZW4pO1xcclxcbiAgICBib3JkZXItY29sb3I6IHZhcigtLWNpdHlncmVlbik7XFxyXFxuICAgIC8qIHBvc2l0aW9uOiBhYnNvbHV0ZTsgKi9cXHJcXG4gICAgcmlnaHQ6IDA7XFxyXFxuICAgIC8qIG1hcmdpbi1sZWZ0OiAyMHB4OyAqL1xcclxcbiAgICAvKiBtYXJnaW4tcmlnaHQ6IGNhbGMoMjB2dyArIDE1cHgpOyAqL1xcclxcbiAgICBtYXJnaW4tdG9wOiAxNXB4O1xcclxcbn1cXHJcXG5cXHJcXG4jc3VwcC10aXRsZSB7XFxyXFxuICAgbWFyZ2luLXRvcDogMjBweDtcXHJcXG59XFxyXFxuXFxyXFxuLmJ0bi1hZGQtaW5pdDpob3ZlciB7XFxyXFxuICAgIGJhY2tncm91bmQtY29sb3I6IHZhcigtLXBhbGVncmVlbik7XFxyXFxufVxcclxcbi5hZGQtaW5pdC1idG4tZGl2e1xcclxcbiAgICB0ZXh0LWFsaWduOiBjZW50ZXI7XFxyXFxufVxcclxcblxcclxcbnNwYW4udG9wLWxpbmUtYW1vdW50IHtcXHJcXG4gICAgZm9udC13ZWlnaHQ6IGJvbGQ7XFxyXFxuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcXHJcXG4gICAgcmlnaHQ6IDA7XFxyXFxuICAgIG1hcmdpbi1yaWdodDogY2FsYygyMHZ3ICsgMjBweCk7XFxyXFxufVwiXSxcInNvdXJjZVJvb3RcIjpcIlwifV0pO1xuLy8gRXhwb3J0c1xuZXhwb3J0IGRlZmF1bHQgX19fQ1NTX0xPQURFUl9FWFBPUlRfX187XG4iXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///995\n')},9:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `body {\r\n background-color: var(--lightGray);\r\n margin: 0;\r\n}`, "",{"version":3,"sources":["webpack://./src/js/components/body/body.css"],"names":[],"mappings":"AAAA;IACI,kCAAkC;IAClC,SAAS;AACb","sourcesContent":["body {\\r\\n background-color: var(--lightGray);\\r\\n margin: 0;\\r\\n}"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiOS5qcyIsIm1hcHBpbmdzIjoiOzs7Ozs7O0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSIsInNvdXJjZXMiOlsid2VicGFjazovL2J1ZGdldC1yZXF1ZXN0LWZvcm0vLi9zcmMvanMvY29tcG9uZW50cy9ib2R5L2JvZHkuY3NzPzM2ZTAiXSwic291cmNlc0NvbnRlbnQiOlsiLy8gSW1wb3J0c1xuaW1wb3J0IF9fX0NTU19MT0FERVJfQVBJX1NPVVJDRU1BUF9JTVBPUlRfX18gZnJvbSBcIi4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvcnVudGltZS9zb3VyY2VNYXBzLmpzXCI7XG5pbXBvcnQgX19fQ1NTX0xPQURFUl9BUElfSU1QT1JUX19fIGZyb20gXCIuLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvYXBpLmpzXCI7XG52YXIgX19fQ1NTX0xPQURFUl9FWFBPUlRfX18gPSBfX19DU1NfTE9BREVSX0FQSV9JTVBPUlRfX18oX19fQ1NTX0xPQURFUl9BUElfU09VUkNFTUFQX0lNUE9SVF9fXyk7XG4vLyBNb2R1bGVcbl9fX0NTU19MT0FERVJfRVhQT1JUX19fLnB1c2goW21vZHVsZS5pZCwgYGJvZHkge1xyXG4gICAgYmFja2dyb3VuZC1jb2xvcjogdmFyKC0tbGlnaHRHcmF5KTtcclxuICAgIG1hcmdpbjogMDtcclxufWAsIFwiXCIse1widmVyc2lvblwiOjMsXCJzb3VyY2VzXCI6W1wid2VicGFjazovLy4vc3JjL2pzL2NvbXBvbmVudHMvYm9keS9ib2R5LmNzc1wiXSxcIm5hbWVzXCI6W10sXCJtYXBwaW5nc1wiOlwiQUFBQTtJQUNJLGtDQUFrQztJQUNsQyxTQUFTO0FBQ2JcIixcInNvdXJjZXNDb250ZW50XCI6W1wiYm9keSB7XFxyXFxuICAgIGJhY2tncm91bmQtY29sb3I6IHZhcigtLWxpZ2h0R3JheSk7XFxyXFxuICAgIG1hcmdpbjogMDtcXHJcXG59XCJdLFwic291cmNlUm9vdFwiOlwiXCJ9XSk7XG4vLyBFeHBvcnRzXG5leHBvcnQgZGVmYXVsdCBfX19DU1NfTE9BREVSX0VYUE9SVF9fXztcbiJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///9\n')},39:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `#file-input {\r\n margin-left: 40%;\r\n}\r\n\r\n/* Spinner Styles */\r\n.spinner {\r\n display: inline-block;\r\n width: 150px;\r\n height: 150px;\r\n border: 20px solid rgba(0, 0, 0, 0.1);\r\n border-radius: 50%;\r\n border-top-color: var(--spiritgreen);\r\n animation: spin 1s ease-in-out infinite;\r\n position: fixed; /* Ensures it stays fixed relative to the viewport */\r\n top: 20%; /* Center vertically */\r\n left: 43%; /* Center horizontally */\r\n transform: translate(-50%, -50%); /* Centralize the spinner exactly */\r\n z-index: 9999; /* Ensures it appears in front of other elements */\r\n background-color: rgba(255, 255, 255, 0.8); /* Optional: light background */\r\n padding: 20px; /* Optional: extra padding for better visibility */\r\n }\r\n \r\n @keyframes spin {\r\n 0% {\r\n transform: rotate(0deg);\r\n }\r\n 100% {\r\n transform: rotate(360deg);\r\n }\r\n }`, "",{"version":3,"sources":["webpack://./src/js/components/file_upload/file_upload.css"],"names":[],"mappings":"AAAA;IACI,gBAAgB;AACpB;;AAEA,mBAAmB;AACnB;IACI,qBAAqB;IACrB,YAAY;IACZ,aAAa;IACb,qCAAqC;IACrC,kBAAkB;IAClB,oCAAoC;IACpC,uCAAuC;IACvC,eAAe,EAAE,oDAAoD;IACrE,QAAQ,EAAE,sBAAsB;IAChC,SAAS,EAAE,wBAAwB;IACnC,gCAAgC,EAAE,mCAAmC;IACrE,aAAa,EAAE,kDAAkD;IACjE,0CAA0C,EAAE,+BAA+B;IAC3E,aAAa,EAAE,kDAAkD;EACnE;;EAEA;IACE;MACE,uBAAuB;IACzB;IACA;MACE,yBAAyB;IAC3B;EACF","sourcesContent":["#file-input {\\r\\n margin-left: 40%;\\r\\n}\\r\\n\\r\\n/* Spinner Styles */\\r\\n.spinner {\\r\\n display: inline-block;\\r\\n width: 150px;\\r\\n height: 150px;\\r\\n border: 20px solid rgba(0, 0, 0, 0.1);\\r\\n border-radius: 50%;\\r\\n border-top-color: var(--spiritgreen);\\r\\n animation: spin 1s ease-in-out infinite;\\r\\n position: fixed; /* Ensures it stays fixed relative to the viewport */\\r\\n top: 20%; /* Center vertically */\\r\\n left: 43%; /* Center horizontally */\\r\\n transform: translate(-50%, -50%); /* Centralize the spinner exactly */\\r\\n z-index: 9999; /* Ensures it appears in front of other elements */\\r\\n background-color: rgba(255, 255, 255, 0.8); /* Optional: light background */\\r\\n padding: 20px; /* Optional: extra padding for better visibility */\\r\\n }\\r\\n \\r\\n @keyframes spin {\\r\\n 0% {\\r\\n transform: rotate(0deg);\\r\\n }\\r\\n 100% {\\r\\n transform: rotate(360deg);\\r\\n }\\r\\n }"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMzkuanMiLCJtYXBwaW5ncyI6Ijs7Ozs7OztBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYnVkZ2V0LXJlcXVlc3QtZm9ybS8uL3NyYy9qcy9jb21wb25lbnRzL2ZpbGVfdXBsb2FkL2ZpbGVfdXBsb2FkLmNzcz9jNTU4Il0sInNvdXJjZXNDb250ZW50IjpbIi8vIEltcG9ydHNcbmltcG9ydCBfX19DU1NfTE9BREVSX0FQSV9TT1VSQ0VNQVBfSU1QT1JUX19fIGZyb20gXCIuLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvc291cmNlTWFwcy5qc1wiO1xuaW1wb3J0IF9fX0NTU19MT0FERVJfQVBJX0lNUE9SVF9fXyBmcm9tIFwiLi4vLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2Nzcy1sb2FkZXIvZGlzdC9ydW50aW1lL2FwaS5qc1wiO1xudmFyIF9fX0NTU19MT0FERVJfRVhQT1JUX19fID0gX19fQ1NTX0xPQURFUl9BUElfSU1QT1JUX19fKF9fX0NTU19MT0FERVJfQVBJX1NPVVJDRU1BUF9JTVBPUlRfX18pO1xuLy8gTW9kdWxlXG5fX19DU1NfTE9BREVSX0VYUE9SVF9fXy5wdXNoKFttb2R1bGUuaWQsIGAjZmlsZS1pbnB1dCB7XHJcbiAgICBtYXJnaW4tbGVmdDogNDAlO1xyXG59XHJcblxyXG4vKiBTcGlubmVyIFN0eWxlcyAqL1xyXG4uc3Bpbm5lciB7XHJcbiAgICBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7XHJcbiAgICB3aWR0aDogMTUwcHg7XHJcbiAgICBoZWlnaHQ6IDE1MHB4O1xyXG4gICAgYm9yZGVyOiAyMHB4IHNvbGlkIHJnYmEoMCwgMCwgMCwgMC4xKTtcclxuICAgIGJvcmRlci1yYWRpdXM6IDUwJTtcclxuICAgIGJvcmRlci10b3AtY29sb3I6IHZhcigtLXNwaXJpdGdyZWVuKTtcclxuICAgIGFuaW1hdGlvbjogc3BpbiAxcyBlYXNlLWluLW91dCBpbmZpbml0ZTtcclxuICAgIHBvc2l0aW9uOiBmaXhlZDsgLyogRW5zdXJlcyBpdCBzdGF5cyBmaXhlZCByZWxhdGl2ZSB0byB0aGUgdmlld3BvcnQgKi9cclxuICAgIHRvcDogMjAlOyAvKiBDZW50ZXIgdmVydGljYWxseSAqL1xyXG4gICAgbGVmdDogNDMlOyAvKiBDZW50ZXIgaG9yaXpvbnRhbGx5ICovXHJcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZSgtNTAlLCAtNTAlKTsgLyogQ2VudHJhbGl6ZSB0aGUgc3Bpbm5lciBleGFjdGx5ICovXHJcbiAgICB6LWluZGV4OiA5OTk5OyAvKiBFbnN1cmVzIGl0IGFwcGVhcnMgaW4gZnJvbnQgb2Ygb3RoZXIgZWxlbWVudHMgKi9cclxuICAgIGJhY2tncm91bmQtY29sb3I6IHJnYmEoMjU1LCAyNTUsIDI1NSwgMC44KTsgLyogT3B0aW9uYWw6IGxpZ2h0IGJhY2tncm91bmQgKi9cclxuICAgIHBhZGRpbmc6IDIwcHg7IC8qIE9wdGlvbmFsOiBleHRyYSBwYWRkaW5nIGZvciBiZXR0ZXIgdmlzaWJpbGl0eSAqL1xyXG4gIH1cclxuICBcclxuICBAa2V5ZnJhbWVzIHNwaW4ge1xyXG4gICAgMCUge1xyXG4gICAgICB0cmFuc2Zvcm06IHJvdGF0ZSgwZGVnKTtcclxuICAgIH1cclxuICAgIDEwMCUge1xyXG4gICAgICB0cmFuc2Zvcm06IHJvdGF0ZSgzNjBkZWcpO1xyXG4gICAgfVxyXG4gIH1gLCBcIlwiLHtcInZlcnNpb25cIjozLFwic291cmNlc1wiOltcIndlYnBhY2s6Ly8uL3NyYy9qcy9jb21wb25lbnRzL2ZpbGVfdXBsb2FkL2ZpbGVfdXBsb2FkLmNzc1wiXSxcIm5hbWVzXCI6W10sXCJtYXBwaW5nc1wiOlwiQUFBQTtJQUNJLGdCQUFnQjtBQUNwQjs7QUFFQSxtQkFBbUI7QUFDbkI7SUFDSSxxQkFBcUI7SUFDckIsWUFBWTtJQUNaLGFBQWE7SUFDYixxQ0FBcUM7SUFDckMsa0JBQWtCO0lBQ2xCLG9DQUFvQztJQUNwQyx1Q0FBdUM7SUFDdkMsZUFBZSxFQUFFLG9EQUFvRDtJQUNyRSxRQUFRLEVBQUUsc0JBQXNCO0lBQ2hDLFNBQVMsRUFBRSx3QkFBd0I7SUFDbkMsZ0NBQWdDLEVBQUUsbUNBQW1DO0lBQ3JFLGFBQWEsRUFBRSxrREFBa0Q7SUFDakUsMENBQTBDLEVBQUUsK0JBQStCO0lBQzNFLGFBQWEsRUFBRSxrREFBa0Q7RUFDbkU7O0VBRUE7SUFDRTtNQUNFLHVCQUF1QjtJQUN6QjtJQUNBO01BQ0UseUJBQXlCO0lBQzNCO0VBQ0ZcIixcInNvdXJjZXNDb250ZW50XCI6W1wiI2ZpbGUtaW5wdXQge1xcclxcbiAgICBtYXJnaW4tbGVmdDogNDAlO1xcclxcbn1cXHJcXG5cXHJcXG4vKiBTcGlubmVyIFN0eWxlcyAqL1xcclxcbi5zcGlubmVyIHtcXHJcXG4gICAgZGlzcGxheTogaW5saW5lLWJsb2NrO1xcclxcbiAgICB3aWR0aDogMTUwcHg7XFxyXFxuICAgIGhlaWdodDogMTUwcHg7XFxyXFxuICAgIGJvcmRlcjogMjBweCBzb2xpZCByZ2JhKDAsIDAsIDAsIDAuMSk7XFxyXFxuICAgIGJvcmRlci1yYWRpdXM6IDUwJTtcXHJcXG4gICAgYm9yZGVyLXRvcC1jb2xvcjogdmFyKC0tc3Bpcml0Z3JlZW4pO1xcclxcbiAgICBhbmltYXRpb246IHNwaW4gMXMgZWFzZS1pbi1vdXQgaW5maW5pdGU7XFxyXFxuICAgIHBvc2l0aW9uOiBmaXhlZDsgLyogRW5zdXJlcyBpdCBzdGF5cyBmaXhlZCByZWxhdGl2ZSB0byB0aGUgdmlld3BvcnQgKi9cXHJcXG4gICAgdG9wOiAyMCU7IC8qIENlbnRlciB2ZXJ0aWNhbGx5ICovXFxyXFxuICAgIGxlZnQ6IDQzJTsgLyogQ2VudGVyIGhvcml6b250YWxseSAqL1xcclxcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZSgtNTAlLCAtNTAlKTsgLyogQ2VudHJhbGl6ZSB0aGUgc3Bpbm5lciBleGFjdGx5ICovXFxyXFxuICAgIHotaW5kZXg6IDk5OTk7IC8qIEVuc3VyZXMgaXQgYXBwZWFycyBpbiBmcm9udCBvZiBvdGhlciBlbGVtZW50cyAqL1xcclxcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKDI1NSwgMjU1LCAyNTUsIDAuOCk7IC8qIE9wdGlvbmFsOiBsaWdodCBiYWNrZ3JvdW5kICovXFxyXFxuICAgIHBhZGRpbmc6IDIwcHg7IC8qIE9wdGlvbmFsOiBleHRyYSBwYWRkaW5nIGZvciBiZXR0ZXIgdmlzaWJpbGl0eSAqL1xcclxcbiAgfVxcclxcbiAgXFxyXFxuICBAa2V5ZnJhbWVzIHNwaW4ge1xcclxcbiAgICAwJSB7XFxyXFxuICAgICAgdHJhbnNmb3JtOiByb3RhdGUoMGRlZyk7XFxyXFxuICAgIH1cXHJcXG4gICAgMTAwJSB7XFxyXFxuICAgICAgdHJhbnNmb3JtOiByb3RhdGUoMzYwZGVnKTtcXHJcXG4gICAgfVxcclxcbiAgfVwiXSxcInNvdXJjZVJvb3RcIjpcIlwifV0pO1xuLy8gRXhwb3J0c1xuZXhwb3J0IGRlZmF1bHQgX19fQ1NTX0xPQURFUl9FWFBPUlRfX187XG4iXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///39\n')},109:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `textarea {height: 100px; width: 100%;}\r\n\r\ntextarea, input, select {\r\n margin-bottom: 5px;\r\n}\r\n\r\n.btn-submit {\r\n margin-top: 20px;\r\n width: 60%;\r\n margin-left: 20%;\r\n background-color: var(--spiritgreen);\r\n}\r\n\r\n#new-form label {\r\n display: block; /* Ensure label is on its own line */\r\n margin-bottom: 0.25em; \r\n}\r\n\r\n#new-form select {\r\n min-width: 300px;\r\n}\r\n\r\n/* Thicker red border when focused */\r\ninput[readonly]:focus {\r\n border: 2px solid red;\r\n outline: none; /* Remove the default outline */\r\n}\r\n\r\n.error-message {\r\n color: red;\r\n}`, "",{"version":3,"sources":["webpack://./src/js/components/form/form.css"],"names":[],"mappings":"AAAA,UAAU,aAAa,EAAE,WAAW,CAAC;;AAErC;IACI,kBAAkB;AACtB;;AAEA;IACI,gBAAgB;IAChB,UAAU;IACV,gBAAgB;IAChB,oCAAoC;AACxC;;AAEA;IACI,cAAc,EAAE,oCAAoC;IACpD,qBAAqB;AACzB;;AAEA;IACI,gBAAgB;AACpB;;AAEA,oCAAoC;AACpC;IACI,qBAAqB;IACrB,aAAa,EAAE,+BAA+B;AAClD;;AAEA;IACI,UAAU;AACd","sourcesContent":["textarea {height: 100px; width: 100%;}\\r\\n\\r\\ntextarea, input, select {\\r\\n margin-bottom: 5px;\\r\\n}\\r\\n\\r\\n.btn-submit {\\r\\n margin-top: 20px;\\r\\n width: 60%;\\r\\n margin-left: 20%;\\r\\n background-color: var(--spiritgreen);\\r\\n}\\r\\n\\r\\n#new-form label {\\r\\n display: block; /* Ensure label is on its own line */\\r\\n margin-bottom: 0.25em; \\r\\n}\\r\\n\\r\\n#new-form select {\\r\\n min-width: 300px;\\r\\n}\\r\\n\\r\\n/* Thicker red border when focused */\\r\\ninput[readonly]:focus {\\r\\n border: 2px solid red;\\r\\n outline: none; /* Remove the default outline */\\r\\n}\\r\\n\\r\\n.error-message {\\r\\n color: red;\\r\\n}"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMTA5LmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYnVkZ2V0LXJlcXVlc3QtZm9ybS8uL3NyYy9qcy9jb21wb25lbnRzL2Zvcm0vZm9ybS5jc3M/ZTM4YSJdLCJzb3VyY2VzQ29udGVudCI6WyIvLyBJbXBvcnRzXG5pbXBvcnQgX19fQ1NTX0xPQURFUl9BUElfU09VUkNFTUFQX0lNUE9SVF9fXyBmcm9tIFwiLi4vLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2Nzcy1sb2FkZXIvZGlzdC9ydW50aW1lL3NvdXJjZU1hcHMuanNcIjtcbmltcG9ydCBfX19DU1NfTE9BREVSX0FQSV9JTVBPUlRfX18gZnJvbSBcIi4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvcnVudGltZS9hcGkuanNcIjtcbnZhciBfX19DU1NfTE9BREVSX0VYUE9SVF9fXyA9IF9fX0NTU19MT0FERVJfQVBJX0lNUE9SVF9fXyhfX19DU1NfTE9BREVSX0FQSV9TT1VSQ0VNQVBfSU1QT1JUX19fKTtcbi8vIE1vZHVsZVxuX19fQ1NTX0xPQURFUl9FWFBPUlRfX18ucHVzaChbbW9kdWxlLmlkLCBgdGV4dGFyZWEge2hlaWdodDogMTAwcHg7IHdpZHRoOiAxMDAlO31cclxuXHJcbnRleHRhcmVhLCBpbnB1dCwgc2VsZWN0IHtcclxuICAgIG1hcmdpbi1ib3R0b206IDVweDtcclxufVxyXG5cclxuLmJ0bi1zdWJtaXQge1xyXG4gICAgbWFyZ2luLXRvcDogMjBweDtcclxuICAgIHdpZHRoOiA2MCU7XHJcbiAgICBtYXJnaW4tbGVmdDogMjAlO1xyXG4gICAgYmFja2dyb3VuZC1jb2xvcjogdmFyKC0tc3Bpcml0Z3JlZW4pO1xyXG59XHJcblxyXG4jbmV3LWZvcm0gbGFiZWwge1xyXG4gICAgZGlzcGxheTogYmxvY2s7IC8qIEVuc3VyZSBsYWJlbCBpcyBvbiBpdHMgb3duIGxpbmUgKi9cclxuICAgIG1hcmdpbi1ib3R0b206IDAuMjVlbTsgXHJcbn1cclxuXHJcbiNuZXctZm9ybSBzZWxlY3Qge1xyXG4gICAgbWluLXdpZHRoOiAzMDBweDtcclxufVxyXG5cclxuLyogVGhpY2tlciByZWQgYm9yZGVyIHdoZW4gZm9jdXNlZCAqL1xyXG5pbnB1dFtyZWFkb25seV06Zm9jdXMge1xyXG4gICAgYm9yZGVyOiAycHggc29saWQgcmVkO1xyXG4gICAgb3V0bGluZTogbm9uZTsgLyogUmVtb3ZlIHRoZSBkZWZhdWx0IG91dGxpbmUgKi9cclxufVxyXG5cclxuLmVycm9yLW1lc3NhZ2Uge1xyXG4gICAgY29sb3I6IHJlZDtcclxufWAsIFwiXCIse1widmVyc2lvblwiOjMsXCJzb3VyY2VzXCI6W1wid2VicGFjazovLy4vc3JjL2pzL2NvbXBvbmVudHMvZm9ybS9mb3JtLmNzc1wiXSxcIm5hbWVzXCI6W10sXCJtYXBwaW5nc1wiOlwiQUFBQSxVQUFVLGFBQWEsRUFBRSxXQUFXLENBQUM7O0FBRXJDO0lBQ0ksa0JBQWtCO0FBQ3RCOztBQUVBO0lBQ0ksZ0JBQWdCO0lBQ2hCLFVBQVU7SUFDVixnQkFBZ0I7SUFDaEIsb0NBQW9DO0FBQ3hDOztBQUVBO0lBQ0ksY0FBYyxFQUFFLG9DQUFvQztJQUNwRCxxQkFBcUI7QUFDekI7O0FBRUE7SUFDSSxnQkFBZ0I7QUFDcEI7O0FBRUEsb0NBQW9DO0FBQ3BDO0lBQ0kscUJBQXFCO0lBQ3JCLGFBQWEsRUFBRSwrQkFBK0I7QUFDbEQ7O0FBRUE7SUFDSSxVQUFVO0FBQ2RcIixcInNvdXJjZXNDb250ZW50XCI6W1widGV4dGFyZWEge2hlaWdodDogMTAwcHg7IHdpZHRoOiAxMDAlO31cXHJcXG5cXHJcXG50ZXh0YXJlYSwgaW5wdXQsIHNlbGVjdCB7XFxyXFxuICAgIG1hcmdpbi1ib3R0b206IDVweDtcXHJcXG59XFxyXFxuXFxyXFxuLmJ0bi1zdWJtaXQge1xcclxcbiAgICBtYXJnaW4tdG9wOiAyMHB4O1xcclxcbiAgICB3aWR0aDogNjAlO1xcclxcbiAgICBtYXJnaW4tbGVmdDogMjAlO1xcclxcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiB2YXIoLS1zcGlyaXRncmVlbik7XFxyXFxufVxcclxcblxcclxcbiNuZXctZm9ybSBsYWJlbCB7XFxyXFxuICAgIGRpc3BsYXk6IGJsb2NrOyAvKiBFbnN1cmUgbGFiZWwgaXMgb24gaXRzIG93biBsaW5lICovXFxyXFxuICAgIG1hcmdpbi1ib3R0b206IDAuMjVlbTsgXFxyXFxufVxcclxcblxcclxcbiNuZXctZm9ybSBzZWxlY3Qge1xcclxcbiAgICBtaW4td2lkdGg6IDMwMHB4O1xcclxcbn1cXHJcXG5cXHJcXG4vKiBUaGlja2VyIHJlZCBib3JkZXIgd2hlbiBmb2N1c2VkICovXFxyXFxuaW5wdXRbcmVhZG9ubHldOmZvY3VzIHtcXHJcXG4gICAgYm9yZGVyOiAycHggc29saWQgcmVkO1xcclxcbiAgICBvdXRsaW5lOiBub25lOyAvKiBSZW1vdmUgdGhlIGRlZmF1bHQgb3V0bGluZSAqL1xcclxcbn1cXHJcXG5cXHJcXG4uZXJyb3ItbWVzc2FnZSB7XFxyXFxuICAgIGNvbG9yOiByZWQ7XFxyXFxufVwiXSxcInNvdXJjZVJvb3RcIjpcIlwifV0pO1xuLy8gRXhwb3J0c1xuZXhwb3J0IGRlZmF1bHQgX19fQ1NTX0xPQURFUl9FWFBPUlRfX187XG4iXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///109\n')},107:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `\r\nh1 {\r\n text-align: center;\r\n font-size: 30px;\r\n}\r\n\r\nh2 {\r\n color: var(--darkGray);\r\n text-align: center;\r\n font-size: 20px;\r\n}\r\n\r\nheader {\r\n align-items: center;\r\n background-color: white;\r\n padding: 5px;\r\n border-bottom: 1px solid var(--citygreen); \r\n /* border: 1px solid var(--citygreen); */\r\n height: var(--header-height);\r\n /* width: calc(100vw - var(--sidebar-width)); */\r\n padding-left: 20px;\r\n}\r\n\r\n/* Logo styling */\r\n#logo {\r\n height: 50px; /* Or your desired size */\r\n margin-right: 20px; /* Optional: space between logo and title */\r\n margin-bottom: -50px;\r\n margin-top: 10px;\r\n}`, "",{"version":3,"sources":["webpack://./src/js/components/header/header.css"],"names":[],"mappings":";AACA;IACI,kBAAkB;IAClB,eAAe;AACnB;;AAEA;IACI,sBAAsB;IACtB,kBAAkB;IAClB,eAAe;AACnB;;AAEA;IACI,mBAAmB;IACnB,uBAAuB;IACvB,YAAY;IACZ,yCAAyC;IACzC,yCAAyC;IACzC,4BAA4B;IAC5B,+CAA+C;IAC/C,kBAAkB;AACtB;;AAEA,iBAAiB;AACjB;IACI,YAAY,EAAE,yBAAyB;IACvC,kBAAkB,EAAE,2CAA2C;IAC/D,oBAAoB;IACpB,gBAAgB;AACpB","sourcesContent":["\\r\\nh1 {\\r\\n text-align: center;\\r\\n font-size: 30px;\\r\\n}\\r\\n\\r\\nh2 {\\r\\n color: var(--darkGray);\\r\\n text-align: center;\\r\\n font-size: 20px;\\r\\n}\\r\\n\\r\\nheader {\\r\\n align-items: center;\\r\\n background-color: white;\\r\\n padding: 5px;\\r\\n border-bottom: 1px solid var(--citygreen); \\r\\n /* border: 1px solid var(--citygreen); */\\r\\n height: var(--header-height);\\r\\n /* width: calc(100vw - var(--sidebar-width)); */\\r\\n padding-left: 20px;\\r\\n}\\r\\n\\r\\n/* Logo styling */\\r\\n#logo {\\r\\n height: 50px; /* Or your desired size */\\r\\n margin-right: 20px; /* Optional: space between logo and title */\\r\\n margin-bottom: -50px;\\r\\n margin-top: 10px;\\r\\n}"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMTA3LmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSIsInNvdXJjZXMiOlsid2VicGFjazovL2J1ZGdldC1yZXF1ZXN0LWZvcm0vLi9zcmMvanMvY29tcG9uZW50cy9oZWFkZXIvaGVhZGVyLmNzcz8yZjRjIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIEltcG9ydHNcbmltcG9ydCBfX19DU1NfTE9BREVSX0FQSV9TT1VSQ0VNQVBfSU1QT1JUX19fIGZyb20gXCIuLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvc291cmNlTWFwcy5qc1wiO1xuaW1wb3J0IF9fX0NTU19MT0FERVJfQVBJX0lNUE9SVF9fXyBmcm9tIFwiLi4vLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2Nzcy1sb2FkZXIvZGlzdC9ydW50aW1lL2FwaS5qc1wiO1xudmFyIF9fX0NTU19MT0FERVJfRVhQT1JUX19fID0gX19fQ1NTX0xPQURFUl9BUElfSU1QT1JUX19fKF9fX0NTU19MT0FERVJfQVBJX1NPVVJDRU1BUF9JTVBPUlRfX18pO1xuLy8gTW9kdWxlXG5fX19DU1NfTE9BREVSX0VYUE9SVF9fXy5wdXNoKFttb2R1bGUuaWQsIGBcclxuaDEge1xyXG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xyXG4gICAgZm9udC1zaXplOiAzMHB4O1xyXG59XHJcblxyXG5oMiB7XHJcbiAgICBjb2xvcjogdmFyKC0tZGFya0dyYXkpO1xyXG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xyXG4gICAgZm9udC1zaXplOiAyMHB4O1xyXG59XHJcblxyXG5oZWFkZXIge1xyXG4gICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcclxuICAgIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xyXG4gICAgcGFkZGluZzogNXB4O1xyXG4gICAgYm9yZGVyLWJvdHRvbTogMXB4IHNvbGlkIHZhcigtLWNpdHlncmVlbik7IFxyXG4gICAgLyogYm9yZGVyOiAxcHggc29saWQgdmFyKC0tY2l0eWdyZWVuKTsgICovXHJcbiAgICBoZWlnaHQ6IHZhcigtLWhlYWRlci1oZWlnaHQpO1xyXG4gICAgLyogd2lkdGg6IGNhbGMoMTAwdncgLSB2YXIoLS1zaWRlYmFyLXdpZHRoKSk7ICovXHJcbiAgICBwYWRkaW5nLWxlZnQ6IDIwcHg7XHJcbn1cclxuXHJcbi8qIExvZ28gc3R5bGluZyAqL1xyXG4jbG9nbyB7XHJcbiAgICBoZWlnaHQ6IDUwcHg7IC8qIE9yIHlvdXIgZGVzaXJlZCBzaXplICovXHJcbiAgICBtYXJnaW4tcmlnaHQ6IDIwcHg7IC8qIE9wdGlvbmFsOiBzcGFjZSBiZXR3ZWVuIGxvZ28gYW5kIHRpdGxlICovXHJcbiAgICBtYXJnaW4tYm90dG9tOiAtNTBweDtcclxuICAgIG1hcmdpbi10b3A6IDEwcHg7XHJcbn1gLCBcIlwiLHtcInZlcnNpb25cIjozLFwic291cmNlc1wiOltcIndlYnBhY2s6Ly8uL3NyYy9qcy9jb21wb25lbnRzL2hlYWRlci9oZWFkZXIuY3NzXCJdLFwibmFtZXNcIjpbXSxcIm1hcHBpbmdzXCI6XCI7QUFDQTtJQUNJLGtCQUFrQjtJQUNsQixlQUFlO0FBQ25COztBQUVBO0lBQ0ksc0JBQXNCO0lBQ3RCLGtCQUFrQjtJQUNsQixlQUFlO0FBQ25COztBQUVBO0lBQ0ksbUJBQW1CO0lBQ25CLHVCQUF1QjtJQUN2QixZQUFZO0lBQ1oseUNBQXlDO0lBQ3pDLHlDQUF5QztJQUN6Qyw0QkFBNEI7SUFDNUIsK0NBQStDO0lBQy9DLGtCQUFrQjtBQUN0Qjs7QUFFQSxpQkFBaUI7QUFDakI7SUFDSSxZQUFZLEVBQUUseUJBQXlCO0lBQ3ZDLGtCQUFrQixFQUFFLDJDQUEyQztJQUMvRCxvQkFBb0I7SUFDcEIsZ0JBQWdCO0FBQ3BCXCIsXCJzb3VyY2VzQ29udGVudFwiOltcIlxcclxcbmgxIHtcXHJcXG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xcclxcbiAgICBmb250LXNpemU6IDMwcHg7XFxyXFxufVxcclxcblxcclxcbmgyIHtcXHJcXG4gICAgY29sb3I6IHZhcigtLWRhcmtHcmF5KTtcXHJcXG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xcclxcbiAgICBmb250LXNpemU6IDIwcHg7XFxyXFxufVxcclxcblxcclxcbmhlYWRlciB7XFxyXFxuICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XFxyXFxuICAgIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xcclxcbiAgICBwYWRkaW5nOiA1cHg7XFxyXFxuICAgIGJvcmRlci1ib3R0b206IDFweCBzb2xpZCB2YXIoLS1jaXR5Z3JlZW4pOyBcXHJcXG4gICAgLyogYm9yZGVyOiAxcHggc29saWQgdmFyKC0tY2l0eWdyZWVuKTsgICovXFxyXFxuICAgIGhlaWdodDogdmFyKC0taGVhZGVyLWhlaWdodCk7XFxyXFxuICAgIC8qIHdpZHRoOiBjYWxjKDEwMHZ3IC0gdmFyKC0tc2lkZWJhci13aWR0aCkpOyAqL1xcclxcbiAgICBwYWRkaW5nLWxlZnQ6IDIwcHg7XFxyXFxufVxcclxcblxcclxcbi8qIExvZ28gc3R5bGluZyAqL1xcclxcbiNsb2dvIHtcXHJcXG4gICAgaGVpZ2h0OiA1MHB4OyAvKiBPciB5b3VyIGRlc2lyZWQgc2l6ZSAqL1xcclxcbiAgICBtYXJnaW4tcmlnaHQ6IDIwcHg7IC8qIE9wdGlvbmFsOiBzcGFjZSBiZXR3ZWVuIGxvZ28gYW5kIHRpdGxlICovXFxyXFxuICAgIG1hcmdpbi1ib3R0b206IC01MHB4O1xcclxcbiAgICBtYXJnaW4tdG9wOiAxMHB4O1xcclxcbn1cIl0sXCJzb3VyY2VSb290XCI6XCJcIn1dKTtcbi8vIEV4cG9ydHNcbmV4cG9ydCBkZWZhdWx0IF9fX0NTU19MT0FERVJfRVhQT1JUX19fO1xuIl0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///107\n')},95:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, ``, "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiOTUuanMiLCJtYXBwaW5ncyI6Ijs7Ozs7OztBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9idWRnZXQtcmVxdWVzdC1mb3JtLy4vc3JjL2pzL2NvbXBvbmVudHMvbW9kYWwvbW9kYWwuY3NzPzZhY2QiXSwic291cmNlc0NvbnRlbnQiOlsiLy8gSW1wb3J0c1xuaW1wb3J0IF9fX0NTU19MT0FERVJfQVBJX1NPVVJDRU1BUF9JTVBPUlRfX18gZnJvbSBcIi4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvcnVudGltZS9zb3VyY2VNYXBzLmpzXCI7XG5pbXBvcnQgX19fQ1NTX0xPQURFUl9BUElfSU1QT1JUX19fIGZyb20gXCIuLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvYXBpLmpzXCI7XG52YXIgX19fQ1NTX0xPQURFUl9FWFBPUlRfX18gPSBfX19DU1NfTE9BREVSX0FQSV9JTVBPUlRfX18oX19fQ1NTX0xPQURFUl9BUElfU09VUkNFTUFQX0lNUE9SVF9fXyk7XG4vLyBNb2R1bGVcbl9fX0NTU19MT0FERVJfRVhQT1JUX19fLnB1c2goW21vZHVsZS5pZCwgYGAsIFwiXCIse1widmVyc2lvblwiOjMsXCJzb3VyY2VzXCI6W10sXCJuYW1lc1wiOltdLFwibWFwcGluZ3NcIjpcIlwiLFwic291cmNlUm9vdFwiOlwiXCJ9XSk7XG4vLyBFeHBvcnRzXG5leHBvcnQgZGVmYXVsdCBfX19DU1NfTE9BREVSX0VYUE9SVF9fXztcbiJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///95\n')},595:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `#nav-btns {\r\n margin: 20px;\r\n text-align: center;\r\n margin-top: 5px;\r\n margin-bottom: 5px;\r\n /* position: absolute; or \'absolute\' depending on use-case */\r\n /* top: 100px; Distance from the top of the viewport or the closest positioned ancestor */\r\n}\r\n\r\n#btn-next, #btn-last {\r\n background-color: var(--darkGray);\r\n}\r\n\r\n#btn-next:hover, #btn-last:hover {\r\n background-color: var(--yellow);\r\n color: var(--darkGray);\r\n}\r\n\r\n/* Add style for when you cannot click the next button */\r\n#btn-next.disabled, #btn-last.disabled,\r\n#btn-next.disabled:hover, #btn-last.disabled:hover {\r\n background-color: gray;\r\n color: white;\r\n pointer-events: none;\r\n \r\n}`, "",{"version":3,"sources":["webpack://./src/js/components/nav_buttons/nav_buttons.css"],"names":[],"mappings":"AAAA;IACI,YAAY;IACZ,kBAAkB;IAClB,eAAe;IACf,kBAAkB;IAClB,4DAA4D;IAC5D,yFAAyF;AAC7F;;AAEA;IACI,iCAAiC;AACrC;;AAEA;IACI,+BAA+B;IAC/B,sBAAsB;AAC1B;;AAEA,wDAAwD;AACxD;;IAEI,sBAAsB;IACtB,YAAY;IACZ,oBAAoB;;AAExB","sourcesContent":["#nav-btns {\\r\\n margin: 20px;\\r\\n text-align: center;\\r\\n margin-top: 5px;\\r\\n margin-bottom: 5px;\\r\\n /* position: absolute; or \'absolute\' depending on use-case */\\r\\n /* top: 100px; Distance from the top of the viewport or the closest positioned ancestor */\\r\\n}\\r\\n\\r\\n#btn-next, #btn-last {\\r\\n background-color: var(--darkGray);\\r\\n}\\r\\n\\r\\n#btn-next:hover, #btn-last:hover {\\r\\n background-color: var(--yellow);\\r\\n color: var(--darkGray);\\r\\n}\\r\\n\\r\\n/* Add style for when you cannot click the next button */\\r\\n#btn-next.disabled, #btn-last.disabled,\\r\\n#btn-next.disabled:hover, #btn-last.disabled:hover {\\r\\n background-color: gray;\\r\\n color: white;\\r\\n pointer-events: none;\\r\\n \\r\\n}"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiNTk1LmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9idWRnZXQtcmVxdWVzdC1mb3JtLy4vc3JjL2pzL2NvbXBvbmVudHMvbmF2X2J1dHRvbnMvbmF2X2J1dHRvbnMuY3NzPzkwNWQiXSwic291cmNlc0NvbnRlbnQiOlsiLy8gSW1wb3J0c1xuaW1wb3J0IF9fX0NTU19MT0FERVJfQVBJX1NPVVJDRU1BUF9JTVBPUlRfX18gZnJvbSBcIi4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvcnVudGltZS9zb3VyY2VNYXBzLmpzXCI7XG5pbXBvcnQgX19fQ1NTX0xPQURFUl9BUElfSU1QT1JUX19fIGZyb20gXCIuLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvYXBpLmpzXCI7XG52YXIgX19fQ1NTX0xPQURFUl9FWFBPUlRfX18gPSBfX19DU1NfTE9BREVSX0FQSV9JTVBPUlRfX18oX19fQ1NTX0xPQURFUl9BUElfU09VUkNFTUFQX0lNUE9SVF9fXyk7XG4vLyBNb2R1bGVcbl9fX0NTU19MT0FERVJfRVhQT1JUX19fLnB1c2goW21vZHVsZS5pZCwgYCNuYXYtYnRucyB7XHJcbiAgICBtYXJnaW46IDIwcHg7XHJcbiAgICB0ZXh0LWFsaWduOiBjZW50ZXI7XHJcbiAgICBtYXJnaW4tdG9wOiA1cHg7XHJcbiAgICBtYXJnaW4tYm90dG9tOiA1cHg7XHJcbiAgICAvKiBwb3NpdGlvbjogYWJzb2x1dGU7IG9yICdhYnNvbHV0ZScgZGVwZW5kaW5nIG9uIHVzZS1jYXNlICovXHJcbiAgICAvKiB0b3A6IDEwMHB4OyBEaXN0YW5jZSBmcm9tIHRoZSB0b3Agb2YgdGhlIHZpZXdwb3J0IG9yIHRoZSBjbG9zZXN0IHBvc2l0aW9uZWQgYW5jZXN0b3IgKi9cclxufVxyXG5cclxuI2J0bi1uZXh0LCAjYnRuLWxhc3Qge1xyXG4gICAgYmFja2dyb3VuZC1jb2xvcjogdmFyKC0tZGFya0dyYXkpO1xyXG59XHJcblxyXG4jYnRuLW5leHQ6aG92ZXIsICNidG4tbGFzdDpob3ZlciB7XHJcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiB2YXIoLS15ZWxsb3cpO1xyXG4gICAgY29sb3I6IHZhcigtLWRhcmtHcmF5KTtcclxufVxyXG5cclxuLyogQWRkIHN0eWxlIGZvciB3aGVuIHlvdSBjYW5ub3QgY2xpY2sgdGhlIG5leHQgYnV0dG9uICovXHJcbiNidG4tbmV4dC5kaXNhYmxlZCwgI2J0bi1sYXN0LmRpc2FibGVkLFxyXG4jYnRuLW5leHQuZGlzYWJsZWQ6aG92ZXIsICNidG4tbGFzdC5kaXNhYmxlZDpob3ZlciB7XHJcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiBncmF5O1xyXG4gICAgY29sb3I6IHdoaXRlO1xyXG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XHJcbiAgICBcclxufWAsIFwiXCIse1widmVyc2lvblwiOjMsXCJzb3VyY2VzXCI6W1wid2VicGFjazovLy4vc3JjL2pzL2NvbXBvbmVudHMvbmF2X2J1dHRvbnMvbmF2X2J1dHRvbnMuY3NzXCJdLFwibmFtZXNcIjpbXSxcIm1hcHBpbmdzXCI6XCJBQUFBO0lBQ0ksWUFBWTtJQUNaLGtCQUFrQjtJQUNsQixlQUFlO0lBQ2Ysa0JBQWtCO0lBQ2xCLDREQUE0RDtJQUM1RCx5RkFBeUY7QUFDN0Y7O0FBRUE7SUFDSSxpQ0FBaUM7QUFDckM7O0FBRUE7SUFDSSwrQkFBK0I7SUFDL0Isc0JBQXNCO0FBQzFCOztBQUVBLHdEQUF3RDtBQUN4RDs7SUFFSSxzQkFBc0I7SUFDdEIsWUFBWTtJQUNaLG9CQUFvQjs7QUFFeEJcIixcInNvdXJjZXNDb250ZW50XCI6W1wiI25hdi1idG5zIHtcXHJcXG4gICAgbWFyZ2luOiAyMHB4O1xcclxcbiAgICB0ZXh0LWFsaWduOiBjZW50ZXI7XFxyXFxuICAgIG1hcmdpbi10b3A6IDVweDtcXHJcXG4gICAgbWFyZ2luLWJvdHRvbTogNXB4O1xcclxcbiAgICAvKiBwb3NpdGlvbjogYWJzb2x1dGU7IG9yICdhYnNvbHV0ZScgZGVwZW5kaW5nIG9uIHVzZS1jYXNlICovXFxyXFxuICAgIC8qIHRvcDogMTAwcHg7IERpc3RhbmNlIGZyb20gdGhlIHRvcCBvZiB0aGUgdmlld3BvcnQgb3IgdGhlIGNsb3Nlc3QgcG9zaXRpb25lZCBhbmNlc3RvciAqL1xcclxcbn1cXHJcXG5cXHJcXG4jYnRuLW5leHQsICNidG4tbGFzdCB7XFxyXFxuICAgIGJhY2tncm91bmQtY29sb3I6IHZhcigtLWRhcmtHcmF5KTtcXHJcXG59XFxyXFxuXFxyXFxuI2J0bi1uZXh0OmhvdmVyLCAjYnRuLWxhc3Q6aG92ZXIge1xcclxcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiB2YXIoLS15ZWxsb3cpO1xcclxcbiAgICBjb2xvcjogdmFyKC0tZGFya0dyYXkpO1xcclxcbn1cXHJcXG5cXHJcXG4vKiBBZGQgc3R5bGUgZm9yIHdoZW4geW91IGNhbm5vdCBjbGljayB0aGUgbmV4dCBidXR0b24gKi9cXHJcXG4jYnRuLW5leHQuZGlzYWJsZWQsICNidG4tbGFzdC5kaXNhYmxlZCxcXHJcXG4jYnRuLW5leHQuZGlzYWJsZWQ6aG92ZXIsICNidG4tbGFzdC5kaXNhYmxlZDpob3ZlciB7XFxyXFxuICAgIGJhY2tncm91bmQtY29sb3I6IGdyYXk7XFxyXFxuICAgIGNvbG9yOiB3aGl0ZTtcXHJcXG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XFxyXFxuICAgIFxcclxcbn1cIl0sXCJzb3VyY2VSb290XCI6XCJcIn1dKTtcbi8vIEV4cG9ydHNcbmV4cG9ydCBkZWZhdWx0IF9fX0NTU19MT0FERVJfRVhQT1JUX19fO1xuIl0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///595\n')},225:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `div#prompt-div {\r\n display: none;\r\n width: 80%;\r\n margin: auto;\r\n text-align: center;\r\n}\r\n\r\nh3#prompt {\r\n text-align: center;\r\n font-size : 1.2em;\r\n}\r\n\r\n#option1, #option2 { \r\n font-size: 1.5em;\r\n border-color: var(--citygreen); \r\n border-width: 2px;\r\n background-color: var(--white);\r\n color: var(--citygreen);\r\n}\r\n\r\n#option1:hover, #option2:hover { \r\n background-color: var(--spiritgreen);\r\n color: white;\r\n}\r\n\r\n#option2.clicked, #option1.clicked {\r\n font-weight: bold;\r\n background-color: var(--spiritgreen);\r\n color: white;\r\n border-width: 3;\r\n}\r\n\r\n#option1.disabled, #option2.disabled, #option1.disabled:hover, #option2.disabled:hover {\r\n background-color: lightgray;\r\n color: gray;\r\n pointer-events: none;\r\n border-color: gray;\r\n}\r\n\r\n`, "",{"version":3,"sources":["webpack://./src/js/components/prompt/prompt.css"],"names":[],"mappings":"AAAA;IACI,aAAa;IACb,UAAU;IACV,YAAY;IACZ,kBAAkB;AACtB;;AAEA;IACI,kBAAkB;IAClB,iBAAiB;AACrB;;AAEA;IACI,gBAAgB;IAChB,8BAA8B;IAC9B,iBAAiB;IACjB,8BAA8B;IAC9B,uBAAuB;AAC3B;;AAEA;IACI,oCAAoC;IACpC,YAAY;AAChB;;AAEA;IACI,iBAAiB;IACjB,oCAAoC;IACpC,YAAY;IACZ,eAAe;AACnB;;AAEA;IACI,2BAA2B;IAC3B,WAAW;IACX,oBAAoB;IACpB,kBAAkB;AACtB","sourcesContent":["div#prompt-div {\\r\\n display: none;\\r\\n width: 80%;\\r\\n margin: auto;\\r\\n text-align: center;\\r\\n}\\r\\n\\r\\nh3#prompt {\\r\\n text-align: center;\\r\\n font-size : 1.2em;\\r\\n}\\r\\n\\r\\n#option1, #option2 { \\r\\n font-size: 1.5em;\\r\\n border-color: var(--citygreen); \\r\\n border-width: 2px;\\r\\n background-color: var(--white);\\r\\n color: var(--citygreen);\\r\\n}\\r\\n\\r\\n#option1:hover, #option2:hover { \\r\\n background-color: var(--spiritgreen);\\r\\n color: white;\\r\\n}\\r\\n\\r\\n#option2.clicked, #option1.clicked {\\r\\n font-weight: bold;\\r\\n background-color: var(--spiritgreen);\\r\\n color: white;\\r\\n border-width: 3;\\r\\n}\\r\\n\\r\\n#option1.disabled, #option2.disabled, #option1.disabled:hover, #option2.disabled:hover {\\r\\n background-color: lightgray;\\r\\n color: gray;\\r\\n pointer-events: none;\\r\\n border-color: gray;\\r\\n}\\r\\n\\r\\n"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMjI1LmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYnVkZ2V0LXJlcXVlc3QtZm9ybS8uL3NyYy9qcy9jb21wb25lbnRzL3Byb21wdC9wcm9tcHQuY3NzP2U2OTkiXSwic291cmNlc0NvbnRlbnQiOlsiLy8gSW1wb3J0c1xuaW1wb3J0IF9fX0NTU19MT0FERVJfQVBJX1NPVVJDRU1BUF9JTVBPUlRfX18gZnJvbSBcIi4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvcnVudGltZS9zb3VyY2VNYXBzLmpzXCI7XG5pbXBvcnQgX19fQ1NTX0xPQURFUl9BUElfSU1QT1JUX19fIGZyb20gXCIuLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvYXBpLmpzXCI7XG52YXIgX19fQ1NTX0xPQURFUl9FWFBPUlRfX18gPSBfX19DU1NfTE9BREVSX0FQSV9JTVBPUlRfX18oX19fQ1NTX0xPQURFUl9BUElfU09VUkNFTUFQX0lNUE9SVF9fXyk7XG4vLyBNb2R1bGVcbl9fX0NTU19MT0FERVJfRVhQT1JUX19fLnB1c2goW21vZHVsZS5pZCwgYGRpdiNwcm9tcHQtZGl2IHtcclxuICAgIGRpc3BsYXk6IG5vbmU7XHJcbiAgICB3aWR0aDogODAlO1xyXG4gICAgbWFyZ2luOiBhdXRvO1xyXG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xyXG59XHJcblxyXG5oMyNwcm9tcHQge1xyXG4gICAgdGV4dC1hbGlnbjogY2VudGVyO1xyXG4gICAgZm9udC1zaXplIDogMS4yZW07XHJcbn1cclxuXHJcbiNvcHRpb24xLCAjb3B0aW9uMiB7IFxyXG4gICAgZm9udC1zaXplOiAxLjVlbTtcclxuICAgIGJvcmRlci1jb2xvcjogdmFyKC0tY2l0eWdyZWVuKTsgXHJcbiAgICBib3JkZXItd2lkdGg6IDJweDtcclxuICAgIGJhY2tncm91bmQtY29sb3I6IHZhcigtLXdoaXRlKTtcclxuICAgIGNvbG9yOiB2YXIoLS1jaXR5Z3JlZW4pO1xyXG59XHJcblxyXG4jb3B0aW9uMTpob3ZlciwgI29wdGlvbjI6aG92ZXIgeyBcclxuICAgIGJhY2tncm91bmQtY29sb3I6IHZhcigtLXNwaXJpdGdyZWVuKTtcclxuICAgIGNvbG9yOiB3aGl0ZTtcclxufVxyXG5cclxuI29wdGlvbjIuY2xpY2tlZCwgI29wdGlvbjEuY2xpY2tlZCB7XHJcbiAgICBmb250LXdlaWdodDogYm9sZDtcclxuICAgIGJhY2tncm91bmQtY29sb3I6IHZhcigtLXNwaXJpdGdyZWVuKTtcclxuICAgIGNvbG9yOiB3aGl0ZTtcclxuICAgIGJvcmRlci13aWR0aDogMztcclxufVxyXG5cclxuI29wdGlvbjEuZGlzYWJsZWQsICNvcHRpb24yLmRpc2FibGVkLCAjb3B0aW9uMS5kaXNhYmxlZDpob3ZlciwgI29wdGlvbjIuZGlzYWJsZWQ6aG92ZXIge1xyXG4gICAgYmFja2dyb3VuZC1jb2xvcjogbGlnaHRncmF5O1xyXG4gICAgY29sb3I6IGdyYXk7XHJcbiAgICBwb2ludGVyLWV2ZW50czogbm9uZTtcclxuICAgIGJvcmRlci1jb2xvcjogZ3JheTtcclxufVxyXG5cclxuYCwgXCJcIix7XCJ2ZXJzaW9uXCI6MyxcInNvdXJjZXNcIjpbXCJ3ZWJwYWNrOi8vLi9zcmMvanMvY29tcG9uZW50cy9wcm9tcHQvcHJvbXB0LmNzc1wiXSxcIm5hbWVzXCI6W10sXCJtYXBwaW5nc1wiOlwiQUFBQTtJQUNJLGFBQWE7SUFDYixVQUFVO0lBQ1YsWUFBWTtJQUNaLGtCQUFrQjtBQUN0Qjs7QUFFQTtJQUNJLGtCQUFrQjtJQUNsQixpQkFBaUI7QUFDckI7O0FBRUE7SUFDSSxnQkFBZ0I7SUFDaEIsOEJBQThCO0lBQzlCLGlCQUFpQjtJQUNqQiw4QkFBOEI7SUFDOUIsdUJBQXVCO0FBQzNCOztBQUVBO0lBQ0ksb0NBQW9DO0lBQ3BDLFlBQVk7QUFDaEI7O0FBRUE7SUFDSSxpQkFBaUI7SUFDakIsb0NBQW9DO0lBQ3BDLFlBQVk7SUFDWixlQUFlO0FBQ25COztBQUVBO0lBQ0ksMkJBQTJCO0lBQzNCLFdBQVc7SUFDWCxvQkFBb0I7SUFDcEIsa0JBQWtCO0FBQ3RCXCIsXCJzb3VyY2VzQ29udGVudFwiOltcImRpdiNwcm9tcHQtZGl2IHtcXHJcXG4gICAgZGlzcGxheTogbm9uZTtcXHJcXG4gICAgd2lkdGg6IDgwJTtcXHJcXG4gICAgbWFyZ2luOiBhdXRvO1xcclxcbiAgICB0ZXh0LWFsaWduOiBjZW50ZXI7XFxyXFxufVxcclxcblxcclxcbmgzI3Byb21wdCB7XFxyXFxuICAgIHRleHQtYWxpZ246IGNlbnRlcjtcXHJcXG4gICAgZm9udC1zaXplIDogMS4yZW07XFxyXFxufVxcclxcblxcclxcbiNvcHRpb24xLCAjb3B0aW9uMiB7IFxcclxcbiAgICBmb250LXNpemU6IDEuNWVtO1xcclxcbiAgICBib3JkZXItY29sb3I6IHZhcigtLWNpdHlncmVlbik7IFxcclxcbiAgICBib3JkZXItd2lkdGg6IDJweDtcXHJcXG4gICAgYmFja2dyb3VuZC1jb2xvcjogdmFyKC0td2hpdGUpO1xcclxcbiAgICBjb2xvcjogdmFyKC0tY2l0eWdyZWVuKTtcXHJcXG59XFxyXFxuXFxyXFxuI29wdGlvbjE6aG92ZXIsICNvcHRpb24yOmhvdmVyIHsgXFxyXFxuICAgIGJhY2tncm91bmQtY29sb3I6IHZhcigtLXNwaXJpdGdyZWVuKTtcXHJcXG4gICAgY29sb3I6IHdoaXRlO1xcclxcbn1cXHJcXG5cXHJcXG4jb3B0aW9uMi5jbGlja2VkLCAjb3B0aW9uMS5jbGlja2VkIHtcXHJcXG4gICAgZm9udC13ZWlnaHQ6IGJvbGQ7XFxyXFxuICAgIGJhY2tncm91bmQtY29sb3I6IHZhcigtLXNwaXJpdGdyZWVuKTtcXHJcXG4gICAgY29sb3I6IHdoaXRlO1xcclxcbiAgICBib3JkZXItd2lkdGg6IDM7XFxyXFxufVxcclxcblxcclxcbiNvcHRpb24xLmRpc2FibGVkLCAjb3B0aW9uMi5kaXNhYmxlZCwgI29wdGlvbjEuZGlzYWJsZWQ6aG92ZXIsICNvcHRpb24yLmRpc2FibGVkOmhvdmVyIHtcXHJcXG4gICAgYmFja2dyb3VuZC1jb2xvcjogbGlnaHRncmF5O1xcclxcbiAgICBjb2xvcjogZ3JheTtcXHJcXG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XFxyXFxuICAgIGJvcmRlci1jb2xvcjogZ3JheTtcXHJcXG59XFxyXFxuXFxyXFxuXCJdLFwic291cmNlUm9vdFwiOlwiXCJ9XSk7XG4vLyBFeHBvcnRzXG5leHBvcnQgZGVmYXVsdCBfX19DU1NfTE9BREVSX0VYUE9SVF9fXztcbiJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///225\n')},263:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `#sidebar-panel {\r\n background-color: lightgrey; \r\n /* min-height: 100vh; Full height of viewport */\r\n }\r\n\r\n#supp-total .stat {\r\n color: var(--yellow);\r\n}\r\n\r\n.stat {\r\n font-weight: bold;\r\n}\r\n\r\n#sidebar-panel {\r\n height: 100%; /* Full height of the viewport */\r\n position: fixed; /* Fixed Sidebar (stay in place on scroll) */\r\n z-index: 1; /* Stay on top */\r\n top: 0; /* Stay at the top */\r\n right: 0; /* Sidebar appears on the left */\r\n background-color: #FFFFFF; /* White background color */\r\n overflow-x: hidden; /* Disable horizontal scroll */\r\n padding: 20px;\r\n box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1); /* Add shadow for some depth */\r\n border-right: 2px solid #DDDDDD; /* Right border */\r\n border-radius: 0px 0px 0px 15px; /* Rounded corners on the left */\r\n /* margin-top: var(--header-height); */\r\n width: var(--sidebar-width);\r\n margin: 0px;\r\n }\r\n \r\n.sidebar a {\r\n padding: 10px 15px;\r\n text-decoration: none;\r\n font-size: 18px;\r\n color: #818181;\r\n display: block;\r\n transition: 0.3s;\r\n}\r\n\r\n#sidebar-title {\r\n color: var(--darkGray);\r\n font-weight: bold;\r\n border-bottom: 1px solid var(--citygreen);\r\n}\r\n\r\nhr {\r\n margin-top: -5px;\r\n margin-bottom: 8px;\r\n}\r\n\r\nh6 {\r\n font-weight: 600;\r\n}\r\n\r\n.section-header {\r\n background-color: var(--mediumGray);\r\n}\r\n\r\n.sidebar-stat-line {\r\n display: flex;\r\n align-items: center;\r\n justify-content: space-between; /* Distribute space between children */\r\n padding: 2px;\r\n border-bottom: 1px solid #ddd;\r\n}\r\n\r\n.edit-icon, .edit-supp {\r\n color: var(--spiritgreen);\r\n cursor: pointer;\r\n font-size: 16px; /* Adjust the size as needed */\r\n margin-left: 10px;\r\n}\r\n\r\n.edit-icon:hover .edit-supp:hover {\r\n color: var(--citygreen);\r\n}\r\n\r\n.stat-label {\r\n margin-right: auto; /* Push next elements to the right */\r\n}\r\n\r\n.stat {\r\n margin-left: 5px; /* Optional: Add some space between currency and edit icon */\r\n}\r\n\r\n.sidebar-stat-line.fund-total .stat {\r\n margin-right: 05px; /* 25px lines up with lines above (with edit symbol) */\r\n}`, "",{"version":3,"sources":["webpack://./src/js/components/sidebar/sidebar.css"],"names":[],"mappings":"AAAA;IACI,2BAA2B;IAC3B,+CAA+C;EACjD;;AAEF;IACI,oBAAoB;AACxB;;AAEA;IACI,iBAAiB;AACrB;;AAEA;IACI,YAAY,EAAE,gCAAgC;IAC9C,eAAe,EAAE,4CAA4C;IAC7D,UAAU,EAAE,gBAAgB;IAC5B,MAAM,EAAE,oBAAoB;IAC5B,QAAQ,EAAE,gCAAgC;IAC1C,yBAAyB,EAAE,2BAA2B;IACtD,kBAAkB,EAAE,8BAA8B;IAClD,aAAa;IACb,wCAAwC,EAAE,8BAA8B;IACxE,+BAA+B,EAAE,iBAAiB;IAClD,+BAA+B,EAAE,gCAAgC;IACjE,sCAAsC;IACtC,2BAA2B;IAC3B,WAAW;EACb;;AAEF;IACI,kBAAkB;IAClB,qBAAqB;IACrB,eAAe;IACf,cAAc;IACd,cAAc;IACd,gBAAgB;AACpB;;AAEA;EACE,sBAAsB;EACtB,iBAAiB;EACjB,yCAAyC;AAC3C;;AAEA;EACE,gBAAgB;EAChB,kBAAkB;AACpB;;AAEA;EACE,gBAAgB;AAClB;;AAEA;EACE,mCAAmC;AACrC;;AAEA;EACE,aAAa;EACb,mBAAmB;EACnB,8BAA8B,EAAE,sCAAsC;EACtE,YAAY;EACZ,6BAA6B;AAC/B;;AAEA;EACE,yBAAyB;EACzB,eAAe;EACf,eAAe,EAAE,8BAA8B;EAC/C,iBAAiB;AACnB;;AAEA;EACE,uBAAuB;AACzB;;AAEA;EACE,kBAAkB,EAAE,oCAAoC;AAC1D;;AAEA;EACE,gBAAgB,EAAE,4DAA4D;AAChF;;AAEA;EACE,kBAAkB,EAAE,sDAAsD;AAC5E","sourcesContent":["#sidebar-panel {\\r\\n background-color: lightgrey; \\r\\n /* min-height: 100vh; Full height of viewport */\\r\\n }\\r\\n\\r\\n#supp-total .stat {\\r\\n color: var(--yellow);\\r\\n}\\r\\n\\r\\n.stat {\\r\\n font-weight: bold;\\r\\n}\\r\\n\\r\\n#sidebar-panel {\\r\\n height: 100%; /* Full height of the viewport */\\r\\n position: fixed; /* Fixed Sidebar (stay in place on scroll) */\\r\\n z-index: 1; /* Stay on top */\\r\\n top: 0; /* Stay at the top */\\r\\n right: 0; /* Sidebar appears on the left */\\r\\n background-color: #FFFFFF; /* White background color */\\r\\n overflow-x: hidden; /* Disable horizontal scroll */\\r\\n padding: 20px;\\r\\n box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1); /* Add shadow for some depth */\\r\\n border-right: 2px solid #DDDDDD; /* Right border */\\r\\n border-radius: 0px 0px 0px 15px; /* Rounded corners on the left */\\r\\n /* margin-top: var(--header-height); */\\r\\n width: var(--sidebar-width);\\r\\n margin: 0px;\\r\\n }\\r\\n \\r\\n.sidebar a {\\r\\n padding: 10px 15px;\\r\\n text-decoration: none;\\r\\n font-size: 18px;\\r\\n color: #818181;\\r\\n display: block;\\r\\n transition: 0.3s;\\r\\n}\\r\\n\\r\\n#sidebar-title {\\r\\n color: var(--darkGray);\\r\\n font-weight: bold;\\r\\n border-bottom: 1px solid var(--citygreen);\\r\\n}\\r\\n\\r\\nhr {\\r\\n margin-top: -5px;\\r\\n margin-bottom: 8px;\\r\\n}\\r\\n\\r\\nh6 {\\r\\n font-weight: 600;\\r\\n}\\r\\n\\r\\n.section-header {\\r\\n background-color: var(--mediumGray);\\r\\n}\\r\\n\\r\\n.sidebar-stat-line {\\r\\n display: flex;\\r\\n align-items: center;\\r\\n justify-content: space-between; /* Distribute space between children */\\r\\n padding: 2px;\\r\\n border-bottom: 1px solid #ddd;\\r\\n}\\r\\n\\r\\n.edit-icon, .edit-supp {\\r\\n color: var(--spiritgreen);\\r\\n cursor: pointer;\\r\\n font-size: 16px; /* Adjust the size as needed */\\r\\n margin-left: 10px;\\r\\n}\\r\\n\\r\\n.edit-icon:hover .edit-supp:hover {\\r\\n color: var(--citygreen);\\r\\n}\\r\\n\\r\\n.stat-label {\\r\\n margin-right: auto; /* Push next elements to the right */\\r\\n}\\r\\n\\r\\n.stat {\\r\\n margin-left: 5px; /* Optional: Add some space between currency and edit icon */\\r\\n}\\r\\n\\r\\n.sidebar-stat-line.fund-total .stat {\\r\\n margin-right: 05px; /* 25px lines up with lines above (with edit symbol) */\\r\\n}"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"263.js","mappings":";;;;;;;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sources":["webpack://budget-request-form/./src/js/components/sidebar/sidebar.css?3b03"],"sourcesContent":["// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../node_modules/css-loader/dist/runtime/sourceMaps.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `#sidebar-panel {\r\n    background-color: lightgrey;  \r\n    /* min-height: 100vh; Full height of viewport */\r\n  }\r\n\r\n#supp-total .stat {\r\n    color: var(--yellow);\r\n}\r\n\r\n.stat {\r\n    font-weight: bold;\r\n}\r\n\r\n#sidebar-panel {\r\n    height: 100%; /* Full height of the viewport */\r\n    position: fixed; /* Fixed Sidebar (stay in place on scroll) */\r\n    z-index: 1; /* Stay on top */\r\n    top: 0; /* Stay at the top */\r\n    right: 0; /* Sidebar appears on the left */\r\n    background-color: #FFFFFF; /* White background color */\r\n    overflow-x: hidden; /* Disable horizontal scroll */\r\n    padding: 20px;\r\n    box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1); /* Add shadow for some depth */\r\n    border-right: 2px solid #DDDDDD; /* Right border */\r\n    border-radius: 0px 0px 0px 15px; /* Rounded corners on the left */\r\n    /* margin-top: var(--header-height); */\r\n    width: var(--sidebar-width);\r\n    margin: 0px;\r\n  }\r\n  \r\n.sidebar a {\r\n    padding: 10px 15px;\r\n    text-decoration: none;\r\n    font-size: 18px;\r\n    color: #818181;\r\n    display: block;\r\n    transition: 0.3s;\r\n}\r\n\r\n#sidebar-title {\r\n  color: var(--darkGray);\r\n  font-weight: bold;\r\n  border-bottom: 1px solid var(--citygreen);\r\n}\r\n\r\nhr {\r\n  margin-top: -5px;\r\n  margin-bottom: 8px;\r\n}\r\n\r\nh6 {\r\n  font-weight: 600;\r\n}\r\n\r\n.section-header {\r\n  background-color: var(--mediumGray);\r\n}\r\n\r\n.sidebar-stat-line {\r\n  display: flex;\r\n  align-items: center;\r\n  justify-content: space-between; /* Distribute space between children */\r\n  padding: 2px;\r\n  border-bottom: 1px solid #ddd;\r\n}\r\n\r\n.edit-icon, .edit-supp {\r\n  color: var(--spiritgreen);\r\n  cursor: pointer;\r\n  font-size: 16px; /* Adjust the size as needed */\r\n  margin-left: 10px;\r\n}\r\n\r\n.edit-icon:hover .edit-supp:hover {\r\n  color: var(--citygreen);\r\n}\r\n\r\n.stat-label {\r\n  margin-right: auto; /* Push next elements to the right */\r\n}\r\n\r\n.stat {\r\n  margin-left: 5px; /* Optional: Add some space between currency and edit icon */\r\n}\r\n\r\n.sidebar-stat-line.fund-total .stat {\r\n  margin-right: 05px; /* 25px lines up with lines above (with edit symbol) */\r\n}`, \"\",{\"version\":3,\"sources\":[\"webpack://./src/js/components/sidebar/sidebar.css\"],\"names\":[],\"mappings\":\"AAAA;IACI,2BAA2B;IAC3B,+CAA+C;EACjD;;AAEF;IACI,oBAAoB;AACxB;;AAEA;IACI,iBAAiB;AACrB;;AAEA;IACI,YAAY,EAAE,gCAAgC;IAC9C,eAAe,EAAE,4CAA4C;IAC7D,UAAU,EAAE,gBAAgB;IAC5B,MAAM,EAAE,oBAAoB;IAC5B,QAAQ,EAAE,gCAAgC;IAC1C,yBAAyB,EAAE,2BAA2B;IACtD,kBAAkB,EAAE,8BAA8B;IAClD,aAAa;IACb,wCAAwC,EAAE,8BAA8B;IACxE,+BAA+B,EAAE,iBAAiB;IAClD,+BAA+B,EAAE,gCAAgC;IACjE,sCAAsC;IACtC,2BAA2B;IAC3B,WAAW;EACb;;AAEF;IACI,kBAAkB;IAClB,qBAAqB;IACrB,eAAe;IACf,cAAc;IACd,cAAc;IACd,gBAAgB;AACpB;;AAEA;EACE,sBAAsB;EACtB,iBAAiB;EACjB,yCAAyC;AAC3C;;AAEA;EACE,gBAAgB;EAChB,kBAAkB;AACpB;;AAEA;EACE,gBAAgB;AAClB;;AAEA;EACE,mCAAmC;AACrC;;AAEA;EACE,aAAa;EACb,mBAAmB;EACnB,8BAA8B,EAAE,sCAAsC;EACtE,YAAY;EACZ,6BAA6B;AAC/B;;AAEA;EACE,yBAAyB;EACzB,eAAe;EACf,eAAe,EAAE,8BAA8B;EAC/C,iBAAiB;AACnB;;AAEA;EACE,uBAAuB;AACzB;;AAEA;EACE,kBAAkB,EAAE,oCAAoC;AAC1D;;AAEA;EACE,gBAAgB,EAAE,4DAA4D;AAChF;;AAEA;EACE,kBAAkB,EAAE,sDAAsD;AAC5E\",\"sourcesContent\":[\"#sidebar-panel {\\r\\n    background-color: lightgrey;  \\r\\n    /* min-height: 100vh; Full height of viewport */\\r\\n  }\\r\\n\\r\\n#supp-total .stat {\\r\\n    color: var(--yellow);\\r\\n}\\r\\n\\r\\n.stat {\\r\\n    font-weight: bold;\\r\\n}\\r\\n\\r\\n#sidebar-panel {\\r\\n    height: 100%; /* Full height of the viewport */\\r\\n    position: fixed; /* Fixed Sidebar (stay in place on scroll) */\\r\\n    z-index: 1; /* Stay on top */\\r\\n    top: 0; /* Stay at the top */\\r\\n    right: 0; /* Sidebar appears on the left */\\r\\n    background-color: #FFFFFF; /* White background color */\\r\\n    overflow-x: hidden; /* Disable horizontal scroll */\\r\\n    padding: 20px;\\r\\n    box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1); /* Add shadow for some depth */\\r\\n    border-right: 2px solid #DDDDDD; /* Right border */\\r\\n    border-radius: 0px 0px 0px 15px; /* Rounded corners on the left */\\r\\n    /* margin-top: var(--header-height); */\\r\\n    width: var(--sidebar-width);\\r\\n    margin: 0px;\\r\\n  }\\r\\n  \\r\\n.sidebar a {\\r\\n    padding: 10px 15px;\\r\\n    text-decoration: none;\\r\\n    font-size: 18px;\\r\\n    color: #818181;\\r\\n    display: block;\\r\\n    transition: 0.3s;\\r\\n}\\r\\n\\r\\n#sidebar-title {\\r\\n  color: var(--darkGray);\\r\\n  font-weight: bold;\\r\\n  border-bottom: 1px solid var(--citygreen);\\r\\n}\\r\\n\\r\\nhr {\\r\\n  margin-top: -5px;\\r\\n  margin-bottom: 8px;\\r\\n}\\r\\n\\r\\nh6 {\\r\\n  font-weight: 600;\\r\\n}\\r\\n\\r\\n.section-header {\\r\\n  background-color: var(--mediumGray);\\r\\n}\\r\\n\\r\\n.sidebar-stat-line {\\r\\n  display: flex;\\r\\n  align-items: center;\\r\\n  justify-content: space-between; /* Distribute space between children */\\r\\n  padding: 2px;\\r\\n  border-bottom: 1px solid #ddd;\\r\\n}\\r\\n\\r\\n.edit-icon, .edit-supp {\\r\\n  color: var(--spiritgreen);\\r\\n  cursor: pointer;\\r\\n  font-size: 16px; /* Adjust the size as needed */\\r\\n  margin-left: 10px;\\r\\n}\\r\\n\\r\\n.edit-icon:hover .edit-supp:hover {\\r\\n  color: var(--citygreen);\\r\\n}\\r\\n\\r\\n.stat-label {\\r\\n  margin-right: auto; /* Push next elements to the right */\\r\\n}\\r\\n\\r\\n.stat {\\r\\n  margin-left: 5px; /* Optional: Add some space between currency and edit icon */\\r\\n}\\r\\n\\r\\n.sidebar-stat-line.fund-total .stat {\\r\\n  margin-right: 05px; /* 25px lines up with lines above (with edit symbol) */\\r\\n}\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///263\n')},279:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `#main-table {\r\n font-size: 12px;\r\n /* calc(0.5vw + 0.5em); */\r\n margin: auto;\r\n width: 100%;\r\n}\r\n\r\n#main-table thead > tr > th {\r\n text-align: left;\r\n background-color: var(--darkGray);\r\n color: white;\r\n position: -webkit-sticky; /* For Safari */\r\n position: sticky;\r\n top: 0;\r\n padding-top: 5px;\r\n padding-bottom: 5px;\r\n padding-left: 5px;\r\n}\r\n\r\nth {\r\n background-color: var(--lightGray);\r\n}\r\n\r\ntr {\r\n border-width: 2px;\r\n background-color: white;\r\n}\r\n\r\ntr td {\r\n border-bottom: 1px solid black;\r\n}\r\n\r\n/* Default to hidden columns */\r\n\r\n#main-table tbody > tr > td {\r\n display : none;\r\n}\r\n#main-table thead > tr > th {\r\n display : none;\r\n}\r\n\r\n/* textbox width in table */\r\ninput {\r\n width: 100%;\r\n}\r\n\r\nselect {\r\n max-width: 100px;\r\n}\r\n\r\ndiv.table-container {\r\n overflow-x: auto; \r\n overflow-y: auto;\r\n margin-left: max(65px, 7vh);\r\n margin-right: max(65px, 7vh);\r\n /* max-width: calc(100vw - var(--sidebar-width)); */\r\n /* margin: auto; */\r\n max-height: max(350px, 6vh);\r\n /* min-height: 350px; */\r\n\r\n}\r\n \r\n/* Buttons */\r\n\r\n/* Add new row button */\r\n.btn-add { \r\n background-color: var(--spiritgreen);\r\n margin-top: 20px;\r\n display: none;\r\n}\r\n\r\n.btn-delete {\r\n background-color: var(--orange);\r\n}\r\n\r\n#add-btn-div {\r\n display: flex;\r\n justify-content: center; /* Aligns horizontally */\r\n align-items: center; /* Aligns vertically */\r\n width: 100%;\r\n}\r\n\r\n.btn-edit {\r\n background-color: var(--spiritgreen);\r\n line-height: 1.1;\r\n}\r\n\r\n.active-editing, .selected {\r\n background-color: var(--palegreen);\r\n}\r\n\r\n.selected { \r\n font-weight: bold;\r\n}\r\n\r\n.btn-confirm {\r\n display: none;\r\n}\r\n\r\n.confirm-btn:hover {\r\n background-color: var(--green);\r\n}\r\n\r\n.hover-effect:hover {\r\n cursor: pointer;\r\n background-color: var(--verypalegreen); \r\n}\r\n\r\n/* Fund table */\r\n.fund-name > .viewed-fund {\r\n color: gray;\r\n}\r\n\r\n/* made edit column narrower */\r\nth.edit, td.edit {\r\n width: 20px;\r\n}\r\n\r\n/* Class for hidden rows on filter */\r\n.hidden {\r\n display: none;\r\n}\r\n\r\n/* Filter styling for the dropdown options */\r\n#filter-container {\r\n display: flex;\r\n flex-wrap: wrap;\r\n gap: 2px; /* Adjust the gap as needed */\r\n font-size: 11px;\r\n margin-left: 10%;\r\n margin-right: 10%;\r\n margin-bottom: 5px;\r\n}\r\n\r\n#filter-container div {\r\n display: flex;\r\n align-items: center;\r\n width: calc(50% - 10px); /* Adjust width to fit two items per row with gap */\r\n}\r\n\r\n#filter-container label {\r\n width: 200px; /* Adjust the width as needed */\r\n text-align: right;\r\n margin-right: 10px; /* Spacing between label and select */\r\n}\r\n\r\n/* Ensure all select boxes are wide enough to show text */\r\n.filter-dropdown {\r\n flex: 1;\r\n min-width: 200px; /* Adjust the min-width as needed */\r\n}\r\n\r\n/* Optional styling to make the select dropdown look better */\r\n.filter-dropdown {\r\n padding: 5px;\r\n border: 1px solid #ccc;\r\n border-radius: 4px;\r\n}\r\n\r\n/* Responsive adjustment: On smaller screens, stack elements vertically */\r\n@media (max-width: 800px) {\r\n #filter-container div {\r\n width: 100%;\r\n }\r\n\r\n #filter-container label {\r\n text-align: left;\r\n width: auto;\r\n margin-right: 0;\r\n margin-bottom: 5px;\r\n }\r\n}`, "",{"version":3,"sources":["webpack://./src/js/components/table/table.css"],"names":[],"mappings":"AAAA;IACI,eAAe;IACf,yBAAyB;IACzB,YAAY;IACZ,WAAW;AACf;;AAEA;IACI,gBAAgB;IAChB,iCAAiC;IACjC,YAAY;IACZ,wBAAwB,EAAE,eAAe;IACzC,gBAAgB;IAChB,MAAM;IACN,gBAAgB;IAChB,mBAAmB;IACnB,iBAAiB;AACrB;;AAEA;IACI,kCAAkC;AACtC;;AAEA;IACI,iBAAiB;IACjB,uBAAuB;AAC3B;;AAEA;IACI,8BAA8B;AAClC;;AAEA,8BAA8B;;AAE9B;IACI,cAAc;AAClB;AACA;IACI,cAAc;AAClB;;AAEA,2BAA2B;AAC3B;IACI,WAAW;AACf;;AAEA;IACI,gBAAgB;AACpB;;AAEA;IACI,gBAAgB;IAChB,gBAAgB;IAChB,2BAA2B;IAC3B,4BAA4B;IAC5B,oDAAoD;IACpD,kBAAkB;IAClB,2BAA2B;IAC3B,uBAAuB;;AAE3B;;AAEA,YAAY;;AAEZ,uBAAuB;AACvB;IACI,oCAAoC;IACpC,gBAAgB;IAChB,aAAa;AACjB;;AAEA;IACI,+BAA+B;AACnC;;AAEA;IACI,aAAa;IACb,uBAAuB,EAAE,wBAAwB;IACjD,mBAAmB,EAAE,sBAAsB;IAC3C,WAAW;AACf;;AAEA;IACI,oCAAoC;IACpC,gBAAgB;AACpB;;AAEA;IACI,kCAAkC;AACtC;;AAEA;IACI,iBAAiB;AACrB;;AAEA;IACI,aAAa;AACjB;;AAEA;IACI,8BAA8B;AAClC;;AAEA;IACI,eAAe;IACf,sCAAsC;AAC1C;;AAEA,eAAe;AACf;IACI,WAAW;AACf;;AAEA,8BAA8B;AAC9B;IACI,WAAW;AACf;;AAEA,oCAAoC;AACpC;IACI,aAAa;AACjB;;AAEA,4CAA4C;AAC5C;IACI,aAAa;IACb,eAAe;IACf,QAAQ,EAAE,6BAA6B;IACvC,eAAe;IACf,gBAAgB;IAChB,iBAAiB;IACjB,kBAAkB;AACtB;;AAEA;IACI,aAAa;IACb,mBAAmB;IACnB,uBAAuB,EAAE,mDAAmD;AAChF;;AAEA;IACI,YAAY,EAAE,+BAA+B;IAC7C,iBAAiB;IACjB,kBAAkB,EAAE,qCAAqC;AAC7D;;AAEA,yDAAyD;AACzD;IACI,OAAO;IACP,gBAAgB,EAAE,mCAAmC;AACzD;;AAEA,6DAA6D;AAC7D;IACI,YAAY;IACZ,sBAAsB;IACtB,kBAAkB;AACtB;;AAEA,yEAAyE;AACzE;IACI;QACI,WAAW;IACf;;IAEA;QACI,gBAAgB;QAChB,WAAW;QACX,eAAe;QACf,kBAAkB;IACtB;AACJ","sourcesContent":["#main-table {\\r\\n font-size: 12px;\\r\\n /* calc(0.5vw + 0.5em); */\\r\\n margin: auto;\\r\\n width: 100%;\\r\\n}\\r\\n\\r\\n#main-table thead > tr > th {\\r\\n text-align: left;\\r\\n background-color: var(--darkGray);\\r\\n color: white;\\r\\n position: -webkit-sticky; /* For Safari */\\r\\n position: sticky;\\r\\n top: 0;\\r\\n padding-top: 5px;\\r\\n padding-bottom: 5px;\\r\\n padding-left: 5px;\\r\\n}\\r\\n\\r\\nth {\\r\\n background-color: var(--lightGray);\\r\\n}\\r\\n\\r\\ntr {\\r\\n border-width: 2px;\\r\\n background-color: white;\\r\\n}\\r\\n\\r\\ntr td {\\r\\n border-bottom: 1px solid black;\\r\\n}\\r\\n\\r\\n/* Default to hidden columns */\\r\\n\\r\\n#main-table tbody > tr > td {\\r\\n display : none;\\r\\n}\\r\\n#main-table thead > tr > th {\\r\\n display : none;\\r\\n}\\r\\n\\r\\n/* textbox width in table */\\r\\ninput {\\r\\n width: 100%;\\r\\n}\\r\\n\\r\\nselect {\\r\\n max-width: 100px;\\r\\n}\\r\\n\\r\\ndiv.table-container {\\r\\n overflow-x: auto; \\r\\n overflow-y: auto;\\r\\n margin-left: max(65px, 7vh);\\r\\n margin-right: max(65px, 7vh);\\r\\n /* max-width: calc(100vw - var(--sidebar-width)); */\\r\\n /* margin: auto; */\\r\\n max-height: max(350px, 6vh);\\r\\n /* min-height: 350px; */\\r\\n\\r\\n}\\r\\n \\r\\n/* Buttons */\\r\\n\\r\\n/* Add new row button */\\r\\n.btn-add { \\r\\n background-color: var(--spiritgreen);\\r\\n margin-top: 20px;\\r\\n display: none;\\r\\n}\\r\\n\\r\\n.btn-delete {\\r\\n background-color: var(--orange);\\r\\n}\\r\\n\\r\\n#add-btn-div {\\r\\n display: flex;\\r\\n justify-content: center; /* Aligns horizontally */\\r\\n align-items: center; /* Aligns vertically */\\r\\n width: 100%;\\r\\n}\\r\\n\\r\\n.btn-edit {\\r\\n background-color: var(--spiritgreen);\\r\\n line-height: 1.1;\\r\\n}\\r\\n\\r\\n.active-editing, .selected {\\r\\n background-color: var(--palegreen);\\r\\n}\\r\\n\\r\\n.selected { \\r\\n font-weight: bold;\\r\\n}\\r\\n\\r\\n.btn-confirm {\\r\\n display: none;\\r\\n}\\r\\n\\r\\n.confirm-btn:hover {\\r\\n background-color: var(--green);\\r\\n}\\r\\n\\r\\n.hover-effect:hover {\\r\\n cursor: pointer;\\r\\n background-color: var(--verypalegreen); \\r\\n}\\r\\n\\r\\n/* Fund table */\\r\\n.fund-name > .viewed-fund {\\r\\n color: gray;\\r\\n}\\r\\n\\r\\n/* made edit column narrower */\\r\\nth.edit, td.edit {\\r\\n width: 20px;\\r\\n}\\r\\n\\r\\n/* Class for hidden rows on filter */\\r\\n.hidden {\\r\\n display: none;\\r\\n}\\r\\n\\r\\n/* Filter styling for the dropdown options */\\r\\n#filter-container {\\r\\n display: flex;\\r\\n flex-wrap: wrap;\\r\\n gap: 2px; /* Adjust the gap as needed */\\r\\n font-size: 11px;\\r\\n margin-left: 10%;\\r\\n margin-right: 10%;\\r\\n margin-bottom: 5px;\\r\\n}\\r\\n\\r\\n#filter-container div {\\r\\n display: flex;\\r\\n align-items: center;\\r\\n width: calc(50% - 10px); /* Adjust width to fit two items per row with gap */\\r\\n}\\r\\n\\r\\n#filter-container label {\\r\\n width: 200px; /* Adjust the width as needed */\\r\\n text-align: right;\\r\\n margin-right: 10px; /* Spacing between label and select */\\r\\n}\\r\\n\\r\\n/* Ensure all select boxes are wide enough to show text */\\r\\n.filter-dropdown {\\r\\n flex: 1;\\r\\n min-width: 200px; /* Adjust the min-width as needed */\\r\\n}\\r\\n\\r\\n/* Optional styling to make the select dropdown look better */\\r\\n.filter-dropdown {\\r\\n padding: 5px;\\r\\n border: 1px solid #ccc;\\r\\n border-radius: 4px;\\r\\n}\\r\\n\\r\\n/* Responsive adjustment: On smaller screens, stack elements vertically */\\r\\n@media (max-width: 800px) {\\r\\n #filter-container div {\\r\\n width: 100%;\\r\\n }\\r\\n\\r\\n #filter-container label {\\r\\n text-align: left;\\r\\n width: auto;\\r\\n margin-right: 0;\\r\\n margin-bottom: 5px;\\r\\n }\\r\\n}"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"279.js","mappings":";;;;;;;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sources":["webpack://budget-request-form/./src/js/components/table/table.css?5237"],"sourcesContent":["// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../node_modules/css-loader/dist/runtime/sourceMaps.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `#main-table {\r\n    font-size: 12px;\r\n    /* calc(0.5vw + 0.5em); */\r\n    margin: auto;\r\n    width: 100%;\r\n}\r\n\r\n#main-table thead > tr > th {\r\n    text-align: left;\r\n    background-color: var(--darkGray);\r\n    color: white;\r\n    position: -webkit-sticky; /* For Safari */\r\n    position: sticky;\r\n    top: 0;\r\n    padding-top: 5px;\r\n    padding-bottom: 5px;\r\n    padding-left: 5px;\r\n}\r\n\r\nth {\r\n    background-color: var(--lightGray);\r\n}\r\n\r\ntr {\r\n    border-width: 2px;\r\n    background-color: white;\r\n}\r\n\r\ntr td {\r\n    border-bottom: 1px solid black;\r\n}\r\n\r\n/* Default to hidden columns */\r\n\r\n#main-table tbody > tr > td {\r\n    display : none;\r\n}\r\n#main-table thead > tr > th {\r\n    display : none;\r\n}\r\n\r\n/* textbox width in table */\r\ninput {\r\n    width: 100%;\r\n}\r\n\r\nselect {\r\n    max-width: 100px;\r\n}\r\n\r\ndiv.table-container {\r\n    overflow-x: auto; \r\n    overflow-y: auto;\r\n    margin-left: max(65px, 7vh);\r\n    margin-right: max(65px, 7vh);\r\n    /* max-width: calc(100vw - var(--sidebar-width));  */\r\n    /* margin: auto; */\r\n    max-height: max(350px, 6vh);\r\n    /* min-height: 350px; */\r\n\r\n}\r\n  \r\n/* Buttons */\r\n\r\n/* Add new row button */\r\n.btn-add { \r\n    background-color: var(--spiritgreen);\r\n    margin-top: 20px;\r\n    display: none;\r\n}\r\n\r\n.btn-delete {\r\n    background-color: var(--orange);\r\n}\r\n\r\n#add-btn-div {\r\n    display: flex;\r\n    justify-content: center; /* Aligns horizontally */\r\n    align-items: center; /* Aligns vertically */\r\n    width: 100%;\r\n}\r\n\r\n.btn-edit {\r\n    background-color: var(--spiritgreen);\r\n    line-height: 1.1;\r\n}\r\n\r\n.active-editing, .selected {\r\n    background-color: var(--palegreen);\r\n}\r\n\r\n.selected { \r\n    font-weight: bold;\r\n}\r\n\r\n.btn-confirm {\r\n    display: none;\r\n}\r\n\r\n.confirm-btn:hover {\r\n    background-color: var(--green);\r\n}\r\n\r\n.hover-effect:hover {\r\n    cursor: pointer;\r\n    background-color: var(--verypalegreen); \r\n}\r\n\r\n/* Fund table */\r\n.fund-name > .viewed-fund {\r\n    color: gray;\r\n}\r\n\r\n/* made edit column narrower */\r\nth.edit, td.edit {\r\n    width: 20px;\r\n}\r\n\r\n/* Class for hidden rows on filter */\r\n.hidden {\r\n    display: none;\r\n}\r\n\r\n/* Filter styling for the dropdown options */\r\n#filter-container {\r\n    display: flex;\r\n    flex-wrap: wrap;\r\n    gap: 2px; /* Adjust the gap as needed */\r\n    font-size: 11px;\r\n    margin-left: 10%;\r\n    margin-right: 10%;\r\n    margin-bottom: 5px;\r\n}\r\n\r\n#filter-container div {\r\n    display: flex;\r\n    align-items: center;\r\n    width: calc(50% - 10px); /* Adjust width to fit two items per row with gap */\r\n}\r\n\r\n#filter-container label {\r\n    width: 200px; /* Adjust the width as needed */\r\n    text-align: right;\r\n    margin-right: 10px; /* Spacing between label and select */\r\n}\r\n\r\n/* Ensure all select boxes are wide enough to show text */\r\n.filter-dropdown {\r\n    flex: 1;\r\n    min-width: 200px; /* Adjust the min-width as needed */\r\n}\r\n\r\n/* Optional styling to make the select dropdown look better */\r\n.filter-dropdown {\r\n    padding: 5px;\r\n    border: 1px solid #ccc;\r\n    border-radius: 4px;\r\n}\r\n\r\n/* Responsive adjustment: On smaller screens, stack elements vertically */\r\n@media (max-width: 800px) {\r\n    #filter-container div {\r\n        width: 100%;\r\n    }\r\n\r\n    #filter-container label {\r\n        text-align: left;\r\n        width: auto;\r\n        margin-right: 0;\r\n        margin-bottom: 5px;\r\n    }\r\n}`, \"\",{\"version\":3,\"sources\":[\"webpack://./src/js/components/table/table.css\"],\"names\":[],\"mappings\":\"AAAA;IACI,eAAe;IACf,yBAAyB;IACzB,YAAY;IACZ,WAAW;AACf;;AAEA;IACI,gBAAgB;IAChB,iCAAiC;IACjC,YAAY;IACZ,wBAAwB,EAAE,eAAe;IACzC,gBAAgB;IAChB,MAAM;IACN,gBAAgB;IAChB,mBAAmB;IACnB,iBAAiB;AACrB;;AAEA;IACI,kCAAkC;AACtC;;AAEA;IACI,iBAAiB;IACjB,uBAAuB;AAC3B;;AAEA;IACI,8BAA8B;AAClC;;AAEA,8BAA8B;;AAE9B;IACI,cAAc;AAClB;AACA;IACI,cAAc;AAClB;;AAEA,2BAA2B;AAC3B;IACI,WAAW;AACf;;AAEA;IACI,gBAAgB;AACpB;;AAEA;IACI,gBAAgB;IAChB,gBAAgB;IAChB,2BAA2B;IAC3B,4BAA4B;IAC5B,oDAAoD;IACpD,kBAAkB;IAClB,2BAA2B;IAC3B,uBAAuB;;AAE3B;;AAEA,YAAY;;AAEZ,uBAAuB;AACvB;IACI,oCAAoC;IACpC,gBAAgB;IAChB,aAAa;AACjB;;AAEA;IACI,+BAA+B;AACnC;;AAEA;IACI,aAAa;IACb,uBAAuB,EAAE,wBAAwB;IACjD,mBAAmB,EAAE,sBAAsB;IAC3C,WAAW;AACf;;AAEA;IACI,oCAAoC;IACpC,gBAAgB;AACpB;;AAEA;IACI,kCAAkC;AACtC;;AAEA;IACI,iBAAiB;AACrB;;AAEA;IACI,aAAa;AACjB;;AAEA;IACI,8BAA8B;AAClC;;AAEA;IACI,eAAe;IACf,sCAAsC;AAC1C;;AAEA,eAAe;AACf;IACI,WAAW;AACf;;AAEA,8BAA8B;AAC9B;IACI,WAAW;AACf;;AAEA,oCAAoC;AACpC;IACI,aAAa;AACjB;;AAEA,4CAA4C;AAC5C;IACI,aAAa;IACb,eAAe;IACf,QAAQ,EAAE,6BAA6B;IACvC,eAAe;IACf,gBAAgB;IAChB,iBAAiB;IACjB,kBAAkB;AACtB;;AAEA;IACI,aAAa;IACb,mBAAmB;IACnB,uBAAuB,EAAE,mDAAmD;AAChF;;AAEA;IACI,YAAY,EAAE,+BAA+B;IAC7C,iBAAiB;IACjB,kBAAkB,EAAE,qCAAqC;AAC7D;;AAEA,yDAAyD;AACzD;IACI,OAAO;IACP,gBAAgB,EAAE,mCAAmC;AACzD;;AAEA,6DAA6D;AAC7D;IACI,YAAY;IACZ,sBAAsB;IACtB,kBAAkB;AACtB;;AAEA,yEAAyE;AACzE;IACI;QACI,WAAW;IACf;;IAEA;QACI,gBAAgB;QAChB,WAAW;QACX,eAAe;QACf,kBAAkB;IACtB;AACJ\",\"sourcesContent\":[\"#main-table {\\r\\n    font-size: 12px;\\r\\n    /* calc(0.5vw + 0.5em); */\\r\\n    margin: auto;\\r\\n    width: 100%;\\r\\n}\\r\\n\\r\\n#main-table thead > tr > th {\\r\\n    text-align: left;\\r\\n    background-color: var(--darkGray);\\r\\n    color: white;\\r\\n    position: -webkit-sticky; /* For Safari */\\r\\n    position: sticky;\\r\\n    top: 0;\\r\\n    padding-top: 5px;\\r\\n    padding-bottom: 5px;\\r\\n    padding-left: 5px;\\r\\n}\\r\\n\\r\\nth {\\r\\n    background-color: var(--lightGray);\\r\\n}\\r\\n\\r\\ntr {\\r\\n    border-width: 2px;\\r\\n    background-color: white;\\r\\n}\\r\\n\\r\\ntr td {\\r\\n    border-bottom: 1px solid black;\\r\\n}\\r\\n\\r\\n/* Default to hidden columns */\\r\\n\\r\\n#main-table tbody > tr > td {\\r\\n    display : none;\\r\\n}\\r\\n#main-table thead > tr > th {\\r\\n    display : none;\\r\\n}\\r\\n\\r\\n/* textbox width in table */\\r\\ninput {\\r\\n    width: 100%;\\r\\n}\\r\\n\\r\\nselect {\\r\\n    max-width: 100px;\\r\\n}\\r\\n\\r\\ndiv.table-container {\\r\\n    overflow-x: auto; \\r\\n    overflow-y: auto;\\r\\n    margin-left: max(65px, 7vh);\\r\\n    margin-right: max(65px, 7vh);\\r\\n    /* max-width: calc(100vw - var(--sidebar-width));  */\\r\\n    /* margin: auto; */\\r\\n    max-height: max(350px, 6vh);\\r\\n    /* min-height: 350px; */\\r\\n\\r\\n}\\r\\n  \\r\\n/* Buttons */\\r\\n\\r\\n/* Add new row button */\\r\\n.btn-add { \\r\\n    background-color: var(--spiritgreen);\\r\\n    margin-top: 20px;\\r\\n    display: none;\\r\\n}\\r\\n\\r\\n.btn-delete {\\r\\n    background-color: var(--orange);\\r\\n}\\r\\n\\r\\n#add-btn-div {\\r\\n    display: flex;\\r\\n    justify-content: center; /* Aligns horizontally */\\r\\n    align-items: center; /* Aligns vertically */\\r\\n    width: 100%;\\r\\n}\\r\\n\\r\\n.btn-edit {\\r\\n    background-color: var(--spiritgreen);\\r\\n    line-height: 1.1;\\r\\n}\\r\\n\\r\\n.active-editing, .selected {\\r\\n    background-color: var(--palegreen);\\r\\n}\\r\\n\\r\\n.selected { \\r\\n    font-weight: bold;\\r\\n}\\r\\n\\r\\n.btn-confirm {\\r\\n    display: none;\\r\\n}\\r\\n\\r\\n.confirm-btn:hover {\\r\\n    background-color: var(--green);\\r\\n}\\r\\n\\r\\n.hover-effect:hover {\\r\\n    cursor: pointer;\\r\\n    background-color: var(--verypalegreen); \\r\\n}\\r\\n\\r\\n/* Fund table */\\r\\n.fund-name > .viewed-fund {\\r\\n    color: gray;\\r\\n}\\r\\n\\r\\n/* made edit column narrower */\\r\\nth.edit, td.edit {\\r\\n    width: 20px;\\r\\n}\\r\\n\\r\\n/* Class for hidden rows on filter */\\r\\n.hidden {\\r\\n    display: none;\\r\\n}\\r\\n\\r\\n/* Filter styling for the dropdown options */\\r\\n#filter-container {\\r\\n    display: flex;\\r\\n    flex-wrap: wrap;\\r\\n    gap: 2px; /* Adjust the gap as needed */\\r\\n    font-size: 11px;\\r\\n    margin-left: 10%;\\r\\n    margin-right: 10%;\\r\\n    margin-bottom: 5px;\\r\\n}\\r\\n\\r\\n#filter-container div {\\r\\n    display: flex;\\r\\n    align-items: center;\\r\\n    width: calc(50% - 10px); /* Adjust width to fit two items per row with gap */\\r\\n}\\r\\n\\r\\n#filter-container label {\\r\\n    width: 200px; /* Adjust the width as needed */\\r\\n    text-align: right;\\r\\n    margin-right: 10px; /* Spacing between label and select */\\r\\n}\\r\\n\\r\\n/* Ensure all select boxes are wide enough to show text */\\r\\n.filter-dropdown {\\r\\n    flex: 1;\\r\\n    min-width: 200px; /* Adjust the min-width as needed */\\r\\n}\\r\\n\\r\\n/* Optional styling to make the select dropdown look better */\\r\\n.filter-dropdown {\\r\\n    padding: 5px;\\r\\n    border: 1px solid #ccc;\\r\\n    border-radius: 4px;\\r\\n}\\r\\n\\r\\n/* Responsive adjustment: On smaller screens, stack elements vertically */\\r\\n@media (max-width: 800px) {\\r\\n    #filter-container div {\\r\\n        width: 100%;\\r\\n    }\\r\\n\\r\\n    #filter-container label {\\r\\n        text-align: left;\\r\\n        width: auto;\\r\\n        margin-right: 0;\\r\\n        margin-bottom: 5px;\\r\\n    }\\r\\n}\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///279\n')},843:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `#tooltip {\r\n position: absolute;\r\n background-color: black;\r\n color: white;\r\n padding: 5px;\r\n border-radius: 3px;\r\n visibility: hidden;\r\n white-space: nowrap;\r\n font-size: 14px;\r\n z-index: 1000;\r\n max-width: 300px;\r\n word-wrap: break-word;\r\n white-space: normal;\r\n}\r\n\r\n.tooltip-cell {\r\n /* color: blue; */\r\n /* text-decoration: underline; */\r\n cursor: pointer;\r\n}\r\n\r\n.tooltip-cell .info-icon {\r\n margin-left: 5px;\r\n color: var(--spiritgreen);\r\n font-size: 15px;\r\n}\r\n\r\n.tooltip-cell:hover {\r\n background-color: #f0f0f0; /* Change background on hover */\r\n}\r\n\r\n.detail {\r\n color: blue;\r\n color: var(--spiritgreen);\r\n text-decoration: underline;\r\n margin-left: 5px;\r\n}`, "",{"version":3,"sources":["webpack://./src/js/components/tooltip/tooltip.css"],"names":[],"mappings":"AAAA;IACI,kBAAkB;IAClB,uBAAuB;IACvB,YAAY;IACZ,YAAY;IACZ,kBAAkB;IAClB,kBAAkB;IAClB,mBAAmB;IACnB,eAAe;IACf,aAAa;IACb,gBAAgB;IAChB,qBAAqB;IACrB,mBAAmB;AACvB;;AAEA;IACI,iBAAiB;IACjB,gCAAgC;IAChC,eAAe;AACnB;;AAEA;IACI,gBAAgB;IAChB,yBAAyB;IACzB,eAAe;AACnB;;AAEA;IACI,yBAAyB,EAAE,+BAA+B;AAC9D;;AAEA;IACI,WAAW;IACX,yBAAyB;IACzB,0BAA0B;IAC1B,gBAAgB;AACpB","sourcesContent":["#tooltip {\\r\\n position: absolute;\\r\\n background-color: black;\\r\\n color: white;\\r\\n padding: 5px;\\r\\n border-radius: 3px;\\r\\n visibility: hidden;\\r\\n white-space: nowrap;\\r\\n font-size: 14px;\\r\\n z-index: 1000;\\r\\n max-width: 300px;\\r\\n word-wrap: break-word;\\r\\n white-space: normal;\\r\\n}\\r\\n\\r\\n.tooltip-cell {\\r\\n /* color: blue; */\\r\\n /* text-decoration: underline; */\\r\\n cursor: pointer;\\r\\n}\\r\\n\\r\\n.tooltip-cell .info-icon {\\r\\n margin-left: 5px;\\r\\n color: var(--spiritgreen);\\r\\n font-size: 15px;\\r\\n}\\r\\n\\r\\n.tooltip-cell:hover {\\r\\n background-color: #f0f0f0; /* Change background on hover */\\r\\n}\\r\\n\\r\\n.detail {\\r\\n color: blue;\\r\\n color: var(--spiritgreen);\\r\\n text-decoration: underline;\\r\\n margin-left: 5px;\\r\\n}"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiODQzLmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYnVkZ2V0LXJlcXVlc3QtZm9ybS8uL3NyYy9qcy9jb21wb25lbnRzL3Rvb2x0aXAvdG9vbHRpcC5jc3M/N2ZkNiJdLCJzb3VyY2VzQ29udGVudCI6WyIvLyBJbXBvcnRzXG5pbXBvcnQgX19fQ1NTX0xPQURFUl9BUElfU09VUkNFTUFQX0lNUE9SVF9fXyBmcm9tIFwiLi4vLi4vLi4vLi4vbm9kZV9tb2R1bGVzL2Nzcy1sb2FkZXIvZGlzdC9ydW50aW1lL3NvdXJjZU1hcHMuanNcIjtcbmltcG9ydCBfX19DU1NfTE9BREVSX0FQSV9JTVBPUlRfX18gZnJvbSBcIi4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvcnVudGltZS9hcGkuanNcIjtcbnZhciBfX19DU1NfTE9BREVSX0VYUE9SVF9fXyA9IF9fX0NTU19MT0FERVJfQVBJX0lNUE9SVF9fXyhfX19DU1NfTE9BREVSX0FQSV9TT1VSQ0VNQVBfSU1QT1JUX19fKTtcbi8vIE1vZHVsZVxuX19fQ1NTX0xPQURFUl9FWFBPUlRfX18ucHVzaChbbW9kdWxlLmlkLCBgI3Rvb2x0aXAge1xyXG4gICAgcG9zaXRpb246IGFic29sdXRlO1xyXG4gICAgYmFja2dyb3VuZC1jb2xvcjogYmxhY2s7XHJcbiAgICBjb2xvcjogd2hpdGU7XHJcbiAgICBwYWRkaW5nOiA1cHg7XHJcbiAgICBib3JkZXItcmFkaXVzOiAzcHg7XHJcbiAgICB2aXNpYmlsaXR5OiBoaWRkZW47XHJcbiAgICB3aGl0ZS1zcGFjZTogbm93cmFwO1xyXG4gICAgZm9udC1zaXplOiAxNHB4O1xyXG4gICAgei1pbmRleDogMTAwMDtcclxuICAgIG1heC13aWR0aDogMzAwcHg7XHJcbiAgICB3b3JkLXdyYXA6IGJyZWFrLXdvcmQ7XHJcbiAgICB3aGl0ZS1zcGFjZTogbm9ybWFsO1xyXG59XHJcblxyXG4udG9vbHRpcC1jZWxsIHtcclxuICAgIC8qIGNvbG9yOiBibHVlOyAqL1xyXG4gICAgLyogdGV4dC1kZWNvcmF0aW9uOiB1bmRlcmxpbmU7ICovXHJcbiAgICBjdXJzb3I6IHBvaW50ZXI7XHJcbn1cclxuXHJcbi50b29sdGlwLWNlbGwgLmluZm8taWNvbiB7XHJcbiAgICBtYXJnaW4tbGVmdDogNXB4O1xyXG4gICAgY29sb3I6IHZhcigtLXNwaXJpdGdyZWVuKTtcclxuICAgIGZvbnQtc2l6ZTogMTVweDtcclxufVxyXG5cclxuLnRvb2x0aXAtY2VsbDpob3ZlciB7XHJcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjZjBmMGYwOyAvKiBDaGFuZ2UgYmFja2dyb3VuZCBvbiBob3ZlciAqL1xyXG59XHJcblxyXG4uZGV0YWlsIHtcclxuICAgIGNvbG9yOiBibHVlO1xyXG4gICAgY29sb3I6IHZhcigtLXNwaXJpdGdyZWVuKTtcclxuICAgIHRleHQtZGVjb3JhdGlvbjogdW5kZXJsaW5lO1xyXG4gICAgbWFyZ2luLWxlZnQ6IDVweDtcclxufWAsIFwiXCIse1widmVyc2lvblwiOjMsXCJzb3VyY2VzXCI6W1wid2VicGFjazovLy4vc3JjL2pzL2NvbXBvbmVudHMvdG9vbHRpcC90b29sdGlwLmNzc1wiXSxcIm5hbWVzXCI6W10sXCJtYXBwaW5nc1wiOlwiQUFBQTtJQUNJLGtCQUFrQjtJQUNsQix1QkFBdUI7SUFDdkIsWUFBWTtJQUNaLFlBQVk7SUFDWixrQkFBa0I7SUFDbEIsa0JBQWtCO0lBQ2xCLG1CQUFtQjtJQUNuQixlQUFlO0lBQ2YsYUFBYTtJQUNiLGdCQUFnQjtJQUNoQixxQkFBcUI7SUFDckIsbUJBQW1CO0FBQ3ZCOztBQUVBO0lBQ0ksaUJBQWlCO0lBQ2pCLGdDQUFnQztJQUNoQyxlQUFlO0FBQ25COztBQUVBO0lBQ0ksZ0JBQWdCO0lBQ2hCLHlCQUF5QjtJQUN6QixlQUFlO0FBQ25COztBQUVBO0lBQ0kseUJBQXlCLEVBQUUsK0JBQStCO0FBQzlEOztBQUVBO0lBQ0ksV0FBVztJQUNYLHlCQUF5QjtJQUN6QiwwQkFBMEI7SUFDMUIsZ0JBQWdCO0FBQ3BCXCIsXCJzb3VyY2VzQ29udGVudFwiOltcIiN0b29sdGlwIHtcXHJcXG4gICAgcG9zaXRpb246IGFic29sdXRlO1xcclxcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiBibGFjaztcXHJcXG4gICAgY29sb3I6IHdoaXRlO1xcclxcbiAgICBwYWRkaW5nOiA1cHg7XFxyXFxuICAgIGJvcmRlci1yYWRpdXM6IDNweDtcXHJcXG4gICAgdmlzaWJpbGl0eTogaGlkZGVuO1xcclxcbiAgICB3aGl0ZS1zcGFjZTogbm93cmFwO1xcclxcbiAgICBmb250LXNpemU6IDE0cHg7XFxyXFxuICAgIHotaW5kZXg6IDEwMDA7XFxyXFxuICAgIG1heC13aWR0aDogMzAwcHg7XFxyXFxuICAgIHdvcmQtd3JhcDogYnJlYWstd29yZDtcXHJcXG4gICAgd2hpdGUtc3BhY2U6IG5vcm1hbDtcXHJcXG59XFxyXFxuXFxyXFxuLnRvb2x0aXAtY2VsbCB7XFxyXFxuICAgIC8qIGNvbG9yOiBibHVlOyAqL1xcclxcbiAgICAvKiB0ZXh0LWRlY29yYXRpb246IHVuZGVybGluZTsgKi9cXHJcXG4gICAgY3Vyc29yOiBwb2ludGVyO1xcclxcbn1cXHJcXG5cXHJcXG4udG9vbHRpcC1jZWxsIC5pbmZvLWljb24ge1xcclxcbiAgICBtYXJnaW4tbGVmdDogNXB4O1xcclxcbiAgICBjb2xvcjogdmFyKC0tc3Bpcml0Z3JlZW4pO1xcclxcbiAgICBmb250LXNpemU6IDE1cHg7XFxyXFxufVxcclxcblxcclxcbi50b29sdGlwLWNlbGw6aG92ZXIge1xcclxcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjZjBmMGYwOyAvKiBDaGFuZ2UgYmFja2dyb3VuZCBvbiBob3ZlciAqL1xcclxcbn1cXHJcXG5cXHJcXG4uZGV0YWlsIHtcXHJcXG4gICAgY29sb3I6IGJsdWU7XFxyXFxuICAgIGNvbG9yOiB2YXIoLS1zcGlyaXRncmVlbik7XFxyXFxuICAgIHRleHQtZGVjb3JhdGlvbjogdW5kZXJsaW5lO1xcclxcbiAgICBtYXJnaW4tbGVmdDogNXB4O1xcclxcbn1cIl0sXCJzb3VyY2VSb290XCI6XCJcIn1dKTtcbi8vIEV4cG9ydHNcbmV4cG9ydCBkZWZhdWx0IF9fX0NTU19MT0FERVJfRVhQT1JUX19fO1xuIl0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///843\n')},235:(module,__webpack_exports__,__webpack_require__)=>{eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ A: () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(354);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(314);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, `/* Welcome page (index.html) */\r\n\r\n.step {\r\n width: 60%;\r\n height: 80px;\r\n font-size: 1.75em;\r\n margin-bottom: 0px; /* Adds spacing between buttons */\r\n margin-left: 20%;\r\n border-color: var(--citygreen);\r\n border-width: 2;\r\n color: var(--citygreen);\r\n background-color: white;\r\n}\r\n\r\n.step:hover {\r\n color: white;\r\n background-color: var(--spiritgreen);\r\n}\r\n\r\n#welcome-page {\r\n justify-content: center; \r\n align-items: center; \r\n padding-top: 20px;\r\n}\r\n.step.disabled {\r\n opacity: 50%;\r\n}`, "",{"version":3,"sources":["webpack://./src/js/components/welcome/welcome.css"],"names":[],"mappings":"AAAA,8BAA8B;;AAE9B;IACI,UAAU;IACV,YAAY;IACZ,iBAAiB;IACjB,kBAAkB,EAAE,iCAAiC;IACrD,gBAAgB;IAChB,8BAA8B;IAC9B,eAAe;IACf,uBAAuB;IACvB,uBAAuB;AAC3B;;AAEA;IACI,YAAY;IACZ,oCAAoC;AACxC;;AAEA;IACI,uBAAuB;IACvB,mBAAmB;IACnB,iBAAiB;AACrB;AACA;IACI,YAAY;AAChB","sourcesContent":["/* Welcome page (index.html) */\\r\\n\\r\\n.step {\\r\\n width: 60%;\\r\\n height: 80px;\\r\\n font-size: 1.75em;\\r\\n margin-bottom: 0px; /* Adds spacing between buttons */\\r\\n margin-left: 20%;\\r\\n border-color: var(--citygreen);\\r\\n border-width: 2;\\r\\n color: var(--citygreen);\\r\\n background-color: white;\\r\\n}\\r\\n\\r\\n.step:hover {\\r\\n color: white;\\r\\n background-color: var(--spiritgreen);\\r\\n}\\r\\n\\r\\n#welcome-page {\\r\\n justify-content: center; \\r\\n align-items: center; \\r\\n padding-top: 20px;\\r\\n}\\r\\n.step.disabled {\\r\\n opacity: 50%;\\r\\n}"],"sourceRoot":""}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMjM1LmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSIsInNvdXJjZXMiOlsid2VicGFjazovL2J1ZGdldC1yZXF1ZXN0LWZvcm0vLi9zcmMvanMvY29tcG9uZW50cy93ZWxjb21lL3dlbGNvbWUuY3NzP2IzNDgiXSwic291cmNlc0NvbnRlbnQiOlsiLy8gSW1wb3J0c1xuaW1wb3J0IF9fX0NTU19MT0FERVJfQVBJX1NPVVJDRU1BUF9JTVBPUlRfX18gZnJvbSBcIi4uLy4uLy4uLy4uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvcnVudGltZS9zb3VyY2VNYXBzLmpzXCI7XG5pbXBvcnQgX19fQ1NTX0xPQURFUl9BUElfSU1QT1JUX19fIGZyb20gXCIuLi8uLi8uLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvYXBpLmpzXCI7XG52YXIgX19fQ1NTX0xPQURFUl9FWFBPUlRfX18gPSBfX19DU1NfTE9BREVSX0FQSV9JTVBPUlRfX18oX19fQ1NTX0xPQURFUl9BUElfU09VUkNFTUFQX0lNUE9SVF9fXyk7XG4vLyBNb2R1bGVcbl9fX0NTU19MT0FERVJfRVhQT1JUX19fLnB1c2goW21vZHVsZS5pZCwgYC8qIFdlbGNvbWUgcGFnZSAoaW5kZXguaHRtbCkgKi9cclxuXHJcbi5zdGVwIHtcclxuICAgIHdpZHRoOiA2MCU7XHJcbiAgICBoZWlnaHQ6IDgwcHg7XHJcbiAgICBmb250LXNpemU6IDEuNzVlbTtcclxuICAgIG1hcmdpbi1ib3R0b206IDBweDsgLyogQWRkcyBzcGFjaW5nIGJldHdlZW4gYnV0dG9ucyAqL1xyXG4gICAgbWFyZ2luLWxlZnQ6IDIwJTtcclxuICAgIGJvcmRlci1jb2xvcjogdmFyKC0tY2l0eWdyZWVuKTtcclxuICAgIGJvcmRlci13aWR0aDogMjtcclxuICAgIGNvbG9yOiB2YXIoLS1jaXR5Z3JlZW4pO1xyXG4gICAgYmFja2dyb3VuZC1jb2xvcjogd2hpdGU7XHJcbn1cclxuXHJcbi5zdGVwOmhvdmVyIHtcclxuICAgIGNvbG9yOiB3aGl0ZTtcclxuICAgIGJhY2tncm91bmQtY29sb3I6IHZhcigtLXNwaXJpdGdyZWVuKTtcclxufVxyXG5cclxuI3dlbGNvbWUtcGFnZSB7XHJcbiAgICBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjsgXHJcbiAgICBhbGlnbi1pdGVtczogY2VudGVyOyBcclxuICAgIHBhZGRpbmctdG9wOiAyMHB4O1xyXG59XHJcbi5zdGVwLmRpc2FibGVkIHtcclxuICAgIG9wYWNpdHk6IDUwJTtcclxufWAsIFwiXCIse1widmVyc2lvblwiOjMsXCJzb3VyY2VzXCI6W1wid2VicGFjazovLy4vc3JjL2pzL2NvbXBvbmVudHMvd2VsY29tZS93ZWxjb21lLmNzc1wiXSxcIm5hbWVzXCI6W10sXCJtYXBwaW5nc1wiOlwiQUFBQSw4QkFBOEI7O0FBRTlCO0lBQ0ksVUFBVTtJQUNWLFlBQVk7SUFDWixpQkFBaUI7SUFDakIsa0JBQWtCLEVBQUUsaUNBQWlDO0lBQ3JELGdCQUFnQjtJQUNoQiw4QkFBOEI7SUFDOUIsZUFBZTtJQUNmLHVCQUF1QjtJQUN2Qix1QkFBdUI7QUFDM0I7O0FBRUE7SUFDSSxZQUFZO0lBQ1osb0NBQW9DO0FBQ3hDOztBQUVBO0lBQ0ksdUJBQXVCO0lBQ3ZCLG1CQUFtQjtJQUNuQixpQkFBaUI7QUFDckI7QUFDQTtJQUNJLFlBQVk7QUFDaEJcIixcInNvdXJjZXNDb250ZW50XCI6W1wiLyogV2VsY29tZSBwYWdlIChpbmRleC5odG1sKSAqL1xcclxcblxcclxcbi5zdGVwIHtcXHJcXG4gICAgd2lkdGg6IDYwJTtcXHJcXG4gICAgaGVpZ2h0OiA4MHB4O1xcclxcbiAgICBmb250LXNpemU6IDEuNzVlbTtcXHJcXG4gICAgbWFyZ2luLWJvdHRvbTogMHB4OyAvKiBBZGRzIHNwYWNpbmcgYmV0d2VlbiBidXR0b25zICovXFxyXFxuICAgIG1hcmdpbi1sZWZ0OiAyMCU7XFxyXFxuICAgIGJvcmRlci1jb2xvcjogdmFyKC0tY2l0eWdyZWVuKTtcXHJcXG4gICAgYm9yZGVyLXdpZHRoOiAyO1xcclxcbiAgICBjb2xvcjogdmFyKC0tY2l0eWdyZWVuKTtcXHJcXG4gICAgYmFja2dyb3VuZC1jb2xvcjogd2hpdGU7XFxyXFxufVxcclxcblxcclxcbi5zdGVwOmhvdmVyIHtcXHJcXG4gICAgY29sb3I6IHdoaXRlO1xcclxcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiB2YXIoLS1zcGlyaXRncmVlbik7XFxyXFxufVxcclxcblxcclxcbiN3ZWxjb21lLXBhZ2Uge1xcclxcbiAgICBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjsgXFxyXFxuICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7IFxcclxcbiAgICBwYWRkaW5nLXRvcDogMjBweDtcXHJcXG59XFxyXFxuLnN0ZXAuZGlzYWJsZWQge1xcclxcbiAgICBvcGFjaXR5OiA1MCU7XFxyXFxufVwiXSxcInNvdXJjZVJvb3RcIjpcIlwifV0pO1xuLy8gRXhwb3J0c1xuZXhwb3J0IGRlZmF1bHQgX19fQ1NTX0xPQURFUl9FWFBPUlRfX187XG4iXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///235\n')},314:module=>{eval('\n\n/*\n MIT License http://www.opensource.org/licenses/mit-license.php\n Author Tobias Koppers @sokra\n*/\nmodule.exports = function (cssWithMappingToString) {\n var list = [];\n\n // return the list of modules as css string\n list.toString = function toString() {\n return this.map(function (item) {\n var content = "";\n var needLayer = typeof item[5] !== "undefined";\n if (item[4]) {\n content += "@supports (".concat(item[4], ") {");\n }\n if (item[2]) {\n content += "@media ".concat(item[2], " {");\n }\n if (needLayer) {\n content += "@layer".concat(item[5].length > 0 ? " ".concat(item[5]) : "", " {");\n }\n content += cssWithMappingToString(item);\n if (needLayer) {\n content += "}";\n }\n if (item[2]) {\n content += "}";\n }\n if (item[4]) {\n content += "}";\n }\n return content;\n }).join("");\n };\n\n // import a list of modules into the list\n list.i = function i(modules, media, dedupe, supports, layer) {\n if (typeof modules === "string") {\n modules = [[null, modules, undefined]];\n }\n var alreadyImportedModules = {};\n if (dedupe) {\n for (var k = 0; k < this.length; k++) {\n var id = this[k][0];\n if (id != null) {\n alreadyImportedModules[id] = true;\n }\n }\n }\n for (var _k = 0; _k < modules.length; _k++) {\n var item = [].concat(modules[_k]);\n if (dedupe && alreadyImportedModules[item[0]]) {\n continue;\n }\n if (typeof layer !== "undefined") {\n if (typeof item[5] === "undefined") {\n item[5] = layer;\n } else {\n item[1] = "@layer".concat(item[5].length > 0 ? " ".concat(item[5]) : "", " {").concat(item[1], "}");\n item[5] = layer;\n }\n }\n if (media) {\n if (!item[2]) {\n item[2] = media;\n } else {\n item[1] = "@media ".concat(item[2], " {").concat(item[1], "}");\n item[2] = media;\n }\n }\n if (supports) {\n if (!item[4]) {\n item[4] = "".concat(supports);\n } else {\n item[1] = "@supports (".concat(item[4], ") {").concat(item[1], "}");\n item[4] = supports;\n }\n }\n list.push(item);\n }\n };\n return list;\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMzE0LmpzIiwibWFwcGluZ3MiOiJBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYnVkZ2V0LXJlcXVlc3QtZm9ybS8uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvcnVudGltZS9hcGkuanM/MjRmYiJdLCJzb3VyY2VzQ29udGVudCI6WyJcInVzZSBzdHJpY3RcIjtcblxuLypcbiAgTUlUIExpY2Vuc2UgaHR0cDovL3d3dy5vcGVuc291cmNlLm9yZy9saWNlbnNlcy9taXQtbGljZW5zZS5waHBcbiAgQXV0aG9yIFRvYmlhcyBLb3BwZXJzIEBzb2tyYVxuKi9cbm1vZHVsZS5leHBvcnRzID0gZnVuY3Rpb24gKGNzc1dpdGhNYXBwaW5nVG9TdHJpbmcpIHtcbiAgdmFyIGxpc3QgPSBbXTtcblxuICAvLyByZXR1cm4gdGhlIGxpc3Qgb2YgbW9kdWxlcyBhcyBjc3Mgc3RyaW5nXG4gIGxpc3QudG9TdHJpbmcgPSBmdW5jdGlvbiB0b1N0cmluZygpIHtcbiAgICByZXR1cm4gdGhpcy5tYXAoZnVuY3Rpb24gKGl0ZW0pIHtcbiAgICAgIHZhciBjb250ZW50ID0gXCJcIjtcbiAgICAgIHZhciBuZWVkTGF5ZXIgPSB0eXBlb2YgaXRlbVs1XSAhPT0gXCJ1bmRlZmluZWRcIjtcbiAgICAgIGlmIChpdGVtWzRdKSB7XG4gICAgICAgIGNvbnRlbnQgKz0gXCJAc3VwcG9ydHMgKFwiLmNvbmNhdChpdGVtWzRdLCBcIikge1wiKTtcbiAgICAgIH1cbiAgICAgIGlmIChpdGVtWzJdKSB7XG4gICAgICAgIGNvbnRlbnQgKz0gXCJAbWVkaWEgXCIuY29uY2F0KGl0ZW1bMl0sIFwiIHtcIik7XG4gICAgICB9XG4gICAgICBpZiAobmVlZExheWVyKSB7XG4gICAgICAgIGNvbnRlbnQgKz0gXCJAbGF5ZXJcIi5jb25jYXQoaXRlbVs1XS5sZW5ndGggPiAwID8gXCIgXCIuY29uY2F0KGl0ZW1bNV0pIDogXCJcIiwgXCIge1wiKTtcbiAgICAgIH1cbiAgICAgIGNvbnRlbnQgKz0gY3NzV2l0aE1hcHBpbmdUb1N0cmluZyhpdGVtKTtcbiAgICAgIGlmIChuZWVkTGF5ZXIpIHtcbiAgICAgICAgY29udGVudCArPSBcIn1cIjtcbiAgICAgIH1cbiAgICAgIGlmIChpdGVtWzJdKSB7XG4gICAgICAgIGNvbnRlbnQgKz0gXCJ9XCI7XG4gICAgICB9XG4gICAgICBpZiAoaXRlbVs0XSkge1xuICAgICAgICBjb250ZW50ICs9IFwifVwiO1xuICAgICAgfVxuICAgICAgcmV0dXJuIGNvbnRlbnQ7XG4gICAgfSkuam9pbihcIlwiKTtcbiAgfTtcblxuICAvLyBpbXBvcnQgYSBsaXN0IG9mIG1vZHVsZXMgaW50byB0aGUgbGlzdFxuICBsaXN0LmkgPSBmdW5jdGlvbiBpKG1vZHVsZXMsIG1lZGlhLCBkZWR1cGUsIHN1cHBvcnRzLCBsYXllcikge1xuICAgIGlmICh0eXBlb2YgbW9kdWxlcyA9PT0gXCJzdHJpbmdcIikge1xuICAgICAgbW9kdWxlcyA9IFtbbnVsbCwgbW9kdWxlcywgdW5kZWZpbmVkXV07XG4gICAgfVxuICAgIHZhciBhbHJlYWR5SW1wb3J0ZWRNb2R1bGVzID0ge307XG4gICAgaWYgKGRlZHVwZSkge1xuICAgICAgZm9yICh2YXIgayA9IDA7IGsgPCB0aGlzLmxlbmd0aDsgaysrKSB7XG4gICAgICAgIHZhciBpZCA9IHRoaXNba11bMF07XG4gICAgICAgIGlmIChpZCAhPSBudWxsKSB7XG4gICAgICAgICAgYWxyZWFkeUltcG9ydGVkTW9kdWxlc1tpZF0gPSB0cnVlO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICAgIGZvciAodmFyIF9rID0gMDsgX2sgPCBtb2R1bGVzLmxlbmd0aDsgX2srKykge1xuICAgICAgdmFyIGl0ZW0gPSBbXS5jb25jYXQobW9kdWxlc1tfa10pO1xuICAgICAgaWYgKGRlZHVwZSAmJiBhbHJlYWR5SW1wb3J0ZWRNb2R1bGVzW2l0ZW1bMF1dKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuICAgICAgaWYgKHR5cGVvZiBsYXllciAhPT0gXCJ1bmRlZmluZWRcIikge1xuICAgICAgICBpZiAodHlwZW9mIGl0ZW1bNV0gPT09IFwidW5kZWZpbmVkXCIpIHtcbiAgICAgICAgICBpdGVtWzVdID0gbGF5ZXI7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgaXRlbVsxXSA9IFwiQGxheWVyXCIuY29uY2F0KGl0ZW1bNV0ubGVuZ3RoID4gMCA/IFwiIFwiLmNvbmNhdChpdGVtWzVdKSA6IFwiXCIsIFwiIHtcIikuY29uY2F0KGl0ZW1bMV0sIFwifVwiKTtcbiAgICAgICAgICBpdGVtWzVdID0gbGF5ZXI7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIGlmIChtZWRpYSkge1xuICAgICAgICBpZiAoIWl0ZW1bMl0pIHtcbiAgICAgICAgICBpdGVtWzJdID0gbWVkaWE7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgaXRlbVsxXSA9IFwiQG1lZGlhIFwiLmNvbmNhdChpdGVtWzJdLCBcIiB7XCIpLmNvbmNhdChpdGVtWzFdLCBcIn1cIik7XG4gICAgICAgICAgaXRlbVsyXSA9IG1lZGlhO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICBpZiAoc3VwcG9ydHMpIHtcbiAgICAgICAgaWYgKCFpdGVtWzRdKSB7XG4gICAgICAgICAgaXRlbVs0XSA9IFwiXCIuY29uY2F0KHN1cHBvcnRzKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBpdGVtWzFdID0gXCJAc3VwcG9ydHMgKFwiLmNvbmNhdChpdGVtWzRdLCBcIikge1wiKS5jb25jYXQoaXRlbVsxXSwgXCJ9XCIpO1xuICAgICAgICAgIGl0ZW1bNF0gPSBzdXBwb3J0cztcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgbGlzdC5wdXNoKGl0ZW0pO1xuICAgIH1cbiAgfTtcbiAgcmV0dXJuIGxpc3Q7XG59OyJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///314\n')},354:module=>{eval('\n\nmodule.exports = function (item) {\n var content = item[1];\n var cssMapping = item[3];\n if (!cssMapping) {\n return content;\n }\n if (typeof btoa === "function") {\n var base64 = btoa(unescape(encodeURIComponent(JSON.stringify(cssMapping))));\n var data = "sourceMappingURL=data:application/json;charset=utf-8;base64,".concat(base64);\n var sourceMapping = "/*# ".concat(data, " */");\n return [content].concat([sourceMapping]).join("\\n");\n }\n return [content].join("\\n");\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMzU0LmpzIiwibWFwcGluZ3MiOiJBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYnVkZ2V0LXJlcXVlc3QtZm9ybS8uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvcnVudGltZS9zb3VyY2VNYXBzLmpzP2FmMTIiXSwic291cmNlc0NvbnRlbnQiOlsiXCJ1c2Ugc3RyaWN0XCI7XG5cbm1vZHVsZS5leHBvcnRzID0gZnVuY3Rpb24gKGl0ZW0pIHtcbiAgdmFyIGNvbnRlbnQgPSBpdGVtWzFdO1xuICB2YXIgY3NzTWFwcGluZyA9IGl0ZW1bM107XG4gIGlmICghY3NzTWFwcGluZykge1xuICAgIHJldHVybiBjb250ZW50O1xuICB9XG4gIGlmICh0eXBlb2YgYnRvYSA9PT0gXCJmdW5jdGlvblwiKSB7XG4gICAgdmFyIGJhc2U2NCA9IGJ0b2EodW5lc2NhcGUoZW5jb2RlVVJJQ29tcG9uZW50KEpTT04uc3RyaW5naWZ5KGNzc01hcHBpbmcpKSkpO1xuICAgIHZhciBkYXRhID0gXCJzb3VyY2VNYXBwaW5nVVJMPWRhdGE6YXBwbGljYXRpb24vanNvbjtjaGFyc2V0PXV0Zi04O2Jhc2U2NCxcIi5jb25jYXQoYmFzZTY0KTtcbiAgICB2YXIgc291cmNlTWFwcGluZyA9IFwiLyojIFwiLmNvbmNhdChkYXRhLCBcIiAqL1wiKTtcbiAgICByZXR1cm4gW2NvbnRlbnRdLmNvbmNhdChbc291cmNlTWFwcGluZ10pLmpvaW4oXCJcXG5cIik7XG4gIH1cbiAgcmV0dXJuIFtjb250ZW50XS5qb2luKFwiXFxuXCIpO1xufTsiXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///354\n')},72:module=>{eval('\n\nvar stylesInDOM = [];\nfunction getIndexByIdentifier(identifier) {\n var result = -1;\n for (var i = 0; i < stylesInDOM.length; i++) {\n if (stylesInDOM[i].identifier === identifier) {\n result = i;\n break;\n }\n }\n return result;\n}\nfunction modulesToDom(list, options) {\n var idCountMap = {};\n var identifiers = [];\n for (var i = 0; i < list.length; i++) {\n var item = list[i];\n var id = options.base ? item[0] + options.base : item[0];\n var count = idCountMap[id] || 0;\n var identifier = "".concat(id, " ").concat(count);\n idCountMap[id] = count + 1;\n var indexByIdentifier = getIndexByIdentifier(identifier);\n var obj = {\n css: item[1],\n media: item[2],\n sourceMap: item[3],\n supports: item[4],\n layer: item[5]\n };\n if (indexByIdentifier !== -1) {\n stylesInDOM[indexByIdentifier].references++;\n stylesInDOM[indexByIdentifier].updater(obj);\n } else {\n var updater = addElementStyle(obj, options);\n options.byIndex = i;\n stylesInDOM.splice(i, 0, {\n identifier: identifier,\n updater: updater,\n references: 1\n });\n }\n identifiers.push(identifier);\n }\n return identifiers;\n}\nfunction addElementStyle(obj, options) {\n var api = options.domAPI(options);\n api.update(obj);\n var updater = function updater(newObj) {\n if (newObj) {\n if (newObj.css === obj.css && newObj.media === obj.media && newObj.sourceMap === obj.sourceMap && newObj.supports === obj.supports && newObj.layer === obj.layer) {\n return;\n }\n api.update(obj = newObj);\n } else {\n api.remove();\n }\n };\n return updater;\n}\nmodule.exports = function (list, options) {\n options = options || {};\n list = list || [];\n var lastIdentifiers = modulesToDom(list, options);\n return function update(newList) {\n newList = newList || [];\n for (var i = 0; i < lastIdentifiers.length; i++) {\n var identifier = lastIdentifiers[i];\n var index = getIndexByIdentifier(identifier);\n stylesInDOM[index].references--;\n }\n var newLastIdentifiers = modulesToDom(newList, options);\n for (var _i = 0; _i < lastIdentifiers.length; _i++) {\n var _identifier = lastIdentifiers[_i];\n var _index = getIndexByIdentifier(_identifier);\n if (stylesInDOM[_index].references === 0) {\n stylesInDOM[_index].updater();\n stylesInDOM.splice(_index, 1);\n }\n }\n lastIdentifiers = newLastIdentifiers;\n };\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiNzIuanMiLCJtYXBwaW5ncyI6IkFBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYnVkZ2V0LXJlcXVlc3QtZm9ybS8uL25vZGVfbW9kdWxlcy9zdHlsZS1sb2FkZXIvZGlzdC9ydW50aW1lL2luamVjdFN0eWxlc0ludG9TdHlsZVRhZy5qcz8yZGJhIl0sInNvdXJjZXNDb250ZW50IjpbIlwidXNlIHN0cmljdFwiO1xuXG52YXIgc3R5bGVzSW5ET00gPSBbXTtcbmZ1bmN0aW9uIGdldEluZGV4QnlJZGVudGlmaWVyKGlkZW50aWZpZXIpIHtcbiAgdmFyIHJlc3VsdCA9IC0xO1xuICBmb3IgKHZhciBpID0gMDsgaSA8IHN0eWxlc0luRE9NLmxlbmd0aDsgaSsrKSB7XG4gICAgaWYgKHN0eWxlc0luRE9NW2ldLmlkZW50aWZpZXIgPT09IGlkZW50aWZpZXIpIHtcbiAgICAgIHJlc3VsdCA9IGk7XG4gICAgICBicmVhaztcbiAgICB9XG4gIH1cbiAgcmV0dXJuIHJlc3VsdDtcbn1cbmZ1bmN0aW9uIG1vZHVsZXNUb0RvbShsaXN0LCBvcHRpb25zKSB7XG4gIHZhciBpZENvdW50TWFwID0ge307XG4gIHZhciBpZGVudGlmaWVycyA9IFtdO1xuICBmb3IgKHZhciBpID0gMDsgaSA8IGxpc3QubGVuZ3RoOyBpKyspIHtcbiAgICB2YXIgaXRlbSA9IGxpc3RbaV07XG4gICAgdmFyIGlkID0gb3B0aW9ucy5iYXNlID8gaXRlbVswXSArIG9wdGlvbnMuYmFzZSA6IGl0ZW1bMF07XG4gICAgdmFyIGNvdW50ID0gaWRDb3VudE1hcFtpZF0gfHwgMDtcbiAgICB2YXIgaWRlbnRpZmllciA9IFwiXCIuY29uY2F0KGlkLCBcIiBcIikuY29uY2F0KGNvdW50KTtcbiAgICBpZENvdW50TWFwW2lkXSA9IGNvdW50ICsgMTtcbiAgICB2YXIgaW5kZXhCeUlkZW50aWZpZXIgPSBnZXRJbmRleEJ5SWRlbnRpZmllcihpZGVudGlmaWVyKTtcbiAgICB2YXIgb2JqID0ge1xuICAgICAgY3NzOiBpdGVtWzFdLFxuICAgICAgbWVkaWE6IGl0ZW1bMl0sXG4gICAgICBzb3VyY2VNYXA6IGl0ZW1bM10sXG4gICAgICBzdXBwb3J0czogaXRlbVs0XSxcbiAgICAgIGxheWVyOiBpdGVtWzVdXG4gICAgfTtcbiAgICBpZiAoaW5kZXhCeUlkZW50aWZpZXIgIT09IC0xKSB7XG4gICAgICBzdHlsZXNJbkRPTVtpbmRleEJ5SWRlbnRpZmllcl0ucmVmZXJlbmNlcysrO1xuICAgICAgc3R5bGVzSW5ET01baW5kZXhCeUlkZW50aWZpZXJdLnVwZGF0ZXIob2JqKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdmFyIHVwZGF0ZXIgPSBhZGRFbGVtZW50U3R5bGUob2JqLCBvcHRpb25zKTtcbiAgICAgIG9wdGlvbnMuYnlJbmRleCA9IGk7XG4gICAgICBzdHlsZXNJbkRPTS5zcGxpY2UoaSwgMCwge1xuICAgICAgICBpZGVudGlmaWVyOiBpZGVudGlmaWVyLFxuICAgICAgICB1cGRhdGVyOiB1cGRhdGVyLFxuICAgICAgICByZWZlcmVuY2VzOiAxXG4gICAgICB9KTtcbiAgICB9XG4gICAgaWRlbnRpZmllcnMucHVzaChpZGVudGlmaWVyKTtcbiAgfVxuICByZXR1cm4gaWRlbnRpZmllcnM7XG59XG5mdW5jdGlvbiBhZGRFbGVtZW50U3R5bGUob2JqLCBvcHRpb25zKSB7XG4gIHZhciBhcGkgPSBvcHRpb25zLmRvbUFQSShvcHRpb25zKTtcbiAgYXBpLnVwZGF0ZShvYmopO1xuICB2YXIgdXBkYXRlciA9IGZ1bmN0aW9uIHVwZGF0ZXIobmV3T2JqKSB7XG4gICAgaWYgKG5ld09iaikge1xuICAgICAgaWYgKG5ld09iai5jc3MgPT09IG9iai5jc3MgJiYgbmV3T2JqLm1lZGlhID09PSBvYmoubWVkaWEgJiYgbmV3T2JqLnNvdXJjZU1hcCA9PT0gb2JqLnNvdXJjZU1hcCAmJiBuZXdPYmouc3VwcG9ydHMgPT09IG9iai5zdXBwb3J0cyAmJiBuZXdPYmoubGF5ZXIgPT09IG9iai5sYXllcikge1xuICAgICAgICByZXR1cm47XG4gICAgICB9XG4gICAgICBhcGkudXBkYXRlKG9iaiA9IG5ld09iaik7XG4gICAgfSBlbHNlIHtcbiAgICAgIGFwaS5yZW1vdmUoKTtcbiAgICB9XG4gIH07XG4gIHJldHVybiB1cGRhdGVyO1xufVxubW9kdWxlLmV4cG9ydHMgPSBmdW5jdGlvbiAobGlzdCwgb3B0aW9ucykge1xuICBvcHRpb25zID0gb3B0aW9ucyB8fCB7fTtcbiAgbGlzdCA9IGxpc3QgfHwgW107XG4gIHZhciBsYXN0SWRlbnRpZmllcnMgPSBtb2R1bGVzVG9Eb20obGlzdCwgb3B0aW9ucyk7XG4gIHJldHVybiBmdW5jdGlvbiB1cGRhdGUobmV3TGlzdCkge1xuICAgIG5ld0xpc3QgPSBuZXdMaXN0IHx8IFtdO1xuICAgIGZvciAodmFyIGkgPSAwOyBpIDwgbGFzdElkZW50aWZpZXJzLmxlbmd0aDsgaSsrKSB7XG4gICAgICB2YXIgaWRlbnRpZmllciA9IGxhc3RJZGVudGlmaWVyc1tpXTtcbiAgICAgIHZhciBpbmRleCA9IGdldEluZGV4QnlJZGVudGlmaWVyKGlkZW50aWZpZXIpO1xuICAgICAgc3R5bGVzSW5ET01baW5kZXhdLnJlZmVyZW5jZXMtLTtcbiAgICB9XG4gICAgdmFyIG5ld0xhc3RJZGVudGlmaWVycyA9IG1vZHVsZXNUb0RvbShuZXdMaXN0LCBvcHRpb25zKTtcbiAgICBmb3IgKHZhciBfaSA9IDA7IF9pIDwgbGFzdElkZW50aWZpZXJzLmxlbmd0aDsgX2krKykge1xuICAgICAgdmFyIF9pZGVudGlmaWVyID0gbGFzdElkZW50aWZpZXJzW19pXTtcbiAgICAgIHZhciBfaW5kZXggPSBnZXRJbmRleEJ5SWRlbnRpZmllcihfaWRlbnRpZmllcik7XG4gICAgICBpZiAoc3R5bGVzSW5ET01bX2luZGV4XS5yZWZlcmVuY2VzID09PSAwKSB7XG4gICAgICAgIHN0eWxlc0luRE9NW19pbmRleF0udXBkYXRlcigpO1xuICAgICAgICBzdHlsZXNJbkRPTS5zcGxpY2UoX2luZGV4LCAxKTtcbiAgICAgIH1cbiAgICB9XG4gICAgbGFzdElkZW50aWZpZXJzID0gbmV3TGFzdElkZW50aWZpZXJzO1xuICB9O1xufTsiXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///72\n')},659:module=>{eval('\n\nvar memo = {};\n\n/* istanbul ignore next */\nfunction getTarget(target) {\n if (typeof memo[target] === "undefined") {\n var styleTarget = document.querySelector(target);\n\n // Special case to return head of iframe instead of iframe itself\n if (window.HTMLIFrameElement && styleTarget instanceof window.HTMLIFrameElement) {\n try {\n // This will throw an exception if access to iframe is blocked\n // due to cross-origin restrictions\n styleTarget = styleTarget.contentDocument.head;\n } catch (e) {\n // istanbul ignore next\n styleTarget = null;\n }\n }\n memo[target] = styleTarget;\n }\n return memo[target];\n}\n\n/* istanbul ignore next */\nfunction insertBySelector(insert, style) {\n var target = getTarget(insert);\n if (!target) {\n throw new Error("Couldn\'t find a style target. This probably means that the value for the \'insert\' parameter is invalid.");\n }\n target.appendChild(style);\n}\nmodule.exports = insertBySelector;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiNjU5LmpzIiwibWFwcGluZ3MiOiJBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYnVkZ2V0LXJlcXVlc3QtZm9ybS8uL25vZGVfbW9kdWxlcy9zdHlsZS1sb2FkZXIvZGlzdC9ydW50aW1lL2luc2VydEJ5U2VsZWN0b3IuanM/YjIxNCJdLCJzb3VyY2VzQ29udGVudCI6WyJcInVzZSBzdHJpY3RcIjtcblxudmFyIG1lbW8gPSB7fTtcblxuLyogaXN0YW5idWwgaWdub3JlIG5leHQgICovXG5mdW5jdGlvbiBnZXRUYXJnZXQodGFyZ2V0KSB7XG4gIGlmICh0eXBlb2YgbWVtb1t0YXJnZXRdID09PSBcInVuZGVmaW5lZFwiKSB7XG4gICAgdmFyIHN0eWxlVGFyZ2V0ID0gZG9jdW1lbnQucXVlcnlTZWxlY3Rvcih0YXJnZXQpO1xuXG4gICAgLy8gU3BlY2lhbCBjYXNlIHRvIHJldHVybiBoZWFkIG9mIGlmcmFtZSBpbnN0ZWFkIG9mIGlmcmFtZSBpdHNlbGZcbiAgICBpZiAod2luZG93LkhUTUxJRnJhbWVFbGVtZW50ICYmIHN0eWxlVGFyZ2V0IGluc3RhbmNlb2Ygd2luZG93LkhUTUxJRnJhbWVFbGVtZW50KSB7XG4gICAgICB0cnkge1xuICAgICAgICAvLyBUaGlzIHdpbGwgdGhyb3cgYW4gZXhjZXB0aW9uIGlmIGFjY2VzcyB0byBpZnJhbWUgaXMgYmxvY2tlZFxuICAgICAgICAvLyBkdWUgdG8gY3Jvc3Mtb3JpZ2luIHJlc3RyaWN0aW9uc1xuICAgICAgICBzdHlsZVRhcmdldCA9IHN0eWxlVGFyZ2V0LmNvbnRlbnREb2N1bWVudC5oZWFkO1xuICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICAvLyBpc3RhbmJ1bCBpZ25vcmUgbmV4dFxuICAgICAgICBzdHlsZVRhcmdldCA9IG51bGw7XG4gICAgICB9XG4gICAgfVxuICAgIG1lbW9bdGFyZ2V0XSA9IHN0eWxlVGFyZ2V0O1xuICB9XG4gIHJldHVybiBtZW1vW3RhcmdldF07XG59XG5cbi8qIGlzdGFuYnVsIGlnbm9yZSBuZXh0ICAqL1xuZnVuY3Rpb24gaW5zZXJ0QnlTZWxlY3RvcihpbnNlcnQsIHN0eWxlKSB7XG4gIHZhciB0YXJnZXQgPSBnZXRUYXJnZXQoaW5zZXJ0KTtcbiAgaWYgKCF0YXJnZXQpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoXCJDb3VsZG4ndCBmaW5kIGEgc3R5bGUgdGFyZ2V0LiBUaGlzIHByb2JhYmx5IG1lYW5zIHRoYXQgdGhlIHZhbHVlIGZvciB0aGUgJ2luc2VydCcgcGFyYW1ldGVyIGlzIGludmFsaWQuXCIpO1xuICB9XG4gIHRhcmdldC5hcHBlbmRDaGlsZChzdHlsZSk7XG59XG5tb2R1bGUuZXhwb3J0cyA9IGluc2VydEJ5U2VsZWN0b3I7Il0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///659\n')},540:module=>{eval('\n\n/* istanbul ignore next */\nfunction insertStyleElement(options) {\n var element = document.createElement("style");\n options.setAttributes(element, options.attributes);\n options.insert(element, options.options);\n return element;\n}\nmodule.exports = insertStyleElement;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiNTQwLmpzIiwibWFwcGluZ3MiOiJBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYnVkZ2V0LXJlcXVlc3QtZm9ybS8uL25vZGVfbW9kdWxlcy9zdHlsZS1sb2FkZXIvZGlzdC9ydW50aW1lL2luc2VydFN0eWxlRWxlbWVudC5qcz9kZTZjIl0sInNvdXJjZXNDb250ZW50IjpbIlwidXNlIHN0cmljdFwiO1xuXG4vKiBpc3RhbmJ1bCBpZ25vcmUgbmV4dCAgKi9cbmZ1bmN0aW9uIGluc2VydFN0eWxlRWxlbWVudChvcHRpb25zKSB7XG4gIHZhciBlbGVtZW50ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcInN0eWxlXCIpO1xuICBvcHRpb25zLnNldEF0dHJpYnV0ZXMoZWxlbWVudCwgb3B0aW9ucy5hdHRyaWJ1dGVzKTtcbiAgb3B0aW9ucy5pbnNlcnQoZWxlbWVudCwgb3B0aW9ucy5vcHRpb25zKTtcbiAgcmV0dXJuIGVsZW1lbnQ7XG59XG5tb2R1bGUuZXhwb3J0cyA9IGluc2VydFN0eWxlRWxlbWVudDsiXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///540\n')},56:(module,__unused_webpack_exports,__webpack_require__)=>{eval('\n\n/* istanbul ignore next */\nfunction setAttributesWithoutAttributes(styleElement) {\n var nonce = true ? __webpack_require__.nc : 0;\n if (nonce) {\n styleElement.setAttribute("nonce", nonce);\n }\n}\nmodule.exports = setAttributesWithoutAttributes;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiNTYuanMiLCJtYXBwaW5ncyI6IkFBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9idWRnZXQtcmVxdWVzdC1mb3JtLy4vbm9kZV9tb2R1bGVzL3N0eWxlLWxvYWRlci9kaXN0L3J1bnRpbWUvc2V0QXR0cmlidXRlc1dpdGhvdXRBdHRyaWJ1dGVzLmpzP2RkY2UiXSwic291cmNlc0NvbnRlbnQiOlsiXCJ1c2Ugc3RyaWN0XCI7XG5cbi8qIGlzdGFuYnVsIGlnbm9yZSBuZXh0ICAqL1xuZnVuY3Rpb24gc2V0QXR0cmlidXRlc1dpdGhvdXRBdHRyaWJ1dGVzKHN0eWxlRWxlbWVudCkge1xuICB2YXIgbm9uY2UgPSB0eXBlb2YgX193ZWJwYWNrX25vbmNlX18gIT09IFwidW5kZWZpbmVkXCIgPyBfX3dlYnBhY2tfbm9uY2VfXyA6IG51bGw7XG4gIGlmIChub25jZSkge1xuICAgIHN0eWxlRWxlbWVudC5zZXRBdHRyaWJ1dGUoXCJub25jZVwiLCBub25jZSk7XG4gIH1cbn1cbm1vZHVsZS5leHBvcnRzID0gc2V0QXR0cmlidXRlc1dpdGhvdXRBdHRyaWJ1dGVzOyJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///56\n')},825:module=>{eval('\n\n/* istanbul ignore next */\nfunction apply(styleElement, options, obj) {\n var css = "";\n if (obj.supports) {\n css += "@supports (".concat(obj.supports, ") {");\n }\n if (obj.media) {\n css += "@media ".concat(obj.media, " {");\n }\n var needLayer = typeof obj.layer !== "undefined";\n if (needLayer) {\n css += "@layer".concat(obj.layer.length > 0 ? " ".concat(obj.layer) : "", " {");\n }\n css += obj.css;\n if (needLayer) {\n css += "}";\n }\n if (obj.media) {\n css += "}";\n }\n if (obj.supports) {\n css += "}";\n }\n var sourceMap = obj.sourceMap;\n if (sourceMap && typeof btoa !== "undefined") {\n css += "\\n/*# sourceMappingURL=data:application/json;base64,".concat(btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap)))), " */");\n }\n\n // For old IE\n /* istanbul ignore if */\n options.styleTagTransform(css, styleElement, options.options);\n}\nfunction removeStyleElement(styleElement) {\n // istanbul ignore if\n if (styleElement.parentNode === null) {\n return false;\n }\n styleElement.parentNode.removeChild(styleElement);\n}\n\n/* istanbul ignore next */\nfunction domAPI(options) {\n if (typeof document === "undefined") {\n return {\n update: function update() {},\n remove: function remove() {}\n };\n }\n var styleElement = options.insertStyleElement(options);\n return {\n update: function update(obj) {\n apply(styleElement, options, obj);\n },\n remove: function remove() {\n removeStyleElement(styleElement);\n }\n };\n}\nmodule.exports = domAPI;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiODI1LmpzIiwibWFwcGluZ3MiOiJBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYnVkZ2V0LXJlcXVlc3QtZm9ybS8uL25vZGVfbW9kdWxlcy9zdHlsZS1sb2FkZXIvZGlzdC9ydW50aW1lL3N0eWxlRG9tQVBJLmpzP2U0NzkiXSwic291cmNlc0NvbnRlbnQiOlsiXCJ1c2Ugc3RyaWN0XCI7XG5cbi8qIGlzdGFuYnVsIGlnbm9yZSBuZXh0ICAqL1xuZnVuY3Rpb24gYXBwbHkoc3R5bGVFbGVtZW50LCBvcHRpb25zLCBvYmopIHtcbiAgdmFyIGNzcyA9IFwiXCI7XG4gIGlmIChvYmouc3VwcG9ydHMpIHtcbiAgICBjc3MgKz0gXCJAc3VwcG9ydHMgKFwiLmNvbmNhdChvYmouc3VwcG9ydHMsIFwiKSB7XCIpO1xuICB9XG4gIGlmIChvYmoubWVkaWEpIHtcbiAgICBjc3MgKz0gXCJAbWVkaWEgXCIuY29uY2F0KG9iai5tZWRpYSwgXCIge1wiKTtcbiAgfVxuICB2YXIgbmVlZExheWVyID0gdHlwZW9mIG9iai5sYXllciAhPT0gXCJ1bmRlZmluZWRcIjtcbiAgaWYgKG5lZWRMYXllcikge1xuICAgIGNzcyArPSBcIkBsYXllclwiLmNvbmNhdChvYmoubGF5ZXIubGVuZ3RoID4gMCA/IFwiIFwiLmNvbmNhdChvYmoubGF5ZXIpIDogXCJcIiwgXCIge1wiKTtcbiAgfVxuICBjc3MgKz0gb2JqLmNzcztcbiAgaWYgKG5lZWRMYXllcikge1xuICAgIGNzcyArPSBcIn1cIjtcbiAgfVxuICBpZiAob2JqLm1lZGlhKSB7XG4gICAgY3NzICs9IFwifVwiO1xuICB9XG4gIGlmIChvYmouc3VwcG9ydHMpIHtcbiAgICBjc3MgKz0gXCJ9XCI7XG4gIH1cbiAgdmFyIHNvdXJjZU1hcCA9IG9iai5zb3VyY2VNYXA7XG4gIGlmIChzb3VyY2VNYXAgJiYgdHlwZW9mIGJ0b2EgIT09IFwidW5kZWZpbmVkXCIpIHtcbiAgICBjc3MgKz0gXCJcXG4vKiMgc291cmNlTWFwcGluZ1VSTD1kYXRhOmFwcGxpY2F0aW9uL2pzb247YmFzZTY0LFwiLmNvbmNhdChidG9hKHVuZXNjYXBlKGVuY29kZVVSSUNvbXBvbmVudChKU09OLnN0cmluZ2lmeShzb3VyY2VNYXApKSkpLCBcIiAqL1wiKTtcbiAgfVxuXG4gIC8vIEZvciBvbGQgSUVcbiAgLyogaXN0YW5idWwgaWdub3JlIGlmICAqL1xuICBvcHRpb25zLnN0eWxlVGFnVHJhbnNmb3JtKGNzcywgc3R5bGVFbGVtZW50LCBvcHRpb25zLm9wdGlvbnMpO1xufVxuZnVuY3Rpb24gcmVtb3ZlU3R5bGVFbGVtZW50KHN0eWxlRWxlbWVudCkge1xuICAvLyBpc3RhbmJ1bCBpZ25vcmUgaWZcbiAgaWYgKHN0eWxlRWxlbWVudC5wYXJlbnROb2RlID09PSBudWxsKSB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG4gIHN0eWxlRWxlbWVudC5wYXJlbnROb2RlLnJlbW92ZUNoaWxkKHN0eWxlRWxlbWVudCk7XG59XG5cbi8qIGlzdGFuYnVsIGlnbm9yZSBuZXh0ICAqL1xuZnVuY3Rpb24gZG9tQVBJKG9wdGlvbnMpIHtcbiAgaWYgKHR5cGVvZiBkb2N1bWVudCA9PT0gXCJ1bmRlZmluZWRcIikge1xuICAgIHJldHVybiB7XG4gICAgICB1cGRhdGU6IGZ1bmN0aW9uIHVwZGF0ZSgpIHt9LFxuICAgICAgcmVtb3ZlOiBmdW5jdGlvbiByZW1vdmUoKSB7fVxuICAgIH07XG4gIH1cbiAgdmFyIHN0eWxlRWxlbWVudCA9IG9wdGlvbnMuaW5zZXJ0U3R5bGVFbGVtZW50KG9wdGlvbnMpO1xuICByZXR1cm4ge1xuICAgIHVwZGF0ZTogZnVuY3Rpb24gdXBkYXRlKG9iaikge1xuICAgICAgYXBwbHkoc3R5bGVFbGVtZW50LCBvcHRpb25zLCBvYmopO1xuICAgIH0sXG4gICAgcmVtb3ZlOiBmdW5jdGlvbiByZW1vdmUoKSB7XG4gICAgICByZW1vdmVTdHlsZUVsZW1lbnQoc3R5bGVFbGVtZW50KTtcbiAgICB9XG4gIH07XG59XG5tb2R1bGUuZXhwb3J0cyA9IGRvbUFQSTsiXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///825\n')},113:module=>{eval("\n\n/* istanbul ignore next */\nfunction styleTagTransform(css, styleElement) {\n if (styleElement.styleSheet) {\n styleElement.styleSheet.cssText = css;\n } else {\n while (styleElement.firstChild) {\n styleElement.removeChild(styleElement.firstChild);\n }\n styleElement.appendChild(document.createTextNode(css));\n }\n}\nmodule.exports = styleTagTransform;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMTEzLmpzIiwibWFwcGluZ3MiOiJBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9idWRnZXQtcmVxdWVzdC1mb3JtLy4vbm9kZV9tb2R1bGVzL3N0eWxlLWxvYWRlci9kaXN0L3J1bnRpbWUvc3R5bGVUYWdUcmFuc2Zvcm0uanM/MWRkZSJdLCJzb3VyY2VzQ29udGVudCI6WyJcInVzZSBzdHJpY3RcIjtcblxuLyogaXN0YW5idWwgaWdub3JlIG5leHQgICovXG5mdW5jdGlvbiBzdHlsZVRhZ1RyYW5zZm9ybShjc3MsIHN0eWxlRWxlbWVudCkge1xuICBpZiAoc3R5bGVFbGVtZW50LnN0eWxlU2hlZXQpIHtcbiAgICBzdHlsZUVsZW1lbnQuc3R5bGVTaGVldC5jc3NUZXh0ID0gY3NzO1xuICB9IGVsc2Uge1xuICAgIHdoaWxlIChzdHlsZUVsZW1lbnQuZmlyc3RDaGlsZCkge1xuICAgICAgc3R5bGVFbGVtZW50LnJlbW92ZUNoaWxkKHN0eWxlRWxlbWVudC5maXJzdENoaWxkKTtcbiAgICB9XG4gICAgc3R5bGVFbGVtZW50LmFwcGVuZENoaWxkKGRvY3VtZW50LmNyZWF0ZVRleHROb2RlKGNzcykpO1xuICB9XG59XG5tb2R1bGUuZXhwb3J0cyA9IHN0eWxlVGFnVHJhbnNmb3JtOyJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///113\n")}},__webpack_module_cache__={};function __webpack_require__(n){var c=__webpack_module_cache__[n];if(void 0!==c)return c.exports;var t=__webpack_module_cache__[n]={id:n,exports:{}};return __webpack_modules__[n](t,t.exports,__webpack_require__),t.exports}__webpack_require__.n=n=>{var c=n&&n.__esModule?()=>n.default:()=>n;return __webpack_require__.d(c,{a:c}),c},__webpack_require__.d=(n,c)=>{for(var t in c)__webpack_require__.o(c,t)&&!__webpack_require__.o(n,t)&&Object.defineProperty(n,t,{enumerable:!0,get:c[t]})},__webpack_require__.o=(n,c)=>Object.prototype.hasOwnProperty.call(n,c),__webpack_require__.nc=void 0;var __webpack_exports__=__webpack_require__(863)})();
\ No newline at end of file
diff --git a/docs/d027d45141f87dc014af.png b/docs/d027d45141f87dc014af.png
new file mode 100644
index 0000000..856d278
Binary files /dev/null and b/docs/d027d45141f87dc014af.png differ
diff --git a/docs/index.html b/docs/index.html
new file mode 100644
index 0000000..16786ec
--- /dev/null
+++ b/docs/index.html
@@ -0,0 +1 @@
+Budget Request Form