diff options
Diffstat (limited to 'build/resources/main/static/plugins/datatables-searchbuilder/js/dataTables.searchBuilder.js')
-rw-r--r-- | build/resources/main/static/plugins/datatables-searchbuilder/js/dataTables.searchBuilder.js | 3797 |
1 files changed, 3797 insertions, 0 deletions
diff --git a/build/resources/main/static/plugins/datatables-searchbuilder/js/dataTables.searchBuilder.js b/build/resources/main/static/plugins/datatables-searchbuilder/js/dataTables.searchBuilder.js new file mode 100644 index 0000000..317013e --- /dev/null +++ b/build/resources/main/static/plugins/datatables-searchbuilder/js/dataTables.searchBuilder.js @@ -0,0 +1,3797 @@ +/*! SearchBuilder 1.3.1 + * ©SpryMedia Ltd - datatables.net/license/mit + */ +(function () { + 'use strict'; + + var $$2; + var dataTable$2; + // eslint-disable-next-line no-extra-parens + var moment = window.moment; + // eslint-disable-next-line no-extra-parens + var luxon = window.luxon; + /** + * Sets the value of jQuery for use in the file + * + * @param jq the instance of jQuery to be set + */ + function setJQuery$2(jq) { + $$2 = jq; + dataTable$2 = jq.fn.dataTable; + } + /** + * The Criteria class is used within SearchBuilder to represent a search criteria + */ + var Criteria = /** @class */ (function () { + function Criteria(table, opts, topGroup, index, depth) { + var _this = this; + if (index === void 0) { index = 0; } + if (depth === void 0) { depth = 1; } + // Check that the required version of DataTables is included + if (!dataTable$2 || !dataTable$2.versionCheck || !dataTable$2.versionCheck('1.10.0')) { + throw new Error('SearchPane requires DataTables 1.10 or newer'); + } + this.classes = $$2.extend(true, {}, Criteria.classes); + // Get options from user and any extra conditions/column types defined by plug-ins + this.c = $$2.extend(true, {}, Criteria.defaults, $$2.fn.dataTable.ext.searchBuilder, opts); + var i18n = this.c.i18n; + this.s = { + condition: undefined, + conditions: {}, + data: undefined, + dataIdx: -1, + dataPoints: [], + dateFormat: false, + depth: depth, + dt: table, + filled: false, + index: index, + origData: undefined, + topGroup: topGroup, + type: '', + value: [] + }; + this.dom = { + buttons: $$2('<div/>') + .addClass(this.classes.buttonContainer), + condition: $$2('<select disabled/>') + .addClass(this.classes.condition) + .addClass(this.classes.dropDown) + .addClass(this.classes.italic) + .attr('autocomplete', 'hacking'), + conditionTitle: $$2('<option value="" disabled selected hidden/>') + .html(this.s.dt.i18n('searchBuilder.condition', i18n.condition)), + container: $$2('<div/>') + .addClass(this.classes.container), + data: $$2('<select/>') + .addClass(this.classes.data) + .addClass(this.classes.dropDown) + .addClass(this.classes.italic), + dataTitle: $$2('<option value="" disabled selected hidden/>') + .html(this.s.dt.i18n('searchBuilder.data', i18n.data)), + defaultValue: $$2('<select disabled/>') + .addClass(this.classes.value) + .addClass(this.classes.dropDown) + .addClass(this.classes.select) + .addClass(this.classes.italic), + "delete": $$2('<button/>') + .html(this.s.dt.i18n('searchBuilder.delete', i18n["delete"])) + .addClass(this.classes["delete"]) + .addClass(this.classes.button) + .attr('title', this.s.dt.i18n('searchBuilder.deleteTitle', i18n.deleteTitle)) + .attr('type', 'button'), + // eslint-disable-next-line no-useless-escape + left: $$2('<button/>') + .html(this.s.dt.i18n('searchBuilder.left', i18n.left)) + .addClass(this.classes.left) + .addClass(this.classes.button) + .attr('title', this.s.dt.i18n('searchBuilder.leftTitle', i18n.leftTitle)) + .attr('type', 'button'), + // eslint-disable-next-line no-useless-escape + right: $$2('<button/>') + .html(this.s.dt.i18n('searchBuilder.right', i18n.right)) + .addClass(this.classes.right) + .addClass(this.classes.button) + .attr('title', this.s.dt.i18n('searchBuilder.rightTitle', i18n.rightTitle)) + .attr('type', 'button'), + value: [ + $$2('<select disabled/>') + .addClass(this.classes.value) + .addClass(this.classes.dropDown) + .addClass(this.classes.italic) + .addClass(this.classes.select) + ], + valueTitle: $$2('<option value="--valueTitle--" disabled selected hidden/>') + .html(this.s.dt.i18n('searchBuilder.value', i18n.value)) + }; + // If the greyscale option is selected then add the class to add the grey colour to SearchBuilder + if (this.c.greyscale) { + this.dom.data.addClass(this.classes.greyscale); + this.dom.condition.addClass(this.classes.greyscale); + this.dom.defaultValue.addClass(this.classes.greyscale); + for (var _i = 0, _a = this.dom.value; _i < _a.length; _i++) { + var val = _a[_i]; + val.addClass(this.classes.greyscale); + } + } + // For responsive design, adjust the criterias properties on the following events + this.s.dt.on('draw.dtsb', function () { + _this._adjustCriteria(); + }); + this.s.dt.on('buttons-action.dtsb', function () { + _this._adjustCriteria(); + }); + $$2(window).on('resize.dtsb', dataTable$2.util.throttle(function () { + _this._adjustCriteria(); + })); + this._buildCriteria(); + return this; + } + /** + * Escape html characters within a string + * + * @param txt the string to be escaped + * @returns the escaped string + */ + Criteria._escapeHTML = function (txt) { + return txt + .toString() + .replace(/&/g, '&') + .replace(/</g, '<') + .replace(/>/g, '>') + .replace(/"/g, '"'); + }; + /** + * Adds the left button to the criteria + */ + Criteria.prototype.updateArrows = function (hasSiblings, redraw) { + if (hasSiblings === void 0) { hasSiblings = false; } + if (redraw === void 0) { redraw = true; } + // Empty the container and append all of the elements in the correct order + this.dom.container.children().detach(); + this.dom.container + .append(this.dom.data) + .append(this.dom.condition) + .append(this.dom.value[0]); + this.setListeners(); + // Trigger the inserted events for the value elements as they are inserted + if (this.dom.value[0] !== undefined) { + this.dom.value[0].trigger('dtsb-inserted'); + } + for (var i = 1; i < this.dom.value.length; i++) { + this.dom.container.append(this.dom.value[i]); + this.dom.value[i].trigger('dtsb-inserted'); + } + // If this is a top level criteria then don't let it move left + if (this.s.depth > 1) { + this.dom.buttons.append(this.dom.left); + } + // If the depthLimit of the query has been hit then don't add the right button + if ((this.c.depthLimit === false || this.s.depth < this.c.depthLimit) && hasSiblings) { + this.dom.buttons.append(this.dom.right); + } + else { + this.dom.right.remove(); + } + this.dom.buttons.append(this.dom["delete"]); + this.dom.container.append(this.dom.buttons); + if (redraw) { + // A different combination of arrows and selectors may lead to a need for responsive to be triggered + this._adjustCriteria(); + } + }; + /** + * Destroys the criteria, removing listeners and container from the dom + */ + Criteria.prototype.destroy = function () { + // Turn off listeners + this.dom.data.off('.dtsb'); + this.dom.condition.off('.dtsb'); + this.dom["delete"].off('.dtsb'); + for (var _i = 0, _a = this.dom.value; _i < _a.length; _i++) { + var val = _a[_i]; + val.off('.dtsb'); + } + // Remove container from the dom + this.dom.container.remove(); + }; + /** + * Passes in the data for the row and compares it against this single criteria + * + * @param rowData The data for the row to be compared + * @returns boolean Whether the criteria has passed + */ + Criteria.prototype.search = function (rowData, rowIdx) { + var condition = this.s.conditions[this.s.condition]; + if (this.s.condition !== undefined && condition !== undefined) { + var filter = rowData[this.s.dataIdx]; + // This check is in place for if a custom decimal character is in place + if (this.s.type.includes('num') && + (this.s.dt.settings()[0].oLanguage.sDecimal !== '' || + this.s.dt.settings()[0].oLanguage.sThousands !== '')) { + var splitRD = [rowData[this.s.dataIdx]]; + if (this.s.dt.settings()[0].oLanguage.sDecimal !== '') { + splitRD = rowData[this.s.dataIdx].split(this.s.dt.settings()[0].oLanguage.sDecimal); + } + if (this.s.dt.settings()[0].oLanguage.sThousands !== '') { + for (var i = 0; i < splitRD.length; i++) { + splitRD[i] = splitRD[i].replace(this.s.dt.settings()[0].oLanguage.sThousands, ','); + } + } + filter = splitRD.join('.'); + } + // If orthogonal data is in place we need to get it's values for searching + if (this.c.orthogonal.search !== 'filter') { + var settings = this.s.dt.settings()[0]; + filter = settings.oApi._fnGetCellData(settings, rowIdx, this.s.dataIdx, typeof this.c.orthogonal === 'string' ? + this.c.orthogonal : + this.c.orthogonal.search); + } + if (this.s.type === 'array') { + // Make sure we are working with an array + if (!Array.isArray(filter)) { + filter = [filter]; + } + filter.sort(); + for (var _i = 0, filter_1 = filter; _i < filter_1.length; _i++) { + var filt = filter_1[_i]; + if (filt && typeof filt === 'string') { + filt = filt.replace(/[\r\n\u2028]/g, ' '); + } + } + } + else if (filter !== null && typeof filter === 'string') { + filter = filter.replace(/[\r\n\u2028]/g, ' '); + } + if (this.s.type.includes('html') && typeof filter === 'string') { + filter = filter.replace(/(<([^>]+)>)/ig, ''); + } + // Not ideal, but jqueries .val() returns an empty string even + // when the value set is null, so we shall assume the two are equal + if (filter === null) { + filter = ''; + } + return condition.search(filter, this.s.value, this); + } + }; + /** + * Gets the details required to rebuild the criteria + */ + Criteria.prototype.getDetails = function (deFormatDates) { + if (deFormatDates === void 0) { deFormatDates = false; } + // This check is in place for if a custom decimal character is in place + if (this.s.type !== null && + this.s.type.includes('num') && + (this.s.dt.settings()[0].oLanguage.sDecimal !== '' || this.s.dt.settings()[0].oLanguage.sThousands !== '')) { + for (var i = 0; i < this.s.value.length; i++) { + var splitRD = [this.s.value[i].toString()]; + if (this.s.dt.settings()[0].oLanguage.sDecimal !== '') { + splitRD = this.s.value[i].split(this.s.dt.settings()[0].oLanguage.sDecimal); + } + if (this.s.dt.settings()[0].oLanguage.sThousands !== '') { + for (var j = 0; j < splitRD.length; j++) { + splitRD[j] = splitRD[j].replace(this.s.dt.settings()[0].oLanguage.sThousands, ','); + } + } + this.s.value[i] = splitRD.join('.'); + } + } + else if (this.s.type !== null && deFormatDates) { + if (this.s.type.includes('date') || + this.s.type.includes('time')) { + for (var i = 0; i < this.s.value.length; i++) { + if (this.s.value[i].match(/^\d{4}-([0]\d|1[0-2])-([0-2]\d|3[01])$/g) === null) { + this.s.value[i] = ''; + } + } + } + else if (this.s.type.includes('moment')) { + for (var i = 0; i < this.s.value.length; i++) { + this.s.value[i] = moment(this.s.value[i], this.s.dateFormat).toISOString(); + } + } + else if (this.s.type.includes('luxon')) { + for (var i = 0; i < this.s.value.length; i++) { + this.s.value[i] = luxon.DateTime.fromFormat(this.s.value[i], this.s.dateFormat).toISO(); + } + } + } + if (this.s.type.includes('num') && this.s.dt.page.info().serverSide) { + for (var i = 0; i < this.s.value.length; i++) { + this.s.value[i] = this.s.value[i].replace(/[^0-9.]/g, ''); + } + } + return { + condition: this.s.condition, + data: this.s.data, + origData: this.s.origData, + type: this.s.type, + value: this.s.value.map(function (a) { return a.toString(); }) + }; + }; + /** + * Getter for the node for the container of the criteria + * + * @returns JQuery<HTMLElement> the node for the container + */ + Criteria.prototype.getNode = function () { + return this.dom.container; + }; + /** + * Populates the criteria data, condition and value(s) as far as has been selected + */ + Criteria.prototype.populate = function () { + this._populateData(); + // If the column index has been found attempt to select a condition + if (this.s.dataIdx !== -1) { + this._populateCondition(); + // If the condittion has been found attempt to select the values + if (this.s.condition !== undefined) { + this._populateValue(); + } + } + }; + /** + * Rebuilds the criteria based upon the details passed in + * + * @param loadedCriteria the details required to rebuild the criteria + */ + Criteria.prototype.rebuild = function (loadedCriteria) { + // Check to see if the previously selected data exists, if so select it + var foundData = false; + var dataIdx; + this._populateData(); + // If a data selection has previously been made attempt to find and select it + if (loadedCriteria.data !== undefined) { + var italic_1 = this.classes.italic; + var data_1 = this.dom.data; + this.dom.data.children('option').each(function () { + if (!foundData && + ($$2(this).text() === loadedCriteria.data || + loadedCriteria.origData && $$2(this).prop('origData') === loadedCriteria.origData)) { + $$2(this).prop('selected', true); + data_1.removeClass(italic_1); + foundData = true; + dataIdx = $$2(this).val(); + } + else { + $$2(this).removeProp('selected'); + } + }); + } + // If the data has been found and selected then the condition can be populated and searched + if (foundData) { + this.s.data = loadedCriteria.data; + this.s.origData = loadedCriteria.origData; + this.s.dataIdx = dataIdx; + this.c.orthogonal = this._getOptions().orthogonal; + this.dom.dataTitle.remove(); + this._populateCondition(); + this.dom.conditionTitle.remove(); + var condition = void 0; + // Check to see if the previously selected condition exists, if so select it + var options = this.dom.condition.children('option'); + // eslint-disable-next-line @typescript-eslint/prefer-for-of + for (var i = 0; i < options.length; i++) { + var option = $$2(options[i]); + if (loadedCriteria.condition !== undefined && + option.val() === loadedCriteria.condition && + typeof loadedCriteria.condition === 'string') { + option.prop('selected', true); + condition = option.val(); + } + else { + option.removeProp('selected'); + } + } + this.s.condition = condition; + // If the condition has been found and selected then the value can be populated and searched + if (this.s.condition !== undefined) { + this.dom.conditionTitle.removeProp('selected'); + this.dom.conditionTitle.remove(); + this.dom.condition.removeClass(this.classes.italic); + // eslint-disable-next-line @typescript-eslint/prefer-for-of + for (var i = 0; i < options.length; i++) { + var option = $$2(options[i]); + if (option.val() !== this.s.condition) { + option.removeProp('selected'); + } + } + this._populateValue(loadedCriteria); + } + else { + this.dom.conditionTitle.prependTo(this.dom.condition).prop('selected', true); + } + } + }; + /** + * Sets the listeners for the criteria + */ + Criteria.prototype.setListeners = function () { + var _this = this; + this.dom.data + .unbind('change') + .on('change.dtsb', function () { + _this.dom.dataTitle.removeProp('selected'); + // Need to go over every option to identify the correct selection + var options = _this.dom.data.children('option.' + _this.classes.option); + // eslint-disable-next-line @typescript-eslint/prefer-for-of + for (var i = 0; i < options.length; i++) { + var option = $$2(options[i]); + if (option.val() === _this.dom.data.val()) { + _this.dom.data.removeClass(_this.classes.italic); + option.prop('selected', true); + _this.s.dataIdx = +option.val(); + _this.s.data = option.text(); + _this.s.origData = option.prop('origData'); + _this.c.orthogonal = _this._getOptions().orthogonal; + // When the data is changed, the values in condition and + // value may also change so need to renew them + _this._clearCondition(); + _this._clearValue(); + _this._populateCondition(); + // If this criteria was previously active in the search then + // remove it from the search and trigger a new search + if (_this.s.filled) { + _this.s.filled = false; + _this.s.dt.draw(); + _this.setListeners(); + } + _this.s.dt.state.save(); + } + else { + option.removeProp('selected'); + } + } + }); + this.dom.condition + .unbind('change') + .on('change.dtsb', function () { + _this.dom.conditionTitle.removeProp('selected'); + // Need to go over every option to identify the correct selection + var options = _this.dom.condition.children('option.' + _this.classes.option); + // eslint-disable-next-line @typescript-eslint/prefer-for-of + for (var i = 0; i < options.length; i++) { + var option = $$2(options[i]); + if (option.val() === _this.dom.condition.val()) { + _this.dom.condition.removeClass(_this.classes.italic); + option.prop('selected', true); + var condDisp = option.val(); + // Find the condition that has been selected and store it internally + for (var _i = 0, _a = Object.keys(_this.s.conditions); _i < _a.length; _i++) { + var cond = _a[_i]; + if (cond === condDisp) { + _this.s.condition = condDisp; + break; + } + } + // When the condition is changed, the value selector may switch between + // a select element and an input element + _this._clearValue(); + _this._populateValue(); + for (var _b = 0, _c = _this.dom.value; _b < _c.length; _b++) { + var val = _c[_b]; + // If this criteria was previously active in the search then remove + // it from the search and trigger a new search + if (_this.s.filled && val !== undefined && _this.dom.container.has(val[0]).length !== 0) { + _this.s.filled = false; + _this.s.dt.draw(); + _this.setListeners(); + } + } + if (_this.dom.value.length === 0 || + _this.dom.value.length === 1 && _this.dom.value[0] === undefined) { + _this.s.dt.draw(); + } + } + else { + option.removeProp('selected'); + } + } + }); + }; + /** + * Adjusts the criteria to make SearchBuilder responsive + */ + Criteria.prototype._adjustCriteria = function () { + // If this criteria is not present then don't bother adjusting it + if ($$2(document).has(this.dom.container).length === 0) { + return; + } + var valRight; + var valWidth; + var outmostval = this.dom.value[this.dom.value.length - 1]; + // Calculate the width and right value of the outmost value element + if (outmostval !== undefined && this.dom.container.has(outmostval[0]).length !== 0) { + valWidth = outmostval.outerWidth(true); + valRight = outmostval.offset().left + valWidth; + } + else { + return; + } + var leftOffset = this.dom.left.offset(); + var rightOffset = this.dom.right.offset(); + var clearOffset = this.dom["delete"].offset(); + var hasLeft = this.dom.container.has(this.dom.left[0]).length !== 0; + var hasRight = this.dom.container.has(this.dom.right[0]).length !== 0; + var buttonsLeft = hasLeft ? + leftOffset.left : + hasRight ? + rightOffset.left : + clearOffset.left; + // Perform the responsive calculations and redraw where necessary + if ((buttonsLeft - valRight < 15 || + hasLeft && leftOffset.top !== clearOffset.top || + hasRight && rightOffset.top !== clearOffset.top) && + !this.dom.container.parent().hasClass(this.classes.vertical)) { + this.dom.container.parent().addClass(this.classes.vertical); + this.s.topGroup.trigger('dtsb-redrawContents'); + } + else if (buttonsLeft - + (this.dom.data.offset().left + + this.dom.data.outerWidth(true) + + this.dom.condition.outerWidth(true) + + valWidth) > 15 + && this.dom.container.parent().hasClass(this.classes.vertical)) { + this.dom.container.parent().removeClass(this.classes.vertical); + this.s.topGroup.trigger('dtsb-redrawContents'); + } + }; + /** + * Builds the elements of the dom together + */ + Criteria.prototype._buildCriteria = function () { + // Append Titles for select elements + this.dom.data.append(this.dom.dataTitle); + this.dom.condition.append(this.dom.conditionTitle); + // Add elements to container + this.dom.container + .append(this.dom.data) + .append(this.dom.condition); + for (var _i = 0, _a = this.dom.value; _i < _a.length; _i++) { + var val = _a[_i]; + val.append(this.dom.valueTitle); + this.dom.container.append(val); + } + // Add buttons to container + this.dom.container + .append(this.dom["delete"]) + .append(this.dom.right); + this.setListeners(); + }; + /** + * Clears the condition select element + */ + Criteria.prototype._clearCondition = function () { + this.dom.condition.empty(); + this.dom.conditionTitle.prop('selected', true).attr('disabled', 'true'); + this.dom.condition.prepend(this.dom.conditionTitle).prop('selectedIndex', 0); + this.s.conditions = {}; + this.s.condition = undefined; + }; + /** + * Clears the value elements + */ + Criteria.prototype._clearValue = function () { + if (this.s.condition !== undefined) { + if (this.dom.value.length > 0 && this.dom.value[0] !== undefined) { + var _loop_1 = function (val) { + if (val !== undefined) { + // Timeout is annoying but because of IOS + setTimeout(function () { + val.remove(); + }, 50); + } + }; + // Remove all of the value elements + for (var _i = 0, _a = this.dom.value; _i < _a.length; _i++) { + var val = _a[_i]; + _loop_1(val); + } + } + // Call the init function to get the value elements for this condition + this.dom.value = [].concat(this.s.conditions[this.s.condition].init(this, Criteria.updateListener)); + if (this.dom.value.length > 0 && this.dom.value[0] !== undefined) { + this.dom.value[0].insertAfter(this.dom.condition).trigger('dtsb-inserted'); + // Insert all of the value elements + for (var i = 1; i < this.dom.value.length; i++) { + this.dom.value[i].insertAfter(this.dom.value[i - 1]).trigger('dtsb-inserted'); + } + } + } + else { + var _loop_2 = function (val) { + if (val !== undefined) { + // Timeout is annoying but because of IOS + setTimeout(function () { + val.remove(); + }, 50); + } + }; + // Remove all of the value elements + for (var _b = 0, _c = this.dom.value; _b < _c.length; _b++) { + var val = _c[_b]; + _loop_2(val); + } + // Append the default valueTitle to the default select element + this.dom.valueTitle + .prop('selected', true); + this.dom.defaultValue + .append(this.dom.valueTitle) + .insertAfter(this.dom.condition); + } + this.s.value = []; + this.dom.value = [ + $$2('<select disabled/>') + .addClass(this.classes.value) + .addClass(this.classes.dropDown) + .addClass(this.classes.italic) + .addClass(this.classes.select) + .append(this.dom.valueTitle.clone()) + ]; + }; + /** + * Gets the options for the column + * + * @returns {object} The options for the column + */ + Criteria.prototype._getOptions = function () { + var table = this.s.dt; + return $$2.extend(true, {}, Criteria.defaults, table.settings()[0].aoColumns[this.s.dataIdx].searchBuilder); + }; + /** + * Populates the condition dropdown + */ + Criteria.prototype._populateCondition = function () { + var conditionOpts = []; + var conditionsLength = Object.keys(this.s.conditions).length; + // If there are no conditions stored then we need to get them from the appropriate type + if (conditionsLength === 0) { + var column = +this.dom.data.children('option:selected').val(); + this.s.type = this.s.dt.columns().type().toArray()[column]; + var colInits = this.s.dt.settings()[0].aoColumns; + if (colInits !== undefined) { + var colInit = colInits[column]; + if (colInit.searchBuilderType !== undefined && colInit.searchBuilderType !== null) { + this.s.type = colInit.searchBuilderType; + } + else if (this.s.type === undefined || this.s.type === null) { + this.s.type = colInit.sType; + } + } + // If the column type is still unknown, call a draw to try reading it again + if (this.s.type === null || this.s.type === undefined) { + $$2.fn.dataTable.ext.oApi._fnColumnTypes(this.s.dt.settings()[0]); + this.s.type = this.s.dt.columns().type().toArray()[column]; + } + // Enable the condition element + this.dom.condition + .removeAttr('disabled') + .empty() + .append(this.dom.conditionTitle) + .addClass(this.classes.italic); + this.dom.conditionTitle + .prop('selected', true); + var decimal = this.s.dt.settings()[0].oLanguage.sDecimal; + // This check is in place for if a custom decimal character is in place + if (decimal !== '' && this.s.type.indexOf(decimal) === this.s.type.length - decimal.length) { + if (this.s.type.includes('num-fmt')) { + this.s.type = this.s.type.replace(decimal, ''); + } + else if (this.s.type.includes('num')) { + this.s.type = this.s.type.replace(decimal, ''); + } + } + // Select which conditions are going to be used based on the column type + var conditionObj = this.c.conditions[this.s.type] !== undefined ? + this.c.conditions[this.s.type] : + this.s.type.includes('moment') ? + this.c.conditions.moment : + this.s.type.includes('luxon') ? + this.c.conditions.luxon : + this.c.conditions.string; + // If it is a moment format then extract the date format + if (this.s.type.includes('moment')) { + this.s.dateFormat = this.s.type.replace(/moment-/g, ''); + } + else if (this.s.type.includes('luxon')) { + this.s.dateFormat = this.s.type.replace(/luxon-/g, ''); + } + // Add all of the conditions to the select element + for (var _i = 0, _a = Object.keys(conditionObj); _i < _a.length; _i++) { + var condition = _a[_i]; + if (conditionObj[condition] !== null) { + // Serverside processing does not supply the options for the select elements + // Instead input elements need to be used for these instead + if (this.s.dt.page.info().serverSide && conditionObj[condition].init === Criteria.initSelect) { + conditionObj[condition].init = Criteria.initInput; + conditionObj[condition].inputValue = Criteria.inputValueInput; + conditionObj[condition].isInputValid = Criteria.isInputValidInput; + } + this.s.conditions[condition] = conditionObj[condition]; + var condName = conditionObj[condition].conditionName; + if (typeof condName === 'function') { + condName = condName(this.s.dt, this.c.i18n); + } + conditionOpts.push($$2('<option>', { + text: condName, + value: condition + }) + .addClass(this.classes.option) + .addClass(this.classes.notItalic)); + } + } + } + // Otherwise we can just load them in + else if (conditionsLength > 0) { + this.dom.condition.empty().removeAttr('disabled').addClass(this.classes.italic); + for (var _b = 0, _c = Object.keys(this.s.conditions); _b < _c.length; _b++) { + var condition = _c[_b]; + var condName = this.s.conditions[condition].conditionName; + if (typeof condName === 'function') { + condName = condName(this.s.dt, this.c.i18n); + } + var newOpt = $$2('<option>', { + text: condName, + value: condition + }) + .addClass(this.classes.option) + .addClass(this.classes.notItalic); + if (this.s.condition !== undefined && this.s.condition === condName) { + newOpt.prop('selected', true); + this.dom.condition.removeClass(this.classes.italic); + } + conditionOpts.push(newOpt); + } + } + else { + this.dom.condition + .attr('disabled', 'true') + .addClass(this.classes.italic); + return; + } + for (var _d = 0, conditionOpts_1 = conditionOpts; _d < conditionOpts_1.length; _d++) { + var opt = conditionOpts_1[_d]; + this.dom.condition.append(opt); + } + this.dom.condition.prop('selectedIndex', 0); + }; + /** + * Populates the data select element + */ + Criteria.prototype._populateData = function () { + var _this = this; + this.dom.data.empty().append(this.dom.dataTitle); + // If there are no datas stored then we need to get them from the table + if (this.s.dataPoints.length === 0) { + this.s.dt.columns().every(function (index) { + // Need to check that the column can be filtered on before adding it + if (_this.c.columns === true || + _this.s.dt.columns(_this.c.columns).indexes().toArray().includes(index)) { + var found = false; + for (var _i = 0, _a = _this.s.dataPoints; _i < _a.length; _i++) { + var val = _a[_i]; + if (val.index === index) { + found = true; + break; + } + } + if (!found) { + var col = _this.s.dt.settings()[0].aoColumns[index]; + var opt = { + index: index, + origData: col.data, + text: (col.searchBuilderTitle === undefined ? + col.sTitle : + col.searchBuilderTitle).replace(/(<([^>]+)>)/ig, '') + }; + _this.s.dataPoints.push(opt); + _this.dom.data.append($$2('<option>', { + text: opt.text, + value: opt.index + }) + .addClass(_this.classes.option) + .addClass(_this.classes.notItalic) + .prop('origData', col.data) + .prop('selected', _this.s.dataIdx === opt.index ? true : false)); + if (_this.s.dataIdx === opt.index) { + _this.dom.dataTitle.removeProp('selected'); + } + } + } + }); + } + // Otherwise we can just load them in + else { + var _loop_3 = function (data) { + this_1.s.dt.columns().every(function (index) { + var col = _this.s.dt.settings()[0].aoColumns[index]; + if ((col.searchBuilderTitle === undefined ? + col.sTitle : + col.searchBuilderTitle).replace(/(<([^>]+)>)/ig, '') === data.text) { + data.index = index; + data.origData = col.data; + } + }); + var newOpt = $$2('<option>', { + text: data.text.replace(/(<([^>]+)>)/ig, ''), + value: data.index + }) + .addClass(this_1.classes.option) + .addClass(this_1.classes.notItalic) + .prop('origData', data.origData); + if (this_1.s.data === data.text) { + this_1.s.dataIdx = data.index; + this_1.dom.dataTitle.removeProp('selected'); + newOpt.prop('selected', true); + this_1.dom.data.removeClass(this_1.classes.italic); + } + this_1.dom.data.append(newOpt); + }; + var this_1 = this; + for (var _i = 0, _a = this.s.dataPoints; _i < _a.length; _i++) { + var data = _a[_i]; + _loop_3(data); + } + } + }; + /** + * Populates the Value select element + * + * @param loadedCriteria optional, used to reload criteria from predefined filters + */ + Criteria.prototype._populateValue = function (loadedCriteria) { + var _this = this; + var prevFilled = this.s.filled; + this.s.filled = false; + // Remove any previous value elements + // Timeout is annoying but because of IOS + setTimeout(function () { + _this.dom.defaultValue.remove(); + }, 50); + var _loop_4 = function (val) { + // Timeout is annoying but because of IOS + setTimeout(function () { + if (val !== undefined) { + val.remove(); + } + }, 50); + }; + for (var _i = 0, _a = this.dom.value; _i < _a.length; _i++) { + var val = _a[_i]; + _loop_4(val); + } + var children = this.dom.container.children(); + if (children.length > 3) { + for (var i = 2; i < children.length - 1; i++) { + $$2(children[i]).remove(); + } + } + // Find the column with the title matching the data for the criteria and take note of the index + if (loadedCriteria !== undefined) { + this.s.dt.columns().every(function (index) { + if (_this.s.dt.settings()[0].aoColumns[index].sTitle === loadedCriteria.data) { + _this.s.dataIdx = index; + } + }); + } + // Initialise the value elements based on the condition + this.dom.value = [].concat(this.s.conditions[this.s.condition].init(this, Criteria.updateListener, loadedCriteria !== undefined ? loadedCriteria.value : undefined)); + if (loadedCriteria !== undefined && loadedCriteria.value !== undefined) { + this.s.value = loadedCriteria.value; + } + // Insert value elements and trigger the inserted event + if (this.dom.value[0] !== undefined) { + this.dom.value[0] + .insertAfter(this.dom.condition) + .trigger('dtsb-inserted'); + } + for (var i = 1; i < this.dom.value.length; i++) { + this.dom.value[i] + .insertAfter(this.dom.value[i - 1]) + .trigger('dtsb-inserted'); + } + // Check if the criteria can be used in a search + this.s.filled = this.s.conditions[this.s.condition].isInputValid(this.dom.value, this); + this.setListeners(); + // If it can and this is different to before then trigger a draw + if (prevFilled !== this.s.filled) { + // If using SSP we want to restrict the amount of server calls that take place + // and this will already have taken place + if (!this.s.dt.page.info().serverSide) { + this.s.dt.draw(); + } + this.setListeners(); + } + }; + /** + * Provides throttling capabilities to SearchBuilder without having to use dt's _fnThrottle function + * This is because that function is not quite suitable for our needs as it runs initially rather than waiting + * + * @param args arguments supplied to the throttle function + * @returns Function that is to be run that implements the throttling + */ + Criteria.prototype._throttle = function (fn, frequency) { + if (frequency === void 0) { frequency = 200; } + var last = null; + var timer = null; + var that = this; + if (frequency === null) { + frequency = 200; + } + return function () { + var args = []; + for (var _i = 0; _i < arguments.length; _i++) { + args[_i] = arguments[_i]; + } + var now = +new Date(); + if (last !== null && now < last + frequency) { + clearTimeout(timer); + } + else { + last = now; + } + timer = setTimeout(function () { + last = null; + fn.apply(that, args); + }, frequency); + }; + }; + Criteria.version = '1.1.0'; + Criteria.classes = { + button: 'dtsb-button', + buttonContainer: 'dtsb-buttonContainer', + condition: 'dtsb-condition', + container: 'dtsb-criteria', + data: 'dtsb-data', + "delete": 'dtsb-delete', + dropDown: 'dtsb-dropDown', + greyscale: 'dtsb-greyscale', + input: 'dtsb-input', + italic: 'dtsb-italic', + joiner: 'dtsp-joiner', + left: 'dtsb-left', + notItalic: 'dtsb-notItalic', + option: 'dtsb-option', + right: 'dtsb-right', + select: 'dtsb-select', + value: 'dtsb-value', + vertical: 'dtsb-vertical' + }; + /** + * Default initialisation function for select conditions + */ + Criteria.initSelect = function (that, fn, preDefined, array) { + if (preDefined === void 0) { preDefined = null; } + if (array === void 0) { array = false; } + var column = that.dom.data.children('option:selected').val(); + var indexArray = that.s.dt.rows().indexes().toArray(); + var settings = that.s.dt.settings()[0]; + that.dom.valueTitle.prop('selected', true); + // Declare select element to be used with all of the default classes and listeners. + var el = $$2('<select/>') + .addClass(Criteria.classes.value) + .addClass(Criteria.classes.dropDown) + .addClass(Criteria.classes.italic) + .addClass(Criteria.classes.select) + .append(that.dom.valueTitle) + .on('change.dtsb', function () { + $$2(this).removeClass(Criteria.classes.italic); + fn(that, this); + }); + if (that.c.greyscale) { + el.addClass(Criteria.classes.greyscale); + } + var added = []; + var options = []; + // Add all of the options from the table to the select element. + // Only add one option for each possible value + for (var _i = 0, indexArray_1 = indexArray; _i < indexArray_1.length; _i++) { + var index = indexArray_1[_i]; + var filter = settings.oApi._fnGetCellData(settings, index, column, typeof that.c.orthogonal === 'string' ? + that.c.orthogonal : + that.c.orthogonal.search); + var value = { + filter: typeof filter === 'string' ? + filter.replace(/[\r\n\u2028]/g, ' ') : // Need to replace certain characters to match search values + filter, + index: index, + text: settings.oApi._fnGetCellData(settings, index, column, typeof that.c.orthogonal === 'string' ? + that.c.orthogonal : + that.c.orthogonal.display) + }; + // If we are dealing with an array type, either make sure we are working with arrays, or sort them + if (that.s.type === 'array') { + value.filter = !Array.isArray(value.filter) ? [value.filter] : value.filter; + value.text = !Array.isArray(value.text) ? [value.text] : value.text; + } + // Function to add an option to the select element + var addOption = function (filt, text) { + if (that.s.type.includes('html') && filt !== null && typeof filt === 'string') { + filt.replace(/(<([^>]+)>)/ig, ''); + } + // Add text and value, stripping out any html if that is the column type + var opt = $$2('<option>', { + type: Array.isArray(filt) ? 'Array' : 'String', + value: filt + }) + .data('sbv', filt) + .addClass(that.classes.option) + .addClass(that.classes.notItalic) + // Have to add the text this way so that special html characters are not escaped - & etc. + .html(typeof text === 'string' ? + text.replace(/(<([^>]+)>)/ig, '') : + text); + var val = opt.val(); + // Check that this value has not already been added + if (added.indexOf(val) === -1) { + added.push(val); + options.push(opt); + if (preDefined !== null && Array.isArray(preDefined[0])) { + preDefined[0] = preDefined[0].sort().join(','); + } + // If this value was previously selected as indicated by preDefined, then select it again + if (preDefined !== null && opt.val() === preDefined[0]) { + opt.prop('selected', true); + el.removeClass(Criteria.classes.italic); + that.dom.valueTitle.removeProp('selected'); + } + } + }; + // If this is to add the individual values within the array we need to loop over the array + if (array) { + for (var i = 0; i < value.filter.length; i++) { + addOption(value.filter[i], value.text[i]); + } + } + // Otherwise the value that is in the cell is to be added + else { + addOption(value.filter, Array.isArray(value.text) ? value.text.join(', ') : value.text); + } + } + options.sort(function (a, b) { + if (that.s.type === 'array' || + that.s.type === 'string' || + that.s.type === 'html') { + if (a.val() < b.val()) { + return -1; + } + else if (a.val() > b.val()) { + return 1; + } + else { + return 0; + } + } + else if (that.s.type === 'num' || + that.s.type === 'html-num') { + if (+a.val().replace(/(<([^>]+)>)/ig, '') < +b.val().replace(/(<([^>]+)>)/ig, '')) { + return -1; + } + else if (+a.val().replace(/(<([^>]+)>)/ig, '') > +b.val().replace(/(<([^>]+)>)/ig, '')) { + return 1; + } + else { + return 0; + } + } + else if (that.s.type === 'num-fmt' || that.s.type === 'html-num-fmt') { + if (+a.val().replace(/[^0-9.]/g, '') < +b.val().replace(/[^0-9.]/g, '')) { + return -1; + } + else if (+a.val().replace(/[^0-9.]/g, '') > +b.val().replace(/[^0-9.]/g, '')) { + return 1; + } + else { + return 0; + } + } + }); + for (var _a = 0, options_1 = options; _a < options_1.length; _a++) { + var opt = options_1[_a]; + el.append(opt); + } + return el; + }; + /** + * Default initialisation function for select array conditions + * + * This exists because there needs to be different select functionality for contains/without and equals/not + */ + Criteria.initSelectArray = function (that, fn, preDefined) { + if (preDefined === void 0) { preDefined = null; } + return Criteria.initSelect(that, fn, preDefined, true); + }; + /** + * Default initialisation function for input conditions + */ + Criteria.initInput = function (that, fn, preDefined) { + if (preDefined === void 0) { preDefined = null; } + // Declare the input element + var searchDelay = that.s.dt.settings()[0].searchDelay; + var el = $$2('<input/>') + .addClass(Criteria.classes.value) + .addClass(Criteria.classes.input) + .on('input.dtsb keypress.dtsb', that._throttle(function (e) { + var code = e.keyCode || e.which; + if (!that.c.enterSearch && + !(that.s.dt.settings()[0].oInit.search !== undefined && + that.s.dt.settings()[0].oInit.search["return"]) || + code === 13) { + return fn(that, this); + } + }, searchDelay === null ? 100 : searchDelay)); + if (that.c.greyscale) { + el.addClass(Criteria.classes.greyscale); + } + // If there is a preDefined value then add it + if (preDefined !== null) { + el.val(preDefined[0]); + } + // This is add responsive functionality to the logic button without redrawing everything else + that.s.dt.one('draw.dtsb', function () { + that.s.topGroup.trigger('dtsb-redrawLogic'); + }); + return el; + }; + /** + * Default initialisation function for conditions requiring 2 inputs + */ + Criteria.init2Input = function (that, fn, preDefined) { + if (preDefined === void 0) { preDefined = null; } + // Declare all of the necessary jQuery elements + var searchDelay = that.s.dt.settings()[0].searchDelay; + var els = [ + $$2('<input/>') + .addClass(Criteria.classes.value) + .addClass(Criteria.classes.input) + .on('input.dtsb keypress.dtsb', that._throttle(function (e) { + var code = e.keyCode || e.which; + if (!that.c.enterSearch && + !(that.s.dt.settings()[0].oInit.search !== undefined && + that.s.dt.settings()[0].oInit.search["return"]) || + code === 13) { + return fn(that, this); + } + }, searchDelay === null ? 100 : searchDelay)), + $$2('<span>') + .addClass(that.classes.joiner) + .html(that.s.dt.i18n('searchBuilder.valueJoiner', that.c.i18n.valueJoiner)), + $$2('<input/>') + .addClass(Criteria.classes.value) + .addClass(Criteria.classes.input) + .on('input.dtsb keypress.dtsb', that._throttle(function (e) { + var code = e.keyCode || e.which; + if (!that.c.enterSearch && + !(that.s.dt.settings()[0].oInit.search !== undefined && + that.s.dt.settings()[0].oInit.search["return"]) || + code === 13) { + return fn(that, this); + } + }, searchDelay === null ? 100 : searchDelay)) + ]; + if (that.c.greyscale) { + els[0].addClass(Criteria.classes.greyscale); + els[2].addClass(Criteria.classes.greyscale); + } + // If there is a preDefined value then add it + if (preDefined !== null) { + els[0].val(preDefined[0]); + els[2].val(preDefined[1]); + } + // This is add responsive functionality to the logic button without redrawing everything else + that.s.dt.one('draw.dtsb', function () { + that.s.topGroup.trigger('dtsb-redrawLogic'); + }); + return els; + }; + /** + * Default initialisation function for date conditions + */ + Criteria.initDate = function (that, fn, preDefined) { + if (preDefined === void 0) { preDefined = null; } + var searchDelay = that.s.dt.settings()[0].searchDelay; + // Declare date element using DataTables dateTime plugin + var el = $$2('<input/>') + .addClass(Criteria.classes.value) + .addClass(Criteria.classes.input) + .dtDateTime({ + attachTo: 'input', + format: that.s.dateFormat ? that.s.dateFormat : undefined + }) + .on('change.dtsb', that._throttle(function () { + return fn(that, this); + }, searchDelay === null ? 100 : searchDelay)) + .on('input.dtsb keypress.dtsb', that.c.enterSearch || + that.s.dt.settings()[0].oInit.search !== undefined && + that.s.dt.settings()[0].oInit.search["return"] ? + function (e) { + that._throttle(function () { + var code = e.keyCode || e.which; + if (code === 13) { + return fn(that, this); + } + }, searchDelay === null ? 100 : searchDelay); + } : + that._throttle(function () { + return fn(that, this); + }, searchDelay === null ? 100 : searchDelay)); + if (that.c.greyscale) { + el.addClass(Criteria.classes.greyscale); + } + // If there is a preDefined value then add it + if (preDefined !== null) { + el.val(preDefined[0]); + } + // This is add responsive functionality to the logic button without redrawing everything else + that.s.dt.one('draw.dtsb', function () { + that.s.topGroup.trigger('dtsb-redrawLogic'); + }); + return el; + }; + Criteria.initNoValue = function (that) { + // This is add responsive functionality to the logic button without redrawing everything else + that.s.dt.one('draw.dtsb', function () { + that.s.topGroup.trigger('dtsb-redrawLogic'); + }); + }; + Criteria.init2Date = function (that, fn, preDefined) { + var _this = this; + if (preDefined === void 0) { preDefined = null; } + var searchDelay = that.s.dt.settings()[0].searchDelay; + // Declare all of the date elements that are required using DataTables dateTime plugin + var els = [ + $$2('<input/>') + .addClass(Criteria.classes.value) + .addClass(Criteria.classes.input) + .dtDateTime({ + attachTo: 'input', + format: that.s.dateFormat ? that.s.dateFormat : undefined + }) + .on('change.dtsb', searchDelay !== null ? + that.s.dt.settings()[0].oApi._fnThrottle(function () { + return fn(that, this); + }, searchDelay) : + function () { + fn(that, _this); + }) + .on('input.dtsb keypress.dtsb', !that.c.enterSearch && + !(that.s.dt.settings()[0].oInit.search !== undefined && + that.s.dt.settings()[0].oInit.search["return"]) && + searchDelay !== null ? + that.s.dt.settings()[0].oApi._fnThrottle(function () { + return fn(that, this); + }, searchDelay) : + that.c.enterSearch || + that.s.dt.settings()[0].oInit.search !== undefined && + that.s.dt.settings()[0].oInit.search["return"] ? + function (e) { + var code = e.keyCode || e.which; + if (code === 13) { + fn(that, _this); + } + } : + function () { + fn(that, _this); + }), + $$2('<span>') + .addClass(that.classes.joiner) + .html(that.s.dt.i18n('searchBuilder.valueJoiner', that.c.i18n.valueJoiner)), + $$2('<input/>') + .addClass(Criteria.classes.value) + .addClass(Criteria.classes.input) + .dtDateTime({ + attachTo: 'input', + format: that.s.dateFormat ? that.s.dateFormat : undefined + }) + .on('change.dtsb', searchDelay !== null ? + that.s.dt.settings()[0].oApi._fnThrottle(function () { + return fn(that, this); + }, searchDelay) : + function () { + fn(that, _this); + }) + .on('input.dtsb keypress.dtsb', !that.c.enterSearch && + !(that.s.dt.settings()[0].oInit.search !== undefined && + that.s.dt.settings()[0].oInit.search["return"]) && + searchDelay !== null ? + that.s.dt.settings()[0].oApi._fnThrottle(function () { + return fn(that, this); + }, searchDelay) : + that.c.enterSearch || + that.s.dt.settings()[0].oInit.search !== undefined && + that.s.dt.settings()[0].oInit.search["return"] ? + function (e) { + var code = e.keyCode || e.which; + if (code === 13) { + fn(that, _this); + } + } : + function () { + fn(that, _this); + }) + ]; + if (that.c.greyscale) { + els[0].addClass(Criteria.classes.greyscale); + els[2].addClass(Criteria.classes.greyscale); + } + // If there are and preDefined values then add them + if (preDefined !== null && preDefined.length > 0) { + els[0].val(preDefined[0]); + els[2].val(preDefined[1]); + } + // This is add responsive functionality to the logic button without redrawing everything else + that.s.dt.one('draw.dtsb', function () { + that.s.topGroup.trigger('dtsb-redrawLogic'); + }); + return els; + }; + /** + * Default function for select elements to validate condition + */ + Criteria.isInputValidSelect = function (el) { + var allFilled = true; + // Check each element to make sure that the selections are valid + for (var _i = 0, el_1 = el; _i < el_1.length; _i++) { + var element = el_1[_i]; + if (element.children('option:selected').length === + element.children('option').length - + element.children('option.' + Criteria.classes.notItalic).length && + element.children('option:selected').length === 1 && + element.children('option:selected')[0] === element.children('option')[0]) { + allFilled = false; + } + } + return allFilled; + }; + /** + * Default function for input and date elements to validate condition + */ + Criteria.isInputValidInput = function (el) { + var allFilled = true; + // Check each element to make sure that the inputs are valid + for (var _i = 0, el_2 = el; _i < el_2.length; _i++) { + var element = el_2[_i]; + if (element.is('input') && element.val().length === 0) { + allFilled = false; + } + } + return allFilled; + }; + /** + * Default function for getting select conditions + */ + Criteria.inputValueSelect = function (el) { + var values = []; + // Go through the select elements and push each selected option to the return array + for (var _i = 0, el_3 = el; _i < el_3.length; _i++) { + var element = el_3[_i]; + if (element.is('select')) { + values.push(Criteria._escapeHTML(element.children('option:selected').data('sbv'))); + } + } + return values; + }; + /** + * Default function for getting input conditions + */ + Criteria.inputValueInput = function (el) { + var values = []; + // Go through the input elements and push each value to the return array + for (var _i = 0, el_4 = el; _i < el_4.length; _i++) { + var element = el_4[_i]; + if (element.is('input')) { + values.push(Criteria._escapeHTML(element.val())); + } + } + return values; + }; + /** + * Function that is run on each element as a call back when a search should be triggered + */ + Criteria.updateListener = function (that, el) { + // When the value is changed the criteria is now complete so can be included in searches + // Get the condition from the map based on the key that has been selected for the condition + var condition = that.s.conditions[that.s.condition]; + that.s.filled = condition.isInputValid(that.dom.value, that); + that.s.value = condition.inputValue(that.dom.value, that); + if (!that.s.filled) { + that.s.dt.draw(); + return; + } + if (!Array.isArray(that.s.value)) { + that.s.value = [that.s.value]; + } + for (var i = 0; i < that.s.value.length; i++) { + // If the value is an array we need to sort it + if (Array.isArray(that.s.value[i])) { + that.s.value[i].sort(); + } + // Otherwise replace the decimal place character for i18n + else if (that.s.type.includes('num') && + (that.s.dt.settings()[0].oLanguage.sDecimal !== '' || + that.s.dt.settings()[0].oLanguage.sThousands !== '')) { + var splitRD = [that.s.value[i].toString()]; + if (that.s.dt.settings()[0].oLanguage.sDecimal !== '') { + splitRD = that.s.value[i].split(that.s.dt.settings()[0].oLanguage.sDecimal); + } + if (that.s.dt.settings()[0].oLanguage.sThousands !== '') { + for (var j = 0; j < splitRD.length; j++) { + splitRD[j] = splitRD[j].replace(that.s.dt.settings()[0].oLanguage.sThousands, ','); + } + } + that.s.value[i] = splitRD.join('.'); + } + } + // Take note of the cursor position so that we can refocus there later + var idx = null; + var cursorPos = null; + for (var i = 0; i < that.dom.value.length; i++) { + if (el === that.dom.value[i][0]) { + idx = i; + if (el.selectionStart !== undefined) { + cursorPos = el.selectionStart; + } + } + } + // Trigger a search + that.s.dt.draw(); + // Refocus the element and set the correct cursor position + if (idx !== null) { + that.dom.value[idx].removeClass(that.classes.italic); + that.dom.value[idx].focus(); + if (cursorPos !== null) { + that.dom.value[idx][0].setSelectionRange(cursorPos, cursorPos); + } + } + }; + // The order of the conditions will make eslint sad :( + // Has to be in this order so that they are displayed correctly in select elements + // Also have to disable member ordering for this as the private methods used are not yet declared otherwise + // eslint-disable-next-line @typescript-eslint/member-ordering + Criteria.dateConditions = { + '=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.equals', i18n.conditions.date.equals); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + value = value.replace(/(\/|-|,)/g, '-'); + return value === comparison[0]; + } + }, + // eslint-disable-next-line sort-keys + '!=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.not', i18n.conditions.date.not); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + value = value.replace(/(\/|-|,)/g, '-'); + return value !== comparison[0]; + } + }, + '<': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.before', i18n.conditions.date.before); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + value = value.replace(/(\/|-|,)/g, '-'); + return value < comparison[0]; + } + }, + '>': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.after', i18n.conditions.date.after); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + value = value.replace(/(\/|-|,)/g, '-'); + return value > comparison[0]; + } + }, + 'between': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.between', i18n.conditions.date.between); + }, + init: Criteria.init2Date, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + value = value.replace(/(\/|-|,)/g, '-'); + if (comparison[0] < comparison[1]) { + return comparison[0] <= value && value <= comparison[1]; + } + else { + return comparison[1] <= value && value <= comparison[0]; + } + } + }, + // eslint-disable-next-line sort-keys + '!between': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.notBetween', i18n.conditions.date.notBetween); + }, + init: Criteria.init2Date, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + value = value.replace(/(\/|-|,)/g, '-'); + if (comparison[0] < comparison[1]) { + return !(comparison[0] <= value && value <= comparison[1]); + } + else { + return !(comparison[1] <= value && value <= comparison[0]); + } + } + }, + 'null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.empty', i18n.conditions.date.empty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return value === null || value === undefined || value.length === 0; + } + }, + // eslint-disable-next-line sort-keys + '!null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.notEmpty', i18n.conditions.date.notEmpty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return !(value === null || value === undefined || value.length === 0); + } + } + }; + // The order of the conditions will make eslint sad :( + // Has to be in this order so that they are displayed correctly in select elements + // Also have to disable member ordering for this as the private methods used are not yet declared otherwise + // eslint-disable-next-line @typescript-eslint/member-ordering + Criteria.momentDateConditions = { + '=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.equals', i18n.conditions.date.equals); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + return moment(value, that.s.dateFormat).valueOf() === + moment(comparison[0], that.s.dateFormat).valueOf(); + } + }, + // eslint-disable-next-line sort-keys + '!=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.not', i18n.conditions.date.not); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + return moment(value, that.s.dateFormat).valueOf() !== + moment(comparison[0], that.s.dateFormat).valueOf(); + } + }, + '<': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.before', i18n.conditions.date.before); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + return moment(value, that.s.dateFormat).valueOf() < moment(comparison[0], that.s.dateFormat).valueOf(); + } + }, + '>': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.after', i18n.conditions.date.after); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + return moment(value, that.s.dateFormat).valueOf() > moment(comparison[0], that.s.dateFormat).valueOf(); + } + }, + 'between': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.between', i18n.conditions.date.between); + }, + init: Criteria.init2Date, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + var val = moment(value, that.s.dateFormat).valueOf(); + var comp0 = moment(comparison[0], that.s.dateFormat).valueOf(); + var comp1 = moment(comparison[1], that.s.dateFormat).valueOf(); + if (comp0 < comp1) { + return comp0 <= val && val <= comp1; + } + else { + return comp1 <= val && val <= comp0; + } + } + }, + // eslint-disable-next-line sort-keys + '!between': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.notBetween', i18n.conditions.date.notBetween); + }, + init: Criteria.init2Date, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + var val = moment(value, that.s.dateFormat).valueOf(); + var comp0 = moment(comparison[0], that.s.dateFormat).valueOf(); + var comp1 = moment(comparison[1], that.s.dateFormat).valueOf(); + if (comp0 < comp1) { + return !(+comp0 <= +val && +val <= +comp1); + } + else { + return !(+comp1 <= +val && +val <= +comp0); + } + } + }, + 'null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.empty', i18n.conditions.date.empty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return value === null || value === undefined || value.length === 0; + } + }, + // eslint-disable-next-line sort-keys + '!null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.notEmpty', i18n.conditions.date.notEmpty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return !(value === null || value === undefined || value.length === 0); + } + } + }; + // The order of the conditions will make eslint sad :( + // Has to be in this order so that they are displayed correctly in select elements + // Also have to disable member ordering for this as the private methods used are not yet declared otherwise + // eslint-disable-next-line @typescript-eslint/member-ordering + Criteria.luxonDateConditions = { + '=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.equals', i18n.conditions.date.equals); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + return luxon.DateTime.fromFormat(value, that.s.dateFormat).ts + === luxon.DateTime.fromFormat(comparison[0], that.s.dateFormat).ts; + } + }, + // eslint-disable-next-line sort-keys + '!=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.not', i18n.conditions.date.not); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + return luxon.DateTime.fromFormat(value, that.s.dateFormat).ts + !== luxon.DateTime.fromFormat(comparison[0], that.s.dateFormat).ts; + } + }, + '<': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.before', i18n.conditions.date.before); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + return luxon.DateTime.fromFormat(value, that.s.dateFormat).ts + < luxon.DateTime.fromFormat(comparison[0], that.s.dateFormat).ts; + } + }, + '>': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.after', i18n.conditions.date.after); + }, + init: Criteria.initDate, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + return luxon.DateTime.fromFormat(value, that.s.dateFormat).ts + > luxon.DateTime.fromFormat(comparison[0], that.s.dateFormat).ts; + } + }, + 'between': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.between', i18n.conditions.date.between); + }, + init: Criteria.init2Date, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + var val = luxon.DateTime.fromFormat(value, that.s.dateFormat).ts; + var comp0 = luxon.DateTime.fromFormat(comparison[0], that.s.dateFormat).ts; + var comp1 = luxon.DateTime.fromFormat(comparison[1], that.s.dateFormat).ts; + if (comp0 < comp1) { + return comp0 <= val && val <= comp1; + } + else { + return comp1 <= val && val <= comp0; + } + } + }, + // eslint-disable-next-line sort-keys + '!between': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.notBetween', i18n.conditions.date.notBetween); + }, + init: Criteria.init2Date, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison, that) { + var val = luxon.DateTime.fromFormat(value, that.s.dateFormat).ts; + var comp0 = luxon.DateTime.fromFormat(comparison[0], that.s.dateFormat).ts; + var comp1 = luxon.DateTime.fromFormat(comparison[1], that.s.dateFormat).ts; + if (comp0 < comp1) { + return !(+comp0 <= +val && +val <= +comp1); + } + else { + return !(+comp1 <= +val && +val <= +comp0); + } + } + }, + 'null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.empty', i18n.conditions.date.empty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return value === null || value === undefined || value.length === 0; + } + }, + // eslint-disable-next-line sort-keys + '!null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.date.notEmpty', i18n.conditions.date.notEmpty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return !(value === null || value === undefined || value.length === 0); + } + } + }; + // The order of the conditions will make eslint sad :( + // Has to be in this order so that they are displayed correctly in select elements + // Also have to disable member ordering for this as the private methods used are not yet declared otherwise + // eslint-disable-next-line @typescript-eslint/member-ordering + Criteria.numConditions = { + '=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.equals', i18n.conditions.number.equals); + }, + init: Criteria.initSelect, + inputValue: Criteria.inputValueSelect, + isInputValid: Criteria.isInputValidSelect, + search: function (value, comparison) { + return +value === +comparison[0]; + } + }, + // eslint-disable-next-line sort-keys + '!=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.not', i18n.conditions.number.not); + }, + init: Criteria.initSelect, + inputValue: Criteria.inputValueSelect, + isInputValid: Criteria.isInputValidSelect, + search: function (value, comparison) { + return +value !== +comparison[0]; + } + }, + '<': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.lt', i18n.conditions.number.lt); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + return +value < +comparison[0]; + } + }, + '<=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.lte', i18n.conditions.number.lte); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + return +value <= +comparison[0]; + } + }, + '>=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.gte', i18n.conditions.number.gte); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + return +value >= +comparison[0]; + } + }, + // eslint-disable-next-line sort-keys + '>': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.gt', i18n.conditions.number.gt); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + return +value > +comparison[0]; + } + }, + 'between': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.between', i18n.conditions.number.between); + }, + init: Criteria.init2Input, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + if (+comparison[0] < +comparison[1]) { + return +comparison[0] <= +value && +value <= +comparison[1]; + } + else { + return +comparison[1] <= +value && +value <= +comparison[0]; + } + } + }, + // eslint-disable-next-line sort-keys + '!between': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.notBetween', i18n.conditions.number.notBetween); + }, + init: Criteria.init2Input, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + if (+comparison[0] < +comparison[1]) { + return !(+comparison[0] <= +value && +value <= +comparison[1]); + } + else { + return !(+comparison[1] <= +value && +value <= +comparison[0]); + } + } + }, + 'null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.empty', i18n.conditions.number.empty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return value === null || value === undefined || value.length === 0; + } + }, + // eslint-disable-next-line sort-keys + '!null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.notEmpty', i18n.conditions.number.notEmpty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return !(value === null || value === undefined || value.length === 0); + } + } + }; + // The order of the conditions will make eslint sad :( + // Has to be in this order so that they are displayed correctly in select elements + // Also have to disable member ordering for this as the private methods used are not yet declared otherwise + // eslint-disable-next-line @typescript-eslint/member-ordering + Criteria.numFmtConditions = { + '=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.equals', i18n.conditions.number.equals); + }, + init: Criteria.initSelect, + inputValue: Criteria.inputValueSelect, + isInputValid: Criteria.isInputValidSelect, + search: function (value, comparison) { + var val = value.indexOf('-') === 0 ? + '-' + value.replace(/[^0-9.]/g, '') : + value.replace(/[^0-9.]/g, ''); + var comp = comparison[0].indexOf('-') === 0 ? + '-' + comparison[0].replace(/[^0-9.]/g, '') : + comparison[0].replace(/[^0-9.]/g, ''); + return +val === +comp; + } + }, + // eslint-disable-next-line sort-keys + '!=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.not', i18n.conditions.number.not); + }, + init: Criteria.initSelect, + inputValue: Criteria.inputValueSelect, + isInputValid: Criteria.isInputValidSelect, + search: function (value, comparison) { + var val = value.indexOf('-') === 0 ? + '-' + value.replace(/[^0-9.]/g, '') : + value.replace(/[^0-9.]/g, ''); + var comp = comparison[0].indexOf('-') === 0 ? + '-' + comparison[0].replace(/[^0-9.]/g, '') : + comparison[0].replace(/[^0-9.]/g, ''); + return +val !== +comp; + } + }, + '<': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.lt', i18n.conditions.number.lt); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + var val = value.indexOf('-') === 0 ? + '-' + value.replace(/[^0-9.]/g, '') : + value.replace(/[^0-9.]/g, ''); + var comp = comparison[0].indexOf('-') === 0 ? + '-' + comparison[0].replace(/[^0-9.]/g, '') : + comparison[0].replace(/[^0-9.]/g, ''); + return +val < +comp; + } + }, + '<=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.lte', i18n.conditions.number.lte); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + var val = value.indexOf('-') === 0 ? + '-' + value.replace(/[^0-9.]/g, '') : + value.replace(/[^0-9.]/g, ''); + var comp = comparison[0].indexOf('-') === 0 ? + '-' + comparison[0].replace(/[^0-9.]/g, '') : + comparison[0].replace(/[^0-9.]/g, ''); + return +val <= +comp; + } + }, + '>=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.gte', i18n.conditions.number.gte); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + var val = value.indexOf('-') === 0 ? + '-' + value.replace(/[^0-9.]/g, '') : + value.replace(/[^0-9.]/g, ''); + var comp = comparison[0].indexOf('-') === 0 ? + '-' + comparison[0].replace(/[^0-9.]/g, '') : + comparison[0].replace(/[^0-9.]/g, ''); + return +val >= +comp; + } + }, + // eslint-disable-next-line sort-keys + '>': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.gt', i18n.conditions.number.gt); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + var val = value.indexOf('-') === 0 ? + '-' + value.replace(/[^0-9.]/g, '') : + value.replace(/[^0-9.]/g, ''); + var comp = comparison[0].indexOf('-') === 0 ? + '-' + comparison[0].replace(/[^0-9.]/g, '') : + comparison[0].replace(/[^0-9.]/g, ''); + return +val > +comp; + } + }, + 'between': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.between', i18n.conditions.number.between); + }, + init: Criteria.init2Input, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + var val = value.indexOf('-') === 0 ? + '-' + value.replace(/[^0-9.]/g, '') : + value.replace(/[^0-9.]/g, ''); + var comp0 = comparison[0].indexOf('-') === 0 ? + '-' + comparison[0].replace(/[^0-9.]/g, '') : + comparison[0].replace(/[^0-9.]/g, ''); + var comp1 = comparison[1].indexOf('-') === 0 ? + '-' + comparison[1].replace(/[^0-9.]/g, '') : + comparison[1].replace(/[^0-9.]/g, ''); + if (+comp0 < +comp1) { + return +comp0 <= +val && +val <= +comp1; + } + else { + return +comp1 <= +val && +val <= +comp0; + } + } + }, + // eslint-disable-next-line sort-keys + '!between': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.notBetween', i18n.conditions.number.notBetween); + }, + init: Criteria.init2Input, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + var val = value.indexOf('-') === 0 ? + '-' + value.replace(/[^0-9.]/g, '') : + value.replace(/[^0-9.]/g, ''); + var comp0 = comparison[0].indexOf('-') === 0 ? + '-' + comparison[0].replace(/[^0-9.]/g, '') : + comparison[0].replace(/[^0-9.]/g, ''); + var comp1 = comparison[1].indexOf('-') === 0 ? + '-' + comparison[1].replace(/[^0-9.]/g, '') : + comparison[1].replace(/[^0-9.]/g, ''); + if (+comp0 < +comp1) { + return !(+comp0 <= +val && +val <= +comp1); + } + else { + return !(+comp1 <= +val && +val <= +comp0); + } + } + }, + 'null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.empty', i18n.conditions.number.empty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return value === null || value === undefined || value.length === 0; + } + }, + // eslint-disable-next-line sort-keys + '!null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.number.notEmpty', i18n.conditions.number.notEmpty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return !(value === null || value === undefined || value.length === 0); + } + } + }; + // The order of the conditions will make eslint sad :( + // Has to be in this order so that they are displayed correctly in select elements + // Also have to disable member ordering for this as the private methods used are not yet declared otherwise + // eslint-disable-next-line @typescript-eslint/member-ordering + Criteria.stringConditions = { + '=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.string.equals', i18n.conditions.string.equals); + }, + init: Criteria.initSelect, + inputValue: Criteria.inputValueSelect, + isInputValid: Criteria.isInputValidSelect, + search: function (value, comparison) { + return value === comparison[0]; + } + }, + // eslint-disable-next-line sort-keys + '!=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.string.not', i18n.conditions.string.not); + }, + init: Criteria.initSelect, + inputValue: Criteria.inputValueSelect, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + return value !== comparison[0]; + } + }, + 'starts': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.string.startsWith', i18n.conditions.string.startsWith); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + return value.toLowerCase().indexOf(comparison[0].toLowerCase()) === 0; + } + }, + // eslint-disable-next-line sort-keys + '!starts': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.string.notStartsWith', i18n.conditions.string.notStartsWith); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + return value.toLowerCase().indexOf(comparison[0].toLowerCase()) !== 0; + } + }, + // eslint-disable-next-line sort-keys + 'contains': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.string.contains', i18n.conditions.string.contains); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + return value.toLowerCase().includes(comparison[0].toLowerCase()); + } + }, + // eslint-disable-next-line sort-keys + '!contains': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.string.notContains', i18n.conditions.string.notContains); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + return !value.toLowerCase().includes(comparison[0].toLowerCase()); + } + }, + 'ends': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.string.endsWith', i18n.conditions.string.endsWith); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + return value.toLowerCase().endsWith(comparison[0].toLowerCase()); + } + }, + // eslint-disable-next-line sort-keys + '!ends': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.string.notEndsWith', i18n.conditions.string.notEndsWith); + }, + init: Criteria.initInput, + inputValue: Criteria.inputValueInput, + isInputValid: Criteria.isInputValidInput, + search: function (value, comparison) { + return !value.toLowerCase().endsWith(comparison[0].toLowerCase()); + } + }, + 'null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.string.empty', i18n.conditions.string.empty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return value === null || value === undefined || value.length === 0; + } + }, + // eslint-disable-next-line sort-keys + '!null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.string.notEmpty', i18n.conditions.string.notEmpty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return !(value === null || value === undefined || value.length === 0); + } + } + }; + // The order of the conditions will make eslint sad :( + // Also have to disable member ordering for this as the private methods used are not yet declared otherwise + // eslint-disable-next-line @typescript-eslint/member-ordering + Criteria.arrayConditions = { + 'contains': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.array.contains', i18n.conditions.array.contains); + }, + init: Criteria.initSelectArray, + inputValue: Criteria.inputValueSelect, + isInputValid: Criteria.isInputValidSelect, + search: function (value, comparison) { + return value.includes(comparison[0]); + } + }, + 'without': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.array.without', i18n.conditions.array.without); + }, + init: Criteria.initSelectArray, + inputValue: Criteria.inputValueSelect, + isInputValid: Criteria.isInputValidSelect, + search: function (value, comparison) { + return value.indexOf(comparison[0]) === -1; + } + }, + // eslint-disable-next-line sort-keys + '=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.array.equals', i18n.conditions.array.equals); + }, + init: Criteria.initSelect, + inputValue: Criteria.inputValueSelect, + isInputValid: Criteria.isInputValidSelect, + search: function (value, comparison) { + if (value.length === comparison[0].length) { + for (var i = 0; i < value.length; i++) { + if (value[i] !== comparison[0][i]) { + return false; + } + } + return true; + } + return false; + } + }, + // eslint-disable-next-line sort-keys + '!=': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.array.not', i18n.conditions.array.not); + }, + init: Criteria.initSelect, + inputValue: Criteria.inputValueSelect, + isInputValid: Criteria.isInputValidSelect, + search: function (value, comparison) { + if (value.length === comparison[0].length) { + for (var i = 0; i < value.length; i++) { + if (value[i] !== comparison[0][i]) { + return true; + } + } + return false; + } + return true; + } + }, + 'null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.array.empty', i18n.conditions.array.empty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return value === null || value === undefined || value.length === 0; + } + }, + // eslint-disable-next-line sort-keys + '!null': { + conditionName: function (dt, i18n) { + return dt.i18n('searchBuilder.conditions.array.notEmpty', i18n.conditions.array.notEmpty); + }, + init: Criteria.initNoValue, + inputValue: function () { + return; + }, + isInputValid: function () { + return true; + }, + search: function (value) { + return value !== null && value !== undefined && value.length !== 0; + } + } + }; + // eslint will be sad because we have to disable member ordering for this as the + // private static properties used are not yet declared otherwise + // eslint-disable-next-line @typescript-eslint/member-ordering + Criteria.defaults = { + columns: true, + conditions: { + 'array': Criteria.arrayConditions, + 'date': Criteria.dateConditions, + 'html': Criteria.stringConditions, + 'html-num': Criteria.numConditions, + 'html-num-fmt': Criteria.numFmtConditions, + 'luxon': Criteria.luxonDateConditions, + 'moment': Criteria.momentDateConditions, + 'num': Criteria.numConditions, + 'num-fmt': Criteria.numFmtConditions, + 'string': Criteria.stringConditions + }, + depthLimit: false, + enterSearch: false, + filterChanged: undefined, + greyscale: false, + i18n: { + add: 'Add Condition', + button: { + 0: 'Search Builder', + _: 'Search Builder (%d)' + }, + clearAll: 'Clear All', + condition: 'Condition', + data: 'Data', + "delete": '×', + deleteTitle: 'Delete filtering rule', + left: '<', + leftTitle: 'Outdent criteria', + logicAnd: 'And', + logicOr: 'Or', + right: '>', + rightTitle: 'Indent criteria', + title: { + 0: 'Custom Search Builder', + _: 'Custom Search Builder (%d)' + }, + value: 'Value', + valueJoiner: 'and' + }, + logic: 'AND', + orthogonal: { + display: 'display', + search: 'filter' + }, + preDefined: false + }; + return Criteria; + }()); + + var $$1; + var dataTable$1; + /** + * Sets the value of jQuery for use in the file + * + * @param jq the instance of jQuery to be set + */ + function setJQuery$1(jq) { + $$1 = jq; + dataTable$1 = jq.fn.dataTable; + } + /** + * The Group class is used within SearchBuilder to represent a group of criteria + */ + var Group = /** @class */ (function () { + function Group(table, opts, topGroup, index, isChild, depth) { + if (index === void 0) { index = 0; } + if (isChild === void 0) { isChild = false; } + if (depth === void 0) { depth = 1; } + // Check that the required version of DataTables is included + if (!dataTable$1 || !dataTable$1.versionCheck || !dataTable$1.versionCheck('1.10.0')) { + throw new Error('SearchBuilder requires DataTables 1.10 or newer'); + } + this.classes = $$1.extend(true, {}, Group.classes); + // Get options from user + this.c = $$1.extend(true, {}, Group.defaults, opts); + this.s = { + criteria: [], + depth: depth, + dt: table, + index: index, + isChild: isChild, + logic: undefined, + opts: opts, + preventRedraw: false, + toDrop: undefined, + topGroup: topGroup + }; + this.dom = { + add: $$1('<button/>') + .addClass(this.classes.add) + .addClass(this.classes.button) + .attr('type', 'button'), + clear: $$1('<button>×</button>') + .addClass(this.classes.button) + .addClass(this.classes.clearGroup) + .attr('type', 'button'), + container: $$1('<div/>') + .addClass(this.classes.group), + logic: $$1('<button><div/></button>') + .addClass(this.classes.logic) + .addClass(this.classes.button) + .attr('type', 'button'), + logicContainer: $$1('<div/>') + .addClass(this.classes.logicContainer) + }; + // A reference to the top level group is maintained throughout any subgroups and criteria that may be created + if (this.s.topGroup === undefined) { + this.s.topGroup = this.dom.container; + } + this._setup(); + return this; + } + /** + * Destroys the groups buttons, clears the internal criteria and removes it from the dom + */ + Group.prototype.destroy = function () { + // Turn off listeners + this.dom.add.off('.dtsb'); + this.dom.logic.off('.dtsb'); + // Trigger event for groups at a higher level to pick up on + this.dom.container + .trigger('dtsb-destroy') + .remove(); + this.s.criteria = []; + }; + /** + * Gets the details required to rebuild the group + */ + // Eslint upset at empty object but needs to be done + // eslint-disable-next-line @typescript-eslint/ban-types + Group.prototype.getDetails = function (deFormatDates) { + if (deFormatDates === void 0) { deFormatDates = false; } + if (this.s.criteria.length === 0) { + return {}; + } + var details = { + criteria: [], + logic: this.s.logic + }; + // NOTE here crit could be either a subgroup or a criteria + for (var _i = 0, _a = this.s.criteria; _i < _a.length; _i++) { + var crit = _a[_i]; + details.criteria.push(crit.criteria.getDetails(deFormatDates)); + } + return details; + }; + /** + * Getter for the node for the container of the group + * + * @returns Node for the container of the group + */ + Group.prototype.getNode = function () { + return this.dom.container; + }; + /** + * Rebuilds the group based upon the details passed in + * + * @param loadedDetails the details required to rebuild the group + */ + Group.prototype.rebuild = function (loadedDetails) { + // If no criteria are stored then just return + if (loadedDetails.criteria === undefined || + loadedDetails.criteria === null || + Array.isArray(loadedDetails.criteria) && loadedDetails.criteria.length === 0) { + return; + } + this.s.logic = loadedDetails.logic; + this.dom.logic.children().first().html(this.s.logic === 'OR' + ? this.s.dt.i18n('searchBuilder.logicOr', this.c.i18n.logicOr) + : this.s.dt.i18n('searchBuilder.logicAnd', this.c.i18n.logicAnd)); + // Add all of the criteria, be it a sub group or a criteria + if (Array.isArray(loadedDetails.criteria)) { + for (var _i = 0, _a = loadedDetails.criteria; _i < _a.length; _i++) { + var crit = _a[_i]; + if (crit.logic !== undefined) { + this._addPrevGroup(crit); + } + else if (crit.logic === undefined) { + this._addPrevCriteria(crit); + } + } + } + // For all of the criteria children, update the arrows incase they require changing and set the listeners + for (var _b = 0, _c = this.s.criteria; _b < _c.length; _b++) { + var crit = _c[_b]; + if (crit.criteria instanceof Criteria) { + crit.criteria.updateArrows(this.s.criteria.length > 1, false); + this._setCriteriaListeners(crit.criteria); + } + } + }; + /** + * Redraws the Contents of the searchBuilder Groups and Criteria + */ + Group.prototype.redrawContents = function () { + if (this.s.preventRedraw) { + return; + } + // Clear the container out and add the basic elements + this.dom.container.children().detach(); + this.dom.container + .append(this.dom.logicContainer) + .append(this.dom.add); + // Sort the criteria by index so that they appear in the correct order + this.s.criteria.sort(function (a, b) { + if (a.criteria.s.index < b.criteria.s.index) { + return -1; + } + else if (a.criteria.s.index > b.criteria.s.index) { + return 1; + } + return 0; + }); + this.setListeners(); + for (var i = 0; i < this.s.criteria.length; i++) { + var crit = this.s.criteria[i].criteria; + if (crit instanceof Criteria) { + // Reset the index to the new value + this.s.criteria[i].index = i; + this.s.criteria[i].criteria.s.index = i; + // Add to the group + this.s.criteria[i].criteria.dom.container.insertBefore(this.dom.add); + // Set listeners for various points + this._setCriteriaListeners(crit); + this.s.criteria[i].criteria.rebuild(this.s.criteria[i].criteria.getDetails()); + } + else if (crit instanceof Group && crit.s.criteria.length > 0) { + // Reset the index to the new value + this.s.criteria[i].index = i; + this.s.criteria[i].criteria.s.index = i; + // Add the sub group to the group + this.s.criteria[i].criteria.dom.container.insertBefore(this.dom.add); + // Redraw the contents of the group + crit.redrawContents(); + this._setGroupListeners(crit); + } + else { + // The group is empty so remove it + this.s.criteria.splice(i, 1); + i--; + } + } + this.setupLogic(); + }; + /** + * Resizes the logic button only rather than the entire dom. + */ + Group.prototype.redrawLogic = function () { + for (var _i = 0, _a = this.s.criteria; _i < _a.length; _i++) { + var crit = _a[_i]; + if (crit instanceof Group) { + crit.redrawLogic(); + } + } + this.setupLogic(); + }; + /** + * Search method, checking the row data against the criteria in the group + * + * @param rowData The row data to be compared + * @returns boolean The result of the search + */ + Group.prototype.search = function (rowData, rowIdx) { + if (this.s.logic === 'AND') { + return this._andSearch(rowData, rowIdx); + } + else if (this.s.logic === 'OR') { + return this._orSearch(rowData, rowIdx); + } + return true; + }; + /** + * Locates the groups logic button to the correct location on the page + */ + Group.prototype.setupLogic = function () { + // Remove logic button + this.dom.logicContainer.remove(); + this.dom.clear.remove(); + // If there are no criteria in the group then keep the logic removed and return + if (this.s.criteria.length < 1) { + if (!this.s.isChild) { + this.dom.container.trigger('dtsb-destroy'); + // Set criteria left margin + this.dom.container.css('margin-left', 0); + } + return; + } + // Set width, take 2 for the border + var height = this.dom.container.height() - 1; + this.dom.clear.height('0px'); + this.dom.logicContainer.append(this.dom.clear).width(height); + // Prepend logic button + this.dom.container.prepend(this.dom.logicContainer); + this._setLogicListener(); + // Set criteria left margin + this.dom.container.css('margin-left', this.dom.logicContainer.outerHeight(true)); + var logicOffset = this.dom.logicContainer.offset(); + // Set horizontal alignment + var currentLeft = logicOffset.left; + var groupLeft = this.dom.container.offset().left; + var shuffleLeft = currentLeft - groupLeft; + var newPos = currentLeft - shuffleLeft - this.dom.logicContainer.outerHeight(true); + this.dom.logicContainer.offset({ left: newPos }); + // Set vertical alignment + var firstCrit = this.dom.logicContainer.next(); + var currentTop = logicOffset.top; + var firstTop = $$1(firstCrit).offset().top; + var shuffleTop = currentTop - firstTop; + var newTop = currentTop - shuffleTop; + this.dom.logicContainer.offset({ top: newTop }); + this.dom.clear.outerHeight(this.dom.logicContainer.height()); + this._setClearListener(); + }; + /** + * Sets listeners on the groups elements + */ + Group.prototype.setListeners = function () { + var _this = this; + this.dom.add.unbind('click'); + this.dom.add.on('click.dtsb', function () { + // If this is the parent group then the logic button has not been added yet + if (!_this.s.isChild) { + _this.dom.container.prepend(_this.dom.logicContainer); + } + _this.addCriteria(); + _this.dom.container.trigger('dtsb-add'); + _this.s.dt.state.save(); + return false; + }); + for (var _i = 0, _a = this.s.criteria; _i < _a.length; _i++) { + var crit = _a[_i]; + crit.criteria.setListeners(); + } + this._setClearListener(); + this._setLogicListener(); + }; + /** + * Adds a criteria to the group + * + * @param crit Instance of Criteria to be added to the group + */ + Group.prototype.addCriteria = function (crit, redraw) { + if (crit === void 0) { crit = null; } + if (redraw === void 0) { redraw = true; } + var index = crit === null ? this.s.criteria.length : crit.s.index; + var criteria = new Criteria(this.s.dt, this.s.opts, this.s.topGroup, index, this.s.depth); + // If a Criteria has been passed in then set the values to continue that + if (crit !== null) { + criteria.c = crit.c; + criteria.s = crit.s; + criteria.s.depth = this.s.depth; + criteria.classes = crit.classes; + } + criteria.populate(); + var inserted = false; + for (var i = 0; i < this.s.criteria.length; i++) { + if (i === 0 && this.s.criteria[i].criteria.s.index > criteria.s.index) { + // Add the node for the criteria at the start of the group + criteria.getNode().insertBefore(this.s.criteria[i].criteria.dom.container); + inserted = true; + } + else if (i < this.s.criteria.length - 1 && + this.s.criteria[i].criteria.s.index < criteria.s.index && + this.s.criteria[i + 1].criteria.s.index > criteria.s.index) { + // Add the node for the criteria in the correct location + criteria.getNode().insertAfter(this.s.criteria[i].criteria.dom.container); + inserted = true; + } + } + if (!inserted) { + criteria.getNode().insertBefore(this.dom.add); + } + // Add the details for this criteria to the array + this.s.criteria.push({ + criteria: criteria, + index: index + }); + this.s.criteria = this.s.criteria.sort(function (a, b) { return a.criteria.s.index - b.criteria.s.index; }); + for (var _i = 0, _a = this.s.criteria; _i < _a.length; _i++) { + var opt = _a[_i]; + if (opt.criteria instanceof Criteria) { + opt.criteria.updateArrows(this.s.criteria.length > 1, redraw); + } + } + this._setCriteriaListeners(criteria); + criteria.setListeners(); + this.setupLogic(); + }; + /** + * Checks the group to see if it has any filled criteria + */ + Group.prototype.checkFilled = function () { + for (var _i = 0, _a = this.s.criteria; _i < _a.length; _i++) { + var crit = _a[_i]; + if (crit.criteria instanceof Criteria && crit.criteria.s.filled || + crit.criteria instanceof Group && crit.criteria.checkFilled()) { + return true; + } + } + return false; + }; + /** + * Gets the count for the number of criteria in this group and any sub groups + */ + Group.prototype.count = function () { + var count = 0; + for (var _i = 0, _a = this.s.criteria; _i < _a.length; _i++) { + var crit = _a[_i]; + if (crit.criteria instanceof Group) { + count += crit.criteria.count(); + } + else { + count++; + } + } + return count; + }; + /** + * Rebuilds a sub group that previously existed + * + * @param loadedGroup The details of a group within this group + */ + Group.prototype._addPrevGroup = function (loadedGroup) { + var idx = this.s.criteria.length; + var group = new Group(this.s.dt, this.c, this.s.topGroup, idx, true, this.s.depth + 1); + // Add the new group to the criteria array + this.s.criteria.push({ + criteria: group, + index: idx, + logic: group.s.logic + }); + // Rebuild it with the previous conditions for that group + group.rebuild(loadedGroup); + this.s.criteria[idx].criteria = group; + this.s.topGroup.trigger('dtsb-redrawContents'); + this._setGroupListeners(group); + }; + /** + * Rebuilds a criteria of this group that previously existed + * + * @param loadedCriteria The details of a criteria within the group + */ + Group.prototype._addPrevCriteria = function (loadedCriteria) { + var idx = this.s.criteria.length; + var criteria = new Criteria(this.s.dt, this.s.opts, this.s.topGroup, idx, this.s.depth); + criteria.populate(); + // Add the new criteria to the criteria array + this.s.criteria.push({ + criteria: criteria, + index: idx + }); + // Rebuild it with the previous conditions for that criteria + criteria.rebuild(loadedCriteria); + this.s.criteria[idx].criteria = criteria; + this.s.topGroup.trigger('dtsb-redrawContents'); + }; + /** + * Checks And the criteria using AND logic + * + * @param rowData The row data to be checked against the search criteria + * @returns boolean The result of the AND search + */ + Group.prototype._andSearch = function (rowData, rowIdx) { + // If there are no criteria then return true for this group + if (this.s.criteria.length === 0) { + return true; + } + for (var _i = 0, _a = this.s.criteria; _i < _a.length; _i++) { + var crit = _a[_i]; + // If the criteria is not complete then skip it + if (crit.criteria instanceof Criteria && !crit.criteria.s.filled) { + continue; + } + // Otherwise if a single one fails return false + else if (!crit.criteria.search(rowData, rowIdx)) { + return false; + } + } + // If we get to here then everything has passed, so return true for the group + return true; + }; + /** + * Checks And the criteria using OR logic + * + * @param rowData The row data to be checked against the search criteria + * @returns boolean The result of the OR search + */ + Group.prototype._orSearch = function (rowData, rowIdx) { + // If there are no criteria in the group then return true + if (this.s.criteria.length === 0) { + return true; + } + // This will check to make sure that at least one criteria in the group is complete + var filledfound = false; + for (var _i = 0, _a = this.s.criteria; _i < _a.length; _i++) { + var crit = _a[_i]; + if (crit.criteria instanceof Criteria && crit.criteria.s.filled) { + // A completed criteria has been found so set the flag + filledfound = true; + // If the search passes then return true + if (crit.criteria.search(rowData, rowIdx)) { + return true; + } + } + else if (crit.criteria instanceof Group && crit.criteria.checkFilled()) { + filledfound = true; + if (crit.criteria.search(rowData, rowIdx)) { + return true; + } + } + } + // If we get here we need to return the inverse of filledfound, + // as if any have been found and we are here then none have passed + return !filledfound; + }; + /** + * Removes a criteria from the group + * + * @param criteria The criteria instance to be removed + */ + Group.prototype._removeCriteria = function (criteria, group) { + if (group === void 0) { group = false; } + // If removing a criteria and there is only then then just destroy the group + if (this.s.criteria.length <= 1 && this.s.isChild) { + this.destroy(); + } + else { + // Otherwise splice the given criteria out and redo the indexes + var last = void 0; + for (var i = 0; i < this.s.criteria.length; i++) { + if (this.s.criteria[i].index === criteria.s.index && + (!group || this.s.criteria[i].criteria instanceof Group)) { + last = i; + } + } + // We want to remove the last element with the desired index, as its replacement will be inserted before it + if (last !== undefined) { + this.s.criteria.splice(last, 1); + } + for (var i = 0; i < this.s.criteria.length; i++) { + this.s.criteria[i].index = i; + this.s.criteria[i].criteria.s.index = i; + } + } + }; + /** + * Sets the listeners in group for a criteria + * + * @param criteria The criteria for the listeners to be set on + */ + Group.prototype._setCriteriaListeners = function (criteria) { + var _this = this; + criteria.dom["delete"] + .unbind('click') + .on('click.dtsb', function () { + _this._removeCriteria(criteria); + criteria.dom.container.remove(); + for (var _i = 0, _a = _this.s.criteria; _i < _a.length; _i++) { + var crit = _a[_i]; + if (crit.criteria instanceof Criteria) { + crit.criteria.updateArrows(_this.s.criteria.length > 1); + } + } + criteria.destroy(); + _this.s.dt.draw(); + _this.s.topGroup.trigger('dtsb-redrawContents'); + return false; + }); + criteria.dom.right + .unbind('click') + .on('click.dtsb', function () { + var idx = criteria.s.index; + var group = new Group(_this.s.dt, _this.s.opts, _this.s.topGroup, criteria.s.index, true, _this.s.depth + 1); + // Add the criteria that is to be moved to the new group + group.addCriteria(criteria); + // Update the details in the current groups criteria array + _this.s.criteria[idx].criteria = group; + _this.s.criteria[idx].logic = 'AND'; + _this.s.topGroup.trigger('dtsb-redrawContents'); + _this._setGroupListeners(group); + return false; + }); + criteria.dom.left + .unbind('click') + .on('click.dtsb', function () { + _this.s.toDrop = new Criteria(_this.s.dt, _this.s.opts, _this.s.topGroup, criteria.s.index); + _this.s.toDrop.s = criteria.s; + _this.s.toDrop.c = criteria.c; + _this.s.toDrop.classes = criteria.classes; + _this.s.toDrop.populate(); + // The dropCriteria event mutates the reference to the index so need to store it + var index = _this.s.toDrop.s.index; + _this.dom.container.trigger('dtsb-dropCriteria'); + criteria.s.index = index; + _this._removeCriteria(criteria); + // By tracking the top level group we can directly trigger a redraw on it, + // bubbling is also possible, but that is slow with deep levelled groups + _this.s.topGroup.trigger('dtsb-redrawContents'); + _this.s.dt.draw(); + return false; + }); + }; + /** + * Set's the listeners for the group clear button + */ + Group.prototype._setClearListener = function () { + var _this = this; + this.dom.clear + .unbind('click') + .on('click.dtsb', function () { + if (!_this.s.isChild) { + _this.dom.container.trigger('dtsb-clearContents'); + return false; + } + _this.destroy(); + _this.s.topGroup.trigger('dtsb-redrawContents'); + return false; + }); + }; + /** + * Sets listeners for sub groups of this group + * + * @param group The sub group that the listeners are to be set on + */ + Group.prototype._setGroupListeners = function (group) { + var _this = this; + // Set listeners for the new group + group.dom.add + .unbind('click') + .on('click.dtsb', function () { + _this.setupLogic(); + _this.dom.container.trigger('dtsb-add'); + return false; + }); + group.dom.container + .unbind('dtsb-add') + .on('dtsb-add.dtsb', function () { + _this.setupLogic(); + _this.dom.container.trigger('dtsb-add'); + return false; + }); + group.dom.container + .unbind('dtsb-destroy') + .on('dtsb-destroy.dtsb', function () { + _this._removeCriteria(group, true); + group.dom.container.remove(); + _this.setupLogic(); + return false; + }); + group.dom.container + .unbind('dtsb-dropCriteria') + .on('dtsb-dropCriteria.dtsb', function () { + var toDrop = group.s.toDrop; + toDrop.s.index = group.s.index; + toDrop.updateArrows(_this.s.criteria.length > 1, false); + _this.addCriteria(toDrop, false); + return false; + }); + group.setListeners(); + }; + /** + * Sets up the Group instance, setting listeners and appending elements + */ + Group.prototype._setup = function () { + this.setListeners(); + this.dom.add.html(this.s.dt.i18n('searchBuilder.add', this.c.i18n.add)); + this.dom.logic.children().first().html(this.c.logic === 'OR' + ? this.s.dt.i18n('searchBuilder.logicOr', this.c.i18n.logicOr) + : this.s.dt.i18n('searchBuilder.logicAnd', this.c.i18n.logicAnd)); + this.s.logic = this.c.logic === 'OR' ? 'OR' : 'AND'; + if (this.c.greyscale) { + this.dom.logic.addClass(this.classes.greyscale); + } + this.dom.logicContainer.append(this.dom.logic).append(this.dom.clear); + // Only append the logic button immediately if this is a sub group, + // otherwise it will be prepended later when adding a criteria + if (this.s.isChild) { + this.dom.container.append(this.dom.logicContainer); + } + this.dom.container.append(this.dom.add); + }; + /** + * Sets the listener for the logic button + */ + Group.prototype._setLogicListener = function () { + var _this = this; + this.dom.logic + .unbind('click') + .on('click.dtsb', function () { + _this._toggleLogic(); + _this.s.dt.draw(); + for (var _i = 0, _a = _this.s.criteria; _i < _a.length; _i++) { + var crit = _a[_i]; + crit.criteria.setListeners(); + } + }); + }; + /** + * Toggles the logic for the group + */ + Group.prototype._toggleLogic = function () { + if (this.s.logic === 'OR') { + this.s.logic = 'AND'; + this.dom.logic.children().first().html(this.s.dt.i18n('searchBuilder.logicAnd', this.c.i18n.logicAnd)); + } + else if (this.s.logic === 'AND') { + this.s.logic = 'OR'; + this.dom.logic.children().first().html(this.s.dt.i18n('searchBuilder.logicOr', this.c.i18n.logicOr)); + } + }; + Group.version = '1.1.0'; + Group.classes = { + add: 'dtsb-add', + button: 'dtsb-button', + clearGroup: 'dtsb-clearGroup', + greyscale: 'dtsb-greyscale', + group: 'dtsb-group', + inputButton: 'dtsb-iptbtn', + logic: 'dtsb-logic', + logicContainer: 'dtsb-logicContainer' + }; + Group.defaults = { + columns: true, + conditions: { + 'date': Criteria.dateConditions, + 'html': Criteria.stringConditions, + 'html-num': Criteria.numConditions, + 'html-num-fmt': Criteria.numFmtConditions, + 'luxon': Criteria.luxonDateConditions, + 'moment': Criteria.momentDateConditions, + 'num': Criteria.numConditions, + 'num-fmt': Criteria.numFmtConditions, + 'string': Criteria.stringConditions + }, + depthLimit: false, + enterSearch: false, + filterChanged: undefined, + greyscale: false, + i18n: { + add: 'Add Condition', + button: { + 0: 'Search Builder', + _: 'Search Builder (%d)' + }, + clearAll: 'Clear All', + condition: 'Condition', + data: 'Data', + "delete": '×', + deleteTitle: 'Delete filtering rule', + left: '<', + leftTitle: 'Outdent criteria', + logicAnd: 'And', + logicOr: 'Or', + right: '>', + rightTitle: 'Indent criteria', + title: { + 0: 'Custom Search Builder', + _: 'Custom Search Builder (%d)' + }, + value: 'Value', + valueJoiner: 'and' + }, + logic: 'AND', + orthogonal: { + display: 'display', + search: 'filter' + }, + preDefined: false + }; + return Group; + }()); + + var $; + var dataTable; + /** + * Sets the value of jQuery for use in the file + * + * @param jq the instance of jQuery to be set + */ + function setJQuery(jq) { + $ = jq; + dataTable = jq.fn.DataTable; + } + /** + * SearchBuilder class for DataTables. + * Allows for complex search queries to be constructed and implemented on a DataTable + */ + var SearchBuilder = /** @class */ (function () { + function SearchBuilder(builderSettings, opts) { + var _this = this; + // Check that the required version of DataTables is included + if (!dataTable || !dataTable.versionCheck || !dataTable.versionCheck('1.10.0')) { + throw new Error('SearchBuilder requires DataTables 1.10 or newer'); + } + var table = new dataTable.Api(builderSettings); + this.classes = $.extend(true, {}, SearchBuilder.classes); + // Get options from user + this.c = $.extend(true, {}, SearchBuilder.defaults, opts); + this.dom = { + clearAll: $('<button type="button">' + table.i18n('searchBuilder.clearAll', this.c.i18n.clearAll) + '</button>') + .addClass(this.classes.clearAll) + .addClass(this.classes.button) + .attr('type', 'button'), + container: $('<div/>') + .addClass(this.classes.container), + title: $('<div/>') + .addClass(this.classes.title), + titleRow: $('<div/>') + .addClass(this.classes.titleRow), + topGroup: undefined + }; + this.s = { + dt: table, + opts: opts, + search: undefined, + topGroup: undefined + }; + // If searchbuilder is already defined for this table then return + if (table.settings()[0]._searchBuilder !== undefined) { + return; + } + table.settings()[0]._searchBuilder = this; + // If using SSP we want to include the previous state in the very first server call + if (this.s.dt.page.info().serverSide) { + this.s.dt.on('preXhr.dtsb', function (e, settings, data) { + var loadedState = _this.s.dt.state.loaded(); + if (loadedState && loadedState.searchBuilder) { + data.searchBuilder = _this._collapseArray(loadedState.searchBuilder); + } + }); + } + // Run the remaining setup when the table is initialised + if (this.s.dt.settings()[0]._bInitComplete) { + this._setUp(); + } + else { + table.one('init.dt', function () { + _this._setUp(); + }); + } + return this; + } + /** + * Gets the details required to rebuild the SearchBuilder as it currently is + */ + // eslint upset at empty object but that is what it is + // eslint-disable-next-line @typescript-eslint/ban-types + SearchBuilder.prototype.getDetails = function (deFormatDates) { + if (deFormatDates === void 0) { deFormatDates = false; } + return this.s.topGroup.getDetails(deFormatDates); + }; + /** + * Getter for the node of the container for the searchBuilder + * + * @returns JQuery<HTMLElement> the node of the container + */ + SearchBuilder.prototype.getNode = function () { + return this.dom.container; + }; + /** + * Rebuilds the SearchBuilder to a state that is provided + * + * @param details The details required to perform a rebuild + */ + SearchBuilder.prototype.rebuild = function (details) { + this.dom.clearAll.click(); + // If there are no details to rebuild then return + if (details === undefined || details === null) { + return this; + } + this.s.topGroup.s.preventRedraw = true; + this.s.topGroup.rebuild(details); + this.s.topGroup.s.preventRedraw = false; + this.s.topGroup.redrawContents(); + this.s.dt.draw(false); + this.s.topGroup.setListeners(); + return this; + }; + /** + * Applies the defaults to preDefined criteria + * + * @param preDef the array of criteria to be processed. + */ + SearchBuilder.prototype._applyPreDefDefaults = function (preDef) { + var _this = this; + if (preDef.criteria !== undefined && preDef.logic === undefined) { + preDef.logic = 'AND'; + } + var _loop_1 = function (crit) { + // Apply the defaults to any further criteria + if (crit.criteria !== undefined) { + crit = this_1._applyPreDefDefaults(crit); + } + else { + this_1.s.dt.columns().every(function (index) { + if (_this.s.dt.settings()[0].aoColumns[index].sTitle === crit.data) { + crit.dataIdx = index; + } + }); + } + }; + var this_1 = this; + for (var _i = 0, _a = preDef.criteria; _i < _a.length; _i++) { + var crit = _a[_i]; + _loop_1(crit); + } + return preDef; + }; + /** + * Set's up the SearchBuilder + */ + SearchBuilder.prototype._setUp = function (loadState) { + var _this = this; + if (loadState === void 0) { loadState = true; } + // Register an Api method for getting the column type + $.fn.DataTable.Api.registerPlural('columns().type()', 'column().type()', function () { + return this.iterator('column', function (settings, column) { + return settings.aoColumns[column].sType; + }, 1); + }); + // Check that DateTime is included, If not need to check if it could be used + // eslint-disable-next-line no-extra-parens + if (!dataTable.DateTime) { + var types = this.s.dt.columns().type().toArray(); + if (types === undefined || types.includes(undefined) || types.includes(null)) { + types = []; + for (var _i = 0, _a = this.s.dt.settings()[0].aoColumns; _i < _a.length; _i++) { + var colInit = _a[_i]; + types.push(colInit.searchBuilderType !== undefined ? colInit.searchBuilderType : colInit.sType); + } + } + var columnIdxs = this.s.dt.columns().toArray(); + // If the types are not yet set then draw to see if they can be retrieved then + if (types === undefined || types.includes(undefined) || types.includes(null)) { + $.fn.dataTable.ext.oApi._fnColumnTypes(this.s.dt.settings()[0]); + types = this.s.dt.columns().type().toArray(); + } + for (var i = 0; i < columnIdxs[0].length; i++) { + var column = columnIdxs[0][i]; + var type = types[column]; + if ( + // Check if this column can be filtered + (this.c.columns === true || + Array.isArray(this.c.columns) && + this.c.columns.includes(i)) && + // Check if the type is one of the restricted types + (type.includes('date') || + type.includes('moment') || + type.includes('luxon'))) { + alert('SearchBuilder Requires DateTime when used with dates.'); + throw new Error('SearchBuilder requires DateTime'); + } + } + } + this.s.topGroup = new Group(this.s.dt, this.c, undefined); + this._setClearListener(); + this.s.dt.on('stateSaveParams.dtsb', function (e, settings, data) { + data.searchBuilder = _this.getDetails(); + data.page = _this.s.dt.page(); + }); + this.s.dt.on('stateLoadParams.dtsb', function (e, settings, data) { + _this.rebuild(data.searchBuilder); + }); + this._build(); + this.s.dt.on('preXhr.dtsb', function (e, settings, data) { + if (_this.s.dt.page.info().serverSide) { + data.searchBuilder = _this._collapseArray(_this.getDetails(true)); + } + }); + this.s.dt.on('column-reorder', function () { + _this.rebuild(_this.getDetails()); + }); + if (loadState) { + var loadedState = this.s.dt.state.loaded(); + // If the loaded State is not null rebuild based on it for statesave + if (loadedState !== null && loadedState.searchBuilder !== undefined) { + this.s.topGroup.rebuild(loadedState.searchBuilder); + this.s.topGroup.dom.container.trigger('dtsb-redrawContents'); + // If using SSP we want to restrict the amount of server calls that take place + // and this information will already have been processed + if (!this.s.dt.page.info().serverSide) { + this.s.dt.page(loadedState.page).draw('page'); + } + this.s.topGroup.setListeners(); + } + // Otherwise load any predefined options + else if (this.c.preDefined !== false) { + this.c.preDefined = this._applyPreDefDefaults(this.c.preDefined); + this.rebuild(this.c.preDefined); + } + } + this._setEmptyListener(); + this.s.dt.state.save(); + }; + SearchBuilder.prototype._collapseArray = function (criteria) { + if (criteria.logic === undefined) { + if (criteria.value !== undefined) { + criteria.value.sort(function (a, b) { + if (!isNaN(+a)) { + a = +a; + b = +b; + } + if (a < b) { + return -1; + } + else if (b < a) { + return 1; + } + else { + return 0; + } + }); + criteria.value1 = criteria.value[0]; + criteria.value2 = criteria.value[1]; + } + } + else { + for (var i = 0; i < criteria.criteria.length; i++) { + criteria.criteria[i] = this._collapseArray(criteria.criteria[i]); + } + } + return criteria; + }; + /** + * Updates the title of the SearchBuilder + * + * @param count the number of filters in the SearchBuilder + */ + SearchBuilder.prototype._updateTitle = function (count) { + this.dom.title.html(this.s.dt.i18n('searchBuilder.title', this.c.i18n.title, count)); + }; + /** + * Builds all of the dom elements together + */ + SearchBuilder.prototype._build = function () { + var _this = this; + // Empty and setup the container + this.dom.clearAll.remove(); + this.dom.container.empty(); + var count = this.s.topGroup.count(); + this._updateTitle(count); + this.dom.titleRow.append(this.dom.title); + this.dom.container.append(this.dom.titleRow); + this.dom.topGroup = this.s.topGroup.getNode(); + this.dom.container.append(this.dom.topGroup); + this._setRedrawListener(); + var tableNode = this.s.dt.table(0).node(); + if (!$.fn.dataTable.ext.search.includes(this.s.search)) { + // Custom search function for SearchBuilder + this.s.search = function (settings, searchData, dataIndex) { + if (settings.nTable !== tableNode) { + return true; + } + return _this.s.topGroup.search(searchData, dataIndex); + }; + // Add SearchBuilder search function to the dataTables search array + $.fn.dataTable.ext.search.push(this.s.search); + } + this.s.dt.on('destroy.dtsb', function () { + _this.dom.container.remove(); + _this.dom.clearAll.remove(); + var searchIdx = $.fn.dataTable.ext.search.indexOf(_this.s.search); + while (searchIdx !== -1) { + $.fn.dataTable.ext.search.splice(searchIdx, 1); + searchIdx = $.fn.dataTable.ext.search.indexOf(_this.s.search); + } + _this.s.dt.off('.dtsb'); + $(_this.s.dt.table().node()).off('.dtsb'); + }); + }; + /** + * Checks if the clearAll button should be added or not + */ + SearchBuilder.prototype._checkClear = function () { + if (this.s.topGroup.s.criteria.length > 0) { + this.dom.clearAll.insertAfter(this.dom.title); + this._setClearListener(); + } + else { + this.dom.clearAll.remove(); + } + }; + /** + * Update the count in the title/button + * + * @param count Number of filters applied + */ + SearchBuilder.prototype._filterChanged = function (count) { + var fn = this.c.filterChanged; + if (typeof fn === 'function') { + fn(count, this.s.dt.i18n('searchBuilder.button', this.c.i18n.button, count)); + } + }; + /** + * Set the listener for the clear button + */ + SearchBuilder.prototype._setClearListener = function () { + var _this = this; + this.dom.clearAll.unbind('click'); + this.dom.clearAll.on('click.dtsb', function () { + _this.s.topGroup = new Group(_this.s.dt, _this.c, undefined); + _this._build(); + _this.s.dt.draw(); + _this.s.topGroup.setListeners(); + _this.dom.clearAll.remove(); + _this._setEmptyListener(); + _this._filterChanged(0); + return false; + }); + }; + /** + * Set the listener for the Redraw event + */ + SearchBuilder.prototype._setRedrawListener = function () { + var _this = this; + this.s.topGroup.dom.container.unbind('dtsb-redrawContents'); + this.s.topGroup.dom.container.on('dtsb-redrawContents.dtsb', function () { + _this._checkClear(); + _this.s.topGroup.redrawContents(); + _this.s.topGroup.setupLogic(); + _this._setEmptyListener(); + var count = _this.s.topGroup.count(); + _this._updateTitle(count); + _this._filterChanged(count); + // If using SSP we want to restrict the amount of server calls that take place + // and this information will already have been processed + if (!_this.s.dt.page.info().serverSide) { + _this.s.dt.draw(); + } + _this.s.dt.state.save(); + }); + this.s.topGroup.dom.container.unbind('dtsb-redrawLogic'); + this.s.topGroup.dom.container.on('dtsb-redrawLogic.dtsb', function () { + _this.s.topGroup.redrawLogic(); + var count = _this.s.topGroup.count(); + _this._updateTitle(count); + _this._filterChanged(count); + }); + this.s.topGroup.dom.container.unbind('dtsb-add'); + this.s.topGroup.dom.container.on('dtsb-add.dtsb', function () { + var count = _this.s.topGroup.count(); + _this._updateTitle(count); + _this._filterChanged(count); + }); + this.s.dt.on('postEdit.dtsb postCreate.dtsb postRemove.dtsb', function () { + _this.s.topGroup.redrawContents(); + }); + this.s.topGroup.dom.container.unbind('dtsb-clearContents'); + this.s.topGroup.dom.container.on('dtsb-clearContents.dtsb', function () { + _this._setUp(false); + _this._filterChanged(0); + _this.s.dt.draw(); + }); + }; + /** + * Sets listeners to check whether clearAll should be added or removed + */ + SearchBuilder.prototype._setEmptyListener = function () { + var _this = this; + this.s.topGroup.dom.add.on('click.dtsb', function () { + _this._checkClear(); + }); + this.s.topGroup.dom.container.on('dtsb-destroy.dtsb', function () { + _this.dom.clearAll.remove(); + }); + }; + SearchBuilder.version = '1.3.1'; + SearchBuilder.classes = { + button: 'dtsb-button', + clearAll: 'dtsb-clearAll', + container: 'dtsb-searchBuilder', + inputButton: 'dtsb-iptbtn', + title: 'dtsb-title', + titleRow: 'dtsb-titleRow' + }; + SearchBuilder.defaults = { + columns: true, + conditions: { + 'date': Criteria.dateConditions, + 'html': Criteria.stringConditions, + 'html-num': Criteria.numConditions, + 'html-num-fmt': Criteria.numFmtConditions, + 'luxon': Criteria.luxonDateConditions, + 'moment': Criteria.momentDateConditions, + 'num': Criteria.numConditions, + 'num-fmt': Criteria.numFmtConditions, + 'string': Criteria.stringConditions + }, + depthLimit: false, + enterSearch: false, + filterChanged: undefined, + greyscale: false, + i18n: { + add: 'Add Condition', + button: { + 0: 'Search Builder', + _: 'Search Builder (%d)' + }, + clearAll: 'Clear All', + condition: 'Condition', + conditions: { + array: { + contains: 'Contains', + empty: 'Empty', + equals: 'Equals', + not: 'Not', + notEmpty: 'Not Empty', + without: 'Without' + }, + date: { + after: 'After', + before: 'Before', + between: 'Between', + empty: 'Empty', + equals: 'Equals', + not: 'Not', + notBetween: 'Not Between', + notEmpty: 'Not Empty' + }, + // eslint-disable-next-line id-blacklist + number: { + between: 'Between', + empty: 'Empty', + equals: 'Equals', + gt: 'Greater Than', + gte: 'Greater Than Equal To', + lt: 'Less Than', + lte: 'Less Than Equal To', + not: 'Not', + notBetween: 'Not Between', + notEmpty: 'Not Empty' + }, + // eslint-disable-next-line id-blacklist + string: { + contains: 'Contains', + empty: 'Empty', + endsWith: 'Ends With', + equals: 'Equals', + not: 'Not', + notContains: 'Does Not Contain', + notEmpty: 'Not Empty', + notEndsWith: 'Does Not End With', + notStartsWith: 'Does Not Start With', + startsWith: 'Starts With' + } + }, + data: 'Data', + "delete": '×', + deleteTitle: 'Delete filtering rule', + left: '<', + leftTitle: 'Outdent criteria', + logicAnd: 'And', + logicOr: 'Or', + right: '>', + rightTitle: 'Indent criteria', + title: { + 0: 'Custom Search Builder', + _: 'Custom Search Builder (%d)' + }, + value: 'Value', + valueJoiner: 'and' + }, + logic: 'AND', + orthogonal: { + display: 'display', + search: 'filter' + }, + preDefined: false + }; + return SearchBuilder; + }()); + + /*! SearchBuilder 1.3.1 + * ©SpryMedia Ltd - datatables.net/license/mit + */ + // DataTables extensions common UMD. Note that this allows for AMD, CommonJS + // (with window and jQuery being allowed as parameters to the returned + // function) or just default browser loading. + (function (factory) { + if (typeof define === 'function' && define.amd) { + // AMD + define(['jquery', 'datatables.net'], function ($) { + return factory($, window, document); + }); + } + else if (typeof exports === 'object') { + // CommonJS + module.exports = function (root, $) { + if (!root) { + root = window; + } + if (!$ || !$.fn.dataTable) { + // eslint-disable-next-line @typescript-eslint/no-var-requires + $ = require('datatables.net')(root, $).$; + } + return factory($, root, root.document); + }; + } + else { + // Browser - assume jQuery has already been loaded + // eslint-disable-next-line no-extra-parens + factory(window.jQuery, window, document); + } + }(function ($, window, document) { + setJQuery($); + setJQuery$1($); + setJQuery$2($); + var dataTable = $.fn.dataTable; + // eslint-disable-next-line no-extra-parens + $.fn.dataTable.SearchBuilder = SearchBuilder; + // eslint-disable-next-line no-extra-parens + $.fn.DataTable.SearchBuilder = SearchBuilder; + // eslint-disable-next-line no-extra-parens + $.fn.dataTable.Group = Group; + // eslint-disable-next-line no-extra-parens + $.fn.DataTable.Group = Group; + // eslint-disable-next-line no-extra-parens + $.fn.dataTable.Criteria = Criteria; + // eslint-disable-next-line no-extra-parens + $.fn.DataTable.Criteria = Criteria; + // eslint-disable-next-line no-extra-parens + var apiRegister = $.fn.dataTable.Api.register; + // Set up object for plugins + $.fn.dataTable.ext.searchBuilder = { + conditions: {} + }; + $.fn.dataTable.ext.buttons.searchBuilder = { + action: function (e, dt, node, config) { + this.popover(config._searchBuilder.getNode(), { + align: 'container', + span: 'container' + }); + // Need to redraw the contents to calculate the correct positions for the elements + if (config._searchBuilder.s.topGroup !== undefined) { + config._searchBuilder.s.topGroup.dom.container.trigger('dtsb-redrawContents'); + } + if (config._searchBuilder.s.topGroup.s.criteria.length === 0) { + $('.' + $.fn.dataTable.Group.classes.add).click(); + } + }, + config: {}, + init: function (dt, node, config) { + var sb = new $.fn.dataTable.SearchBuilder(dt, $.extend({ + filterChanged: function (count, text) { + dt.button(node).text(text); + } + }, config.config)); + dt.button(node).text(config.text || dt.i18n('searchBuilder.button', sb.c.i18n.button, 0)); + config._searchBuilder = sb; + }, + text: null + }; + apiRegister('searchBuilder.getDetails()', function (deFormatDates) { + if (deFormatDates === void 0) { deFormatDates = false; } + var ctx = this.context[0]; + // If SearchBuilder has not been initialised on this instance then return + return ctx._searchBuilder ? + ctx._searchBuilder.getDetails(deFormatDates) : + null; + }); + apiRegister('searchBuilder.rebuild()', function (details) { + var ctx = this.context[0]; + // If SearchBuilder has not been initialised on this instance then return + if (ctx._searchBuilder === undefined) { + return null; + } + ctx._searchBuilder.rebuild(details); + return this; + }); + apiRegister('searchBuilder.container()', function () { + var ctx = this.context[0]; + // If SearchBuilder has not been initialised on this instance then return + return ctx._searchBuilder ? + ctx._searchBuilder.getNode() : + null; + }); + /** + * Init function for SearchBuilder + * + * @param settings the settings to be applied + * @param options the options for SearchBuilder + * @returns JQUERY<HTMLElement> Returns the node of the SearchBuilder + */ + function _init(settings, options) { + var api = new dataTable.Api(settings); + var opts = options + ? options + : api.init().searchBuilder || dataTable.defaults.searchBuilder; + var searchBuilder = new SearchBuilder(api, opts); + var node = searchBuilder.getNode(); + return node; + } + // Attach a listener to the document which listens for DataTables initialisation + // events so we can automatically initialise + $(document).on('preInit.dt.dtsp', function (e, settings) { + if (e.namespace !== 'dt') { + return; + } + if (settings.oInit.searchBuilder || + dataTable.defaults.searchBuilder) { + if (!settings._searchBuilder) { + _init(settings); + } + } + }); + // DataTables `dom` feature option + dataTable.ext.feature.push({ + cFeature: 'Q', + fnInit: _init + }); + // DataTables 2 layout feature + if (dataTable.ext.features) { + dataTable.ext.features.register('searchBuilder', _init); + } + })); + +}()); |