The currency mentioned in the description doesn’t seem to match the base currency of your company. Please edit your company settings or chose to proceed with the current description.
\"\n }\n }\n return result\n }\n\n selectedCurrencyCouldBeWrong() {\n for (const currencyCode in this.currencyKeywords) {\n const keywords = this.currencyKeywords[currencyCode]\n for (const k in keywords) {\n for (const w in this.wordsInName) {\n if (this.wordsInName[w].toLowerCase() === keywords[k].toLowerCase()) {\n if (this.selectedCurrencyCode !== currencyCode) { return true }\n }\n }\n }\n }\n\n return false\n }\n\n get form() {\n return $(this.formTarget)\n }\n\n get warningOptions() {\n return JSON.parse(this.data.get('warningOptions'))\n }\n\n get companyId() {\n return this.data.get('companyId')\n }\n\n get currencyKeywords() {\n return JSON.parse(this.data.get('currencyKeywords'))\n }\n\n get wordsInName() {\n return this.nameTarget.value.toLowerCase().split(' ')\n }\n\n get selectedCurrencyCode() {\n return this.currencyCodes[this.currencyIdTarget.value]\n }\n\n get currencyCodes() {\n return JSON.parse(this.data.get('currencyCodes'))\n }\n}\n","// FARMPLAN: added to prevent totals from being affected by tax code changes\n// (https://yozudev.atlassian.net/browse/FP-1089)\nimport { LineItemRowBaseController } from './line_item_row_base_controller'\nimport { isNull } from './helpers/_calculations'\n\nexport default class extends LineItemRowBaseController {\n splitCloneFirstRowInputFields() {\n return ['descriptionTarget']\n }\n\n updateTaxTotalFromNet() {\n // Overrides the base controller due to imported bank transactions\n // having a slightly distinct behaviour from other transaction rows:\n // in non split transactions, the total amount is a fixed value\n // and should **never** be recalculated when other fields change.\n if (this.isElementFromSplitRow()) {\n // If is a split row behaves like all the other transactions\n super.updateTaxTotalFromNet()\n } else if (!isNull(this.getNet())) {\n // If is not a split row, do not recalculate the total and update the net and tax instead\n this.updateNetTaxFromTotal()\n }\n }\n\n updateNetAndPriceFromTax() {\n this.setNet(this.getTotal() - this.getTax())\n if (isNull(this.getPrice())) return\n this.setPrice(this.getNet() / this.getQuantity())\n }\n\n checkForPriceGreaterThanTotal() {\n const net = this.getNet()\n\n if (this.getPrice() > net + this.getTax()) {\n this.setPrice(net / this.getQuantity())\n }\n }\n\n // FARMPLAN: Overrided to check for price greater than imported total\n updateNetAndQuantityFromPrice() {\n const quantity = this.getQuantity()\n const net = this.getNet()\n\n if (!net && !quantity) {\n return\n }\n\n this.checkForPriceGreaterThanTotal()\n this.calculateQuantity()\n this.calculateNet()\n }\n}\n","import { Controller } from '@hotwired/stimulus'\nimport { sanitizeAmount } from './helpers/_calculations'\nimport { confirmationModal } from './helpers/_modals'\n\nexport default class BankReconciliationController extends Controller {\n static values = {\n currency: String\n }\n\n static targets = [\n 'selectAllCheckbox',\n 'saveAllButton',\n 'backButton',\n 'filterSelectAllCheckbox',\n 'filterSelectZeroCheckbox'\n ]\n\n connect() {\n $('#modal-lg').on('shown.bs.modal', this.handleOpeningModal.bind(this))\n $('#modal-lg').on('hide.bs.modal', this.handleClosingModal.bind(this))\n this.reloadCurrentBootgrid()\n }\n\n initialize() {\n this.initializeVariables()\n }\n\n reInitialize() {\n this.initializeVariables()\n this.reloadCurrentBootgrid()\n }\n\n initializeVariables() {\n this.isBankReconciliationOpened = false\n this.reconciledTotal = 0\n this.initialReconciledTotal = 0\n this.isBankReconciliationInitialized = false\n this.isBankReconciliationOpened = false\n this.matchedTransactionsToSave = []\n }\n\n setInitialStatus() {\n this.insertElements([\n {\n selector: '#modal-lg .actionBar .actions',\n position: 'beforeend',\n template: this.bankReconciliationButtonTemplate()\n },\n {\n selector: '#modal-lg .pagination',\n position: 'afterend',\n template: this.backButtonTemplate()\n },\n {\n selector: '#modal-lg .pagination',\n position: 'afterend',\n template: this.saveAllButtonTemplate()\n }\n ])\n }\n\n setBootGridEventListeners() {\n const bootgrid = $('#account_table').bootgrid()\n bootgrid.on(\n 'loaded.rs.jquery.bootgrid',\n this.handleBootGridLoaded.bind(this)\n )\n bootgrid.on(\n 'load.rs.jquery.bootgrid',\n this.handleBootGridWillLoad.bind(this)\n )\n }\n\n setInitialReconciledAmounts() {\n this.updateElements('#modal-lg .checkbox .form-control', (el) => {\n el.checked ? this.updateMatchedAmount(el) : this.setZeroAmount(el)\n })\n }\n\n formatModalTableHeader() {\n this.updateElement(\n \"[data-column-id='matched_to_statement']\",\n '' +\n 'There are currently no files uploaded against this invoice.' +\n '
'\n }\n}\n","import { Controller } from '@hotwired/stimulus'\n\nexport default class extends Controller {\n static targets = ['reg_no']\n\n connect() {\n this.prohibitedCountries = ['GB', 'XI']\n this.numbersRegex = /\\D/g\n }\n\n restrictVatNumberToNumbersOnly(event) {\n if (!window.isFeatureToggleActive('vat_number_uk_restriction')) { return }\n const companyUnLocode = window.companySettings.country.un_locode\n const isUK = this.prohibitedCountries.includes(companyUnLocode)\n\n if (isUK && this.valueContainsNonDigits()) {\n const newValue = this.reg_noTarget.value.replace(this.numbersRegex, '')\n this.reg_noTarget.value = newValue\n }\n }\n\n valueContainsNonDigits() {\n const regNoValue = this.reg_noTarget.value\n return Boolean(regNoValue.match(this.numbersRegex))\n }\n}\n","var map = {\n\t\"./_bootgrid.js\": 248,\n\t\"./_bootgrid_converters.js\": 6,\n\t\"./_bootgrid_csv_exporter.js\": 249,\n\t\"./_bootgrid_preferences.js\": 250,\n\t\"./_init_bootgrid_tables.js\": 251,\n\t\"./accounts_show.js\": 252,\n\t\"./balance_sheet_report.js\": 253,\n\t\"./bank_feeds_row_controller.js\": 254,\n\t\"./bank_imports.js\": 255,\n\t\"./index.js\": 165,\n\t\"./opening_balance_wizard.js\": 256,\n\t\"./profit_and_loss_report.js\": 257,\n\t\"./trial_balance_report.js\": 258,\n\t\"./vat_returns__invoices.js\": 259\n};\n\n\nfunction webpackContext(req) {\n\tvar id = webpackContextResolve(req);\n\treturn __webpack_require__(id);\n}\nfunction webpackContextResolve(req) {\n\tif(!__webpack_require__.o(map, req)) {\n\t\tvar e = new Error(\"Cannot find module '\" + req + \"'\");\n\t\te.code = 'MODULE_NOT_FOUND';\n\t\tthrow e;\n\t}\n\treturn map[req];\n}\nwebpackContext.keys = function webpackContextKeys() {\n\treturn Object.keys(map);\n};\nwebpackContext.resolve = webpackContextResolve;\nmodule.exports = webpackContext;\nwebpackContext.id = 247;","/*!\n * jQuery Bootgrid v1.3.1 - 09/11/2015\n * Copyright (c) 2014-2015 Rafael Staib (http://www.jquery-bootgrid.com)\n * Licensed under MIT http://www.opensource.org/licenses/MIT\n */\nwindow.Bootgrid = (function($, window, undefined) {\n /*jshint validthis: true */\n \"use strict\";\n\n // GRID INTERNAL FIELDS\n // ====================\n\n var namespace = \".rs.jquery.bootgrid\";\n\n // GRID INTERNAL FUNCTIONS\n // =====================\n\n function appendRow(row) {\n var that = this;\n\n function exists(item) {\n return that.identifier && item[ that.identifier ] === row[ that.identifier ];\n }\n\n if (!this.rows.find(exists)) {\n this.rows.push(row);\n return true;\n }\n\n return false;\n }\n\n function findFooterAndHeaderItems(selector) {\n var footer = (this.footer) ? this.footer.find(selector) : $(),\n header = (this.header) ? this.header.find(selector) : $();\n return $.merge(footer, header);\n }\n\n function getRequest() {\n var resetCurrent = false;\n\n var request = {\n current : this.current,\n rowCount : this.rowCount,\n sort : this.sortDictionary,\n searchPhrase : this.searchPhrase,\n filterOptions : this.options.filterOptions,\n transactionFilters : this.element.attr('data-transaction-filters'),\n recordFilters : this.element.attr('data-record-filters')\n },\n post = this.options.post;\n\n if (this.element.data().defaultRowCount != this.rowCount) {\n this.element.data('default-row-count', this.rowCount);\n UserPreferences.set(this.element.attr('id') + '_bootgrid_row_count',\n this.rowCount);\n }\n\n //For some reason due to the way .data() caching works .attr() has to be used for the dictionary sort data attribute\n if (this.element.attr('data-sort-dictionary-user') != JSON.stringify(this.sortDictionary)) {\n resetCurrent = true;\n this.element.attr('data-sort-dictionary-user', JSON.stringify(this.sortDictionary));\n UserPreferences.set(this.element.attr('id') + '_sort_dictionary',\n JSON.stringify(this.sortDictionary));\n }\n if (this.element.data().currentTablePage != this.current && typeof(this.current)== \"number\") {\n this.element.data('current-table-page', this.current);\n }\n\n //This is used to detect any changes in the filter options - attribute not used elsewhere\n if ((this.element.attr('data-filter-options') == undefined)) {\n this.element.attr('data-filter-options', JSON.stringify(this.options.filterOptions));\n }\n if (this.element.attr('data-filter-options') != JSON.stringify(this.options.filterOptions)) {\n resetCurrent = true;\n this.element.attr('data-filter-options', JSON.stringify(this.options.filterOptions));\n }\n if (this.element.data('bootgrid-extra-request-data-function')){\n this.element.data('bootgrid-extra-request-data-function')(request);\n }\n\n // Force the bg table to switch to the first page when sorting options have been changed\n if (resetCurrent) {\n request[\"current\"] = 1;\n }\n\n post = ($.isFunction(post)) ? post() : post;\n return this.options.requestHandler($.extend(true, request, post));\n }\n\n function getCssSelector(css) {\n return \".\" + $.trim(css).replace(/\\s+/gm, \".\");\n }\n\n function getUrl() {\n var url = this.options.url;\n return ($.isFunction(url)) ? url() : url;\n }\n\n function init() {\n this.element.trigger(\"initialize\" + namespace);\n\n loadColumns.call(this); // Loads columns from HTML thead tag\n this.selection = this.options.selection && this.identifier != null;\n loadRows.call(this); // Loads rows from HTML tbody tag if ajax is false\n prepareTable.call(this);\n renderTableHeader.call(this);\n renderSearchField.call(this);\n renderActions.call(this);\n loadData.call(this);\n initSweetAlertListener.call(this);\n this.element.trigger(\"initialized\" + namespace);\n }\n\n function highlightAppendedRows(rows) {\n if (this.options.highlightRows) {\n // todo: implement\n }\n }\n\n function isVisible(column) {\n return column.visible;\n }\n\n function createCell(content, cellCss, style) {\n var that = this,\n tpl = this.options.templates;\n\n return tpl.cell.resolve(that.getParams({\n content : content,\n css : cellCss,\n style : style\n }));\n }\n\n function getContent(element){\n return (element == null || element === \"\") ? \" \" : element\n }\n\n function getColumnAlign(css, alignment){\n if (alignment === \"right\") {\n return css.right\n } else if (alignment === \"center\") {\n return css.center\n } else {\n return css.left\n }\n }\n\n function getCssClass(cssClass){\n if (cssClass.length > 0) {\n return \" \" + cssClass\n } else {\n return \"\"\n }\n }\n\n function loadColumns() {\n var that = this,\n firstHeadRow = this.element.find(\"thead > tr\").first(),\n sorted = false;\n\n /*jshint -W018*/\n firstHeadRow.children().each(function() {\n var $this = $(this),\n data = $this.data(),\n column = {\n id : data.columnId,\n identifier : that.identifier == null && data.identifier || false,\n converter : that.options.converters[ data.converter || data.type ] ||\n that.options.converters[ \"string\" ],\n text : $this.text(),\n align : data.align || \"left\",\n headerAlign : data.headerAlign || \"left\",\n footerAlign : data.footerAlign || \"left\", // Added for renderTfoot\n cssClass : data.cssClass || \"\",\n headerCssClass : data.headerCssClass || \"\",\n footerCssClass : data.footerCssClass || \"\", // Added for renderTfoot\n formatter : that.options.formatters[ data.formatter ] || null,\n order : (!sorted && (data.order === \"asc\" || data.order === \"desc\")) ? data.order : null,\n searchable : !(data.searchable === false), // default: true\n sortable : !(data.sortable === false), // default: true\n visible : !(data.visible === false), // default: true\n visibleInSelection : !(data.visibleInSelection === false), // default: true\n width : ($.isNumeric(data.width)) ? data.width + \"px\" :\n (typeof(data.width) === \"string\") ? data.width : null\n };\n\n that.columns.push(column);\n if (column.order != null) {\n that.sortDictionary[ column.id ] = column.order;\n }\n\n // Prevents multiple identifiers\n if (column.identifier) {\n that.identifier = column.id;\n that.converter = column.converter;\n }\n\n // ensures that only the first order will be applied in case of multi sorting is disabled\n if (!that.options.multiSort && column.order !== null) {\n sorted = true;\n }\n });\n /*jshint +W018*/\n }\n\n /*\n response = {\n current: 1,\n rowCount: 10,\n rows: [{}, {}],\n sort: [{ \"columnId\": \"asc\" }],\n total: 101\n }\n */\n\n function loadData() {\n var that = this;\n\n this.element._bgBusyAria(true).trigger(\"load\" + namespace);\n showLoading.call(this);\n\n function containsPhrase(row) {\n var column,\n searchPattern = new RegExp(that.searchPhrase, (that.options.caseSensitive) ? \"g\" : \"gi\");\n\n for (var i = 0; i < that.columns.length; i++) {\n column = that.columns[ i ];\n if (column.searchable && column.visible &&\n column.converter.to(row[ column.id ]).search(searchPattern) > -1) {\n return true;\n }\n }\n\n return false;\n }\n\n function update(that, rows, total, tfoot_rows, total_filtered) {\n that.currentRows = rows;\n setTotals.call(that, total, total_filtered);\n\n if (!that.options.keepSelection) {\n that.selectedRows = [];\n }\n\n that.removeLoadingAnimation();\n renderRows.call(that, rows);\n renderInfos.call(that);\n renderPagination.call(that);\n\n if (typeof(tfoot_rows) !== \"undefined\") {\n renderTfoot.call(that, tfoot_rows); // Added function to populate tfoot data\n }\n\n that.element._bgBusyAria(false).trigger(\"loaded\" + namespace);\n }\n\n function renderTfoot(rows) {\n if (typeof rows != undefined && rows.length > 0) {\n var that = this,\n css = this.options.css,\n tpl = this.options.templates,\n tfoot = this.element.children(\"tfoot\").first(),\n html = \"\";\n\n $.each(rows, function(index, row) {\n var cells = \"\",\n rowAttr = \"\";\n if (that.selection) {\n cells += createCell.call(that, \" \", '', '')\n }\n $.each(that.columns, function(j, column) {\n if (column.visible) {\n var value = ($.isFunction(column.formatter)) ?\n column.formatter.call(that, column, row) :\n column.converter.to(row[ column.id ]),\n content = (hideTamountLabel.call(that)) ? \"\" : getContent(value),\n cellCss = getColumnAlign(css, column.footerAlign) + getCssClass(column.cssClass) + getCssClass(column.footerCssClass),\n style = (column.width == null) ? \"\" : \"width:\" + column.width + \";\";\n\n cells += createCell.call(that, content, cellCss, style)\n }\n });\n html += tpl.row.resolve(that.getParams({ attr : rowAttr, cells : cells }));\n });\n tfoot.html(html);\n }\n }\n\n function hideTamountLabel() {\n if (this.element.attr('data-hide-tfoot-when-amounts-hidden')) {\n var invisibleCount = 0;\n $.each(this.columns, function(i, column) {\n if ((column.id == \"debit_amount\" || column.id == \"credit_amount\") && column.visible == false) {\n invisibleCount += 1;\n };\n })\n return invisibleCount == 2\n }\n }\n\n if (this.options.ajax) {\n var request = getRequest.call(this),\n url = getUrl.call(this);\n\n if (url == null || typeof url !== \"string\" || url.length === 0) {\n throw new Error(\"Url setting must be a none empty string or a function that returns one.\");\n }\n\n // aborts the previous ajax request if not already finished or failed\n if (this.xqr) {\n this.xqr.abort();\n }\n\n var showNoResults = function(caller) {\n renderNoResultsRow.call(caller); // overrides loading mask\n caller.element._bgBusyAria(false).trigger(\"loaded\" + namespace);\n }\n var settings = {\n url : url,\n data : request,\n success : function(response) {\n that.xqr = null;\n\n if (typeof (response) === \"string\") {\n response = JSON.showMaintenanceModalOrParse(response);\n }\n\n if ( response == 'maintenance_mode' ){\n showNoResults(that);\n } else if ( 'error' in response ) {\n showError(response[\"error\"]);\n showNoResults(that);\n } else {\n var total = response.total;\n\n if (total > 0 && total <= response.rowCount * (response.current - 1)) {\n that.current = 1;\n loadData.call(that);\n } else {\n response = that.options.responseHandler(response);\n\n that.current = response.current;\n update(that, response.rows, response.total, response.tfoot_rows, response.total_filtered);\n }\n }\n\n },\n error : function(jqXHR, textStatus, errorThrown) {\n that.xqr = null;\n\n if (textStatus !== \"abort\") {\n showNoResults(that);\n }\n }\n };\n settings = $.extend(this.options.ajaxSettings, settings);\n\n this.xqr = $.ajax(settings);\n } else {\n var rows = (this.searchPhrase.length > 0) ? this.rows.filter(containsPhrase) : this.rows,\n total = rows.length;\n if (this.rowCount !== -1) {\n rows = pageArray(rows, this.current, this.rowCount);\n }\n\n // todo: improve the following comment\n // setTimeout decouples the initialization so that adding event handlers happens before\n window.setTimeout(function() {\n update(that, rows, total);\n }, 10);\n }\n }\n\n function loadRows() {\n if (!this.options.ajax) {\n var that = this,\n rows = this.element.find(\"tbody > tr\");\n\n rows.each(function() {\n var $this = $(this),\n cells = $this.children(\"td\"),\n row = {};\n\n $.each(that.columns, function(i, column) {\n // Allow HTML content in non-AJAX tables (eg links)\n row[ column.id ] = column.converter.from(cells.eq(i).html());\n });\n\n appendRow.call(that, row);\n });\n\n setTotals.call(this, this.rows.length);\n sortRows.call(this);\n }\n }\n\n function setTotals(total, total_filtered) {\n this.total = total;\n this.totalFiltered = total_filtered;\n\n if (total_filtered === 0) {\n this.totalPages = 1;\n } else if (total_filtered) {\n this.totalPages = (this.rowCount === -1) ? 1 : Math.ceil(this.totalFiltered / this.rowCount);\n } else {\n this.totalPages = (this.rowCount === -1) ? 1 : Math.ceil(this.total / this.rowCount);\n }\n }\n\n function prepareTable() {\n var tpl = this.options.templates,\n wrapper = (this.element.parent().hasClass(this.options.css.responsiveTable)) ?\n this.element.parent() : this.element;\n\n this.element.addClass(this.options.css.table);\n\n // checks whether there is an tbody element; otherwise creates one\n if (this.element.children(\"tbody\").length === 0) {\n this.element.append(tpl.body);\n }\n\n if (this.options.navigation & 1) {\n this.header = $(tpl.header.resolve(this.getParams({ id : this.element._bgId() + \"-header\" })));\n // Allow us to put the search and actions where we like, when there are multiple tables on the page\n if ($('#bootgrid_actions__' + this.element._bgId()).length) {\n $('#bootgrid_actions__' + this.element._bgId()).html(this.header);\n // Allow us to put the search and actions where we like\n } else if ($('#bootgrid_actions').length) {\n $('#bootgrid_actions').html(this.header);\n } else {\n wrapper.before(this.header);\n }\n }\n\n if (this.options.navigation & 2) {\n this.footer = $(tpl.footer.resolve(this.getParams({ id : this.element._bgId() + \"-footer\" })));\n wrapper.after(this.footer);\n }\n }\n\n function renderActions() {\n if (this.options.navigation !== 0) {\n var css = this.options.css,\n selector = getCssSelector(css.actions),\n actionItems = findFooterAndHeaderItems.call(this, selector);\n\n if (actionItems.length > 0) {\n var tpl = this.options.templates,\n actions = $(tpl.actions.resolve(this.getParams()));\n\n renderRefreshButton.call(this, actions);\n renderCsvExport.call(this, actions);\n renderPdfExport.call(this, actions);\n renderAdvancedSearch.call(this, actions);\n renderRowCountSelection.call(this, actions);\n renderColumnSelection.call(this, actions);\n renderFilterOptions.call(this, actions);\n replacePlaceHolder.call(this, actionItems, actions);\n }\n }\n bootgridPreferences.decorate(this);\n }\n\n function renderRefreshButton(actions) {\n if (this.options.ajax) {\n var that = this,\n tpl = this.options.templates,\n css = this.options.css,\n refreshIcon = tpl.icon.resolve(this.getParams({ iconCss : css.iconRefresh }));\n\n var refresh = $(tpl.actionButton.resolve(this.getParams(\n {\n content : refreshIcon,\n text : this.options.labels.refresh\n })))\n .addClass ('bgm-blue')\n .on(\"click\" + namespace, function(e) {\n // todo: prevent multiple fast clicks (fast click detection)\n e.stopPropagation();\n that.current = 1;\n loadData.call(that);\n });\n\n actions.append(refresh);\n }\n }\n\n function renderCsvExport(actions){\n if (this.options.csvExport) {\n bootgridCsvExporter.decorate(this, actions);\n }\n }\n\n function renderPdfExport(actions) {\n if (this.options.pdfExport) {\n var tpl = this.options.templates,\n css = this.options.css,\n pdfIcon = tpl.icon.resolve(this.getParams({ iconCss : css.iconPdf }));\n\n if (this.options.pdfExportTargetBlank) {\n var pdfTemplate = tpl.link;\n } else {\n var pdfTemplate = tpl.actionButton;\n }\n var pdf = $(pdfTemplate.resolve(this.getParams(\n {\n content : pdfIcon,\n text : this.options.labels.pdfExport\n })))\n .addClass ('bgm-red')\n .attr('href', this.options.pdfUrl)\n .on('click'+namespace, function(e) {\n // Yes, really. Clicking the button doesn't trigger a request to its href by itself.\n $.get($(e.currentTarget).attr('href'));\n });\n actions.append(pdf);\n }\n }\n\n function renderAdvancedSearch(actions){\n if (this.options.advancedSearch){\n var that = this,\n tpl = this.options.templates,\n css = this.options.css;\n\n var search = $(tpl.actionAdvancedSearch.resolve(this.getParams({ iconCss : css.iconAdvancedSearch })));\n actions.append(search);\n }\n }\n\n function renderFilterOptions(actions) {\n if (this.options.filterOptions) {\n var that = this,\n css = this.options.css,\n tpl = this.options.templates,\n icon = tpl.icon.resolve(\n this.getParams({ iconCss : css.iconFilterOptions })),\n dropDown = $(tpl.actionDropDown.resolve(this.getParams({\n content : icon,\n dropDownLabel : this.options.labels.filterOptions\n }))),\n selector = getCssSelector(css.dropDownItem),\n checkboxSelector = getCssSelector(css.dropDownItemCheckbox),\n itemsSelector = getCssSelector(css.dropDownMenuItems);\n\n $.each(this.options.filterOptions, function(i, filter_option) {\n var item = $(tpl.actionDropDownCheckboxItem.resolve(\n that.getParams(\n {\n id : filter_option.id + \"_filter\",\n name : filter_option.id,\n label : filter_option.name,\n checked : filter_option.visible\n })))\n .addClass('p-r-15 p-l-15')\n .on(\"click\", selector, function(e) {\n e.stopPropagation();\n\n var $this = $(this),\n checkbox = $this.find(checkboxSelector);\n\n if (!checkbox.prop(\"disabled\")) {\n filter_option.visible = checkbox.prop(\"checked\");\n\n $this.parents(itemsSelector).find(selector + \":has(\" + checkboxSelector + \":checked)\")\n ._bgEnableAria(true).find(checkboxSelector)._bgEnableField(true);\n\n that.element.find(\"tbody\").empty(); // Fixes an column visualization bug\n\n loadData.call(that);\n }\n });\n if (filter_option.visible) {\n item.find('input').attr('checked', true);\n }\n dropDown.find(getCssSelector(css.dropDownMenuItems)).append(item);\n });\n actions.append(dropDown);\n }\n }\n\n function renderColumnSelection(actions) {\n if (this.options.columnSelection && this.columns.length > 1) {\n var that = this,\n css = this.options.css,\n tpl = this.options.templates,\n icon = tpl.icon.resolve(this.getParams({ iconCss : css.iconColumns })),\n dropDown = $(tpl.actionDropDown.resolve(this.getParams({\n content : icon,\n dropDownLabel : this.options.labels.columnSelect\n }))),\n selector = getCssSelector(css.dropDownItem),\n checkboxSelector = getCssSelector(css.dropDownItemCheckbox),\n itemsSelector = getCssSelector(css.dropDownMenuItems);\n\n $.each(this.columns, function(i, column) {\n if (column.visibleInSelection) {\n var item = $(tpl.actionDropDownCheckboxItem.resolve(that.getParams(\n {\n id : that.element[0].id + \"_\" + column.id + \"_col_select\",\n name : column.id,\n label : column.text,\n checked : column.visible\n })))\n .addClass('p-r-15 p-l-15')\n .on(\"click\" + namespace, selector, function(e) {\n e.stopPropagation();\n\n var $this = $(this),\n checkbox = $this.find(checkboxSelector);\n if (!checkbox.prop(\"disabled\")) {\n column.visible = checkbox.prop(\"checked\");\n var enable = that.columns.filter(isVisible).length > 1;\n $this.parents(itemsSelector).find(selector + \":has(\" + checkboxSelector + \":checked)\")\n ._bgEnableAria(enable).find(checkboxSelector)._bgEnableField(enable);\n\n that.element.find(\"tbody\").empty(); // Fixes an column visualization bug\n removeSorting.call(that, column);\n renderTableHeader.call(that);\n loadData.call(that);\n }\n });\n dropDown.find(getCssSelector(css.dropDownMenuItems)).append(item);\n }\n });\n actions.append(dropDown);\n }\n }\n\n function removeSorting(column) {\n var that = this;\n if (!column.visible && that.sortDictionary[column.id]) {\n if (that.options.multiSort){\n that.sortDictionary[column.id] = null;\n } else {\n that.sortDictionary = {};\n };\n };\n }\n\n function renderInfos() {\n if (this.options.navigation !== 0) {\n var selector = getCssSelector(this.options.css.infos),\n infoItems = findFooterAndHeaderItems.call(this, selector),\n totalResults = _totalOrTotalFiltered(this);\n\n if (infoItems.length > 0) {\n var infoText;\n if (this.totalFiltered === 0 || this.total === 0 || this.rowCount === -1) {\n infoText = this.options.labels.infosAllRowsShown.resolve(\n this.getParams({\n total : totalResults\n })\n );\n } else {\n var end = (this.current * this.rowCount),\n infoText = this.options.labels.infos.resolve(\n this.getParams({\n start : (this.total === 0) ? 0 : (end - this.rowCount + 1),\n end : (totalResults === 0 || end === -1 || end > totalResults) ? totalResults : end,\n total : totalResults\n })\n );\n }\n if (typeof(this.totalFiltered) !== 'undefined' && this.totalFiltered != this.total) {\n infoText += this.options.labels.infosFiltered.resolve(\n this.getParams({\n total : this.total\n })\n );\n }\n if (this.selectedRows && this.selectedRows.length !== 0) {\n infoText += this.options.labels.infosSelected.resolve(\n this.getParams({\n selected : this.selectedRows.length\n })\n );\n }\n\n var infos = $(this.options.templates.infos.resolve(\n this.getParams({\n infoText : infoText\n })\n ));\n\n replacePlaceHolder.call(this, infoItems, infos);\n }\n }\n }\n\n function _totalOrTotalFiltered(that) {\n return typeof(that.totalFiltered) === 'undefined' ? that.total : that.totalFiltered;\n }\n\n function renderNoResultsRow() {\n var tbody = this.element.children(\"tbody\").first(),\n tpl = this.options.templates,\n count = this.columns.filter(isVisible).length;\n\n if (this.selection) {\n count = count + 1;\n }\n tbody.html(tpl.noResults.resolve(this.getParams({ columns : count })));\n }\n\n function renderPagination() {\n if (this.options.navigation !== 0) {\n var selector = getCssSelector(this.options.css.pagination),\n paginationItems = findFooterAndHeaderItems.call(this, selector)._bgShowAria(this.rowCount !== -1);\n\n if (this.rowCount !== -1 && paginationItems.length > 0) {\n var tpl = this.options.templates,\n current = this.current,\n totalPages = this.totalPages,\n pagination = $(tpl.pagination.resolve(this.getParams())),\n offsetRight = totalPages - current,\n offsetLeft = (this.options.padding - current) * -1,\n startWith = ((offsetRight >= this.options.padding) ?\n Math.max(offsetLeft, 1) :\n Math.max((offsetLeft - this.options.padding + offsetRight), 1)),\n maxCount = this.options.padding * 2 + 1,\n count = (totalPages >= maxCount) ? maxCount : totalPages;\n\n renderPaginationItem.call(this, pagination, \"first\", \"').css({ backgroundColor : 'transparent' }).append($(''));\n }\n\n this.loadingContainer.find('td')\n .attr('colspan', columns)\n .append(this.preloader.getElement());\n\n return this.loadingContainer;\n };\n\n Grid.prototype.removeLoadingAnimation = function() {\n if (this.loadingContainer) {\n this.loadingContainer.remove();\n }\n\n if (this.preloader) {\n this.preloader.dispose();\n this.preloader = null;\n }\n\n $.rails.enableElement($('.bg-filters-button'));\n\n };\n\n // GRID COMMON TYPE EXTENSIONS\n // ============\n\n $.fn.extend({\n _bgAria : function(name, value) {\n return (value) ? this.attr(\"aria-\" + name, value) : this.attr(\"aria-\" + name);\n },\n\n _bgBusyAria : function(busy) {\n return (busy == null || busy) ?\n this._bgAria(\"busy\", \"true\") :\n this._bgAria(\"busy\", \"false\");\n },\n\n _bgRemoveAria : function(name) {\n return this.removeAttr(\"aria-\" + name);\n },\n\n _bgEnableAria : function(enable) {\n return (enable == null || enable) ?\n this.removeClass(\"disabled\")._bgAria(\"disabled\", \"false\") :\n this.addClass(\"disabled\")._bgAria(\"disabled\", \"true\");\n },\n\n _bgEnableField : function(enable) {\n return (enable == null || enable) ?\n this.removeAttr(\"disabled\") :\n this.attr(\"disabled\", \"disable\");\n },\n\n _bgShowAria : function(show) {\n return (show == null || show) ?\n this.show()._bgAria(\"hidden\", \"false\") :\n this.hide()._bgAria(\"hidden\", \"true\");\n },\n\n _bgSelectAria : function(select) {\n return (select == null || select) ?\n this.addClass(\"active\")._bgAria(\"selected\", \"true\") :\n this.removeClass(\"active\")._bgAria(\"selected\", \"false\");\n },\n\n _bgId : function(id) {\n return (id) ? this.attr(\"id\", id) : this.attr(\"id\");\n }\n });\n\n if (!String.prototype.resolve) {\n var formatter = {\n \"checked\" : function(value) {\n if (typeof value === \"boolean\") {\n return (value) ? \"checked=\\\"checked\\\"\" : \"\";\n }\n return value;\n }\n };\n\n String.prototype.resolve = function(substitutes, prefixes) {\n var result = this;\n $.each(substitutes, function(key, value) {\n if (value != null && typeof value !== \"function\") {\n if (typeof value === \"object\") {\n var keys = (prefixes) ? $.extend([], prefixes) : [];\n keys.push(key);\n result = result.resolve(value, keys) + \"\";\n }\n else {\n if (formatter && formatter[ key ] && typeof formatter[ key ] === \"function\") {\n value = formatter[ key ](value);\n }\n key = (prefixes) ? prefixes.join(\".\") + \".\" + key : key;\n var pattern = new RegExp(\"\\\\{\\\\{\" + key + \"\\\\}\\\\}\", \"gm\");\n result = result.replace(pattern, (value.replace) ? value.replace(/\\$/gi, \"$\") : value);\n }\n }\n });\n return result;\n };\n }\n\n function pageArray(array, page, size) {\n var offset = (page - 1) * size;\n\n if (array.length > offset) {\n return array.slice(offset, offset + size);\n } else {\n return [];\n }\n }\n\n function getObjectValuesFromArray(array, key) {\n return array.map(function(val) {\n return val[ key ];\n });\n }\n\n // GRID PLUGIN DEFINITION\n // =====================\n\n var old = $.fn.bootgrid;\n\n $.fn.bootgrid = function(option) {\n var args = Array.prototype.slice.call(arguments, 1),\n returnValue = null,\n elements = this.each(function(index) {\n var $this = $(this),\n instance = $this.data(namespace),\n options = typeof option === \"object\" && option;\n\n if (!instance && option === \"destroy\") {\n return;\n }\n if (!instance) {\n $this.data(namespace, (instance = new Grid(this, options)));\n init.call(instance);\n }\n if (typeof option === \"string\") {\n if (option.indexOf(\"get\") === 0 && index === 0) {\n returnValue = instance[ option ].apply(instance, args);\n }\n else if (option.indexOf(\"get\") !== 0) {\n return instance[ option ].apply(instance, args);\n }\n }\n });\n return (typeof option === \"string\" && option.indexOf(\"get\") === 0) ? returnValue : elements;\n };\n\n $.fn.bootgrid.namespace = namespace;\n\n $.fn.bootgrid.Constructor = Grid;\n\n // GRID NO CONFLICT\n // ===============\n\n $.fn.bootgrid.noConflict = function() {\n $.fn.bootgrid = old;\n return this;\n };\n\n // GRID DATA-API\n // ============\n\n $(\"[data-toggle=\\\"bootgrid\\\"]\").bootgrid();\n\n return {\n getRequest: getRequest,\n getUrl: getUrl\n }\n})(jQuery, window);\n","(function(){\n\n window.bootgridCsvExporter = {};\n\n bootgridCsvExporter.decorate = function($bootgrid, actions){\n var templates = $bootgrid.options.templates;\n var css = $bootgrid.options.css;\n var exportIcon = templates.icon.resolve(\n $bootgrid.getParams({ iconCss: css.iconExport }));\n\n var csvExport = $(templates.actionButton.resolve($bootgrid.getParams({\n content: exportIcon,\n text: $bootgrid.options.labels.csvExport\n })))\n .addClass('bgm-green')\n .on(\"click\" + $.fn.bootgrid.namespace, onExportButtonClick($bootgrid));\n\n actions.append(csvExport);\n };\n\n function onExportButtonClick ($bootgrid){\n return function(e){\n if ($bootgrid.options.ajax) {\n loadCsvExport($bootgrid, e.target);\n }\n }\n }\n\n function loadCsvExport($bootgrid, button) {\n var css = $bootgrid.options.css;\n var button = getCsvExportButton(button, css);\n var initial_button_class = button.attr('class');\n\n displayCsvLoadingIconAndDisableButton(button, css);\n abortCurrentXhr($bootgrid);\n sendXhr($bootgrid, button, initial_button_class);\n }\n\n function getCsvExportButton(button, css) {\n return $(button).hasClass(css.icon) ? $(button) : $(button).find('.' + css.iconExport);\n }\n\n function displayCsvLoadingIconAndDisableButton (export_button_span, css) {\n export_button_span.attr('class', css.icon + ' ' + css.iconLoading);\n export_button_span.parent().attr('disabled', 'disabled');\n }\n\n function abortCurrentXhr ($bootgrid) {\n if ($bootgrid.xqr) {\n $bootgrid.xqr.abort();\n }\n }\n\n function sendXhr ($bootgrid, button, initial_button_class) {\n return ($bootgrid.xqr = $.ajax($.extend($bootgrid.options.ajaxSettings, {\n url: getUrl($bootgrid),\n data: getRequest($bootgrid),\n success: onXhrSuccess($bootgrid, button, initial_button_class),\n error: onXhrFail()\n })));\n }\n\n function getUrl ($bootgrid){\n return Bootgrid.getUrl.call($bootgrid).replace(\".bootgrid\", \".csv\");\n }\n\n function getRequest ($bootgrid){\n var request = Bootgrid.getRequest.call($bootgrid);\n request.rowCount = \"-1\"; // We want all the rows\n return request;\n }\n\n function onXhrFail () {\n return function() {\n notify('An error occurred downloading your export, please try again or contact support', 'danger', 'Error ');\n }\n }\n\n function onXhrSuccess ($bootgrid, button, initial_button_class) {\n return function(response) {\n $bootgrid.xqr = null;\n response = JSON.parse(response);\n // We need to make a distinction between new background jobs processes for now - TB-12379\n if (Number.isInteger(response)) {\n connectToChannel(response, button, initial_button_class);\n } else {\n saveData(response.filename || getDefaultFilename(), response.csv);\n restoreCsvButton(button, initial_button_class);\n }\n }\n }\n\n function getDefaultFilename() {\n return branding.product_name + \" Data.csv\";\n }\n\n function saveData(filename, csv){\n saveAs(new Blob([csv], { type: \"text/plain;charset=utf-8\"}), filename);\n }\n\n // This needs to stay in until all csv exports have been moved to background jobs - TB-12379\n function restoreCsvButton (export_button_span, old_button_class) {\n export_button_span.attr('class', old_button_class);\n export_button_span.parent().attr('disabled', false);\n }\n\n function connectToChannel(response, button, initial_button_class) {\n notify('Your report is being calculated and will soon be available for download.', 'success', 'Success! ');\n createSubscriptionToBackgroundExports(response, button, initial_button_class, function() {\n Routes.bootgrid_table_background_export_path(response)\n });\n }\n })();\n","/**\n * @fileoverview Exposes a js interface for storing a user's bootgrid table\n * settings on the server.\n */\n\n(function(){\n /**\n * @namespace\n */\n window.bootgridPreferences = {};\n /**\n * @param $grid\n */\n bootgridPreferences.decorate = function($grid) {\n _decorateColumnSelection($grid);\n };\n\n function _decorateColumnSelection($grid) {\n if(_canDecorateColumnSelection($grid)){\n _addColumnSelectionChangeEventListener($grid);\n }\n }\n\n function _canDecorateColumnSelection($grid) {\n return ($grid.header\n && $grid.options.columnSelection\n && $grid.columns.length > 1);\n }\n\n function _addColumnSelectionChangeEventListener($grid) {\n $grid.header.on('change', _onColumnSelectionChange(_getId($grid)));\n }\n\n function _getId($grid){\n return $grid.element.attr('id');\n }\n\n /**\n * Here we assume that any change event will be triggered by a column\n * selection checkbox. Additional checks will need to be performed if other\n * input elements are added to the bootgrid templates.\n *\n * @returns {Function}\n * @private\n */\n function _onColumnSelectionChange(table_id) {\n return function (e) {\n var elem = e.target;\n UserPreferences.set(getPreferenceKey(table_id, elem.name), elem.checked);\n }\n }\n\n function getPreferenceKey(table_id, elem_name){\n return ['col_select', table_id, elem_name].join('_');\n }\n\n})();\n","var RowSelected = {\n transactions: []\n};\n\nwindow.initBootgridTable = function(table, row_count, csv_export, filter_options, labels) {\n filter_options = useInjectedFilterOptions(filter_options, getFilterOptionsFromPage(table.attr('id')));\n\n return errors.try(function() {\n row_count = typeof row_count !== 'undefined' ? row_count : [ 10, 25, 50, 100 ]\n if ( table.length === 0 ) { return; }\n return table.bootgrid({\n css : {\n icon : 'zmdi icon',\n iconColumns : 'zmdi-view-module',\n iconDown : 'zmdi-chevron-down',\n iconRefresh : 'zmdi-refresh',\n iconUp : 'zmdi-chevron-up',\n iconExport : 'csv-export-icon',\n iconFilterOptions : 'zmdi-filter-list',\n iconLoading : 'zmdi-rotate-right zmdi-hc-spin'\n },\n ajaxSettings : {\n method : 'GET'\n },\n multiSort : false,\n rowCount : row_count,\n csvExport : csv_export,\n filterOptions : filter_options,\n labels: labels\n });\n });\n}\n\nfunction useInjectedFilterOptions(defaults, injected) {\n return injected ? injected : defaults;\n}\n\nfunction getFilterOptionsFromPage(table_id) {\n var settings = window.bootgrid_settings;\n return settings && settings.filter_options[table_id];\n}\n\nwindow.initBootgridTableWithCheckboxes = function(table, csv_export, filter_options) {\n filter_options = useInjectedFilterOptions(filter_options, getFilterOptionsFromPage(table.attr('id')));\n\n return errors.try(function() {\n table.bootgrid({\n css : {\n icon : 'zmdi icon',\n iconColumns : 'zmdi-view-module',\n iconDown : 'zmdi-chevron-down',\n iconRefresh : 'zmdi-refresh',\n iconUp : 'zmdi-chevron-up',\n iconExport : 'csv-export-icon',\n iconFilterOptions : 'zmdi-filter-list',\n iconLoading : 'zmdi-rotate-right zmdi-hc-spin'\n },\n ajaxSettings : {\n method : 'GET'\n },\n multiSort : false,\n selection : true,\n multiSelect : true,\n keepSelection : true,\n csvExport : csv_export,\n filterOptions : filter_options\n }).on('selected.rs.jquery.bootgrid', errors.tryFn(function(e, rows) {\n for (var i = 0; i < rows.length; i++) {\n RowSelected.transactions.push(rows[ i ].id);\n rowClick();\n }\n })).on('deselected.rs.jquery.bootgrid', errors.tryFn(function(e, rows) {\n for (var i = 0; i < rows.length; i++) {\n var id = rows[ i ].id;\n var index = $.inArray(id, RowSelected.transactions);\n RowSelected.transactions.splice(index, 1);\n rowClick();\n }\n }));\n disableButtonsWhenNoSelection();\n });\n}\n\nfunction toggleGroupOptionWhenApplicable() {\n if ($(\"tr.active\").find(\".converted\").length > 0) {\n $('[data-convertable-quotes]').attr('disabled', true);\n } else {\n $('[data-convertable-quotes]').attr('disabled', false);\n }\n}\n\nfunction rowClick() {\n // Need to use .split(',') in controller to make an array of IDs\n $('#id_field').val(RowSelected.transactions);\n disableButtonsWhenNoSelection();\n toggleGroupOptionWhenApplicable();\n}\n\nwindow.initTransactionUpdates = function() {\n RowSelected.transactions = [];\n}\n\nfunction disableButtonsWhenNoSelection() {\n if ($('.require-selection').length === 0) { return }\n $('.require-selection').attr('disabled', RowSelected.transactions.length === 0);\n}\n","import CONVERTER from './_bootgrid_converters.js'\n(function() {\n 'use strict';\n\n window.initAccountTable = function() {\n $('#account_table').bootgrid({\n css: {\n icon: 'zmdi icon',\n iconColumns: 'zmdi-view-module',\n iconDown: 'zmdi-caret-down',\n iconRefresh: 'zmdi-refresh',\n iconUp: 'zmdi-caret-up',\n iconExport: 'csv-export-icon',\n iconLoading: 'zmdi-rotate-right zmdi-hc-spin'\n },\n ajaxSettings: { method: 'GET' },\n multiSort: false,\n csvExport: true,\n labels: { noResults: 'No amounts found' }\n }).on(\"loaded.rs.jquery.bootgrid\", function(e) {\n balanceColumnVisibility();\n })\n }\n\n function balanceColumnVisibility() {\n var balanceColumn = $(\"#account_table\").find('[data-column-id=\"balance\"]');\n if (balanceColumn.length === 0 ) { return; }\n var colIndex = balanceColumn[0].cellIndex + 1;\n\n if (balanceValuesAvailable()) {\n balanceColumn.show();\n $('#account_table').find(\"tr > td:nth-child(\"+colIndex+\")\").show();\n $('#account_table_balance_col_select').removeClass(\"disable-check\");\n $('#account_table_balance_col_select_inp').attr(\"disabled\", false);\n } else {\n balanceColumn.hide();\n $('#account_table').find(\"tr > td:nth-child(\"+colIndex+\")\").hide();\n $('#account_table_balance_col_select').addClass(\"disable-check\");\n $('#account_table_balance_col_select_inp').attr(\"disabled\", true);\n }\n }\n\n function balanceValuesAvailable() {\n var colIndex = $('#account_table').find('[data-column-id=\"balance\"]')[0].cellIndex + 1,\n balanceValues = $.unique(\n $(\"#account_table > tbody > tr td:nth-child(\"+colIndex+\")\").map(function() { return this.innerHTML; })\n )\n return !(balanceValues.length == 1 && balanceValues[0] == \" \");\n }\n\n window.initNonPersistedAccountTable = function() {\n $('#account_table').bootgrid({\n css: {\n icon: 'zmdi icon',\n iconColumns: 'zmdi-view-module',\n iconDown: 'zmdi-caret-down',\n iconRefresh: 'zmdi-refresh',\n iconUp: 'zmdi-caret-up'\n },\n converters: {\n order_by_link_text: CONVERTER.order_by_link_text,\n order_by_date: CONVERTER.order_by_date\n }\n });\n }\n})();\n","import CONVERTER from './_bootgrid_converters.js'\n\nwindow.initBalanceSheetTables = function() {\n balanceSheetTableSetup($('#assets_table'));\n balanceSheetTableSetup($('#liabilities_table'));\n balanceSheetTableSetup($('#capital_table'));\n}\n\nfunction balanceSheetTableSetup(table) {\n table.bootgrid({\n css : {\n icon: 'zmdi icon',\n iconColumns: 'zmdi-view-module',\n iconDown: 'zmdi-caret-down',\n iconRefresh: 'zmdi-refresh',\n iconUp: 'zmdi-caret-up'\n },\n columnSelection : false,\n navigation : 0,\n rowCount : -1,\n converters : {\n reports_numeric_ignore_zero: CONVERTER.reports_numeric_ignore_zero,\n order_by_link_text: CONVERTER.order_by_link_text\n },\n labels : { noResults : ' ' }\n });\n}\n","window.BankFeedsRowController = function(cellMap) {\n this._cell_map = cellMap;\n this._data = null;\n};\n\nBankFeedsRowController.prototype.update = function(row) {\n this._data = row;\n Object.keys(row).forEach(this._updateColumn, this);\n};\n\nBankFeedsRowController.prototype._updateColumn = function(col_id) {\n var cell = this._cell_map[ col_id ];\n\n if (cell) {\n cell.html(this._data[ col_id ]);\n }\n};\n","(function() {\n window.BankFeedsTableController = function($grid) {\n this._$table = $grid;\n this._column_ids = null;\n this._row_controllers = {};\n this._channel = null;\n this._last_request_data = null;\n this._bound_function = null;\n };\n /**\n * Entry point, add a global event listener for all incoming xhr responses.\n */\n BankFeedsTableController.prototype.init = function() {\n this._bound_function = this._onBootgridXhrComplete.bind(this);\n $(document).ajaxSuccess(this._bound_function);\n };\n /***\n\n */\n BankFeedsTableController.prototype.destroy = function() {\n $(document).unbind('ajaxSuccess', this._bound_function);\n };\n /**\n * This is a bit of a hack: here we check the xhr responses to see if they're for this bootgrid table by testing the\n * url. Proceed when such a response is found.\n *\n * @param event\n * @param jqXhr\n * @param ajaxOptions\n * @private\n */\n BankFeedsTableController.prototype._onBootgridXhrComplete = function(event, jqXhr, ajaxOptions) {\n if (isImportsUrl(ajaxOptions.url)) {\n this._handleInitialBootgridXhrResponse(getRowsFromResponseText(jqXhr.responseText));\n }\n };\n /**\n * If the user has made changes to the bootgrid table's settings since we last initialised then we need to disconnect\n * the channel and reinitialise, as our row mappings will now be invalid.\n *\n * If we aren't currently watching any rows, check to see if any are pending bank feed imports and poll the server\n * if so.\n *\n * @param event\n * @param rows The rows parsed from the bootgrid xhr response\n * @private\n */\n BankFeedsTableController.prototype._handleInitialBootgridXhrResponse = function(rows) {\n if (this._hasBootgridStateChanged()) {\n this._reset();\n }\n\n this._initialiseIfNoRowControllers(rows);\n };\n\n BankFeedsTableController.prototype._hasBootgridStateChanged = function() {\n return this._last_request_data && !areBootgridStatesEqual(this._getRequestData(), this._last_request_data);\n };\n\n BankFeedsTableController.prototype._reset = function() {\n this._column_ids = null;\n this._row_controllers = {};\n this._last_request_data = null;\n\n this._channel.disconnect();\n };\n\n BankFeedsTableController.prototype._initialiseIfNoRowControllers = function(rows) {\n if (!this._hasRowControllers()) {\n this._decorateRowsAndOpenChannel(rows);\n }\n };\n\n BankFeedsTableController.prototype._decorateRowsAndOpenChannel = function(rows) {\n rows.forEach(this._decorateRowIfPendingOrWaiting, this);\n\n if (this._hasRowControllers()) {\n this._openChannel();\n }\n };\n /**\n * Creates a BankFeedsRowController that can be used to update the required cells when the bank feed pending state\n * changes, and store it against the row's index number in the _row_controllers map.\n *\n * @param row\n * @param row_index\n * @private\n */\n BankFeedsTableController.prototype._decorateRowIfPendingOrWaiting = function(row, row_index) {\n if (isRowPendingOrWaiting(row)) {\n this._row_controllers[ row_index ] = this._createRowController(row_index);\n }\n };\n /**\n * Searches the table for the cells we need to update and passes them to a new BankFeedsRowController.\n *\n * @param row_index\n * @returns {Window.BankFeedsRowController}\n * @private\n */\n BankFeedsTableController.prototype._createRowController = function(row_index) {\n return new BankFeedsRowController(this._createRowElementMap(row_index));\n };\n\n BankFeedsTableController.prototype._createRowElementMap = function(row_index) {\n return bootgrid.queryHelpers.getRowCellMap(row_index, this._getColumnIds(), this._$table);\n };\n\n BankFeedsTableController.prototype._getColumnIds = function() {\n if (this._column_ids === null) {\n this._column_ids = bootgrid.queryHelpers.getHeaderColumnIds(this._$table);\n }\n\n return this._column_ids;\n };\n\n BankFeedsTableController.prototype._hasRowControllers = function() {\n return Object.keys(this._row_controllers).length > 0\n };\n /**\n * Creates a polling channel using the bootgrid table's current request data to watch for server side changes.\n * @private\n */\n BankFeedsTableController.prototype._openChannel = function() {\n var requestData = this._getRequestData();\n\n this._last_request_data = extractBootgridStateFromRequest(requestData);\n\n this._channel = new PollingChannel(\n {\n url : this._$table.data('url'),\n data : requestData\n }\n );\n\n this._channel.subscribe(this._onChannelUpdate.bind(this));\n this._channel.connect();\n };\n\n BankFeedsTableController.prototype._getRequestData = function() {\n return Bootgrid.getRequest.call(this._$table.data('.rs.jquery.bootgrid'));\n };\n\n BankFeedsTableController.prototype._onChannelUpdate = function(responseText) {\n this._updateRows(getRowsFromResponseText(responseText));\n };\n\n BankFeedsTableController.prototype._updateRows = function(rows) {\n rows.forEach(this._updateRow, this);\n\n if (!rows.some(isRowPendingOrWaiting)) {\n this._reset(rows);\n }\n };\n\n BankFeedsTableController.prototype._updateRow = function(row, i) {\n var controller = this._row_controllers[ i ];\n\n if (controller) {\n controller.update(row);\n }\n };\n\n function isImportsUrl(url) {\n return url.indexOf('imports.bootgrid') > 0;\n }\n\n function getRowsFromResponseText(text) {\n var rows = JSON.parse(text).rows;\n\n if (Array.isArray(rows)) {\n return rows;\n } else {\n throw new Error('Bootgrid xhr response does not contain an array of rows');\n }\n }\n\n /**\n * See method 'status_text' in\n * app/bootgrid_tables/bootgrid_table/imports_index/hash_builder.rb\n *\n * @param {Object} row\n * @returns {boolean}\n */\n function isRowPendingOrWaiting(row) {\n var status = row.status;\n return status === 'Waiting' || status.indexOf('Pending') === 0;\n }\n\n function extractBootgridStateFromRequest(ob) {\n return {\n current : ob.current,\n rowCount : ob.rowCount,\n filterOptions : ob.filterOptions.map(extractFilterOptionState)\n };\n }\n\n function extractFilterOptionState(ob) {\n return {\n visible : ob.visible\n }\n }\n\n function areBootgridStatesEqual(a, b) {\n return (\n a.current === b.current &&\n a.rowCount === b.rowCount &&\n areFilterOptionStatesEqual(a.filterOptions, b.filterOptions)\n )\n }\n\n function areFilterOptionStatesEqual(a, b) {\n return a.every(\n function(item, i) {\n return item.visible === b[ i ].visible;\n }\n );\n }\n})()\n","import CONVERTER from './_bootgrid_converters.js'\n\nwindow.initObTables = function() {\n initBootgridTable($('#customer_table'), undefined, true);\n initBootgridTable($('#supplier_table'), undefined, true);\n initBootgridTable($('#vat_scheme_journals_table'));\n \n $('#ob_nominal_table').bootgrid({\n css: {\n icon: 'zmdi icon',\n iconColumns: 'zmdi-view-module',\n iconDown: 'zmdi-caret-down',\n iconRefresh: 'zmdi-refresh',\n iconUp: 'zmdi-caret-up'\n },\n converters: {\n order_by_link_text: CONVERTER.order_by_link_text\n },\n columnSelection: false,\n navigation: 0,\n rowCount: -1\n }).on(\"loaded.rs.jquery.bootgrid\", function(e){\n // NB: any change listeners on form fields in a bootgrid table have to be added after it has loaded\n initObBalancesForm();\n });\n\n updateStockAccountDebitAmount();\n}","import CONVERTER from './_bootgrid_converters.js'\n\nwindow.initPnLTables = function() {\n PnlTableSetup($('#sales_table'));\n PnlTableSetup($('#direct_costs_table'));\n PnlTableSetup($('#expenses_table'));\n}\n\nfunction PnlTableSetup(table) {\n table.bootgrid({\n css: {\n icon: 'zmdi icon',\n iconColumns: 'zmdi-view-module',\n iconDown: 'zmdi-caret-down',\n iconRefresh: 'zmdi-refresh',\n iconUp: 'zmdi-caret-up'\n },\n columnSelection: false,\n navigation: 0,\n rowCount: -1,\n converters: {\n reports_numeric_ignore_zero: CONVERTER.reports_numeric_ignore_zero,\n order_by_link_text: CONVERTER.order_by_link_text\n },\n labels: { noResults: ' ' }\n });\n}","import CONVERTER from './_bootgrid_converters.js'\n\nwindow.initTrialBalanceDataTable = function() {\n $('#trial_balance_report_table').bootgrid({\n css: {\n icon: 'zmdi icon',\n iconColumns: 'zmdi-view-module',\n iconDown: 'zmdi-caret-down',\n iconRefresh: 'zmdi-refresh',\n iconUp: 'zmdi-caret-up'\n },\n columnSelection: false,\n navigation: 0,\n rowCount: -1,\n converters: {\n numeric_ignore_zero: CONVERTER.numeric_ignore_zero,\n order_by_link_text: CONVERTER.order_by_link_text\n },\n labels: { noResults: ' ' }\n });\n}","window.initModalInvoicesTable = function() {\n $('#vat_return_invoices_transactions_table').bootgrid({\n css: {\n icon: 'zmdi icon',\n iconColumns: 'zmdi-view-module',\n iconDown: 'zmdi-caret-down',\n iconRefresh: 'zmdi-refresh',\n iconUp: 'zmdi-caret-up'\n },\n ajaxSettings: { method: 'GET' },\n multiSort: false,\n labels: { search: 'Search by description or ref' }\n });\n}\n\nfunction _vatReturnTransactionTable() {\n return $('#vat_return_invoices_transactions_table')\n}\n\nwindow.alertVatModalAfterDestroy = function() {\n if(_vatReturnTransactionTable().length != 0 && document.querySelectorAll(\"div.alert.alert-warning[data-growl]\").length == 0) {\n notify('Please recalculate your VAT report to ensure changes are reported', \"warning\", \"Warning! \");\n }\n}\n\nwindow.reloadVatModal = function() {\n if(_vatReturnTransactionTable().length != 0) { _vatReturnTransactionTable().bootgrid('reload') }\n}\n","var map = {\n\t\"./advanced_search.js\": 10,\n\t\"./boolean_parser.js\": 25,\n\t\"./bootgrid_helper.js\": 20,\n\t\"./date_helper.js\": 11,\n\t\"./dom_parser.js\": 26,\n\t\"./import_all.js\": 13,\n\t\"./index.js\": 166,\n\t\"./multipicker.js\": 12,\n\t\"./normalizeDateString.js\": 14\n};\n\n\nfunction webpackContext(req) {\n\tvar id = webpackContextResolve(req);\n\treturn __webpack_require__(id);\n}\nfunction webpackContextResolve(req) {\n\tif(!__webpack_require__.o(map, req)) {\n\t\tvar e = new Error(\"Cannot find module '\" + req + \"'\");\n\t\te.code = 'MODULE_NOT_FOUND';\n\t\tthrow e;\n\t}\n\treturn map[req];\n}\nwebpackContext.keys = function webpackContextKeys() {\n\treturn Object.keys(map);\n};\nwebpackContext.resolve = webpackContextResolve;\nmodule.exports = webpackContext;\nwebpackContext.id = 260;","var map = {\n\t\"./_FormChangeObserver.js\": 262,\n\t\"./_alerts.js\": 263,\n\t\"./_analytics.js\": 264,\n\t\"./_are_you_sure.js\": 265,\n\t\"./_bootgrid_query_helpers.js\": 266,\n\t\"./_bootstrap.js\": 267,\n\t\"./_bootstrap_wizard.js\": 268,\n\t\"./_breach_limit.js\": 269,\n\t\"./_buttons.js\": 270,\n\t\"./_checkboxes.js\": 271,\n\t\"./_dates.js\": 272,\n\t\"./_deselect.js\": 273,\n\t\"./_dom.js\": 274,\n\t\"./_dropzone.js\": 275,\n\t\"./_feed_import_before_locked_date.js\": 276,\n\t\"./_fixed_element.js\": 277,\n\t\"./_forms.js\": 278,\n\t\"./_hide_target_if_selected.js\": 279,\n\t\"./_infinite_scrolling.js\": 280,\n\t\"./_keyboard_codes.js\": 281,\n\t\"./_maths.js\": 282,\n\t\"./_memoize_jquery.js\": 283,\n\t\"./_memoized_request.js\": 284,\n\t\"./_modal.js\": 285,\n\t\"./_nav_tabs.js\": 286,\n\t\"./_nested_fields.js\": 287,\n\t\"./_non-bootgrid.js\": 288,\n\t\"./_pageloader.js\": 289,\n\t\"./_polling_channel.js\": 290,\n\t\"./_popovers.js\": 291,\n\t\"./_preloader.js\": 292,\n\t\"./_split_transaction_modal.js\": 293,\n\t\"./_stripe_card_form.js\": 294,\n\t\"./_summernote.js\": 295,\n\t\"./_telephone_field.js\": 296,\n\t\"./_text_inserts.js\": 297,\n\t\"./_throttle.js\": 298,\n\t\"./_tooltips.js\": 299,\n\t\"./_typeahead.js\": 300,\n\t\"./_user_preferences.js\": 301,\n\t\"./addresses.js\": 302,\n\t\"./advanced_search.js\": 303,\n\t\"./announcements.js\": 304,\n\t\"./bank_account_form_controller.js\": 305,\n\t\"./bank_accounts.js\": 306,\n\t\"./bank_accounts/plaid_api_actions.js\": 307,\n\t\"./bank_accounts/plaid_feeds_connect_setup.js\": 308,\n\t\"./bank_accounts/plaid_feeds_refresh_setup.js\": 309,\n\t\"./bank_accounts/plaid_feeds_setup.js\": 310,\n\t\"./bank_rules.js\": 311,\n\t\"./bank_transactions/batch_bt_field_state_selector.js\": 312,\n\t\"./bank_transactions/batch_bt_row.js\": 313,\n\t\"./bank_transactions/batch_form.js\": 314,\n\t\"./bank_transactions/bt_field_states.js\": 315,\n\t\"./bank_transactions/duplication_checks.js\": 316,\n\t\"./bank_transactions/invoice_allocation.js\": 317,\n\t\"./bank_transactions/invoice_selector_modal.js\": 318,\n\t\"./bank_transactions/net_tax_total.js\": 319,\n\t\"./bank_transactions/quick_conversion.js\": 320,\n\t\"./bank_transactions/running_balances.js\": 321,\n\t\"./bank_transactions/sanity_checks.js\": 322,\n\t\"./bank_transactions/split_modal.js\": 323,\n\t\"./bank_transactions/split_net_tax_total.js\": 324,\n\t\"./batch_form_paging.js\": 325,\n\t\"./batch_forms.js\": 326,\n\t\"./calculations/net_tax_total.js\": 327,\n\t\"./charts/_bar_chart.js\": 328,\n\t\"./charts/_line_chart.js\": 329,\n\t\"./charts/_pie_chart.js\": 330,\n\t\"./companies.js\": 331,\n\t\"./conversion_rate.js\": 332,\n\t\"./customer_documents.js\": 333,\n\t\"./customers.js\": 334,\n\t\"./customers_suppliers.js\": 335,\n\t\"./feature_checker.js\": 336,\n\t\"./financial_transactions.js\": 337,\n\t\"./help/error_listeners.js\": 338,\n\t\"./help/general.js\": 339,\n\t\"./help/notifications.js\": 340,\n\t\"./html_generator.js\": 341,\n\t\"./industry.js\": 342,\n\t\"./instrumentation.js\": 343,\n\t\"./invoice_payment.js\": 344,\n\t\"./invoices.js\": 345,\n\t\"./invoices/batch_inv_field_states.js\": 346,\n\t\"./invoices/batch_invoice_row.js\": 347,\n\t\"./invoices/cr_invoice_selector_modal.js\": 348,\n\t\"./invoices/field_states_batch_invoice_row.js\": 349,\n\t\"./invoices/field_states_non_batch_invoice.js\": 350,\n\t\"./invoices/line_items_modal.js\": 351,\n\t\"./invoices/modal_line_item_row.js\": 352,\n\t\"./invoices/non_batch_inv_field_states.js\": 353,\n\t\"./invoices/sanity_checks.js\": 354,\n\t\"./invoices/tax_code_defaulting.js\": 355,\n\t\"./invoices/tax_code_filtering.js\": 356,\n\t\"./journal_entry_currency.js\": 357,\n\t\"./modals/_are_you_sure_modal.js\": 358,\n\t\"./modals/_bootgrid_actions_modal.js\": 359,\n\t\"./modals/_confirm_modal.js\": 360,\n\t\"./modals/_maximise_modal.js\": 361,\n\t\"./modals/_modal_chosen_select.js\": 362,\n\t\"./modals/_remove_selected_from.js\": 363,\n\t\"./new_select_options/customer_creation.js\": 364,\n\t\"./new_select_options/customer_group_load.js\": 365,\n\t\"./new_select_options/customer_single_load.js\": 366,\n\t\"./new_select_options/nominal_account_creation.js\": 367,\n\t\"./new_select_options/stock_item_creation.js\": 368,\n\t\"./new_select_options/supplier_creation.js\": 369,\n\t\"./new_select_options/tax_code_creation.js\": 370,\n\t\"./number_comparison/number_comparison.js\": 371,\n\t\"./number_comparison/number_comparison_result.js\": 372,\n\t\"./number_comparison/number_comparison_service.js\": 373,\n\t\"./quotes/field_states_quotes.js\": 374,\n\t\"./quotes/quote_field_states.js\": 375,\n\t\"./reports.js\": 376,\n\t\"./skin_select.js\": 377,\n\t\"./stock_item.js\": 378,\n\t\"./string_helper.js\": 379,\n\t\"./suppliers.js\": 380,\n\t\"./table_helper.js\": 381,\n\t\"./third_party/_companies_house.js\": 382,\n\t\"./third_party/_companies_house_contact_details.js\": 383,\n\t\"./third_party/_companies_house_officers.js\": 384,\n\t\"./third_party/_mailgun.js\": 385,\n\t\"./third_party/companies_house_api_actions.js\": 386,\n\t\"./third_party/companies_house_search.js\": 387,\n\t\"./third_party/hmrc_header_data.js\": 388,\n\t\"./third_party/mailgun_validator.js\": 389,\n\t\"./toggle_card_width.js\": 390,\n\t\"./uploads.js\": 391,\n\t\"./uploads_modal.js\": 392,\n\t\"./vat_schemes.js\": 393,\n\t\"./year_end.js\": 394\n};\n\n\nfunction webpackContext(req) {\n\tvar id = webpackContextResolve(req);\n\treturn __webpack_require__(id);\n}\nfunction webpackContextResolve(req) {\n\tif(!__webpack_require__.o(map, req)) {\n\t\tvar e = new Error(\"Cannot find module '\" + req + \"'\");\n\t\te.code = 'MODULE_NOT_FOUND';\n\t\tthrow e;\n\t}\n\treturn map[req];\n}\nwebpackContext.keys = function webpackContextKeys() {\n\treturn Object.keys(map);\n};\nwebpackContext.resolve = webpackContextResolve;\nmodule.exports = webpackContext;\nwebpackContext.id = 261;","/**\n * Creates an object for detecting form changes. Written as an ES5 class here\n * so we can clean up the event listeners after usage.\n *\n * We may also want to extend this behaviour, e.g. by supporting different\n * types of components, or choosing which fields to observe.\n *\n * @param formParent {jQuery} Jquery instance of the modal\n * @constructor\n */\nwindow.FormChangeObserver = function(formParent) {\n this.formParent = this._validateFormParent(formParent);\n this._isChanged = false;\n this._isSubmitted = false;\n this._listeners = [];\n this._isDisposed = false;\n this.addEventListeners();\n}\n/**\n * Does the user have unsaved data in the form?\n * @returns {boolean}\n */\nFormChangeObserver.prototype.isStale = function() {\n return (this._isChanged && !this._isSubmitted);\n};\n/**\n * Currently we handle all change events that bubble from input elements and\n * datetimepickers.\n * @protected\n */\nFormChangeObserver.prototype.addEventListeners = function() {\n this._addEventListener('submit', this._onSubmit);\n this._addEventListener('change', this._onChange);\n this._addEventListener('dp.change', this._datepickerOnChange);\n};\n/**\n * Removes all event listeners and deletes the object's keys for garbage\n * collection.\n *\n * N.B. This should always be called after the observer has been used.\n */\nFormChangeObserver.prototype.dispose = function() {\n if(!this._isDisposed) {\n this._removeListeners();\n this._deleteKeys();\n this._isDisposed = true;\n }\n};\n/**\n * Ensures that only one JQuery element is given\n * @param formParent {jQuery}\n * @returns {jQuery}\n * @private\n */\nFormChangeObserver.prototype._validateFormParent = function(formParent) {\n if(!(formParent instanceof jQuery)) {\n throw new Error('formParent ' + formParent + 'is not a JQuery object!');\n } else if(formParent.length !== 1) {\n throw new Error('Exactly one form element should be given. Received formParent ' + formParent + ' with length ' + formParent.length);\n }\n\n return formParent;\n}\n/**\n * @param eventType\n * @param fn\n * @private\n */\nFormChangeObserver.prototype._addEventListener = function(eventType, fn) {\n fn = fn.bind(this);\n this._listeners.push({ eventType: eventType, fn: fn });\n this.formParent.on(eventType, fn);\n};\n/**\n * @returns {function(this:FormChangeObserver)}\n * @private\n */\nFormChangeObserver.prototype._onSubmit = function() {\n this._isSubmitted = true;\n};\n/**\n * @returns {function(this:FormChangeObserver)}\n * @private\n */\nFormChangeObserver.prototype._onChange = function() {\n this._isChanged = true;\n};\n/**\n * @returns {function(this:FormChangeObserver)}\n * @private\n */\nFormChangeObserver.prototype._datepickerOnChange = function(e) {\n if ( !this._datepickerIsBeingInitialised(e) ) {\n this._onChange();\n }\n};\n/**\n * See https://github.com/Eonasdan/bootstrap-datetimepicker/issues/1118\n * Ignores the 'change' event if this is just initialising a datepicker\n * field that already has a value\n * @returns {function(this:FormChangeObserver)}\n * @private\n */\nFormChangeObserver.prototype._datepickerIsBeingInitialised = function(e) {\n var old_date_is_null = e.oldDate === null;\n var datepicker_tf = $(e.target).find('.form-control:first');\n var dp_val_is_orig_val = datepicker_tf && datepicker_tf.val() === datepicker_tf.prop(\"defaultValue\");\n return old_date_is_null && dp_val_is_orig_val;\n};\n/**\n * @private\n */\nFormChangeObserver.prototype._removeListeners = function() {\n this._listeners.forEach(this._removeEventListener, this);\n this._listeners = [];\n};\n/**\n * @param binding {Object}\n */\nFormChangeObserver.prototype._removeEventListener = function(binding) {\n this.formParent.off(binding.eventType, binding.fn);\n};\n/**\n *\n * @private\n */\nFormChangeObserver.prototype._deleteKeys = function() {\n Object.keys(this).forEach(function(k) {\n delete this[k];\n }, this);\n};","(function() {\n 'use strict';\n\n /*\n * Bootstrap Growl - Notifications popups\n\n Type: should be info, success, danger, warning or inverse\n Message: the message text\n Title: title text (optional)\n Icon: zmdi icon (optional)\n Url: makes the whole popup a link (optional)\n\n */\n window.notify = function(message, type, title, icon, url, delay) {\n $.growl(\n {\n message: message,\n icon: icon,\n title: title,\n url: url\n }, {\n type: type,\n allow_dismiss: false,\n label: 'Cancel',\n className: 'btn-xs btn-inverse',\n placement: {\n from: 'top',\n align: 'right'\n },\n delay: delay || 7500,\n animate: {\n enter: 'animated fadeInUp',\n exit: 'animated fadeOut'\n },\n offset: {\n x: 20,\n y: ($(this).scrollTop() > 163) ? 15 : 163\n // 163 = Menu size + 15px\n }\n }\n );\n };\n\n /*\n\n Displays a sweetalert confirmation, and only submits the form if the user confirms\n\n @param alert_text {String} Text (or HTML) to display\n @param $form {jQuery} Form element to submit\n\n */\n window.confirmFormSubmission = function(alert_text, $form, custom_class) {\n sweetAlert({\n title: 'Warning!',\n text: alert_text,\n html: true,\n type: 'warning',\n allowEscapeKey: false,\n showCancelButton: true,\n customClass: custom_class || ''\n }, function(isConfirmed) {\n if (isConfirmed) {\n $form.submit();\n }\n });\n }\n\n /*\n\n Displays a sweetalert error message\n\n @param alert_text {String} Text (or HTML) to display\n\n */\n window.showError = function(error_text) {\n sweetAlert({\n title: 'Error',\n text: error_text,\n html: true,\n type: 'warning',\n allowEscapeKey: true,\n showCancelButton: false\n });\n }\n\n window.sweetDeleteConfirmation = function(button) {\n sweetAlert({\n title: 'Are you sure?',\n text: button.dataset.confirmationMessage,\n html: true,\n type: 'warning',\n allowEscapeKey: true,\n showCancelButton: true,\n customClass: 'sweetalert-pandle-theme',\n }, function(isConfirmed) {\n if (isConfirmed) {\n $.ajax({\n dataType: 'script',\n url: button.href,\n method: button.dataset['method'] || \"delete\"\n });\n }\n });\n }\n\n window.loadSweetDeleteActions = function() {\n var $deleteButtons = $(\"[data-confirmation-style='sweet']\");\n if ($deleteButtons.length === 0) { return }\n $deleteButtons.off(\"click\").on(\"click\", function(e) {\n e.stopImmediatePropagation();\n e.preventDefault();\n sweetDeleteConfirmation(this);\n });\n }\n})();\n","(function() {\n 'use strict';\n\n // Use this when we want to trigger a Google conversion in a js.erb response instead of an html.erb response\n window.trackGoogleConversion = function(google_conversion_id, google_conversion_label, user_id) {\n var image = new Image(1, 1);\n image.src = '//www.googleadservices.com/pagead/conversion/' + google_conversion_id +\n '/?value=1.00¤cy_code=GBP&label=' + google_conversion_label +\n '&guid=ON&oid=' + user_id;\n }\n\n // Use this when we want to trigger a LinkedIn conversion in a js.erb response instead of an html.erb response\n window.trackLinkedInConversion = function(linkedin_conversion_id) {\n var image = new Image(1, 1);\n image.src = 'https://dc.ads.linkedin.com/collect/?pid=55726&conversionId=' + linkedin_conversion_id + '&fmt=gif';\n }\n})();\n","/**\n* @fileoverview Class and helper method to trigger browser 'are you sure' alerts\n* when the user attempts to leave a page with unsaved data.\n*/\n\n(function() {\n 'use strict';\n\n /**\n * Legacy interface\n * @param form_parent_id\n */\n window.checkFormChangedOnPageUnload = function(form_parent_id) {\n // return new AreYouSureController($(form_parent_id)).initialize();\n };\n /**\n * Class that takes a form element and FormChangeObserver object and adds an\n * event listener to trigger a browser dialogue requesting user confirmation\n * if they try to unload the page after the form has been changed.\n *\n * This class can be disposed, which will remove any attached event listeners\n * and those of the FormChangeObserver via its own dispose method.\n *\n * @param $form_parent {jQuery}\n * @param observer {FormChangeObserver=} defaults to FormChangeObserver\n * @constructor\n */\n window.AreYouSureController = function($form_parent, observer) {\n observer = observer || new FormChangeObserver($form_parent);\n this._observer = observer;\n this._pageExitMessage = null;\n this._listeners = [];\n this._isDisposed = false;\n }\n /**\n * @type {string}\n */\n AreYouSureController.defaultPageExitMessage = 'You have entered data into a form on this page.';\n /**\n *\n * @returns {AreYouSureController}\n */\n AreYouSureController.prototype.initialize = function() {\n this._addPageUnloadEventListener(); // for non-turbo requests\n this._addTurboEventListener(); // for turbo requests\n return this;\n };\n /**\n * Set a non-default page exit message\n */\n AreYouSureController.prototype.setPageExitMessage = function(message) {\n this._pageExitMessage = message;\n }\n /**\n * Clean up form observer.\n */\n AreYouSureController.prototype.dispose = function() {\n if(!this._isDisposed) {\n this._observer.dispose();\n this._observer = null;\n this._pageExitMessage = null;\n this._removeEventListeners();\n this._isDisposed = true;\n }\n };\n /**\n *\n * @private\n */\n AreYouSureController.prototype._addPageUnloadEventListener = function() {\n this._addEventListener(window, 'beforeunload', this._onBeforeUnload.bind(this));\n };\n /**\n * When this callback is fired, the browser will display a confirmation\n * dialogue so the user can interrupt page unloading. This only happens if a\n * string is returned. Some browsers will display the contents of this string,\n * however the latest versions of Chrome and Firefox will display a generic\n * message instead.\n *\n * @param event\n * @returns {string|void} Message to display in browser alert window, if\n * supported\n * @private\n */\n AreYouSureController.prototype._onBeforeUnload = function(event) {\n if (this._observer.isStale()) {\n this.dispose();\n\n var msg = AreYouSureController.defaultPageExitMessage;\n AreYouSureController._setEventReturnValue(event, msg);\n\n return msg;\n }\n };\n /**\n * This is to cope with browser quirks.\n *\n * @param event\n * @param msg {string}\n * @returns {string}\n * @private\n * @static\n */\n AreYouSureController._setEventReturnValue = function(event, msg) {\n (event || window.event).returnValue = msg;\n };\n /**\n * @private\n */\n AreYouSureController.prototype._addTurboEventListener = function() {\n this._addEventListener(document, 'turbo:before-visit', this._onBeforeVisit.bind(this));\n };\n /**\n * When this callback is fired, the browser will display a confirmation\n * dialogue so the user can interrupt page unloading.\n *\n * @param event\n * @private\n */\n AreYouSureController.prototype._onBeforeVisit = function(event) {\n if (this._observer.isStale()) {\n var msg = this._pageExitMessage || AreYouSureController.defaultPageExitMessage;\n this.dispose();\n\n if ( window.hasOwnProperty('confirm')) {\n // window.confirm is optional, but exists in all modern browsers\n // In IE7, we just won't have an 'are you sure' - seems acceptable?\n // https://developer.mozilla.org/en-US/docs/Web/API/Window/confirm\n var are_they_sure = confirm('Are you sure you want to leave this page? \\n ' + msg);\n\n AreYouSureController._setEventReturnValue(event, are_they_sure);\n }\n }\n };\n /**\n * Add an event listener and store the arguments so we can remove it later\n * @param target\n * @param type\n * @param handler\n * @private\n */\n AreYouSureController.prototype._addEventListener = function(target, type, handler) {\n target.addEventListener(type, handler);\n this._listeners.push({ target: target, type: type, handler: handler});\n };\n /**\n *\n * @private\n */\n AreYouSureController.prototype._removeEventListeners = function() {\n this._listeners.forEach(function(ob) {\n ob.target.removeEventListener(ob.type, ob.handler);\n });\n };\n})();\n","(function(){\n window.bootgrid = window.bootgrid || {};\n \n bootgrid.queryHelpers = {\n \n getHeaderColumnIds : function($table){\n return $table.find('thead tr').children().get().map(getColumnId);\n },\n \n getRowCellMap: function(row_index, col_ids, $table){\n var ob = {};\n col_ids.forEach(assignEachRowCellToMap(ob, row_index, $table));\n return ob;\n },\n \n getCellAt : function($table, col, row) {\n return $(this.getRowAt($table, row).children()[ col ]);\n },\n \n getRowAt : function($table, index) {\n return $($table.find('tbody tr')[ index ]);\n }\n \n };\n \n function getColumnId(elem) {\n var col_id = elem.dataset.columnId;\n \n if(col_id && col_id.length){\n return col_id\n } else {\n console.error(elem);\n throw new Error('Could get column id from bootgrid header column element');\n }\n }\n \n function assignEachRowCellToMap(ob, row_index, $table){\n return function(col_id, col_index){\n ob[ col_id ] = bootgrid.queryHelpers.getCellAt($table, col_index, row_index);\n };\n }\n \n})();\n","(function() {\n 'use strict';\n\n window.bootstrap = {\n show: function($elem) {\n return $elem.removeClass('hidden');\n },\n hide: function($elem) {\n return $elem.addClass('hidden');\n }\n }\n})();\n","(function() {\n 'use strict';\n\n // Function to initialize a Bootstrap Wizard\n // Optionally pass in a function that should be executed when a tab is shown\n window.initBootstrapWizard = function(on_tab_show_function) {\n $('.form-wizard-basic').bootstrapWizard({\n onTabShow: function(tab, navigation, index) {\n _wizardProgressBar(navigation, index);\n\n if ( typeof on_tab_show_function !== 'undefined' ){\n on_tab_show_function(tab, navigation, index);\n }\n },\n tabClass: 'fw-nav',\n 'nextSelector': '.next',\n 'previousSelector': '.previous',\n 'finishSelector': '.finish'\n });\n _focusFirstTabWithError();\n };\n\n function _wizardProgressBar(navigation, index) {\n var total = navigation.find('li').length;\n var current = index + 1;\n var percent = (current / total) * 100;\n $('.form-wizard-basic').find('.bar').css({ width: percent + '%' });\n };\n\n function _focusFirstTabWithError() {\n // This method isn't very nice :( but the plugin doesn't currently expose any nicer way\n\n var first_problem_field;\n\n if ($('#error_explanation').length > 0) {\n first_problem_field = $('#error_explanation');\n } else {\n return;\n }\n\n var first_problem_step = $(first_problem_field).closest('.tab-pane');\n var step_id = $(first_problem_step).attr('id');\n\n $(\"a[href*='\"+step_id+\"']\").click();\n };\n})();\n","\nwindow.checkCreditLimitOnSubmit = function() {\n $('.credit_limit_check_button').on('click', function(e) {\n e.stopImmediatePropagation();\n e.preventDefault();\n\n if ($('.credit_limit_check_button').prop('disabled') === true) {\n return\n } // They double-clicked and the previous request hasn't finished yet\n\n // Disable the buttons so they can't click again\n $.rails.disableFormElements($('#invoice-save-options'))\n $.rails.disableElement($('#save_and_recur_button'))\n \n var transactions = getAllRows().map(function(row){ return {\n 'invoice_id': row[0],\n 'customer_id': row[1],\n 'amount': formatWithoutDelimiter(row[2])\n }\n });\n\n $.post(\"/credit_checks\", { \"credit_checks_attributes\": transactions},function(resp) {showCreditLimitWarning(resp, e)} ,'json')\n });\n}\n\nfunction getAllRows(){\n if (isBatchForm()){\n return $('tr.invoice_row').map(function(index, row){\n var wrappedRow = wrapBatchInvoiceRow(row)\n return [[wrappedRow.getRow().data('invoice-id'), wrappedRow.getCustomerSupplierSelect().val(), wrappedRow.getTotalAmountTf().val()]]\n }).toArray();\n }else{\n return [[$('.non_batch_invoices_form').data('invoice-id'), $('#customer_select_single').val(), $('#cd_total_amount_tf').val()]]\n }\n}\n\nfunction showCreditLimitWarning(resp, e){\n if (resp.length > 0){\n if (isRecurringSubmit(e)) {\n initRecurringBreachAlert(e, e.target.value, resp)\n return;\n }\n\n $(\"#modal .modal-header h4\").html(\"Exceeding Credit Limit\");\n\n // Content Body\n if (resp.length > 1) { // http://stackoverflow.com/a/6700/3767922\n $(\"#modal .modal-body-and-footer\").html(\n BATCH_BREACH_MODAL_CONTENT + BREACH_MODAL_CONTENT_SHARED\n );\n multipleCustomerUpdateList(resp);\n } else {\n $(\"#modal .modal-body-and-footer\").html(\n NON_BATCH_BREACH_MODAL_CONTENT + BREACH_MODAL_CONTENT_SHARED\n );\n singleCustomerUpdateList(resp);\n }\n\n // Init button and show modal\n initAllowBreachButton(e, e.target.value); // After .html(BREACH_MODAL_CONTENT)\n\n $('#modal').on('hidden.bs.modal', function(){\n $('.credit_limit_check_button').removeAttr(\"disabled\");\n $.rails.enableElement($('.credit_limit_check_button'))\n });\n\n $(\"#modal\").modal();\n }else{\n continueSubmit(e, e.target.value);\n }\n}\n\nfunction isBatchForm() {\n if(typeof window.isBatchCache !== 'undefined'){\n return window.isBatchCache\n }\n\n if ($('#invoice_form').length > 0) {\n window.isBatchCache = true\n return window.isBatchCache\n }\n\n if ( _isNonBatchNewForm() || isNonBatchEditForm() ) {\n window.isBatchCache = false\n return window.isBatchCache\n }\n\n throw new Error('isBatchForm() did not find an expected form');\n}\n\nfunction isRecurringSubmit(event) {\n return event.target.classList.contains('recurring_submit_button')\n}\n\nfunction _isNonBatchNewForm(){\n return $('#new_sales_invoice').length > 0 || $('#new_sales_credit_note').length > 0;\n}\n\nwindow.isNonBatchEditForm = function(){\n return $('[id^=\"edit_sales_invoice\"], [id^=\"edit_sales_credit_note\"], [id^=\"edit_customer_group_invoice\"]').length > 0;\n}\n\nfunction initAllowBreachButton(event, button_text) {\n $('#allow_breach').on('click', function() {\n continueSubmit(event, button_text);\n });\n $('#do_not_breach').on('click', resetSubmitButtons)\n}\n\nfunction initRecurringBreachAlert(event, button_text, customers) {\n var alert_text = getSweetAlertText(customers)\n\n resetSubmitButtons()\n sweetAlert({\n title: \"Exceeding Credit Limit\",\n text: alert_text,\n html: true,\n allowEscapeKey: false,\n showCancelButton: true,\n confirmButtonText: 'Continue',\n cancelButtonText: 'Cancel',\n customClass: 'sweetalert-pandle-theme'\n }, function(isConfirmed) {\n if(isConfirmed) { continueSubmit(event, button_text); } else { return; }\n });\n}\n\nfunction getSweetAlertText(customers) {\n if (customers.length > 1) {\n var alert_text = \"Continuing will put the following customers over their credit limit: \"\n alert_text += customers.map(function(customer) { return customer.name }).join(', ')\n alert_text += \". Are you sure you want to continue?\"\n } else {\n var customer = customers[0];\n var alert_text = \"Continuing will put \" + customer.name +\n \" over their credit limit of \" + currencySymbol() + customer.credit_limit + \". \" +\n customer.name + \"'s new balance will be \" + currencySymbol() + newAccountCurrencyBalance(customer) +\n \". Are you sure you want to continue?\"\n }\n return alert_text\n}\n\nfunction resetSubmitButtons() {\n $.rails.enableFormElements($('#invoice-save-options'))\n $('.credit_limit_check_button').removeClass('waves-effect')\n $.rails.enableElement($('#save_and_recur_button'))\n}\n\nfunction continueSubmit(event, button_text) {\n if (isRecurringSubmit(event)) {\n resetSubmitButtons();\n return recurringContinueSubmit()\n } else if (isBatchForm()) {\n return batchContinueSubmit();\n } else {\n return nonBatchContinueSubmit(button_text);\n }\n}\n\nfunction batchContinueSubmit() {\n $('#invoice_form').submit();\n}\n\nfunction nonBatchContinueSubmit(button_text) {\n if (_isNonBatchNewForm() || isNonBatchEditForm()) {\n // Submit with commit param for Save/Next Save/View, etc\n // http://stackoverflow.com/a/5097295/3767922\n var $form = $('#new_sales_invoice, #new_sales_credit_note, [id^=\"edit_sales_invoice\"], [id^=\"edit_sales_credit_note\"], [id^=\"edit_customer_group_invoice\"]');\n $form.append(\"\");\n $form.submit();\n }\n}\n\nfunction recurringContinueSubmit() {\n $('#recurring_transactions_form').submit();\n}\n\nfunction singleCustomerUpdateList(customers) {\n $.each(customers, function(index, customer) {\n $('span#customer_name').text(customer.name);\n $('span#customer_amount').text(currencySymbol() + formatAsCurrency(customer.credit_limit));\n $('span#account_currency_balance').text(currencySymbol() + newAccountCurrencyBalance(customer));\n });\n}\n\nfunction multipleCustomerUpdateList(customers) {\n $.each(customers, function(index, customer) {\n $('table.customer_list_table').append(\n \" |
\" +\n \"
Continuing will put \" +\n \" \" +\n \"over their credit limit of \" +\n \".
\" +\n \"\" +\n \"'s new balance will be \" +\n \".\" +\n \"
\"\n);\n\nvar BATCH_BREACH_MODAL_CONTENT = (\n \"
\" +\n \"
Continuing will put the following customers over their credit limit:
\" +\n \"
\" +\n \"
\" +\n \"\" +\n \"Customer | \" +\n \"Credit Limit | \" +\n \"New Account Balance | \" +\n \"
\" +\n \"
\" +\n \"\"\n);\n\nvar BREACH_MODAL_CONTENT_SHARED = (\n \"
Are you sure you want to continue?
\" +\n \"
\" +\n \"\"\n);\n","(function() {\n\n window.copyToClipboardOnClick = function() {\n if ($('.copy-link-button').length === 0) { return; }\n $('.copy-link-button').click(function(event) {\n event.preventDefault();\n event.stopImmediatePropagation();\n var link = $(this).attr('data-transaction-link');\n var dummy_span = $('
').val(link).appendTo('body').select();\n document.execCommand(\"Copy\");\n $(dummy_span).remove();\n notify('Link Copied to Clipboard!', 'success');\n });\n }\n})();","(function() {\n 'use strict';\n\n // Methods to do useful things involving checkboxes\n\n var SELECTED_PATTERN = /\\d+/;\n\n window.updateSelectedCount = function() {\n var $elem = $('#selected_count');\n updateSelectedCountContentAndVisibility($elem, $elem.text(), getSelectedCount());\n }\n\n function getSelectedCount() {\n return $('.checkboxes:checked').not(':disabled').length\n }\n\n function updateSelectedCountContentAndVisibility($elem, text, count) {\n if (count > 0) {\n $elem.text(text.replace(SELECTED_PATTERN, count));\n bootstrap.show($elem);\n } else {\n bootstrap.hide($elem);\n }\n }\n\n /**\n * Global helper to show a section of a page when a checkbox is checked,\n * and hide it when unchecked\n *\n * @param parent_selector {string} jQuery selector for the parent containing the checkbox and section\n * @param checkbox_selector {string} jQuery selector for the checkbox\n * @param section_selector {string} jQuery selector for the section to show/hide\n * @param fields_selector {string} (Optional) jQuery selector for the form fields to enable/disable (if not inside\n * section_selector)\n */\n window.showOrHideSectionWithCheckbox = function(parent_selector, checkbox_selector, section_selector, fields_selector) {\n var $parent = $(parent_selector);\n var disabled = !$parent.find(checkbox_selector).is(':checked');\n var $section = $parent.find(section_selector);\n var $fields_container = typeof(fields_selector) === 'undefined' ? $section : $(fields_selector);\n\n if ($section.is(':visible') == disabled) {\n $section.slideToggle();\n }\n\n $fields_container.find('input:not(.always-disabled, .hidden_date_format_field)').prop('disabled', disabled);\n $fields_container.find('textarea').prop('disabled', disabled);\n $fields_container.find('select').prop('disabled', disabled).trigger('chosen:updated');\n }\n\n window.showOrHideAndNullSectionWithCheckbox = function(parent_selector, checkbox_selector, section_selector, fields_selector) {\n var $parent = $(parent_selector);\n var disabled = !$parent.find(checkbox_selector).is(':checked');\n var $section = $parent.find(section_selector);\n var $fields_container = typeof(fields_selector) === 'undefined' ? $section : $(fields_selector);\n\n if ($section.is(':visible') == disabled) {\n $section.slideToggle();\n }\n\n if (disabled) {\n $fields_container.find('input:not(.always-disabled, .hidden_date_format_field)').val(null);\n $fields_container.find('textarea').val(null);\n $fields_container.find('select').val(null).trigger('chosen:updated');\n }\n\n $fields_container.find('input:not(.always-disabled, .hidden_date_format_field)').prop('disabled', disabled);\n $fields_container.find('textarea').prop('disabled', disabled);\n $fields_container.find('select').prop('disabled', disabled).trigger('chosen:updated');\n }\n\n window.checkboxReadOnly = function($checkbox, readOnly){\n var $hiddenHelper = $checkbox.parent().find(':hidden');\n if (readOnly) {\n $hiddenHelper.val($checkbox.val());\n $checkbox.prop('disabled','true');\n }else{\n $hiddenHelper.val(\"0\");\n $checkbox.removeProp('disabled');\n }\n }\n\n window.addCheckboxAreYouSure = function($checkbox, message){\n $checkbox.click(areYouSureThisIsFinal(message))\n }\n\n /* FARMPLAN OVERRIDE */\n window.addCheckboxAreYouDoubleSure = function($checkbox, message, second_message) {\n $checkbox.click(areYouDoubleSureThisIsFinal(message, second_message))\n }\n\n function areYouSureThisIsFinal(message){\n return function(event){\n if ( $(event.target).is(':checked') ){\n return confirmCheckboxSubmission(message, $(event.target));\n } else {\n // It was checked, and we're unchecking - no confirmation needed\n return true;\n }\n }\n }\n\n /* FARMPLAN OVERRIDE */\n function areYouDoubleSureThisIsFinal(message, second_message) {\n return function(event){\n if ($(event.target).is(':checked')) {\n return doubleConfirmCheckboxSubmission(message, $(event.target), second_message)\n } else {\n // It was checked, and we're unchecking - no confirmation needed\n return true\n }\n }\n }\n\n /*\n\n Displays a sweetalert confirmation, and only leaves the checkbox checked if the user confirms\n\n @param alert_text {String} Text (or HTML) to display\n @param $checkbox {jQuery} Checkbox to check they meant to check\n\n */\n window.confirmCheckboxSubmission = function(alert_text, $checkbox, css) {\n sweetAlert({\n title: 'Warning!',\n text: alert_text,\n html: true,\n type: 'warning',\n allowEscapeKey: false,\n showCancelButton: true,\n confirmButtonText: 'Yes',\n cancelButtonText: 'No',\n customClass: 'sweetalert-scary-warning ' + css\n }, function(isConfirmed) {\n if ( isConfirmed ){\n return true;\n } else {\n $checkbox.removeAttr('checked');\n }\n });\n }\n\n /* FARMPLAN OVERRIDE*/\n window.doubleConfirmCheckboxSubmission = function(alert_text, $checkbox, second_alert_text) {\n sweetAlert({\n title: 'Warning!',\n text: alert_text,\n html: true,\n type: 'warning',\n allowEscapeKey: false,\n showCancelButton: true,\n confirmButtonText: 'Yes',\n cancelButtonText: 'No',\n customClass: 'sweetalert-scary-warning'\n }, function(isConfirmed) {\n if ( isConfirmed ){\n setTimeout(\n () => window.confirmCheckboxSubmission(\n second_alert_text, $checkbox, 'sweetalert-extra-scary-warning'\n ), 500\n )\n } else {\n $checkbox.removeAttr('checked');\n }\n });\n }\n\n})();\n","// TEMPORARY OVERRIDE - THIS FILE CAN BE REMOVED ONCE THE CHANGES COME THROUGH FROM PANDLE\n(function() {\n 'use strict';\n\n window.DATE_POPOVER_ERROR_CONTENT = '
The date entered is invalid or incomplete
';\n\n var DATE_HELP_WARNING_CONTENT = '
The date entered is outside of the current program period.
' +\n '
Did you mean to enter a date within the range specified on the Categories page?
';\n var INVOICE_DATE_WARNING_CONTENT = '
The date entered is prior to the invoice date
';\n var date_warning_enabled;\n\n window.addDatePeriodWarning = function(datepicker_div) {\n try {\n _setDatePeriodWarning(getDateFromDateTimePicker($(datepicker_div)));\n } catch (e) {\n if (e instanceof DatePickerError) {\n console.warn('addDatePeriodWarning called before date picker initialised');\n } else {\n throw e\n }\n }\n };\n\n window.setDateWarningEnabled = function(enabled) {\n date_warning_enabled = enabled;\n };\n\n window.addInvoiceDateWarning = function(datepicker_div) {\n try {\n var current_row = $(datepicker_div).closest('tr')[0];\n var bt_type = getBankTransactionType(current_row);\n\n if(bt_type === 'SalesReceipt' || bt_type === 'PurchasePayment' ) {\n _setInvoiceDateWarning(getDateFromDateTimePicker($(datepicker_div)), current_row);\n }\n } catch(e) {\n if (e instanceof DatePickerError) {\n console.warn('addInvoiceDateWarning called before date picker initialised');\n } else {\n throw e\n }\n }\n };\n\n window.dateIsValid = function(datepicker) {\n //inputmask doesn't allow entering an invalid date, so we only need to check for blank dates here\n return datepicker && datepicker.date();\n };\n\n window.dateInFuture = function(date) {\n return date.isAfter(moment());\n };\n\n window.addDateInFutureWarning = function(datepicker_div) {\n try {\n var date = getDateFromDateTimePicker(datepicker_div);\n\n if (date && dateInFuture(date)) {\n addHelpMessage($(datepicker_div), HELP_WARNING, \"The transaction date is in the future i.e. after today's date\");\n }\n } catch (e) {\n if (e instanceof DatePickerError) {\n console.warn('addDateInFutureWarning called before date picker initialised');\n } else {\n throw e\n }\n }\n };\n\n // Use this helper to get a Moment date object from an input that *doesn't* have a DateTimePicker\n window.dateFromDateInput = function(obj) {\n var format = obj.data('format');\n return moment.utc(obj.val(), format);\n };\n\n window.getClosestDateTimePicker = function($elem) {\n return getDateTimePicker($elem.closest('.datepicker-component'));\n };\n\n // Use this when we want to send a date in DD/MM/YYYY format as a URL param for an ajax request\n window.getGbDateStringFromClosestDateTimePicker = function($elem) {\n var dtp = getClosestDateTimePicker($elem);\n if (dtp) {\n return formatDateGb(dtp.date());\n } else {\n return;\n }\n }\n\n // Use this helper to get a DateTimePicker object from an input with a DateTimePicker\n window.getDateTimePicker = function($elem) {\n return $elem.data('DateTimePicker') || $elem.find('input').data('DateTimePicker');\n };\n\n window.adjustDatesToFitMinMax = function(datePicker) {\n if(datePicker.minDate() && (datePicker.date() < datePicker.minDate())){\n datePicker.date(datePicker.minDate())\n }else if (datePicker.maxDate() && (datePicker.date() > datePicker.maxDate())) {\n datePicker.date(datePicker.maxDate())\n }\n };\n\n window.getDateFromDateTimePicker = function($elem) {\n onceAfterDatepickersComponentHaveBeenInitialised(function(){\n var dp = getDateTimePicker($elem);\n\n if(typeof dp == 'object' && dp.date) {\n return dp.date();\n } else {\n throw new DatePickerError();\n }\n })\n };\n\n window.formatDate = function(date, format) {\n if ( date && typeof date == 'object' && date._isAMomentObject ) {\n return date.format(format);\n }\n };\n\n window.formatDateGb = function(date) {\n return formatDate(date, 'DD/MM/YYYY');\n };\n\n window.formatDateForCompany = function(date) {\n return formatDate(date, companySettings.company_date_format);\n };\n\n window.getBankTransactionType = function(row) {\n return $(row).find('.bt_type_field').val();\n };\n\n // Use this helper to get a Moment date object from an input that might or might not have a DateTimePicker\n // (eg the BT form where the date field might be disabled)\n window.dateFromElement = function(element) {\n var picker = getClosestDateTimePicker(element);\n\n if (picker) {\n return picker.date();\n } else {\n return dateFromDateInput(element);\n }\n }\n\n /**\n * Raised when anything goes wrong to do with datepickers.\n *\n * @constructor\n * @extends Error\n */\n\n function DatePickerError(message) {\n if(typeof message !== 'undefined') {\n this.message = message;\n }\n\n this.stack = new Error().stack;\n }\n\n DatePickerError.prototype = Object.create(Error.prototype);\n DatePickerError.prototype.constructor = DatePickerError;\n DatePickerError.prototype.name = 'DatePickerError';\n DatePickerError.prototype.message = 'Element does not have a datepicker attached';\n\n window.DatePickerError = DatePickerError;\n\n /**\n * Add an event listener to a start date picker so that, when changed, the end\n * date picker's value will be updated according to the given interval.\n *\n * For the interval parameter,\n * @see{http://momentjs.com/docs/#/parsing/object/}\n *\n * @param {jQuery} $start jQuery selector that contains a DateTimePicker\n * @param {Object} interval\n * @param {jQuery} $end jQuery selector that contains a DateTimePicker\n */\n window.setEndDateOnStartDateChange = function($start, interval, $end) {\n $start.on('dp.change', function() {\n setDatePickerWithInterval(getDateFromDateTimePicker($start), $end, interval);\n });\n };\n\n function _setDatePeriodWarning(date) {\n if(date) {\n getJsonOrShowMaintenanceModal('/companies/date_in_period?date=' + encodeURIComponent(date), function(isInPeriod) {\n if (!isInPeriod && date_warning_enabled) {\n addHelpMessage(null, HELP_WARNING,DATE_HELP_WARNING_CONTENT);\n }\n }).fail(function() {\n addHelpMessage(null, HELP_ERROR,DATE_POPOVER_ERROR_CONTENT.substr(3, DATE_POPOVER_ERROR_CONTENT.length - 4));\n });\n }\n }\n\n function _getLatestInvoiceDate(row_id) {\n // Finds the latest invoice date with a non-zero amount\n var lastDate = 0;\n var rows = $('#inv-tbody'+row_id).children('tr');\n rows.each(function() {\n var uid = $(this).data('id');\n var payment_field = $('#total_payment'+uid);\n if ( payment_field.val() && payment_field.val() > 0 ) {\n var inv_date = dateFromDateInput($('#date'+uid));\n if ( inv_date > lastDate ) {\n lastDate = inv_date;\n }\n }\n });\n return lastDate;\n }\n\n function _setInvoiceDateWarning(date, current_row) {\n var invoiceDate = _getLatestInvoiceDate($(current_row).data('id'));\n\n if(date && invoiceDate && date.isBefore(invoiceDate)) {\n addHelpMessage(null,HELP_ERROR,INVOICE_DATE_WARNING_CONTENT.substr(3,INVOICE_DATE_WARNING_CONTENT.length-4));\n }\n }\n\n function setDatePickerWithInterval(base, $dp, interval) {\n if (base) {\n getDateTimePicker($dp).date(base.add(interval).subtract({ days: 1 }));\n }\n }\n\n document.addEventListener('turbo:load', errors.tryFn(function() {\n setDateWarningEnabled(true);\n }));\n})();\n","(function(window) {\n 'use strict';\n $(document).on(\"turbo:load show.bs.modal nested:fieldAdded\", function(e) {\n $(\"[data-deselect]\").on(\"change\", function(e) {\n var toggleTarget = $(this).data('deselect');\n if ($(this).prop('checked')) {\n $(toggleTarget).prop('checked', false);\n }\n });\n });\n})(this);\n","(function() {\n 'use strict';\n\n window.dom = window.dom || {};\n\n dom.createButton = function(colour, icon, label) {\n var button = $('