\n
\n ${arg.event.title} ${ arg.event.extendedProps.onTrial ? `T` : \"\" }\n ${ renderSharedStatus(arg.event.extendedProps.sharedWithClient) }\n
\n
\n `\n };\n },\n eventClick: function(info) {\n const event = info.event;\n $.ajax({\n method: \"GET\",\n url: `/admin/assessments/${event.id}`,\n dataType: \"script\"\n })\n }\n })\n calendar.render()\n}\n\nfunction renderSharedStatus(status) {\n if(!status) return ''\n if(status == 'shared') return ``\n return ``\n}\n\nfunction parseResponse(response) {\n const events = [];\n \n response.forEach(item => {\n const event = {\n id: item.id,\n backgroundColor: item.backgroundColor,\n borderColor: item.borderColor,\n textColor: item.textColor,\n title: item.title,\n start: item.start,\n extendedProps: {\n subtitle: item.subtitle,\n statusClass: item.statusClass,\n onTrial: item.onTrial,\n finishedTrial: item.finishedTrial,\n sharedWithClient: item.sharedWithClient\n }\n };\n \n events.push(event);\n });\n \n return events;\n}\n\n","window.initAssessorResourcesForm = function() {\n tinyMceFull();\n\n $(\".js-resource-type\").on('change', function(){\n $('[data-type]').addClass('is-hidden');\n $(`[data-type=\"${$(this).val()}\"]`).removeClass('is-hidden');\n })\n}\n","window.scrollToError = function() {\n const firstError = $('.error-message:first');\n if(firstError.length > 0){\n window.scrollTo({ top: $(firstError).offset().top - (window.innerHeight / 3), behavior: 'smooth' });\n }\n}\n\nwindow.scrollToChapter = function(id) {\n const chapter = $(`#${id}`);\n if(chapter.length > 0){\n window.scrollTo({ top: $(chapter).offset().top - (window.innerHeight / 6), behavior: 'smooth' });\n }\n}\n\nwindow.scrollToHighlighted = function(id) {\n const firstHighlighted = $('.is-highlighted:first');\n if(firstHighlighted.length > 0){\n window.scrollTo({ top: $(firstHighlighted).offset().top - (window.innerHeight / 3), behavior: 'smooth' });\n }\n}\n\nwindow.scrollToRefutable = function(dom_id) {\n const refutable = $(`[data-refutable=\"${dom_id}\"]`);\n if(refutable.length > 0){\n window.scrollTo({ top: $(refutable).offset().top - (window.innerHeight / 3), behavior: 'smooth' });\n }\n}\n\n","window.initTrainingItemQuestionForm = function() {\n initFormSort();\n $(document).on('change', '.js-correct', function(){\n if($(this).is(':checked')){\n $('.js-correct').not(this).prop('checked', false);\n } else {\n $(this).prop('checked', true);\n }\n })\n}","window.renderFileName = function(file_field){\n $('#file-name').text(file_field.files.item(0).name.replace(/ /g,\"_\"));\n}","window.adaptResidenceFilters = function(){\n if($('.js-city').val() == \"\"){\n $('[data-city]').show();\n }\n\n if($('.js-country').val() == \"\"){\n $('[data-country]').show();\n }\n\n if($('.js-country').val()){\n let pickedCountry = $('.js-country').val()\n\n $('[data-country]').hide();\n $(`[data-country=\"${pickedCountry}\"]`).show();\n }\n\n if($('.js-city').val()){\n let pickedCity = $('.js-city').val()\n\n $('[data-city]').hide();\n $(`[data-city=\"${pickedCity}\"]`).show();\n }\n}","window.initSettingAssessorDocumentsForm = function(){\n\t$('[name=\"setting_assessor_document[document_type]\"]').on('change', function(){\n\t\t$('.js-type-field').addClass('is-hidden');\n\t\t$('.js-type-field').find('input').prop('disabled', true);\n\t\tif($(this).val() == 'pandadoc'){\n\t\t\t$('.js-pandadoc-field').removeClass('is-hidden');\n\t\t\t$('.js-pandadoc-field').find('input').prop('disabled', false);\n\t\t} else if($(this).val() == 'custom') {\n\t\t\t$('.js-custom-field').removeClass('is-hidden');\n\t\t\t$('.js-custom-field').find('input').prop('disabled', false);\n\t\t}\n\t})\n\ttinyMceText();\n}","window.initClientAccountForm = function() {\n $('[name=\"client_account[brand_access]\"]').on('change', function(){\n if($(this).is(':checked')){\n $('.js-clients').addClass('is-hidden');\n } else {\n $('.js-clients').removeClass('is-hidden');\n }\n })\n}","window.initRequestFromParam = function(path, id_param) {\n if(getParameterByName(id_param)){\n $.get(`${path}/${getParameterByName(id_param)}`);\n }\n}\n\n","window.initClientExternalLinkForm = function() {\n $(\".js-link-for\").on('change', function(){\n $('[data-for]').addClass('is-hidden');\n $(`[data-for=\"${$(this).val()}\"]`).removeClass('is-hidden');\n })\n}\n","window.initRefutable = function(){\n $('.refute-indicator').hover(\n function(){\n $('[data-refutable]').removeClass('is-selected');\n $(`[data-refutable=\"${$(this).attr('data-refute-for')}\"]`).addClass('is-selected');\n },\n function(){\n $('[data-refutable]').removeClass('is-selected');\n }\n )\n}\n","window.changeExpensesNoticeButtonDisabledState = function(){\n $('#read_expenses_notice').prop('disabled', !$('#i_understand').is(':checked'));\n}\n","window.initPayoutXLSXExportForm = function(){\n $('[name=\"columns[]\"]').on('change', function(){\n console.log($('[name=\"columns[]\"]:checked').length);\n $('#submit').prop('disabled', $('[name=\"columns[]\"]:checked').length < 1)\n })\n}","window.initRequestPayoutForm = function() {\n\tsetCurrency();\n\t$('[name=\"assessor_payout[assessment_id]\"]').on('change', function(){\n\t\tsetCurrency();\n\t})\n}\n\nfunction setCurrency(default_currency='AED') {\n\tlet currency = $('[name=\"assessor_payout[assessment_id]\"]').find('option:selected').attr('data-currency');\n\tif(currency) {\n\t\t$('.js-currency').text(currency);\n\t} else {\n\t\t$('.js-currency').text(default_currency);\n\t}\n}","document.addEventListener(\"turbolinks:load\", function() {\n const activeTab = document.querySelector('.tabs__button.is-active');\n const scrollContainer = document.querySelector('.tabs__button-wrapper');\n \n if (activeTab && scrollContainer) {\n const tabOffset = activeTab.getBoundingClientRect().left;\n const containerOffset = scrollContainer.getBoundingClientRect().left;\n const offset = tabOffset - containerOffset - 16; // Assuming 1rem = 16px\n scrollContainer.scrollLeft = offset;\n }\n});\n","// wiki: https://gitlab.com/bamboolab/bamboo_datatable\nexport class BambooDatatable {\n constructor(table, options = {}) {\n this.table = table;\n // this.className = options.className || 'datatable';\n this.tableClassName = options.tableClassName || \"js-datatable__table\";\n this.theadClassName = options.theadClassName || \"js-datatable__head\";\n this.tbodyClassName = options.tbodyClassName || \"js-datatable__body\";\n this.rowClassName = options.rowClassName || \"js-row\";\n this.cellClassName = options.cellClassName || \"js-cell\";\n this.searchClassName = options.searchClassName || \"js-datatable__search\";\n this.paginationClassName = options.paginationClassName || \"js-datatable__pagination\";\n this.paginationTag = options.paginationTag || \"a\";\n this.perPage = options.perPage || 5;\n this.paginationPreviousText = options.paginationPreviousText || \"‹\";\n this.paginationNextText = options.paginationNextText || \"›\";\n\n // commonClasses are used for element creation\n this.commonClasses = {\n tableClassName: document.querySelector(`.${this.tableClassName}`).className,\n theadClassName: document.querySelector(`.${this.theadClassName}`).className,\n tbodyClassName: document.querySelector(`.${this.tbodyClassName}`).className,\n rowClassName: document.querySelector(`.${this.rowClassName}`).className,\n cellClassName: document.querySelector(`.${this.cellClassName}`).className,\n searchClassName: document.querySelector(`.${this.searchClassName}`).className,\n paginationClassName: document.querySelector(`.${this.paginationClassName}`).className\n }\n\n this.ignore_order = \"js-ignore_order\";\n\n this.search = options.search || false;\n this.searchKeyword = \"\";\n this.order = options.order || false;\n this.orderByColumn = \"\";\n this.orderDirection = \"\";\n this.pagination = options.pagination || false;\n\n this.remote = (options.remoteUrl && options.remoteUrl.length); // add ajax call and display only returned data\n this.remoteUrl = options.remoteUrl;\n this.currentPage = options.currentPage || 1;\n this.totalPages = options.totalPages; // controller can determine the number off pages\n \n this.disableFetch = false;\n\n // strings for queries\n this.queries = {\n pagination: `.${this.paginationClassName}`,\n paginationNumbers: `.${this.paginationClassName} > .pagination_numbers`,\n activePaginationNumber: (page) =>\n `${this.queries.paginationNumbers}[data-show-page='${page}']`,\n searchInput: `.${this.searchClassName} > input[type=text]`,\n rows: (headOrBody) =>\n `.${this.tableClassName} > .${headOrBody} > .${this.rowClassName}`,\n cells: (ignore) => `.${this.cellClassName}:not(.${ignore})`,\n xBttn: `.${this.searchClassName} > .js-clear`,\n tBody: `.${this.tableClassName} > .${this.tbodyClassName}`,\n cell: `.${this.cellClassName}`,\n legendSpan: (spanClassName) =>\n `.${this.cellClassName}:not(.${this.ignore_order}) .${spanClassName}`,\n };\n\n this.columnPositions = this.getColumnPositions(); \n\n this.rowType = \"ul\"; \n this.celltype = \"li\";\n this.setRowAndCellType();\n\n // run main methods\n if (this.search) {\n this.addSearch();\n }\n if (this.order) {\n this.addOrder();\n }\n if (this.pagination) {\n this.addPagination();\n }\n if (this) {\n this.addMobileLabels();\n }\n }\n\n // MAIN METHODS\n addSearch() {\n var input = this.table.querySelector(this.queries.searchInput);\n input.onkeyup = (event) => {\n var keyword = input.value.toLowerCase();\n this.searchKeyword = keyword;\n this.runSearch();\n };\n this.addXBttn(input);\n }\n\n addOrder() {\n var orderRows = this.findRows(this.theadClassName);\n var orderCells = this.findOrderCells(orderRows);\n\n this.addDefaultOrder(orderCells);\n }\n\n addPagination() {\n var visibleRows = this.findRows(this.tbodyClassName);\n this.runPagination(visibleRows);\n }\n\n // SHARDE METHODS\n findRows(headOrBody) {\n return Array.from(\n this.table.querySelectorAll(this.queries.rows(headOrBody))\n );\n }\n\n getColumnPositions() {\n var positions = [];\n this.findRows(this.theadClassName).forEach((row) => {\n var columns = row.children;\n for (var i =0; i < columns.length; i++) {\n positions.push(columns[i].dataset.name)\n }\n });\n return positions;\n }\n\n setRowAndCellType() {\n // row and cell types are default ul and li but setRowAndCellType will find the first row and first cell and get their types\n var allRows = this.table.querySelectorAll(this.queries.rows(this.tbodyClassName));\n if (allRows && allRows.length) {\n this.rowType = allRows[0].tagName;\n var cells = allRows[0].children;\n if (cells && cells.length) {\n this.celltype = cells[0].tagName;\n }\n }\n }\n\n createRows(rows) {\n // used for ajax\n var visibleRows = [];\n for (let r in rows) {\n var row = rows[r];\n var rowElement = document.createElement(this.rowType);\n rowElement.className = this.commonClasses.rowClassName;\n\n if (row instanceof Array) {\n // create columns\n for (let c in row){\n var cell = row[c]\n var cellElement = document.createElement(this.celltype );\n\n if (typeof cell === 'object' && !Array.isArray(cell) && cell !== null) {\n cellElement.dataset.order = cell.order;\n cellElement.innerHTML = cell.text;\n } else {\n cellElement.innerHTML = cell;\n }\n\n cellElement.className = this.commonClasses.cellClassName;\n rowElement.appendChild(cellElement); \n }\n } else {\n // create columns\n for (var c = 0; c < this.columnPositions.length; c++){\n var cell = row[this.columnPositions[c]];\n var cellElement = document.createElement(this.celltype );\n cellElement.innerHTML = cell;\n cellElement.className = this.commonClasses.cellClassName;\n rowElement.appendChild(cellElement); \n }\n }\n visibleRows.push(rowElement);\n }\n return visibleRows;\n }\n\n filterVisible(element) {\n return (\n window.getComputedStyle(element).getPropertyValue(\"display\") != \"none\"\n );\n }\n\n // METHODS FOR SEARCHING\n hideAll(arr) {\n for (var item in arr) {\n arr[item].style.display = \"none\";\n }\n }\n showAll(arr) {\n for (var item in arr) {\n arr[item].style.display = \"\";\n }\n }\n\n runSearch() {\n if (this.remote) {\n this.removeAllFromDom(this.queries.tBody);\n this.ajaxCall(1); \n } else {\n var searchableRows = this.findRows(this.tbodyClassName);\n this.hideAll(searchableRows);\n var rowContainingKeyword = this.findRowContaining(searchableRows);\n this.showAll(rowContainingKeyword);\n if (this.pagination) {\n this.runPagination(rowContainingKeyword);\n }\n this.currentPage = 1;\n }\n }\n\n findRowContaining(searchableRows) {\n var keyword = this.searchKeyword;\n var foundRows = [];\n for (var i in searchableRows) {\n var row = searchableRows[i];\n var cells = row.querySelectorAll(this.queries.cells(\"ignore_search\"));\n var cellsContainingKeyword = this.contains(cells, keyword);\n var cellsContainingKeywordParents = [];\n for (var j in cellsContainingKeyword) {\n var cell = cellsContainingKeyword[j];\n var safeExit = 10;\n /// climb the DOM up to the right parent (default .row)\n while (cell && safeExit > 0) {\n cellsContainingKeywordParents.unshift(cell);\n cell = cell.parentElement;\n safeExit--;\n\n if (cell && cell.classList.contains(`${this.rowClassName}`)) {\n foundRows.push(cell);\n break;\n }\n }\n }\n }\n return foundRows.filter(this.onlyUnique);\n }\n contains(cells, keyword) {\n /*\n contains('div', 'sometext'); // find \"div\" that contain \"sometext\"\n contains('div', /^sometext/); // find \"div\" that start with \"sometext\"\n contains('div', /sometext$/i); // find \"div\" that end with \"sometext\", case-insensitive\n */\n return Array.prototype.filter.call(cells, (cell) => {\n var cellContent = cell.textContent.replace(/ /g, \"\").toLowerCase();\n var keywordContent = keyword.replace(/ /g, \"\").toLowerCase();\n return RegExp(keywordContent).test(cellContent);\n });\n }\n onlyUnique(value, index, self) {\n return self.indexOf(value) === index;\n }\n\n addXBttn(input) {\n var xBttn = this.table.querySelector(this.queries.xBttn);\n xBttn.onclick = (event) => {\n input.value = \"\";\n this.searchKeyword = \"\";\n this.runSearch();\n };\n }\n\n // METHODS FOR ORDERING\n findOrderCells(orderRows) {\n var orderRow = this.getOrderRow(orderRows);\n var cells = Array.from(\n orderRow.querySelectorAll(this.queries.cells(`${this.ignore_order}`))\n );\n for (var i in cells) {\n this.addOrderEvent(cells[i], i);\n }\n return cells;\n }\n\n addOrderEvent(cell, i) {\n cell.onclick = (event) => {\n if (!this.remote) {\n this.currentPage = 1;\n }\n\n this.changeOrderType(cell);\n this.reorderBy(cell, i);\n\n this.orderDirection = cell.dataset.order;\n this.orderByColumn = cell.dataset.name;\n\n if (this.remote) { \n // run sort and show pagination page 1\n this.ajaxCall(1); \n };\n };\n }\n\n addDefaultOrder(orderCells) {\n for (var i in orderCells) {\n this.changeOrderType(orderCells[i]);\n }\n }\n\n changeOrderType(cell) {\n if (cell.dataset.order == \"asc\") {\n cell.dataset.order = \"desc\";\n } else {\n cell.dataset.order = \"asc\";\n }\n }\n\n reorderBy(cell, index) {\n var direction = cell.dataset.order;\n var index = index;\n\n this.addOrderLegend(cell, direction);\n\n var orderableRows = this.findRows(this.tbodyClassName);\n\n var newTBodyRows = this.getNewOrder(orderableRows, index);\n var newTBodyRows = this.applyDirection(newTBodyRows, direction);\n\n // need to show all for pagination\n if (this.pagination) {\n this.showAll(orderableRows);\n }\n\n // empty and add new rows\n this.table.querySelector(this.queries.tBody).innerHTML = \"\";\n for (var i in newTBodyRows) {\n var row = newTBodyRows[i];\n this.table.querySelector(this.queries.tBody).appendChild(row);\n }\n\n if (this.pagination) {\n this.runPagination(newTBodyRows);\n }\n\n if (this.search) {\n var input = this.table.querySelector(this.queries.searchInput);\n var keyword = input.value.toLowerCase();\n this.searchKeyword = keyword;\n // this.runSearch(); <-- commented bc it breaks order - makes double request\n }\n }\n\n getNewOrder(rows, index) {\n return rows.sort((a, b) => {\n var nameA = this.getOrderValues(a, index);\n var nameB = this.getOrderValues(b, index);\n if (nameA < nameB) {\n return -1;\n }\n if (nameA > nameB) {\n return 1;\n }\n return 0; // names must be equal\n });\n }\n\n getOrderValues(x, index) {\n var cell = x.querySelectorAll(this.queries.cell)[index];\n if (cell.dataset.order) {\n // order by custom value\n return cell.dataset.order.toUpperCase();\n } else {\n // order by text\n return cell.textContent.toUpperCase();\n }\n }\n\n applyDirection(rows, direction) {\n if (direction == \"desc\") {\n return (rows = rows.reverse());\n } else {\n return rows;\n }\n }\n\n addOrderLegend(cell, direction) {\n var spanClassName = \"datatable__icon\";\n\n var arrows = {\n up: \"↑\",\n down: \"↓\",\n };\n\n var orderRows = this.findRows(this.theadClassName);\n var orderRow = this.getOrderRow(orderRows);\n var legendSpans = orderRow.querySelectorAll(\n this.queries.legendSpan(spanClassName)\n );\n for (const s of legendSpans) {\n s.remove();\n }\n\n var legendSpan = document.createElement(\"span\");\n legendSpan.className = spanClassName;\n if (direction == \"asc\") {\n legendSpan.innerHTML = arrows.down;\n } else if (direction == \"desc\") {\n legendSpan.innerHTML = arrows.up;\n }\n\n cell.appendChild(legendSpan);\n orderRows[0]\n .querySelectorAll(`.${this.cellClassName}`)\n .forEach((item) => item.classList.remove(\"is-active\"));\n cell.classList.add(\"is-active\");\n }\n\n getOrderRow(orderRows) {\n return orderRows[0];\n }\n\n\n // PAGINATION METHODS\n runPagination(rows, showPage = this.currentPage) {\n this.applyPagination(showPage, rows);\n var numberBttns = this.makePaginationBttns(rows, this.perPage);\n this.addToDOM(this.queries.pagination, numberBttns);\n }\n\n rmPagination() {\n var pagination = this.table.querySelector(this.queries.pagination);\n pagination.innerHTML = \"\";\n }\n\n makePaginationBttns(rows) {\n this.rmPagination();\n\n if (this.totalPages) {\n // controller from backend should say how many pages\n var numOfPages = this.totalPages;\n } else {\n var rowsCount = rows.length;\n var numOfPages = Math.ceil(rowsCount / this.perPage);\n }\n\n var bttns = [];\n // add prev bttn\n if (this.currentPage > 1) {\n bttns.push(\n this.createPaginationBttn({\n tag: this.paginationTag,\n text: this.paginationPreviousText,\n showPage: this.currentPage - 1,\n rows: rows,\n })\n );\n }\n\n // add number buttons\n if (numOfPages > 1) {\n for (var i = 1; i <= numOfPages; i++) {\n bttns.push(\n this.createPaginationBttn({\n tag: this.paginationTag,\n text: i,\n showPage: i,\n rows: rows,\n })\n );\n }\n }\n\n // add next bttn\n if (this.currentPage < numOfPages) {\n bttns.push(\n this.createPaginationBttn({\n tag: this.paginationTag,\n text: this.paginationNextText,\n showPage: this.currentPage + 1,\n rows: rows,\n })\n );\n }\n\n return bttns;\n }\n\n createPaginationBttn(options) {\n var bttn = document.createElement(options.tag);\n bttn.innerHTML = options.text;\n bttn.className = \"pagination_numbers\";\n bttn.dataset.showPage = options.showPage;\n if (options.showPage == this.currentPage) {\n bttn.classList.add(\"is-active\");\n }\n bttn.onclick = (event) => {\n if (this.remote) {\n this.ajaxCall(options.showPage);\n } else {\n this.runPagination(options.rows, options.showPage);\n }\n };\n return bttn;\n }\n\n addToDOM(target, elementsToAdd) {\n var container = this.table.querySelector(target);\n for (var i in elementsToAdd) {\n var element = elementsToAdd[i];\n container.appendChild(element);\n }\n }\n\n removeAllFromDom(tBody) {\n document.querySelectorAll(tBody).forEach((item) => item.innerHTML = \"\");\n }\n\n applyPagination(page, rows = this.findRows(this.tbodyClassName)) {\n this.hideAll(rows);\n var start = (page - 1) * this.perPage;\n var end = start + this.perPage;\n var currentPageRows = rows.slice(start, end);\n this.showAll(currentPageRows);\n this.currentPage = page;\n }\n\n applyRemotePagination(page, rows = this.findRows(this.tbodyClassName)) {\n this.removeAllFromDom(this.queries.tBody);\n for (let r in rows) {\n document.querySelectorAll(this.queries.tBody).forEach((item) => item.appendChild(rows[r]));\n }\n\n this.currentPage = page;\n }\n\n\n // AJAX methods\n ajaxCall(nextPage=null) {\n // break if a fetch is running\n if (this.disableFetch) {\n return false\n }\n this.disableFetch = true\n\n this.disableFetch = true;\n var showPage = nextPage ? nextPage : this.currentPage+1;\n\n // console.log(this.orderByColumn)\n fetch(`${this.remoteUrl}?current_page=${showPage}&order_by_column=${this.orderByColumn}&order_direction=${this.orderDirection}&search_keyword=${this.searchKeyword}`, {\n headers: {\n 'Content-Type': 'application/json',\n \"Accept\": \"application/json\"\n },\n credentials: 'same-origin',\n method: 'get',\n })\n .then(response => response.json())\n .then(jsonData => {\n this.disableFetch = false;\n this.perPage = jsonData.perPage || this.perPage;\n this.totalPages = parseInt(jsonData.totalPages) || this.totalPages;\n var visibleRows = this.createRows(jsonData.data);\n\n var newCurrentPage = parseInt(jsonData.currentPage) || this.currentPage;\n this.applyRemotePagination(newCurrentPage, visibleRows);\n\n var numberBttns = this.makePaginationBttns(visibleRows);\n this.addToDOM(this.queries.pagination, numberBttns);\n })\n .catch(err => {\n //error block\n });\n }\n\n addMobileLabels(selector) {}\n};\n","import { BambooDatatable } from \"../modules/bamboo_datatable\"\n\ndocument.addEventListener(\"turbolinks:load\", () => {\n let datatables = document.querySelectorAll(\".js-datatable\");\n\n if (datatables) {\n datatables.forEach((datatable) => {\n var table = new BambooDatatable(datatable, {\n search: true,\n order: true,\n pagination: true,\n });\n\n if (isDevEnvironment() && table) {\n console.info(`[DEV] Bamboo Datatable Initialized`);\n }\n });\n }\n});\n"],"sourceRoot":""}