@progress/kendo-ui
Version:
This package is part of the [Kendo UI for jQuery](http://www.telerik.com/kendo-ui) suite.
1,132 lines (971 loc) • 86.1 kB
JavaScript
import './kendo.datepicker.js';
import './kendo.numerictextbox.js';
import './kendo.dropdownlist.js';
import './kendo.buttongroup.js';
import './kendo.binder.js';
import './kendo.html.button.js';
import './kendo.icons.js';
import './kendo.pane.js';
import './kendo.actionsheet.js';
import './kendo.calendar.js';
import './kendo.core.js';
import './kendo.licensing.js';
import '@progress/kendo-licensing';
import './kendo.selectable.js';
import './kendo.userevents.js';
import './kendo.html.icon.js';
import './kendo.html.base.js';
import '@progress/kendo-svg-icons';
import './kendo.popup.js';
import './kendo.dateinput.js';
import './kendo.label.js';
import './kendo.floatinglabel.js';
import '@progress/kendo-dateinputs-common';
import './kendo.actionsheet.view.js';
import './kendo.button.js';
import './kendo.badge.js';
import './prefix-suffix-containers-Cid0cOEy.js';
import './kendo.list.js';
import './kendo.data.js';
import './kendo.data.odata.js';
import './kendo.data.xml.js';
import './dropdowns-loader-00xUvouJ.js';
import './kendo.mobile.scroller.js';
import './kendo.fx.js';
import './kendo.draganddrop.js';
import './kendo.virtuallist.js';
import './valueMapper-CXgI6HWc.js';
import './kendo.togglebutton.js';
import './kendo.view.js';
const __meta__ = {
id: "filtermenu",
name: "Filtering Menu",
category: "framework",
depends: [ "datepicker", "numerictextbox", "dropdownlist", "buttongroup", "binder", "html.button", "icons", "actionsheet" ],
advanced: true
};
(function($, undefined$1) {
var kendo = window.kendo,
ui = kendo.ui,
support = kendo.support,
encode = kendo.htmlEncode,
AUTOCOMPLETEVALUE = support.browser.chrome ? "disabled" : "off",
POPUP = "kendoPopup",
ACTIONSHEET = "kendoActionSheet",
INIT = "init",
OPEN = "open",
REFRESH = "refresh",
CHANGE = "change",
NS = ".kendoFilterMenu",
COLUMN_HEADER_SELECTOR = ".k-table-th",
EQ = "Is equal to",
NEQ = "Is not equal to",
roles = {
"number": "numerictextbox",
"date": "datepicker"
},
mobileRoles = {
"string": "text",
"number": "number",
"date": "date"
},
isFunction = kendo.isFunction,
Widget = ui.Widget;
var actionsFilterButtonsContainer = ({ actionsCssClass, messages, size }) =>
`<div class="k-actions-stretched ${actionsCssClass ? actionsCssClass : "k-actions"}">` +
kendo.html.renderButton(`<button title="${messages.filter}">${encode(messages.filter)}</button>`, { type: "submit", themeColor: "primary", icon: "filter", size: size }) +
kendo.html.renderButton(`<button title="${messages.clear}">${encode(messages.clear)}</button>`, { type: "reset", icon: "filter-clear", size: size }) +
'</div>';
var booleanTemplate = ({ field, format, ns, messages, extra, operators, type, role, values, componentType, isAdaptive }) =>
'<div class="k-filter-menu-container">' +
`<div class="k-filter-help-text">${encode(messages.info)}</div>` +
'<label>' +
`<input type="radio" class="k-radio ${isAdaptive ? "k-radio-lg" : "k-radio-md"}" data-${ns}bind="checked: filters[0].value" value="true" name="filters[0].value"/>` +
`${encode(messages.isTrue)}` +
'</label>' +
'<label>' +
`<input type="radio" class="k-radio ${isAdaptive ? "k-radio-lg" : "k-radio-md"}" data-${ns}bind="checked: filters[0].value" value="false" name="filters[0].value"/>` +
`${encode(messages.isFalse)}` +
'</label>' +
`${!isAdaptive ? actionsFilterButtonsContainer({ messages }) : ""}` +
'</div>';
var modernBooleanTemplate = ({ field, format, ns, messages, extra, operators, type, role, values, componentType, isAdaptive }) => {
var inputIdForTrue = kendo.guid(), inputIdForFalse = kendo.guid();
return '<div class="k-filter-menu-container">' +
'<div>' +
'<ul class="k-radio-list k-reset">' +
'<li>' +
`<input type="radio" class="k-radio ${isAdaptive ? "k-radio-lg" : "k-radio-md"}" id="${inputIdForTrue}" ${isAdaptive ? 'data-size="large"' : ''} data-${ns}bind="checked: filters[0].value" value="true" name="filters[0].value" />` +
`<label class="k-radio-label" for="${inputIdForTrue}">${encode(messages.isTrue)}</label>` +
'</li>' +
'<li>' +
`<input type="radio" class="k-radio ${isAdaptive ? "k-radio-lg" : "k-radio-md"}" id="${inputIdForFalse}" data-${ns}bind="checked: filters[0].value" value="false" name="filters[0].value" />` +
`<label class="k-radio-label" for="${inputIdForFalse}">${encode(messages.isFalse)}</label>` +
'</li>' +
'</ul>' +
`${!isAdaptive ? actionsFilterButtonsContainer({ actionsCssClass: "k-columnmenu-actions", messages }) : ""}` +
'</div>' +
'</div>';
};
var customBooleanTemplate = ({ field, format, ns, messages, extra, operators, type, role, values, componentType, isAdaptive }) =>
'<div class="k-filter-menu-container">' +
`<div class="k-filter-help-text">${encode(messages.info)}</div>` +
'<label>' +
`<span class="k-textbox k-input ${isAdaptive ? 'k-input-lg' : 'k-input-md'} k-input-md k-rounded-md k-input-solid"><input class="k-input-inner" ${isAdaptive ? 'data-size="large"' : ''} data-${ns}bind="value: filters[0].value adaptiveMode: ${isAdaptive ? "auto" : "none"}" name="filters[0].value"/></span>` +
'</label>' +
`${!isAdaptive ? actionsFilterButtonsContainer({ messages }) : ""}` +
'</div>';
var logicTemplate = ({ ns, messages, componentType, isAdaptive }) => (componentType === "modern" ?
`<ul data-${ns}role="buttongroup" data-bind="events: { select: onLogicChange }">` +
`<li data-${ns}value="and">${encode(messages.and)}</li>` +
`<li data-${ns}value="or">${encode(messages.or)}</li>` +
'</ul>'
:
`<select title="${messages.logic}" class="k-filter-and" ${isAdaptive ? 'data-size="large"' : ''} data-${ns}adaptive-mode="${isAdaptive ? "auto" : "none"}" data-${ns}bind="value: logic" data-${ns}role="dropdownlist">` +
`<option value="and">${encode(messages.and)}</option>` +
`<option value="or">${encode(messages.or)}</option>` +
'</select>'
);
var defaultTemplate = ({ field, format, ns, messages, extra, operators, type, role, values, componentType, isAdaptive }) =>
'<div class="k-filter-menu-container">' +
(componentType === "classic" ?
`<div class="k-filter-help-text">${encode(messages.info)}</div>`
: '') +
`<select title="${messages.operator}" data-${ns}bind="value: filters[0].operator" ${isAdaptive ? 'data-size="large"' : ''} data-${ns}adaptive-mode="${isAdaptive ? "auto" : "none"}" data-${ns}role="dropdownlist">` +
`${Object.keys(operators || {}).map((op) =>
`<option value="${op}">${operators[op]}</option>`
)}` +
'</select>' +
(values ?
`<select title="${messages.value}" data-${ns}bind="value:filters[0].value" data-${ns}text-field="text" ${isAdaptive ? 'data-size="large"' : ''} data-${ns}adaptive-mode="${isAdaptive ? "auto" : "none"}" data-${ns}value-field="value" data-${ns}source='${kendo.stringify(values).replace(/\'/g,"'")}' data-${ns}role="dropdownlist" data-${ns}option-label="${messages.selectValue}" data-${ns}value-primitive="true">` +
'</select>'
:
`<input title="${messages.value}" data-${ns}bind="value:filters[0].value" ${isAdaptive ? 'data-size="large"' : ''} class="k-input-inner" type="text" data-${ns}adaptive-mode="${isAdaptive ? "auto" : "none"}" ${role ? `data-${ns}role="${role}"` : ""} />`
) +
(extra ?
logicTemplate({ ns, messages, componentType, isAdaptive }) +
`<select title="${messages.additionalOperator}" ${isAdaptive ? 'data-size="large"' : ''} data-${ns}bind="value: filters[1].operator" data-${ns}adaptive-mode="${isAdaptive ? "auto" : "none"}" data-${ns}role="dropdownlist">` +
`${Object.keys(operators || {}).map((op) =>
`<option value="${op}">${encode(operators[op])}</option>`
)}` +
'</select>' +
(values ?
`<select title="${messages.additionalValue}" data-${ns}bind="value:filters[1].value" ${isAdaptive ? 'data-size="large"' : ''} data-${ns}text-field="text" data-${ns}adaptive-mode="${isAdaptive ? "auto" : "none"}" data-${ns}value-field="value" data-${ns}source='${kendo.stringify(values).replace(/\'/g,"'")}' data-${ns}role="dropdownlist" data-${ns}option-label="${messages.selectValue}" data-${ns}value-primitive="true">` +
'</select>'
:
`<input title="${messages.additionalValue}" data-${ns}bind="value: filters[1].value" ${isAdaptive ? 'data-size="large"' : ''} class="k-input-inner" type="text" data-${ns}adaptive-mode="${isAdaptive ? "auto" : "none"}" ${role ? `data-${ns}role="${role}"` : ""}/>`
)
: '') +
`${!isAdaptive ? actionsFilterButtonsContainer({ messages }) : ""}` +
'</div>';
var defaultMobileTemplate = ({ field, title, format, ns, messages, extra, operators, filterMenuGuid, type, role, inputType, values }) =>
`<div data-${ns}role="view" class="k-grid-filter-menu">` +
`<div data-${ns}role="header" class="k-appbar k-appbar-primary">` +
kendo.html.renderButton(`<button class="k-header-cancel" title="${messages.cancel}" aria-label="${messages.cancel}"></button>`, { icon: "chevron-left", fillMode: "flat" }) +
`<span class="k-spacer"></span>` +
`<span>${encode(messages.filter)} ${encode(messages.into)} ${encode(title)}</span>` +
`<span class="k-spacer"></span>` +
kendo.html.renderButton(`<button class="k-header-done" title="${messages.done}" aria-label="${messages.done}"></button>`, { icon: "check", fillMode: "flat" }) +
'</div>' +
`<form title="${messages.title}" class="k-filter-menu">` +
'<ul class="k-reset">' +
'<li>' +
`<span class="k-list-title k-filter-help-text">${encode(messages.info)}</span>` +
'<ul class="k-listgroup k-listgroup-flush">' +
'<li class="k-item k-listgroup-item">' +
'<label class="k-listgroup-form-row k-label">' +
`<span class="k-listgroup-form-field-label k-filter-operator-text">${messages.operator}</span>` +
'<span class="k-listgroup-form-field-wrapper">' +
`<select id="operator_${filterMenuGuid}" title="${messages.operator}" class="k-filter-operator" data-${ns}bind="value: filters[0].operator" autocomplete="${AUTOCOMPLETEVALUE}" >` +
`${Object.keys(operators || {}).map((op) =>
`<option value="${op}">${encode(operators[op])}</option>`
)}` +
'</select>' +
'</span>' +
'</label>' +
'</li>' +
'<li class="k-item k-listgroup-item">' +
'<label class="k-listgroup-form-row k-label">' +
`<span class="k-listgroup-form-field-label k-filter-input-text">${messages.value}</span>` +
'<span class="k-listgroup-form-field-wrapper">' +
(values ?
`<select id="value_${filterMenuGuid}" title="${messages.value}" data-${ns}bind="value:filters[0].value" autocomplete="${AUTOCOMPLETEVALUE}" >` +
`<option value="">${messages.selectValue}</option>` +
`${Object.keys(values || {}).map((val) =>
`<option value="${values[val].value}">${encode(values[val].text)}</option>`
)}` +
'</select>'
:
`<input id="value_${filterMenuGuid}" title="${messages.value}" data-${ns}bind="value:filters[0].value" class="k-value-input" type="${inputType}" autocomplete="${AUTOCOMPLETEVALUE}" />`
) +
'</span>' +
'</label>' +
'</li>' +
'</ul>' +
(extra ?
'<ul class="k-listgroup k-listgroup-flush">' +
'<li class="k-item k-listgroup-item">' +
'<label class="k-listgroup-form-row k-label">' +
`<span class="k-listgroup-form-field-label k-filter-logic-and-text">${messages.and}</span>` +
'<span class="k-listgroup-form-field-wrapper">' +
`<input id="and_${filterMenuGuid}" title="${messages.and}" type="radio" name="logic"data-${ns}bind="checked: logic" value="and" autocomplete="${AUTOCOMPLETEVALUE}" />` +
'</span>' +
'</label>' +
'</li>' +
'<li class="k-item k-listgroup-item">' +
'<label class="k-listgroup-form-row k-label">' +
`<span class="k-listgroup-form-field-label k-filter-logic-or-text">${messages.or}</span>` +
'<span class="k-listgroup-form-field-wrapper">' +
`<input id="or_${filterMenuGuid}" title="${messages.or}" type="radio" name="logic" data-${ns}bind="checked: logic" value="or" autocomplete="${AUTOCOMPLETEVALUE}" />` +
'</span>' +
'</label>' +
'</li>' +
'</ul>' +
'<ul class="k-listgroup k-listgroup-flush">' +
'<li class="k-item k-listgroup-item">' +
'<label class="k-listgroup-form-row k-label">' +
`<span class="k-listgroup-form-field-label k-filter-operator-text">${messages.additionalOperator}</span>` +
'<span class="k-listgroup-form-field-wrapper">' +
`<select id="additionalOperator_${filterMenuGuid}" title="${messages.additionalOperator}" class="k-filter-operator" data-${ns}bind="value: filters[1].operator" autocomplete="${AUTOCOMPLETEVALUE}" >` +
`${Object.keys(operators || {}).map((op) =>
`<option value="${op}">${operators[op]}</option>`
)}` +
'</select>' +
'</span>' +
'</label>' +
'</li>' +
'<li class="k-item k-listgroup-item">' +
'<label class="k-listgroup-form-row k-label">' +
`<span class="k-listgroup-form-field-label k-filter-input-text">${messages.additionalValue}</span>` +
'<span class="k-listgroup-form-field-wrapper">' +
(values ?
`<select id="additionalValue_${filterMenuGuid}" title="${messages.additionalValue}" data-${ns}bind="value:filters[1].value" autocomplete="${AUTOCOMPLETEVALUE}" >` +
`<option value="">${messages.selectValue}</option>` +
`${Object.keys(values || {}).map((val) =>
`<option value="${values[val].value}">${encode(values[val].text)}</option>`
)}` +
'</select>'
:
`<input id="additionalValue_${filterMenuGuid}" title="${messages.additionalValue}" data-${ns}bind="value:filters[1].value" class="k-value-input" type="${inputType}" autocomplete="${AUTOCOMPLETEVALUE}" />`
) +
'</span>' +
'</label>' +
'</li>' +
'</ul>'
: '') +
'</li>' +
'<li class="k-item k-clear-wrap">' +
'<span class="k-list-title"> </span>' +
'<ul class="k-listgroup k-listgroup-flush">' +
'<li class="k-listgroup-item">' +
`<span class="k-link k-label k-clear" title="${messages.clear}" aria-label="${messages.clear}">` +
`${encode(messages.clear)}` +
'</span>' +
'</li>' +
'</ul>' +
'</li>' +
'</ul>' +
'</form>' +
'</div>';
var booleanMobileTemplate = ({ field, title, format, ns, messages, extra, operators, filterMenuGuid, type, role, inputType, values }) =>
`<div data-${ns}role="view" class="k-grid-filter-menu">` +
`<div data-${ns}role="header" class="k-appbar k-appbar-primary">` +
kendo.html.renderButton(`<button class="k-header-cancel" title="${messages.cancel}" aria-label="${messages.cancel}"></button>`, { icon: "chevron-left", fillMode: "flat" }) +
`<span class="k-spacer"></span>` +
`<span>${encode(messages.filter)} ${encode(messages.into)} ${encode(title)}</span>` +
`<span class="k-spacer"></span>` +
kendo.html.renderButton(`<button class="k-header-done" title="${messages.done}" aria-label="${messages.done}"></button>`, { icon: "check", fillMode: "flat" }) +
'</div>' +
`<form title="${messages.title}" class="k-filter-menu">` +
'<ul class="k-reset">' +
'<li>' +
`<span class="k-list-title k-filter-help-text">${encode(messages.info)}</span>` +
'<ul class="k-listgroup k-listgroup-flush k-multicheck-bool-wrap">' +
'<li class="k-item k-listgroup-item">' +
'<label class="k-listgroup-form-row k-label">' +
`<span class="k-listgroup-form-field-label k-item-title">${encode(messages.isTrue)}</span>` +
'<span class="k-listgroup-form-field-wrapper"></span>' +
`<input id="true_${filterMenuGuid}" title="${messages.isTrue}" type="radio" data-${ns}bind="checked: filters[0].value" value="true" name="filters[0].value" autocomplete="${AUTOCOMPLETEVALUE}" />` +
'</span>' +
'</label>' +
'</li>' +
'<li class="k-item k-listgroup-item">' +
'<label class="k-listgroup-form-row k-label">' +
`<span for="false_${filterMenuGuid}" class="k-listgroup-form-field-label k-item-title">${encode(messages.isFalse)}</span>` +
'<span class="k-listgroup-form-field-wrapper">' +
`<input id="false_${filterMenuGuid}" title="${messages.isFalse}" type="radio" data-${ns}bind="checked: filters[0].value" value="false" name="filters[0].value" autocomplete="${AUTOCOMPLETEVALUE}" />` +
'</span>' +
'</label>' +
'</li>' +
'</ul>' +
'</li>' +
'<li class="k-item k-clear-wrap">' +
'<span class="k-list-title"> </span>' +
'<ul class="k-listgroup k-listgroup-flush">' +
'<li class="k-listgroup-item">' +
`<span class="k-link k-label k-clear" title="${messages.clear}" aria-label="${messages.clear}">` +
`${encode(messages.clear)}` +
'</span>' +
'</li>' +
'</ul>' +
'</li>' +
'</ul>' +
'</form>' +
'</div>';
function removeFiltersForField(expression, field) {
if (expression.filters) {
expression.filters = $.grep(expression.filters, function(filter) {
removeFiltersForField(filter, field);
if (filter.filters) {
return filter.filters.length;
} else {
return filter.field != field;
}
});
}
}
function convertItems(items) {
var idx,
length,
item,
value,
text,
result;
if (items && items.length) {
result = [];
for (idx = 0, length = items.length; idx < length; idx++) {
item = items[idx];
text = item.text !== "" ? item.text || item.value || item : item.text;
value = item.value == null ? (item.text || item) : item.value;
result[idx] = { text: text, value: value };
}
}
return result;
}
function clearFilter(filters, field) {
return $.grep(filters, function(expr) {
if (expr.filters) {
expr.filters = $.grep(expr.filters, function(nested) {
return nested.field != field;
});
return expr.filters.length;
}
return expr.field != field;
});
}
var FilterMenu = Widget.extend({
init: function(element, options) {
var that = this,
type = "string",
operators,
initial,
field,
columnHeader;
options = options || {};
options.componentType = options.componentType || "classic";
Widget.fn.init.call(that, element, options);
operators = that.operators = options.operators || {};
element = that.element;
options = that.options;
that.dataSource = DataSource.create(options.dataSource);
that.field = options.field || element.attr(kendo.attr("field"));
columnHeader = $(element.closest(COLUMN_HEADER_SELECTOR));
if (columnHeader.length) {
that.appendTo = columnHeader.find(options.appendTo);
} else {
that.appendTo = $(options.appendTo);
}
that.link = that._createLink() || $();
that.model = that.dataSource.reader.model;
that._bindMediaQueries();
that._parse = function(value) {
return value != null ? (value + "") : value;
};
if (that.model && that.model.fields) {
field = that.model.fields[that.field];
if (field) {
type = field.type || "string";
if (field.parse) {
that._parse = field.parse.bind(field);
}
}
}
if (options.values) {
type = "enums";
}
that.type = type;
operators = operators[type] || options.operators[type];
for (initial in operators) { // get the first operator
break;
}
that._defaultFilter = function(isExtra) {
const defaultFilterModel = { field: that.field, operator: initial || "eq", value: "" };
if (!that.form) {
return defaultFilterModel;
}
const fromChildren = that.form.find(".k-filter-menu-container").children();
const inputs = fromChildren.find("input");
const firstInput = inputs.first();
const lastInput = inputs.last();
if (firstInput.is(lastInput) && isExtra) {
return defaultFilterModel;
}
const targetElement = isExtra ? lastInput : firstInput;
const widget = kendo.widgetInstance(targetElement);
if (widget) {
return { field: that.field, operator: initial || "eq", value: widget.value() };
}
return defaultFilterModel;
};
that._refreshHandler = that.refresh.bind(that);
that.dataSource.bind(CHANGE, that._refreshHandler);
if (options.appendToElement) { // force creation if used in column menu
that._init();
} else {
that.refresh(); //refresh if DataSource is fitered before menu is created
}
},
_init: function() {
var that = this,
ui = that.options.ui,
setUI = isFunction(ui),
attrRole = kendo.attr("role"),
role;
that.pane = that.options.pane;
if (that.pane) {
that._isMobile = true;
}
if (!setUI) {
role = ui || roles[that.type];
}
if (that._isMobile) {
that._createMobileForm(role);
} else {
that._createForm(role);
}
that.form
.on("submit" + NS, that._submit.bind(that))
.on("reset" + NS, that._reset.bind(that));
if (setUI) {
that.form.find(".k-input-inner")
.removeClass("k-input-inner")
.each(function() {
ui($(this));
});
} else {
that.form
.find(".k-input-inner[" + attrRole + "]")
.removeClass("k-input-inner");
that.form
.find(".k-input-inner:not([data-role]):not(.k-numerictextbox>.k-input-inner)")
.wrap(`<span class='k-textbox k-input ${that._showAdaptiveView ? "k-input-lg" : "k-input-md"} k-rounded-md k-input-solid'></span>`);
}
that.refresh();
that.trigger(INIT, { field: that.field, container: that.form });
if (that.options.cycleForm) {
kendo.cycleForm(that.form);
}
},
_adaptiveView: function() {
const that = this;
if (that.popup) {
that.popup.close();
that.popup.wrapper && that.popup.wrapper.remove();
that.popup.destroy();
that.popup = null;
}
},
_createForm: function(role) {
var that = this,
options = that.options,
operators = that.operators || {},
type = that.type;
const isAdaptive = that._showAdaptiveView;
operators = operators[type] || options.operators[type];
that.form = $('<form title="' + encode(that.options.messages.title) + '" class="k-filter-menu"/>')
.html(kendo.template(that._getTemplate())({
field: that.field,
format: options.format,
ns: kendo.ns,
messages: options.messages,
extra: options.extra,
operators: operators,
type: type,
role: role,
values: convertItems(options.values),
componentType: that.options.componentType,
isAdaptive: isAdaptive
}));
if (!options.appendToElement) {
if (isAdaptive) {
const actionsheetContainer = $("<div></div>").append(that.form).appendTo("body");
that.popup = actionsheetContainer[ACTIONSHEET]({
anchor: that.link,
copyAnchorStyles: false,
open: that._open.bind(that),
activate: that._activate.bind(that),
adaptive: true,
closeButton: true,
title: options.adaptiveTitle || "Filter by " + that.field,
subtitle: options.adaptiveSubtitle,
actionButtons: [
{
text: "Clear",
icon: "filter-clear",
},
{
text: "Filter",
icon: "filter",
themeColor: "primary",
}
],
close: function() {
if (that.options.closeCallback) {
that.options.closeCallback(that.element);
}
}
}).data(ACTIONSHEET);
that.popup.fullscreen(that.smallMQL.mediaQueryList.matches);
} else {
that.popup = that.form[POPUP]({
anchor: that.link,
copyAnchorStyles: false,
open: that._open.bind(that),
activate: that._activate.bind(that),
close: function() {
if (that.options.closeCallback) {
that.options.closeCallback(that.element);
}
}
}).data(POPUP);
}
} else {
that.element.append(that.form);
that.popup = that.element.closest(".k-column-menu.k-popup").data(POPUP);
}
that.form
.on("keydown" + NS, that._keydown.bind(that));
},
_getTemplate: function() {
var that = this,
hasCustomTemplate = isFunction(that.options.ui);
if (that.type === 'boolean') {
if (hasCustomTemplate) {
return customBooleanTemplate;
} else if (that.options.componentType === 'modern') {
return modernBooleanTemplate;
} else {
return booleanTemplate;
}
} else {
return defaultTemplate;
}
},
_createMobileForm: function(role) {
var that = this,
options = that.options,
operators = that.operators || {},
filterMenuGuid = kendo.guid(),
type = that.type;
operators = operators[type] || options.operators[type];
that.form = $("<div />")
.html(kendo.template(type === "boolean" ? booleanMobileTemplate : defaultMobileTemplate)({
field: that.field,
title: options.title || that.field,
format: options.format,
ns: kendo.ns,
messages: options.messages,
extra: options.extra,
operators: operators,
filterMenuGuid: filterMenuGuid,
type: type,
role: role,
inputType: mobileRoles[type],
values: convertItems(options.values)
}));
that.view = that.pane.append(that.form.html());
that.form = that.view.element.find("form");
that.view.element
.on("click", ".k-header-done", function(e) {
that.form.submit();
e.preventDefault();
})
.on("click", ".k-header-cancel", function(e) {
that._closeForm();
e.preventDefault();
})
.on("click", ".k-clear", function(e) {
that._mobileClear();
e.preventDefault();
});
that.view.bind("showStart", function() {
that.refresh();
});
that.view.bind("transitionEnd", function(e) {
if (e.type === "show") {
that.trigger(OPEN, { field: that.field, container: that.form });
}
});
},
_createLink: function() {
var that = this,
element = that.element,
appendTarget = that.appendTo.length ? element.find(that.appendTo) : element,
options = that.options,
title = kendo.format(options.messages.buttonTitle, that.options.title || that.field),
link;
if (options.appendToElement) {
return;
}
link = element.addClass("k-filterable").find(".k-grid-filter-menu");
if (!link[0]) {
link = appendTarget
.append('<a class="k-grid-filter-menu k-grid-header-menu" href="#" aria-hidden="true" title="' +
title + '" >' + kendo.ui.icon("filter") + '</a>')
.find(".k-grid-filter-menu");
}
link.attr("tabindex", -1)
.on("click" + NS, that._click.bind(that));
return link;
},
_bindMediaQueries: function() {
const that = this;
const isAdaptive = that.options.adaptiveMode === "auto";
if (isAdaptive) {
that.largeMQL = kendo.mediaQuery("large");
that.mediumMQL = kendo.mediaQuery("medium");
that.smallMQL = kendo.mediaQuery("small");
that.smallMQL
.onEnter(() => {
that._showAdaptiveView = true;
if (that.popup && that.popup.visible() && that.popup.fullscreen) {
that.popup.fullscreen(true);
} else {
that._adaptiveView();
}
});
that.mediumMQL
.onEnter(() => {
that._showAdaptiveView = true;
if (that.popup && that.popup.visible() && that.popup.fullscreen) {
that.popup.fullscreen(false);
} else {
that._adaptiveView();
}
});
that.largeMQL
.onEnter(() => {
that._showAdaptiveView = false;
that._adaptiveView();
});
} else {
that.smallMQL && that.smallMQL.destroy();
that.mediumMQL && that.mediumMQL.destroy();
that.largeMQL && that.largeMQL.destroy();
that._showAdaptiveView = false;
}
},
refresh: function() {
var that = this,
expression = that.dataSource.filter() || { filters: [], logic: "and" };
var defaultFilters = [ that._defaultFilter() ];
var defaultOperator = that._defaultFilter().operator;
if (that.options.extra || (defaultOperator !== "isnull" && defaultOperator !== "isnullorempty" && defaultOperator !== "isnotnullorempty" && defaultOperator !== "isnotnull" && defaultOperator !== "isempty" && defaultOperator !== "isnotempty")) {
defaultFilters.push(that._defaultFilter(true));
}
that.filterModel = kendo.observable({
logic: "and",
filters: defaultFilters
});
if (that.form) {
//NOTE: binding the form element directly causes weird error in IE when grid is bound through MVVM and column is sorted
kendo.bind(that.form.children().first(), that.filterModel);
if (that.options.componentType === "modern" && that.options.extra && that.type !== "boolean" && !that._isMobile) {
that.filterModel.bind("change", function() {
var roleAttribute = kendo.attr("role");
var buttongroup = that.form.find("[" + roleAttribute + "='buttongroup']").data('kendoButtonGroup');
var index = this.logic === "and" ? 0 : 1;
buttongroup.select(buttongroup.element.children().eq(index));
});
that.filterModel.set("onLogicChange",that._logicChangeHandler);
}
}
if (that._bind(expression)) {
that.link.addClass("k-active");
} else {
that.link.removeClass("k-active");
}
},
_logicChangeHandler: function(e) {
var valueAttribute = kendo.attr('value');
var logic = e.sender.current().attr(valueAttribute);
this.set('logic', logic);
},
destroy: function() {
var that = this;
Widget.fn.destroy.call(that);
if (that.form) {
kendo.unbind(that.form);
kendo.destroy(that.form);
that.form.off(NS);
if (that.popup) {
that.popup.destroy();
that.popup = null;
}
that.form = null;
}
if (that.view) {
that.view.purge();
that.view = null;
}
that.link.off(NS);
if (that._refreshHandler) {
that.dataSource.unbind(CHANGE, that._refreshHandler);
that.dataSource = null;
}
that.element = that.link = that._refreshHandler = that.filterModel = null;
},
_bind: function(expression) {
var that = this,
filters = expression.filters,
idx,
length,
found = false,
current = 0,
filterModel = that.filterModel,
currentFilter,
filter;
for (idx = 0, length = filters.length; idx < length; idx++) {
filter = filters[idx];
if (filter.field == that.field) {
filterModel.set("logic", expression.logic);
currentFilter = filterModel.filters[current];
if (!currentFilter) {
filterModel.filters.push({ field: that.field });
currentFilter = filterModel.filters[current];
}
currentFilter.set("value", that._parse(filter.value));
currentFilter.set("operator", filter.operator);
current++;
found = true;
} else if (filter.filters) {
found = found || that._bind(filter);
}
}
return found;
},
_stripFilters: function(filters) {
return $.grep(filters, function(filter) {
return (filter.value !== "" && filter.value != null) ||
(filter.operator === "isnull" || filter.operator === "isnotnull" ||
filter.operator === "isempty" || filter.operator === "isnotempty" ||
filter.operator == "isnullorempty" || filter.operator == "isnotnullorempty");
});
},
_merge: function(expression) {
var that = this,
logic = expression.logic || "and",
filters = this._stripFilters(expression.filters),
filter,
result = that.dataSource.filter() || { filters: [], logic: "and" },
idx,
length;
removeFiltersForField(result, that.field);
for (idx = 0, length = filters.length; idx < length; idx++) {
filter = filters[idx];
filter.value = that._parse(filter.value);
}
if (filters.length) {
if (result.filters.length) {
expression.filters = filters;
if (result.logic !== "and") {
result.filters = [ { logic: result.logic, filters: result.filters }];
result.logic = "and";
}
if (filters.length > 1) {
result.filters.push(expression);
} else {
result.filters.push(filters[0]);
}
} else {
result.filters = filters;
result.logic = logic;
}
}
return result;
},
filter: function(expression) {
var filters = this._stripFilters(expression.filters);
if (filters.length && this.trigger("change", {
filter: { logic: expression.logic, filters: filters },
field: this.field
})) {
return;
}
expression = this._merge(expression);
if (expression.filters.length) {
this.dataSource.filter(expression);
}
},
clear: function(expression) {
var that = this;
expression = expression || $.extend(true, {}, { filters: [] }, that.dataSource.filter()) || { filters: [] };
if (this.trigger("change", { filter: null, field: that.field })) {
return;
}
that._removeFilter(expression);
},
_mobileClear: function() {
var that = this;
var viewElement = that.view.element;
if (that.type === "boolean") {
var booleanRadioButton = viewElement.find("[type='radio']:checked");
var booleanRadioButtonValue = booleanRadioButton.val();
booleanRadioButton.val("");
booleanRadioButton.trigger("change");
booleanRadioButton.val(booleanRadioButtonValue);
booleanRadioButton.prop("checked", false);
} else {
var operatorSelects = viewElement.find("select");
operatorSelects.each(function(i, e) {
var input = $(e);
input.val(input.find("option").first().val());
input.trigger("change");
});
if (that.type === "string" || that.type === "date" || that.type === "number") {
var valueInputs = viewElement.find(".k-value-input");
valueInputs.each(function(i, e) {
var input = $(e);
input.val("");
input.trigger("change");
});
}
if (that.options.extra) {
var andLogicRadio = viewElement.find("[name=logic]").first();
andLogicRadio.prop("checked", true);
andLogicRadio.trigger("change");
}
}
},
_removeFilter: function(expression) {
var that = this;
expression.filters = $.grep(expression.filters, function(filter) {
if (filter.filters) {
filter.filters = clearFilter(filter.filters, that.field);
return filter.filters.length;
}
return filter.field != that.field;
});
if (!expression.filters.length) {
expression = null;
}
that.dataSource.filter(expression);
},
_submit: function(e) {
e.preventDefault();
e.stopPropagation();
var expression = this.filterModel.toJSON();
var containsFilters = $.grep(expression.filters, function(filter) {
return filter.value !== "" && filter.value !== null;
});
if (this._checkForNullOrEmptyFilter(expression) || (containsFilters && containsFilters.length)) {
this.filter(expression);
} else {
var currentExpression = this.dataSource.filter();
if (currentExpression) {
currentExpression.filters.push(expression);
expression = currentExpression;
}
this.clear(expression);
}
this._closeForm();
},
_checkForNullOrEmptyFilter: function(expression) {
if (!expression || !expression.filters || !expression.filters.length) {
return false;
}
var firstNullOrEmpty = false;
var secondNullOrEmpty = false;
var operator;
if (expression.filters[0]) {
operator = expression.filters[0].operator;
firstNullOrEmpty = operator == "isnull" || operator == "isnotnull" || operator == "isnotempty" ||
operator == "isempty" || operator == "isnullorempty" || operator == "isnotnullorempty";
}
if (expression.filters[1]) {
operator = expression.filters[1].operator;
secondNullOrEmpty = operator == "isnull" || operator == "isnotnull" || operator == "isnotempty" ||
operator == "isempty" || operator == "isnullorempty" || operator == "isnotnullorempty";
}
return (!this.options.extra && firstNullOrEmpty) || (this.options.extra && (firstNullOrEmpty || secondNullOrEmpty));
},
_reset: function() {
this.clear();
if (this.options.search && this.container) {
this.container.find("label").parent().show();
}
this._closeForm();
},
_closeForm: function() {
if (this._isMobile) {
this.pane.navigate("", this.options.animations.right);
} else {
this.popup && this.popup.close();
}
},
_click: function(e) {
e.preventDefault();
e.stopPropagation();
if (!this.popup && !this.pane) {
this._init();
}
if (this._isMobile) {
this.pane.navigate(this.view, this.options.animations.left);
} else {
this.popup.toggle();
}
},
_open: function() {
var popup;
$(".k-filter-menu").not(this.form).each(function() {
popup = $(this).data(POPUP);
if (popup) {
popup.close();
}
});
},
_activate: function() {
const that = this;
that.form.find(":kendoFocusable").first().trigger("focus");
if (that.popup) {
that.popup.wrapper.find("[ref-actionsheet-action-button]").bind("click", function(e) {
const button = $(e.currentTarget);
if (button.hasClass("k-button-solid-primary")) {
that.form.trigger("submit");
} else {
that.form.trigger("reset");
}
});
}
that.trigger(OPEN, { field: that.field, container: that.form });
},
_keydown: function(e) {
var target = $(e.target),
instance;
if (e.keyCode == kendo.keys.ESC) {
instance = kendo.widgetInstance(target.find("select"));
if (target.hasClass("k-picker") &&
instance &&
instance.popup.visible()) {
e.stopPropagation();
return;
}
target.closest(".k-popup").getKendoPopup().close();
}
},
events: [ INIT, "change", OPEN ],
options: {
name: "FilterMenu",
extra: true,
appendToElement: false,
adaptiveMode: "n