UNPKG

@syncfusion/ej2-splitbuttons

Version:

A package of feature-rich Essential JS 2 components such as DropDownButton, SplitButton, ProgressButton and ButtonGroup.

1,308 lines (1,305 loc) 78.5 kB
import { extend, deleteObject, addClass, isNullOrUndefined, ChildProperty, Property, Component, attributes, getComponent, select, getUniqueID, append, SanitizeHtmlHelper, compile, detach, removeClass, EventHandler, rippleEffect, isRippleEnabled, closest, Animation, formatUnit, classList, Collection, Complex, Event, NotifyPropertyChanges, getValue, setValue, remove, KeyboardEvents, getInstance, createElement, animationMode } from '@syncfusion/ej2-base'; import { Button } from '@syncfusion/ej2-buttons'; import { Popup, createSpinner, showSpinner, hideSpinner } from '@syncfusion/ej2-popups'; var __decorate = (undefined && undefined.__decorate) || function (decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; }; /** * @param {Object} props - Specifies the properties * @param {string[]} model - Specifies the model * @returns {Object} Component Model */ function getModel(props, model) { const obj = extend({}, props); for (const prop of Object.keys(obj)) { if ((model).indexOf(prop) < 0) { deleteObject(obj, prop); } } return obj; } /** @hidden * @param {HTMLElement} ul - Specifies the UL element * @param {number} keyCode - Specifies the keycode * @returns {void} */ function upDownKeyHandler(ul, keyCode) { const defaultIdx = keyCode === 40 ? 0 : ul.childElementCount - 1; let liIdx = defaultIdx; let li; const selectedLi = ul.querySelector('.e-selected'); if (selectedLi) { selectedLi.classList.remove('e-selected'); } for (let i = 0, len = ul.children.length; i < len; i++) { if (ul.children[i].classList.contains('e-focused')) { li = ul.children[i]; liIdx = i; li.classList.remove('e-focused'); if (keyCode === 40) { liIdx++; } else { liIdx--; } if (liIdx === (keyCode === 40 ? ul.childElementCount : -1)) { liIdx = defaultIdx; } } } li = ul.children[liIdx]; liIdx = isValidLI(ul, li, liIdx, keyCode); if (liIdx !== -1) { addClass([ul.children[liIdx]], 'e-focused'); ul.children[liIdx].focus(); } } /** * Get Valid LI element * * @param {HTMLElement} ul - Specifies the UL element * @param {Element} li - Specifies the LI element * @param {number} index - Specifies the index * @param {number} keyCode - Specifies the keycode * @param {number} count - Specifies the count * @returns {number} - Index */ function isValidLI(ul, li, index, keyCode, count = 0) { if (li.classList.contains('e-separator') || li.classList.contains('e-disabled')) { if (index === (keyCode === 40 ? ul.childElementCount - 1 : 0)) { index = keyCode === 40 ? 0 : ul.childElementCount - 1; } else { if (keyCode === 40) { index++; } else { index--; } } } li = ul.children[index]; if (li.classList.contains('e-separator') || li.classList.contains('e-disabled')) { count++; if (count === ul.childElementCount) { return index = -1; } index = isValidLI(ul, li, index, keyCode, count); } return index; } /** @hidden * @param {HTMLElement} popup - Specifies the popup element. * @param {boolean} blankIcon - Specifies the blankIcon value. * @returns {void} */ function setBlankIconStyle(popup, blankIcon) { const blankIconList = [].slice.call(popup.getElementsByClassName('e-blank-icon')); if (blankIcon) { const menuItem = [].slice.call(popup.getElementsByClassName('e-item')); menuItem.forEach((li) => { if (li.style.paddingLeft || li.style.paddingRight) { li.removeAttribute('style'); } }); } if (!blankIconList.length) { return; } let iconLi = popup.querySelector('.e-item:not(.e-blank-icon):not(.e-separator)'); if (isNullOrUndefined(iconLi)) { return; } if (iconLi.classList.contains('e-url')) { iconLi = iconLi.querySelector('.e-menu-url'); } const icon = iconLi.querySelector('.e-menu-icon'); let cssProp; const enableRtl = popup.classList.contains('e-rtl'); if (enableRtl) { cssProp = { padding: 'paddingRight', margin: 'marginLeft' }; } else { cssProp = { padding: 'paddingLeft', margin: 'marginRight' }; } /* eslint-disable */ let size = `${parseInt(getComputedStyle(icon).fontSize, 10) + parseInt((enableRtl ? getComputedStyle(icon)[cssProp.margin] : getComputedStyle(icon)[cssProp.margin]), 10) + parseInt(getComputedStyle(iconLi).paddingLeft, 10)}px`; blankIconList.forEach((li) => { if (li.classList.contains('e-url') && li.querySelector('.e-menu-url')) { li.querySelector('.e-menu-url').style[cssProp.padding] = size; } else { li.style[cssProp.padding] = size; } }); /* eslint-enable */ } /** * Defines the items of Split Button/DropDownButton. */ class Item extends ChildProperty { } __decorate([ Property('') ], Item.prototype, "iconCss", void 0); __decorate([ Property('') ], Item.prototype, "id", void 0); __decorate([ Property(false) ], Item.prototype, "separator", void 0); __decorate([ Property('') ], Item.prototype, "text", void 0); __decorate([ Property('') ], Item.prototype, "url", void 0); __decorate([ Property(false) ], Item.prototype, "disabled", void 0); var __decorate$1 = (undefined && undefined.__decorate) || function (decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; }; const classNames = { DISABLED: 'e-disabled', FOCUS: 'e-focused', ICON: 'e-menu-icon', ITEM: 'e-item', POPUP: 'e-dropdown-popup', RTL: 'e-rtl', SEPARATOR: 'e-separator', VERTICAL: 'e-vertical', POPUPWIDTH: 'e-dropdown-popup-width' }; /** * Animation configuration settings. */ class DropDownMenuAnimationSettings extends ChildProperty { } __decorate$1([ Property('SlideDown') ], DropDownMenuAnimationSettings.prototype, "effect", void 0); __decorate$1([ Property(400) ], DropDownMenuAnimationSettings.prototype, "duration", void 0); __decorate$1([ Property('ease') ], DropDownMenuAnimationSettings.prototype, "easing", void 0); /** * DropDownButton component is used to toggle contextual overlays for displaying list of action items. * It can contain both text and images. * ```html * <button id="element">DropDownButton</button> * ``` * ```typescript * <script> * var dropDownButtonObj = new DropDownButton({items: [{ text: 'Action1' }, { text: 'Action2' },{ text: 'Action3' }]); * dropDownButtonObj.appendTo("#element"); * </script> * ``` */ let DropDownButton = class DropDownButton extends Component { /** * Constructor for creating the widget * * @param {DropDownButtonModel} options - Specifies dropdown button model * @param {string|HTMLButtonElement} element - Specifies element * @hidden */ constructor(options, element) { super(options, element); this.isPopupCreated = true; } preRender() { /** */ } /** * Get the properties to be maintained in the persisted state. * * @returns {string} - Persist data */ getPersistData() { return this.addOnPersist([]); } /** * To open/close DropDownButton popup based on current state of the DropDownButton. * * @returns {void} */ toggle() { if (this.canOpen()) { this.openPopUp(); } else if (this.createPopupOnClick && !this.isPopupCreated) { this.createPopup(); this.openPopUp(); } else { this.closePopup(); } } /** * Initialize the Component rendering * * @returns {void} * @private */ render() { this.initialize(); if (!this.disabled) { this.wireEvents(); } this.renderComplete(); } /** * Adds a new item to the menu. By default, new item appends to the list as the last item, * but you can insert based on the text parameter. * * @param { ItemModel[] } items - Specifies an array of JSON data. * @param { string } text - Specifies the text to insert the newly added item in the menu. * @returns {void}. */ addItems(items, text) { let newItem; let idx = this.items.length; for (let j = 0, len = this.items.length; j < len; j++) { if (text === this.items[j].text) { idx = j; break; } } for (let i = items.length - 1; i >= 0; i--) { // eslint-disable-next-line @typescript-eslint/no-explicit-any newItem = new Item(this, 'items', items[i], true); this.items.splice(idx, 0, newItem); } if (!this.canOpen()) { this.createItems(); } } /** * Removes the items from the menu. * * @param { string[] } items - Specifies an array of string to remove the items. * @param { string } isUniqueId - Set `true` if specified items is a collection of unique id. * @returns {void}. */ removeItems(items, isUniqueId) { let refresh = false; for (let i = 0, len = items.length; i < len; i++) { for (let j = 0, len = this.items.length; j < len; j++) { if (items[i] === (isUniqueId ? this.items[j].id : this.items[j].text)) { this.items.splice(j, 1); refresh = true; break; } } } if (refresh && this.getULElement()) { this.createItems(); } } createPopup() { const div = this.createElement('div', { className: this.popupWidth !== 'auto' ? `${classNames.POPUP} ${classNames.POPUPWIDTH}` : classNames.POPUP, id: this.element.id + '-popup' }); document.body.appendChild(div); this.dropDown = new Popup(div, { width: this.popupWidth, relateTo: this.element, collision: { X: 'fit', Y: 'flip' }, position: { X: 'left', Y: 'bottom' }, targetType: 'relative', content: this.target ? this.getTargetElement() : '', enableRtl: this.enableRtl }); this.dropDown.element.setAttribute('role', 'dialog'); this.dropDown.element.setAttribute('aria-label', 'dropdown menu'); if (!isNullOrUndefined(this.popupContent)) { this.popupContent.style.display = ''; } if (this.dropDown.element.style.position === 'fixed') { this.dropDown.refreshPosition(this.element); } this.dropDown.hide(); attributes(this.element, { ['aria-haspopup']: this.items.length || this.target ? 'true' : 'false', ['aria-expanded']: 'false', ['type']: 'button' }); if (this.cssClass) { addClass([div], this.cssClass.replace(/\s+/g, ' ').trim().split(' ')); } this.isPopupCreated = true; if (this.createPopupOnClick) { const splitButton = getComponent(this.activeElem[0], 'split-btn'); if (splitButton) { splitButton.isPopupCreated = true; } } } getTargetElement() { if (this.createPopupOnClick && !this.isColorPicker() && !isNullOrUndefined(this.popupContent)) { return this.popupContent; } return typeof (this.target) === 'string' ? select(this.target) : this.target; } createItems(appendItems) { const items = this.items; const showIcon = this.hasIcon(this.items, 'iconCss'); let span; let item; let li; let eventArgs; let ul = this.getULElement(); if (ul) { ul.innerHTML = ''; } else { ul = this.createElement('ul', { attrs: { 'role': 'menu', 'tabindex': '0' } }); } if (this.itemTemplate) { const compiledTemplate = this.compiletemplate(this.itemTemplate); items.forEach((item) => { const li = this.createElement('li', { className: item.separator ? classNames.ITEM + ' ' + classNames.SEPARATOR : classNames.ITEM, attrs: item.separator ? { 'role': 'separator', 'tabindex': '-1', 'aria-label': 'separator', 'aria-hidden': 'true' } : { 'role': 'menuitem', 'tabindex': '-1', 'aria-label': item.text }, id: item.id ? item.id : getUniqueID('e-' + this.getModuleName() + '-item') }); const compiledElement = compiledTemplate(item, this, 'template', null, false, null, li); if (compiledElement) { append(compiledElement, li); } if (item.disabled) { li.classList.add('e-disabled'); } const eventArgs = { item, element: li }; this.trigger('beforeItemRender', eventArgs); if (eventArgs.item.disabled !== item.disabled) { li.classList[eventArgs.item.disabled ? 'add' : 'remove']('e-disabled'); } ul.appendChild(li); }); if (this.isReact) { this.renderReactTemplates(); } } else { for (let i = 0; i < items.length; i++) { item = items[i]; if (this.enableHtmlSanitizer) { item.text = SanitizeHtmlHelper.sanitize(item.text); } const tempItem = item.text; li = this.createElement('li', { innerHTML: item.url ? '' : tempItem, className: item.separator ? classNames.ITEM + ' ' + classNames.SEPARATOR : classNames.ITEM, attrs: item.separator ? { 'role': 'separator', 'tabindex': '-1', 'aria-label': 'separator', 'aria-hidden': 'true' } : { 'role': 'menuitem', 'tabindex': '-1', 'aria-label': tempItem }, id: item.id ? item.id : getUniqueID('e-' + this.getModuleName() + '-item') }); if (this.enableHtmlSanitizer) { li.textContent = item.url ? '' : tempItem; } else { li.innerHTML = item.url ? '' : tempItem; } if (item.url) { li.appendChild(this.createAnchor(item)); li.classList.add('e-url'); } if (item.iconCss) { span = this.createElement('span', { className: classNames.ICON + ' ' + item.iconCss }); if (item.url) { li.childNodes[0].appendChild(span); } else { li.insertBefore(span, li.childNodes[0]); } } else { if (showIcon && !item.separator) { li.classList.add('e-blank-icon'); } } const beforeDisabled = item.disabled; if (item.disabled) { li.classList.add('e-disabled'); } eventArgs = { item: item, element: li }; this.trigger('beforeItemRender', eventArgs); const afterDisabled = eventArgs.item.disabled; if (beforeDisabled !== afterDisabled) { if (eventArgs.item.disabled) { li.classList.add('e-disabled'); } else { li.classList.remove('e-disabled'); } } ul.appendChild(li); } } if (appendItems) { this.getPopUpElement().appendChild(ul); } if (showIcon) { setBlankIconStyle(this.getPopUpElement()); } } // eslint-disable-next-line @typescript-eslint/no-explicit-any compiletemplate(template) { if (!this.itemTemplate) { return undefined; } try { if (typeof this.itemTemplate !== 'function') { const templateElement = document.querySelector(this.itemTemplate); if (templateElement) { return compile(templateElement.innerHTML.trim()); } } return compile(template); } catch (_a) { return compile(template); } } hasIcon(items, field) { for (let i = 0, len = items.length; i < len; i++) { if (items[i][`${field}`]) { return true; } } return false; } createAnchor(item) { const tempItem = item.text; const anchor = this.createElement('a', { className: 'e-menu-text e-menu-url', attrs: { 'href': item.url } }); if (this.enableHtmlSanitizer) { anchor.textContent = tempItem; } else { anchor.innerHTML = tempItem; } return anchor; } initialize() { this.button = new Button({ iconCss: this.iconCss, iconPosition: this.iconPosition, cssClass: this.cssClass, content: this.content, disabled: this.disabled, enableRtl: this.enableRtl, enablePersistence: this.enablePersistence }); this.button.createElement = this.createElement; this.button.appendTo(this.element); if (!this.element.id) { this.element.id = getUniqueID('e-' + this.getModuleName()); } this.appendArrowSpan(); this.setActiveElem([this.element]); this.element.setAttribute('tabindex', '0'); this.element.setAttribute('aria-label', this.element.textContent ? this.element.textContent : 'dropdownbutton'); if ((this.target && !this.isColorPicker() && !this.createPopupOnClick) || !this.createPopupOnClick) { this.createPopup(); } else { this.isPopupCreated = false; if (this.target && !this.isColorPicker() && this.createPopupOnClick) { this.popupContent = this.getTargetElement(); this.popupContent.style.display = 'none'; } } } isColorPicker() { if (!this.element) { return false; } const prevElem = this.element.previousSibling; if (prevElem && prevElem.classList && prevElem.classList.contains('e-split-colorpicker')) { return true; } return false; } appendArrowSpan() { this.cssClass = isNullOrUndefined(this.cssClass) ? '' : this.cssClass; this.element.appendChild(this.createElement('span', { className: 'e-btn-icon e-icons ' + 'e-icon-' + (this.cssClass.indexOf(classNames.VERTICAL) > -1 ? 'bottom' : 'right') + ' e-caret' })); } setActiveElem(elem) { this.activeElem = elem; } /** * Get component name. * * @returns {string} - Module Name * @private */ getModuleName() { return 'dropdown-btn'; } canOpen() { let val = false; if (this.isPopupCreated) { val = this.getPopUpElement().classList.contains('e-popup-close'); } return val; } /** * Destroys the widget. * * @returns {void} */ destroy() { super.destroy(); if (this.getModuleName() === 'dropdown-btn') { let classList; if (this.element.querySelector('span.e-caret')) { detach(this.element.querySelector('span.e-caret')); } if (this.cssClass) { classList = this.cssClass.split(' '); } this.button.destroy(); if (classList) { removeClass([this.element], classList); } removeClass(this.activeElem, ['e-active']); const attrList = this.element.getAttribute('class') ? ['aria-haspopup', 'aria-expanded', 'aria-owns', 'type'] : ['aria-haspopup', 'aria-expanded', 'aria-owns', 'type', 'class']; attrList.forEach((key) => { this.element.removeAttribute(key); }); this.popupUnWireEvents(); this.destroyPopup(); this.isPopupCreated = false; if (!this.disabled) { this.unWireEvents(); } } } destroyPopup() { if (this.isPopupCreated) { this.dropDown.destroy(); if (this.getPopUpElement()) { const popupEle = document.getElementById(this.getPopUpElement().id); if (popupEle) { removeClass([popupEle], ['e-popup-open', 'e-popup-close']); detach(popupEle); } } EventHandler.remove(this.getPopUpElement(), 'click', this.clickHandler); EventHandler.remove(this.getPopUpElement(), 'keydown', this.keyBoardHandler); if (this.isPopupCreated && this.dropDown) { this.dropDown.element = null; this.dropDown = undefined; } } this.isPopupCreated = false; const splitButton = getComponent(this.activeElem[0], 'split-btn'); if (this.createPopupOnClick && splitButton) { const dropDownButton = getComponent(this.activeElem[1], 'dropdown-btn'); if (dropDownButton) { dropDownButton.isPopupCreated = false; } } } getPopUpElement() { let val = null; if (!this.dropDown && this.activeElem[0].classList.contains('e-split-btn')) { const dropDownBtn = getComponent(this.activeElem[1], 'dropdown-btn'); if (dropDownBtn) { this.dropDown = dropDownBtn.dropDown; } } if (this.dropDown) { val = this.dropDown.element; } return val; } getULElement() { let val = null; if (this.getPopUpElement()) { val = this.getPopUpElement().children[0]; } return val; } wireEvents() { this.delegateMousedownHandler = this.mousedownHandler.bind(this); if (!this.createPopupOnClick) { EventHandler.add(document, 'mousedown touchstart', this.delegateMousedownHandler, this); } EventHandler.add(this.element, 'click', this.clickHandler, this); EventHandler.add(this.element, 'keydown', this.keyBoardHandler, this); // eslint-disable-next-line @typescript-eslint/no-explicit-any EventHandler.add(window, 'resize', this.windowResize, this); } windowResize() { if (!this.canOpen() && this.dropDown) { this.dropDown.refreshPosition(this.element); } } popupWireEvents() { if (!this.delegateMousedownHandler) { this.delegateMousedownHandler = this.mousedownHandler.bind(this); } const popupElement = this.getPopUpElement(); if (this.createPopupOnClick) { EventHandler.add(document, 'mousedown touchstart', this.delegateMousedownHandler, this); } if (popupElement) { EventHandler.add(popupElement, 'click', this.clickHandler, this); EventHandler.add(popupElement, 'keydown', this.keyBoardHandler, this); if (this.closeActionEvents) { EventHandler.add(popupElement, this.closeActionEvents, this.focusoutHandler, this); } } this.rippleFn = rippleEffect(popupElement, { selector: '.' + classNames.ITEM }); } popupUnWireEvents() { const popupElement = this.getPopUpElement(); if (this.createPopupOnClick) { EventHandler.remove(document, 'mousedown touchstart', this.delegateMousedownHandler); } if (popupElement && popupElement.parentElement) { EventHandler.remove(popupElement, 'click', this.clickHandler); EventHandler.remove(popupElement, 'keydown', this.keyBoardHandler); if (this.closeActionEvents) { EventHandler.remove(popupElement, this.closeActionEvents, this.focusoutHandler); } } if (isRippleEnabled && this.rippleFn) { this.rippleFn(); } } /** * Handles the keyboard interactions. * * @param {KeyboardEventArgs} e - Specifies keyboard event args. * @returns {void} * @hidden */ keyBoardHandler(e) { if (e.target === this.element && (e.keyCode === 9 || (!e.altKey && e.keyCode === 40) || e.keyCode === 38)) { return; } if (e.target && (e.target.classList.contains('e-item') || (e.target.parentElement && e.target.parentElement.classList.contains('e-split-btn-wrapper'))) && e.keyCode === 13) { e.preventDefault(); } switch (e.keyCode) { case 38: case 40: if (e.altKey && (e.keyCode === 38 || e.keyCode === 40)) { this.keyEventHandler(e); } else { this.upDownKeyHandler(e); } break; case 9: case 13: case 27: case 32: this.keyEventHandler(e); break; } } isSafari() { return /^((?!chrome|android).)*safari/i.test(navigator.userAgent); } upDownKeyHandler(e) { if (this.target && (e.keyCode === 38 || e.keyCode === 40)) { return; } e.preventDefault(); upDownKeyHandler(this.getULElement(), e.keyCode); } keyEventHandler(e) { if (this.target && (e.keyCode === 13 || e.keyCode === 9)) { return; } if (e.keyCode === 13 && this.activeElem[0].classList.contains('e-split-btn')) { this.triggerSelect(e); this.activeElem[0].focus(); return; } if (e.target && e.target.className.indexOf('e-edit-template') > -1 && e.keyCode === 32) { return; } if (e.keyCode !== 9) { e.preventDefault(); } if (e.keyCode === 27 || e.keyCode === 38 || e.keyCode === 9) { if (!this.canOpen()) { this.closePopup(e, this.element); } } else { this.clickHandler(e); } } getLI(elem) { return elem.tagName === 'LI' ? elem : closest(elem, 'li'); } mousedownHandler(e) { const trgt = e.target; if (this.dropDown && !this.canOpen() && this.getPopUpElement() && !(closest(trgt, '[id="' + this.getPopUpElement().id + '"]') || closest(trgt, '[id="' + this.element.id + '"]'))) { this.closePopup(e); } } focusoutHandler(e) { if (this.isPopupCreated && !this.canOpen()) { const liTarget = (e.relatedTarget || e.target); if (liTarget && liTarget.className.indexOf('e-item') > -1) { const li = this.getLI(liTarget); if (li) { const liIdx = Array.prototype.indexOf.call(this.getULElement().children, li); const item = this.items[liIdx]; if (item) { const selectEventArgs = { element: li, item: item, event: e }; this.trigger('select', selectEventArgs); } } } this.closePopup(e); } } clickHandler(e) { const trgt = e.target; if (closest(trgt, '[id="' + this.element.id + '"]')) { if (!this.createPopupOnClick || (this.target && this.target !== '' && !this.isColorPicker() && !this.createPopupOnClick)) { if (this.getPopUpElement().classList.contains('e-popup-close')) { this.openPopUp(e); } else { this.closePopup(e); } } else if (this.isPopupCreated) { this.closePopup(e, this.activeElem[0]); } else { this.createPopup(); this.openPopUp(e); } } else { if (closest(trgt, '[id="' + this.getPopUpElement().id + '"]')) { const li = this.getLI(e.target); if (li) { this.triggerSelect(e); this.closePopup(e, this.activeElem[0]); } } } } triggerSelect(e) { let eventArgs; let liIdx; let item; const li = this.getLI(e.target); if (li) { liIdx = Array.prototype.indexOf.call(this.getULElement().children, li); item = this.items[liIdx]; if (item) { eventArgs = { element: li, item: item, event: e }; this.trigger('select', eventArgs); } } } openPopUp(e = null) { const popupElem = this.getPopUpElement(); if (this.activeElem[0] && this.activeElem[0].classList.contains('e-dropdown-btn') && popupElem.style.width && popupElem.style.width !== 'auto') { this.setWidth(popupElem); } if (!this.target) { this.createItems(true); } else { if (this.activeElem.length > 1) { const splitButton = getComponent(this.activeElem[0], 'split-btn'); // eslint-disable-next-line @typescript-eslint/no-explicit-any if (splitButton.isReact && popupElem.childNodes.length < 1) { // eslint-disable-next-line @typescript-eslint/no-explicit-any if (splitButton.appendReactElement) { // eslint-disable-next-line @typescript-eslint/no-explicit-any splitButton.appendReactElement(this.getTargetElement(), this.getPopUpElement()); } this.renderReactTemplates(); } } else { // eslint-disable-next-line @typescript-eslint/no-explicit-any if (this.isReact && popupElem.childNodes.length < 1) { // eslint-disable-next-line @typescript-eslint/no-explicit-any if (this.appendReactElement) { // eslint-disable-next-line @typescript-eslint/no-explicit-any this.appendReactElement(this.getTargetElement(), this.getPopUpElement()); } this.renderReactTemplates(); } } } const ul = this.getULElement(); this.popupWireEvents(); const beforeOpenArgs = { element: ul, items: this.items, event: e, cancel: false }; this.trigger('beforeOpen', beforeOpenArgs, (observedArgs) => { if (!observedArgs.cancel) { const ul = this.getULElement(); const animationOptions = this.animationSettings.effect !== 'None' ? { name: this.animationSettings.effect, duration: this.animationSettings.duration, timingFunction: this.animationSettings.easing } : null; if (animationOptions) { this.animatePopup(animationOptions, ul); } this.dropDown.show(null, this.element); addClass([this.element], 'e-active'); this.element.setAttribute('aria-expanded', 'true'); this.element.setAttribute('aria-owns', this.getPopUpElement().id); if (ul && !this.isSafari()) { ul.focus(); } if (this.enableRtl && ul.parentElement.style.left !== '0px') { let wrapperWidth; if (this.element.parentElement && this.element.parentElement.classList.contains('e-split-btn-wrapper')) { wrapperWidth = this.element.parentElement.offsetWidth; } else { wrapperWidth = this.element.offsetWidth; } const popupRect = ul.parentElement.offsetWidth - wrapperWidth; let popupLeft = parseFloat(ul.parentElement.style.left) - popupRect; if (popupLeft < 0) { popupLeft = 0; } ul.parentElement.style.left = popupLeft + 'px'; } const openArgs = { element: ul, items: this.items }; this.trigger('open', openArgs); if (ul && this.isSafari()) { ul.focus(); } } }); } animatePopup(animationOptions, element) { new Animation(animationOptions).animate(element, { begin: (args) => { args.element.parentElement.style.height = args.element.parentElement.offsetHeight + 'px'; } }); } setWidth(popupElem) { const width = formatUnit(popupElem.style.width); if (width.indexOf('%') > -1) { const btnWidth = this.element.offsetWidth * parseFloat(width) / 100; popupElem.style.width = `${btnWidth}px`; } } closePopup(e = null, focusEle) { const ul = this.getULElement(); const beforeCloseArgs = { element: ul, items: this.items, event: e, cancel: false }; this.trigger('beforeClose', beforeCloseArgs, (observedArgs) => { if (!observedArgs.cancel) { const popupElement = this.getPopUpElement(); if (popupElement) { EventHandler.remove(popupElement, 'keydown', this.keyBoardHandler); } this.popupUnWireEvents(); const ul = this.getULElement(); let selectedLi; if (ul) { selectedLi = ul.querySelector('.e-selected'); } if (selectedLi) { selectedLi.classList.remove('e-selected'); } if (this.dropDown) { this.dropDown.hide(); } removeClass(this.activeElem, 'e-active'); this.element.setAttribute('aria-expanded', 'false'); this.element.removeAttribute('aria-owns'); if (focusEle) { if (!this.isSafari()) { focusEle.focus(); } else { focusEle.focus({ preventScroll: true }); } } const closeArgs = { element: ul, items: this.items }; this.trigger('close', closeArgs); if (!this.target && ul) { detach(ul); } if (!this.target || this.isColorPicker() || (this.target && !this.isColorPicker())) { if (this.createPopupOnClick) { this.destroyPopup(); } } } else { if (ul) { ul.focus(); } } }); } unWireEvents() { if (!this.createPopupOnClick) { EventHandler.remove(document, 'mousedown touchstart', this.delegateMousedownHandler); } EventHandler.remove(this.element, 'click', this.clickHandler); EventHandler.remove(this.element, 'keydown', this.keyBoardHandler); if (this.isPopupCreated) { EventHandler.remove(this.getPopUpElement(), 'click', this.clickHandler); EventHandler.remove(this.getPopUpElement(), 'keydown', this.keyBoardHandler); } // eslint-disable-next-line @typescript-eslint/no-explicit-any EventHandler.remove(window, 'resize', this.windowResize); } /** * Called internally if any of the property value changed. * * @param {DropDownButtonModel} newProp - Specifies new properties * @param {DropDownButtonModel} oldProp - Specifies old properties * @returns {void} * @private */ onPropertyChanged(newProp, oldProp) { const btnModel = ['content', 'cssClass', 'iconCss', 'iconPosition', 'disabled', 'enableRtl']; this.button.setProperties(getModel(newProp, btnModel)); let popupElement; if (this.isPopupCreated) { popupElement = this.getPopUpElement(); this.dropDown.setProperties(getModel(newProp, ['enableRtl'])); } for (const prop of Object.keys(newProp)) { switch (prop) { case 'content': if (!this.element.querySelector('span.e-caret')) { this.appendArrowSpan(); } break; case 'disabled': if (newProp.disabled) { this.unWireEvents(); if (this.isPopupCreated && !this.canOpen()) { this.closePopup(); } } else { this.wireEvents(); } break; case 'cssClass': oldProp.cssClass = isNullOrUndefined(oldProp.cssClass) ? '' : oldProp.cssClass; if (newProp.cssClass.indexOf(classNames.VERTICAL) > -1 || oldProp.cssClass.indexOf(classNames.VERTICAL) > -1) { if (!this.element.querySelector('span.e-caret')) { this.appendArrowSpan(); } const arrowSpan = this.element.querySelector('span.e-caret'); // eslint-disable-next-line @typescript-eslint/no-unused-expressions newProp.cssClass.indexOf(classNames.VERTICAL) > -1 ? classList(arrowSpan, ['e-icon-bottom'], ['e-icon-right']) : classList(arrowSpan, ['e-icon-right'], ['e-icon-bottom']); } if (this.isPopupCreated) { if (oldProp.cssClass) { removeClass([popupElement], oldProp.cssClass.split(' ')); } if (newProp.cssClass) { addClass([popupElement], newProp.cssClass.replace(/\s+/g, ' ').trim().split(' ')); } } break; case 'target': this.dropDown.content = this.getTargetElement(); this.dropDown.dataBind(); break; case 'items': if (this.isPopupCreated && this.getULElement()) { this.createItems(); } break; case 'createPopupOnClick': if (newProp.createPopupOnClick) { this.destroyPopup(); } else { this.createPopup(); } break; } } } /** * Sets the focus to DropDownButton * its native method * * @public * @returns {void} */ focusIn() { this.element.focus(); } }; __decorate$1([ Property('') ], DropDownButton.prototype, "content", void 0); __decorate$1([ Property('') ], DropDownButton.prototype, "cssClass", void 0); __decorate$1([ Property(false) ], DropDownButton.prototype, "disabled", void 0); __decorate$1([ Property('') ], DropDownButton.prototype, "iconCss", void 0); __decorate$1([ Property('Left') ], DropDownButton.prototype, "iconPosition", void 0); __decorate$1([ Property(true) ], DropDownButton.prototype, "enableHtmlSanitizer", void 0); __decorate$1([ Collection([], Item) ], DropDownButton.prototype, "items", void 0); __decorate$1([ Property(false) ], DropDownButton.prototype, "createPopupOnClick", void 0); __decorate$1([ Property('') ], DropDownButton.prototype, "target", void 0); __decorate$1([ Property('') ], DropDownButton.prototype, "closeActionEvents", void 0); __decorate$1([ Property(null) ], DropDownButton.prototype, "itemTemplate", void 0); __decorate$1([ Property('auto') ], DropDownButton.prototype, "popupWidth", void 0); __decorate$1([ Complex({ effect: 'None' }, DropDownMenuAnimationSettings) ], DropDownButton.prototype, "animationSettings", void 0); __decorate$1([ Event() ], DropDownButton.prototype, "beforeItemRender", void 0); __decorate$1([ Event() ], DropDownButton.prototype, "beforeOpen", void 0); __decorate$1([ Event() ], DropDownButton.prototype, "beforeClose", void 0); __decorate$1([ Event() ], DropDownButton.prototype, "close", void 0); __decorate$1([ Event() ], DropDownButton.prototype, "open", void 0); __decorate$1([ Event() ], DropDownButton.prototype, "select", void 0); __decorate$1([ Event() ], DropDownButton.prototype, "created", void 0); DropDownButton = __decorate$1([ NotifyPropertyChanges ], DropDownButton); var __decorate$2 = (undefined && undefined.__decorate) || function (decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; }; const RTL = 'e-rtl'; const TAGNAME = 'EJS-SPLITBUTTON'; /** * SplitButton component has primary and secondary button. Primary button is used to select * default action and secondary button is used to toggle contextual overlays for displaying list of * action items. It can contain both text and images. * ```html * <button id="element"></button> * ``` * ```typescript * <script> * var splitBtnObj = new SplitButton({content: 'SplitButton'}); * splitBtnObj.appendTo("#element"); * </script> * ``` */ let SplitButton = class SplitButton extends DropDownButton { /** * Constructor for creating the widget * * @param {SplitButtonModel} options - Specifies the splitbutton model * @param {string|HTMLButtonElement} element - Specifies the element * @hidden */ constructor(options, element) { super(options, element); } /** * Initialize Angular support. * * @private * @returns {void} */ preRender() { let ele = this.element; if (ele.tagName === TAGNAME) { const ejInstance = getValue('ej2_instances', ele); const btn = this.createElement('button', { attrs: { 'type': 'button' } }); const wrapper = this.createElement(TAGNAME, { className: 'e-' + this.getModuleName() + '-wrapper' }); for (let idx = 0, len = ele.attributes.length; idx < len; idx++) { btn.setAttribute(ele.attributes[idx].nodeName, ele.attributes[idx].nodeValue); } ele.parentNode.insertBefore(wrapper, ele); detach(ele); ele = btn; wrapper.appendChild(ele); setValue('ej2_instances', ejInstance, ele); this.wrapper = wrapper; this.element = ele; } if (!this.element.id) { this.element.id = getUniqueID('e-' + this.getModuleName()); } } /** * Initialize the Component rendering. * * @returns {void} * @private */ render() { this.initWrapper(); this.createPrimaryButton(); this.renderControl(); } renderControl() { this.createSecondaryButton(); this.setActiveElem([this.element, this.secondaryBtnObj.element]); this.setAria(); this.wireEvents(); this.renderComplete(); } /** * Adds a new item to the menu. By default, new item appends to the list as the last item, * but you can insert based on the text parameter. * * @param { ItemModel[] } items - Specifies an array of JSON data. * @param { string } text - Specifies the text to insert the newly added item in the menu. * @returns {void}. */ addItems(items, text) { super.addItems(items, text); this.secondaryBtnObj.items = this.items; } /** * Removes the items from the menu. * * @param { string[] } items - Specifies an array of string to remove the items. * @param { string } isUniqueId - Set `true` if specified items is a collection of unique id. * @returns {void}. */ removeItems(items, isUniqueId) { super.removeItems(items, isUniqueId); this.secondaryBtnObj.items = this.items; } initWrapper() { if (!this.wrapper) { this.wrapper = this.createElement('div', { className: 'e-' + this.getModuleName() + '-wrapper' }); this.element.parentNode.insertBefore(this.wrapper, this.element); } this.element.classList.remove('e-' + this.getModuleName()); if (this.enableRtl) { this.wrapper.classList.add(RTL); } if (this.cssClass) { addClass([this.wrapper], this.cssClass.replace(/\s+/g, ' ').trim().split(' ')); } } createPrimaryButton() { const btnModel = { cssClass: this.cssClass, enableRtl: this.enableRtl, iconCss: this.iconCss, iconPosition: this.iconPosition, content: this.content, disabled: this.disabled }; this.primaryBtnObj = new Button(btnModel); this.primaryBtnObj.createElement = this.createElement; this.primaryBtnObj.appendTo(this.element); this.element.classList.add('e-' + this.getModuleName()); this.element.type = 'button'; this.wrapper.appendChild(this.element); } createSecondaryButton() { const btnElem = this.createElement('button', { className: 'e-icon-btn', attrs: { 'tabindex': '-1' }, id: this.element.id + '_dropdownbtn' }); this.wrapper.appendChild(btnElem); const dropDownBtnModel = { cssClass: this.cssClass, disabled: this.disabled, enableRtl: this.enableRtl, items: this.items, target: this.target, createPopupOnClick: this.createPopupOnClick }; dropDownBtnModel.beforeItemRender = (args) => { if (this.createPopupOnClick) { this.secondaryBtnObj.dropDown.relateTo = this.wrapper; this.dropDown = this.secondaryBtnObj.dropDown; } this.trigger('beforeItemRender', args); }; dropDownBtnModel.open = (args) => { this.trigger('open', args); }; dropDownBtnModel.close = (args) => { this.trigger('close', args); }; dropDownBtnModel.select = (args) => { this.trigger('select', args); }; dropDownBtnModel.beforeOpen = (args) => { if (this.createPopupOnClick && this.items.length === 0) { this.secondaryBtnObj.dropDown.relateTo = this.wrapper; this.dropDown = this.secondaryBtnObj.dropDown; } const callBackPromise = new Deferred(); this.trigger('beforeOpen', args, (observedArgs) => { callBackPromise.resolve(observedArgs); }); return callBackPromise; }; dropDownBtnModel.beforeClose = (args) => { const callBackPromise = new Deferred(); this.trigger('beforeClose', args, (observedArgs) => { callBackPromise.resolve(observedArgs); }); return callBackPromise; }; this.secondaryBtnObj = new DropDownButton(dropDownBtnModel); this.secondaryBtnObj.createElement = this.createElement; this.secondaryBtnObj.appendTo(btnElem); if (!this.createPopupOnClick) { this.secondaryBtnObj.dropDown.relateTo = this.wrapper