UNPKG

devextreme

Version:

HTML5 JavaScript Component Suite for Responsive Web Development

665 lines (664 loc) • 24.8 kB
/** * DevExtreme (ui/select_box.js) * Version: 18.1.3 * Build date: Tue May 15 2018 * * Copyright (c) 2012 - 2018 Developer Express Inc. ALL RIGHTS RESERVED * Read about DevExtreme licensing here: https://js.devexpress.com/Licensing/ */ "use strict"; var $ = require("../core/renderer"), eventsEngine = require("../events/core/events_engine"), commonUtils = require("../core/utils/common"), typeUtils = require("../core/utils/type"), isDefined = typeUtils.isDefined, isPromise = typeUtils.isPromise, extend = require("../core/utils/extend").extend, inArray = require("../core/utils/array").inArray, each = require("../core/utils/iterator").each, deferredUtils = require("../core/utils/deferred"), getPublicElement = require("../core/utils/dom").getPublicElement, Deferred = deferredUtils.Deferred, errors = require("../core/errors"), inkRipple = require("./widget/utils.ink_ripple"), messageLocalization = require("../localization/message"), registerComponent = require("../core/component_registrator"), eventUtils = require("../events/utils"), dataQuery = require("../data/query"), DropDownList = require("./drop_down_editor/ui.drop_down_list"), themes = require("./themes"), clickEvent = require("../events/click"); var DISABLED_STATE_SELECTOR = ".dx-state-disabled", SELECTBOX_CLASS = "dx-selectbox", SELECTBOX_POPUP_CLASS = "dx-selectbox-popup", SELECTBOX_CONTAINER_CLASS = "dx-selectbox-container", SELECTBOX_POPUP_WRAPPER_CLASS = "dx-selectbox-popup-wrapper"; var SelectBox = DropDownList.inherit({ _supportedKeys: function() { var that = this, parent = this.callBase(), clearSelectBox = function(e) { var isEditable = this._isEditable(); if (!isEditable) { if (this.option("showClearButton")) { e.preventDefault(); this.reset() } } else { if (this._valueSubstituted()) { this._preventFiltering = true } } this._preventSubstitution = true }; var searchIfNeeded = function() { if (that.option("searchEnabled") && that._valueSubstituted()) { that._searchHandler() } }; return extend({}, parent, { tab: function() { if (this.option("opened") && "instantly" === this.option("applyValueMode")) { this._cleanInputSelection() } if (this._wasSearch()) { this._clearFilter() } parent.tab.apply(this, arguments) }, upArrow: function() { if (parent.upArrow.apply(this, arguments)) { if (!this.option("opened")) { this._setNextValue(-1) } return true } }, downArrow: function() { if (parent.downArrow.apply(this, arguments)) { if (!this.option("opened")) { this._setNextValue(1) } return true } }, leftArrow: function() { searchIfNeeded(); parent.leftArrow.apply(this, arguments) }, rightArrow: function() { searchIfNeeded(); parent.rightArrow.apply(this, arguments) }, home: function() { searchIfNeeded(); parent.home.apply(this, arguments) }, end: function() { searchIfNeeded(); parent.end.apply(this, arguments) }, escape: function() { parent.escape.apply(this, arguments); if (!this._isEditable() && this._list) { this._focusListElement(null); this._updateField(this.option("selectedItem")) } }, enter: function(e) { if ("" === this._input().val() && this.option("value") && this.option("allowClearing")) { this.option({ selectedItem: null, value: null }); this.close() } else { if (this.option("acceptCustomValue")) { e.preventDefault(); return this.option("opened") } if (parent.enter.apply(this, arguments)) { return this.option("opened") } } }, backspace: clearSelectBox, del: clearSelectBox }) }, _getDefaultOptions: function() { return extend(this.callBase(), { placeholder: messageLocalization.format("Select"), fieldTemplate: null, valueChangeEvent: "change", acceptCustomValue: false, onCustomItemCreating: function(e) { if (!isDefined(e.customItem)) { e.customItem = e.text } }, showSelectionControls: false, autocompletionEnabled: true, allowClearing: true, tooltipEnabled: false, openOnFieldClick: true, showDropDownButton: true, displayCustomValue: false, _isAdaptablePopupPosition: false, useInkRipple: false }) }, _defaultOptionsRules: function() { return this.callBase().concat([{ device: function() { return /win8/.test(themes.current()) }, options: { _isAdaptablePopupPosition: true, popupPosition: { at: "left top", offset: { h: 0, v: 0 } } } }, { device: function() { return /android5/.test(themes.current()) }, options: { _isAdaptablePopupPosition: true, popupPosition: { offset: { h: -16, v: -8 } }, useInkRipple: true } }]) }, _init: function() { this.callBase(); this._initCustomItemCreatingAction() }, _initMarkup: function() { this._renderSubmitElement(); this.$element().addClass(SELECTBOX_CLASS); this._renderTooltip(); this.option("useInkRipple") && this._renderInkRipple(); this.callBase(); this._$container.addClass(SELECTBOX_CONTAINER_CLASS) }, _renderSubmitElement: function() { this._$submitElement = $("<input>").attr("type", "hidden").appendTo(this.$element()) }, _renderInkRipple: function() { this._inkRipple = inkRipple.render() }, _toggleActiveState: function($element, value, e) { this.callBase.apply(this, arguments); if (!this._inkRipple || this._isEditable()) { return } var config = { element: this._inputWrapper(), event: e }; if (value) { this._inkRipple.showWave(config) } else { this._inkRipple.hideWave(config) } }, _createPopup: function() { this.callBase(); this._popup.$element().addClass(SELECTBOX_POPUP_CLASS) }, _popupWrapperClass: function() { return this.callBase() + " " + SELECTBOX_POPUP_WRAPPER_CLASS }, _renderOpenedState: function() { this.callBase(); if (this.option("opened")) { this._scrollToSelectedItem(); this._focusSelectedElement() } }, _focusSelectedElement: function() { var searchValue = this._searchValue(); if (!searchValue) { this._focusListElement(null); return } var $listItems = this._list._itemElements(), index = inArray(this.option("selectedItem"), this.option("items")), focusedElement = index >= 0 && !this._isCustomItemSelected() ? $listItems.eq(index) : null; this._focusListElement(focusedElement) }, _renderFocusedElement: function() { if (!this._list) { return } var searchValue = this._searchValue(); if (!searchValue || this.option("acceptCustomValue")) { this._focusListElement(null); return } var $listItems = this._list._itemElements(), focusedElement = $listItems.not(DISABLED_STATE_SELECTOR).eq(0); this._focusListElement(focusedElement) }, _focusListElement: function(element) { this._preventInputValueRender = true; this._list.option("focusedElement", getPublicElement(element)); delete this._preventInputValueRender }, _scrollToSelectedItem: function() { this._list.scrollToItem(this._list.option("selectedItem")) }, _listContentReadyHandler: function() { this.callBase(); var isPaginate = this._dataSource && this._dataSource.paginate(); if (isPaginate && this._needPopupRepaint()) { return } this._scrollToSelectedItem() }, _renderValue: function() { this._renderInputValue(); this._setSubmitValue() }, _setSubmitValue: function() { var value = this.option("value"), submitValue = "this" === this.option("valueExpr") ? this._displayGetter(value) : value; this._$submitElement.val(submitValue) }, _getSubmitElement: function() { return this._$submitElement }, _renderInputValue: function() { return this.callBase().always(function() { this._renderInputValueAsync() }.bind(this)) }, _renderInputValueAsync: function() { this._renderTooltip(); this._renderInputValueImpl(); this._refreshSelected() }, _renderInputValueImpl: function() { this._renderInputAddons() }, _fitIntoRange: function(value, start, end) { if (value > end) { return start } if (value < start) { return end } return value }, _setNextValue: function(step) { var dataSourceIsLoaded = this._dataSource.isLoaded() ? (new Deferred).resolve() : this._dataSource.load(); dataSourceIsLoaded.done(function() { var item = this._calcNextItem(step), value = this._valueGetter(item); this._setValue(value) }.bind(this)) }, _calcNextItem: function(step) { var items = this._items(); var nextIndex = this._fitIntoRange(this._getSelectedIndex() + step, 0, items.length - 1); return items[nextIndex] }, _items: function() { var items = this._list ? this.option("items") : this._dataSource.items(); var availableItems = new dataQuery(items).filter("disabled", "<>", true).toArray(); return availableItems }, _getSelectedIndex: function() { var items = this._items(); var selectedItem = this.option("selectedItem"); var result = -1; each(items, function(index, item) { if (this._isValueEquals(item, selectedItem)) { result = index; return false } }.bind(this)); return result }, _setSelectedItem: function(item) { var isUnknownItem = !this._isCustomValueAllowed() && void 0 === item; this.callBase(isUnknownItem ? null : item) }, _isCustomValueAllowed: function() { return this.option("acceptCustomValue") || this.callBase() }, _displayValue: function(item) { item = !isDefined(item) && this._isCustomValueAllowed() ? this.option("value") : item; return this.callBase(item) }, _listConfig: function() { var result = extend(this.callBase(), { pageLoadMode: "scrollBottom", onSelectionChanged: this._getSelectionChangeHandler(), selectedItem: this.option("selectedItem"), onFocusedItemChanged: this._listFocusedItemChangeHandler.bind(this) }); if (this.option("showSelectionControls")) { extend(result, { showSelectionControls: true, selectionByClick: true }) } return result }, _listFocusedItemChangeHandler: function(e) { if (this._preventInputValueRender) { return } var list = e.component, focusedElement = $(list.option("focusedElement")), focusedItem = list._getItemData(focusedElement); this._updateField(focusedItem) }, _updateField: function(item) { var fieldTemplate = this._getTemplateByOption("fieldTemplate"); if (!(fieldTemplate && this.option("fieldTemplate"))) { this._renderDisplayText(this._displayGetter(item)); return } this._renderInputAddons() }, _getSelectionChangeHandler: function() { return this.option("showSelectionControls") ? this._selectionChangeHandler.bind(this) : commonUtils.noop }, _selectionChangeHandler: function(e) { each(e.addedItems || [], function(_, addedItem) { this._setValue(this._valueGetter(addedItem)) }.bind(this)) }, _getActualSearchValue: function() { return this._dataSource.searchValue() }, _toggleOpenState: function(isVisible) { if (this.option("disabled")) { return } isVisible = arguments.length ? isVisible : !this.option("opened"); if (!isVisible) { this._restoreInputText() } if (this._wasSearch() && isVisible) { this._wasSearch(false); if (this.option("showDataBeforeSearch") || 0 === this.option("minSearchLength")) { if (this._searchTimer) { return } var searchValue = this._getActualSearchValue(); searchValue && this._wasSearch(true); this._filterDataSource(searchValue || null) } else { this._setListOption("items", []) } } this.callBase(isVisible) }, _renderTooltip: function() { if (this.option("tooltipEnabled")) { this.$element().attr("title", this.option("displayValue")) } }, _renderDimensions: function() { this.callBase(); this._setPopupOption("width") }, _restoreInputText: function() { this._loadItemDeferred && this._loadItemDeferred.always(function() { if (this.option("acceptCustomValue")) { return } if (this.option("searchEnabled")) { if (!this._searchValue() && this.option("allowClearing")) { this._clearTextValue(); return } } var oldSelectedItem = this.option("selectedItem"); if (this._displayGetter(oldSelectedItem) === this._searchValue()) { return } this._renderInputValue().always(function(selectedItem) { var newSelectedItem = commonUtils.ensureDefined(selectedItem, oldSelectedItem); this._setSelectedItem(newSelectedItem); this._updateField(newSelectedItem); this._clearFilter() }.bind(this)) }.bind(this)) }, _focusOutHandler: function(e) { this.callBase(e); this._restoreInputText() }, _clearTextValue: function() { this.option("value", null) }, _renderValueChangeEvent: function() { if (this._isEditable()) { this.callBase() } }, _isEditable: function() { return this.option("acceptCustomValue") || this.option("searchEnabled") }, _fieldRenderData: function() { var $listFocused = this._list && this.option("opened") && $(this._list.option("focusedElement")); if ($listFocused && $listFocused.length) { return this._list._getItemData($listFocused) } return this.option("selectedItem") }, _readOnlyPropValue: function() { return !this._isEditable() || this.option("readOnly") }, _isSelectedValue: function(value) { return this._isValueEquals(value, this.option("value")) }, _shouldCloseOnItemClick: function() { return !(this.option("showSelectionControls") && "single" !== this.option("selectionMode")) }, _listItemClickHandler: function(e) { var previousValue = this._getCurrentValue(); this._saveValueChangeEvent(e.event); if (this._wasSearch()) { this._clearFilter() } this._completeSelection(this._valueGetter(e.itemData)); if (this._shouldCloseOnItemClick()) { this.option("opened", false) } if (this.option("searchEnabled") && previousValue === this._valueGetter(e.itemData)) { this._updateField(e.itemData) } }, _completeSelection: function(value) { this._setValue(value) }, _clearValueHandler: function() { this.reset() }, _loadItem: function(value, cache) { var that = this, deferred = new Deferred; this.callBase(value, cache).done(function(item) { deferred.resolve(item) }.bind(this)).fail(function() { var selectedItem = that.option("selectedItem"); if (that.option("acceptCustomValue") && value === that._valueGetter(selectedItem)) { deferred.resolve(selectedItem) } else { deferred.reject() } }.bind(this)); return deferred.promise() }, _loadInputValue: function(value, callback) { this._loadItemDeferred = this._loadItem(value).always(callback); return this._loadItemDeferred }, _isCustomItemSelected: function() { var selectedItem = this.option("selectedItem"), searchValue = this._searchValue(), selectedItemText = this._displayGetter(selectedItem); return !selectedItemText || searchValue !== selectedItemText.toString() }, _valueChangeEventHandler: function() { if (this.option("acceptCustomValue") && this._isCustomItemSelected()) { this._customItemAddedHandler() } }, _initCustomItemCreatingAction: function() { this._customItemCreatingAction = this._createActionByOption("onCustomItemCreating") }, _createCustomItem: function(text) { var params = { text: text }, actionResult = this._customItemCreatingAction(params), item = commonUtils.ensureDefined(actionResult, params.customItem); if (isDefined(actionResult)) { errors.log("W0015", "onCustomItemCreating", "customItem") } return item }, _customItemAddedHandler: function() { var searchValue = this._searchValue(), item = this._createCustomItem(searchValue); if (void 0 === item) { this._renderValue(); throw errors.Error("E0121") } if (isPromise(item)) { deferredUtils.fromPromise(item).done(this._setCustomItem.bind(this)).fail(this._setCustomItem.bind(this, null)) } else { this._setCustomItem(item) } }, _setCustomItem: function(item) { if (this._disposed) { return } item = item || null; this.option("selectedItem", item); this._setValue(this._valueGetter(item)); this._renderDisplayText(this._displayGetter(item)); if (null === item && this._wasSearch()) { this._filterDataSource(null) } }, _createClearButton: function() { var eventName = eventUtils.addNamespace(clickEvent.name, this.NAME); var $clearButton = this.callBase(); eventsEngine.on($clearButton, eventName, function() { return false }); return $clearButton }, _wasSearch: function(value) { if (!arguments.length) { return this._wasSearchValue } this._wasSearchValue = value }, _searchHandler: function(e) { if (e && this._isControlKey(e.key) || this._preventFiltering) { delete this._preventFiltering; return } if (this._needPassDataSourceToList()) { this._wasSearch(true) } this.callBase(e) }, _dataSourceFiltered: function(searchValue) { this.callBase(); if (null !== searchValue) { this._renderInputSubstitution(); this._renderFocusedElement() } }, _valueSubstituted: function() { var input = this._input().get(0), isAllSelected = 0 === input.selectionStart && input.selectionEnd === this._searchValue().length, inputHasSelection = input.selectionStart !== input.selectionEnd; return this._wasSearch() && inputHasSelection && !isAllSelected }, _shouldSubstitutionBeRendered: function() { return this.option("autocompletionEnabled") && !this._preventSubstitution && this.option("searchEnabled") && !this.option("acceptCustomValue") && "startswith" === this.option("searchMode") }, _renderInputSubstitution: function() { if (!this._shouldSubstitutionBeRendered()) { delete this._preventSubstitution; return } var item = this._list && this._getPlainItems(this._list.option("items"))[0]; if (!item) { return } var $input = this._input(), valueLength = $input.val().length; if (0 === valueLength) { return } var inputElement = $input.get(0), displayValue = this._displayGetter(item); inputElement.value = displayValue; inputElement.selectionStart = valueLength; inputElement.selectionEnd = displayValue.length }, _cleanInputSelection: function() { var inputElement = this._input().get(0), endPosition = inputElement.value.length; inputElement.selectionStart = endPosition; inputElement.selectionEnd = endPosition }, _dispose: function() { this._renderInputValueAsync = commonUtils.noop; delete this._loadItemDeferred; this.callBase() }, _optionChanged: function(args) { switch (args.name) { case "_isAdaptablePopupPosition": case "autocompletionEnabled": break; case "onCustomItemCreating": this._initCustomItemCreatingAction(); break; case "tooltipEnabled": this._renderTooltip(); break; case "displayCustomValue": case "acceptCustomValue": case "showSelectionControls": case "useInkRipple": this._invalidate(); break; case "selectedItem": if (args.previousValue !== args.value) { this.callBase(args) } break; case "allowClearing": break; default: this.callBase(args) } } }); registerComponent("dxSelectBox", SelectBox); module.exports = SelectBox; module.exports.default = module.exports;