UNPKG

@progress/kendo-angular-common

Version:

Kendo UI for Angular - Utility Package

1,255 lines (1,207 loc) 68.3 kB
/**----------------------------------------------------------------------------------------- * Copyright © 2025 Progress Software Corporation. All rights reserved. * Licensed under commercial license. See LICENSE.md in the project root for more information *-------------------------------------------------------------------------------------------*/ import * as i0 from '@angular/core'; import { EventEmitter, Directive, Input, Output, Injectable, Component, HostBinding, ViewChild, Optional, isDevMode } from '@angular/core'; import { detectDesktopBrowser, detectMobileOS } from '@progress/kendo-common'; import { take, auditTime } from 'rxjs/operators'; import { Draggable } from '@progress/kendo-draggable'; import { merge, fromEvent, from, Subscription } from 'rxjs'; import { NgIf, NgStyle } from '@angular/common'; import { getLicenseStatus } from '@progress/kendo-licensing'; /** * @hidden */ const isDocumentAvailable = () => typeof document !== 'undefined'; /** * @hidden */ const isChanged = (propertyName, changes, skipFirstChange = true) => (typeof changes[propertyName] !== 'undefined' && (!changes[propertyName].isFirstChange() || !skipFirstChange) && changes[propertyName].previousValue !== changes[propertyName].currentValue); /** * @hidden */ const anyChanged = (propertyNames, changes, skipFirstChange = true) => propertyNames.some(name => isChanged(name, changes, skipFirstChange)); /** * @hidden */ const hasObservers = (emitter) => emitter && emitter.observers.length > 0; /** * @hidden */ const guid = () => { let id = ""; for (let i = 0; i < 32; i++) { const random = Math.random() * 16 | 0; // eslint-disable-line no-bitwise if (i === 8 || i === 12 || i === 16 || i === 20) { id += "-"; } let charValue; if (i === 12) { charValue = 4; } else if (i === 16) { // eslint-disable-next-line no-bitwise charValue = random & 3 | 8; } else { charValue = random; } id += charValue.toString(16); } return id; }; /** * @hidden * * Returns true if the used browser is Safari. */ const isSafari = (userAgent) => { return detectDesktopBrowser(userAgent).safari || (detectMobileOS(userAgent) && detectMobileOS(userAgent).browser === 'mobilesafari'); }; /** * @hidden * * Returns true if the used browser is Firefox. */ const isFirefox = (userAgent) => { const desktopBrowser = detectDesktopBrowser(userAgent); const mobileOS = detectMobileOS(userAgent); return (desktopBrowser && desktopBrowser.mozilla) || (mobileOS && mobileOS.browser === 'firefox'); }; /** * @hidden */ const firefoxMaxHeight = 17895697; /* eslint-disable @typescript-eslint/no-explicit-any */ /** * @hidden */ const isPresent = (value) => value !== null && value !== undefined; /** * @hidden */ const isObjectPresent = (value) => { return isObject(value) && Object.keys(value).length > 0; }; /** * @hidden */ const isString = (value) => value instanceof String || typeof value === 'string'; /** * @hidden */ const isObject = (value) => isPresent(value) && !Array.isArray(value) && typeof value === 'object'; /** * @hidden */ const splitStringToArray = (value) => value.trim().replace(/\s+/g, " ").split(' '); /** * Receives CSS class declarations either as an object, string or array and returns an array of the class names. * * @hidden */ const parseCSSClassNames = (value) => { if (Array.isArray(value)) { return parseArrayClassNames(value); } if (isObject(value)) { return parseObjectClassNames(value); } if (isString(value)) { return parseStringClassNames(value); } }; const parseObjectClassNames = (value) => { const classes = []; Object.keys(value).forEach((className) => { const currentClassName = splitStringToArray(className); if (value[className] && currentClassName.length) { classes.push(...currentClassName); } }); return classes; }; const parseStringClassNames = (value) => { const classes = []; const classesArray = splitStringToArray(value); classesArray.forEach((className) => { classes.push(className); }); return classes; }; const parseArrayClassNames = (value) => { const classes = []; value.forEach((className) => { const current = splitStringToArray(className); if (current[0]) { classes.push(...current); } }); return classes; }; /** * @hidden */ const setHTMLAttributes = (attributes, renderer, element, zone) => { zone ? zone.onStable.pipe(take(1)).subscribe(() => { applyAttributes(attributes, renderer, element); }) : applyAttributes(attributes, renderer, element); }; /** * @hidden */ const removeHTMLAttributes = (attributes, renderer, element) => { for (const attribute in attributes) { if (attribute) { renderer.removeAttribute(element, attribute); } } }; /** * @hidden */ const parseAttributes = (target, source) => { const targetObj = target; Object.keys(source).forEach(key => { delete targetObj[key]; }); return targetObj; }; /** * @hidden */ const applyAttributes = (attributes, renderer, element) => { for (const attribute in attributes) { if (attribute && isPresent(attributes[attribute])) { renderer.setAttribute(element, attribute, attributes[attribute]); } } }; /** * @hidden */ const isControlRequired = (control) => { if (!control?.validator) { return false; } return control.validator(control)?.hasOwnProperty('required'); }; const areObjectsEqual = (firstObject, secondObject) => { if (Object.keys(firstObject).length !== Object.keys(secondObject).length) { return false; } const equalSettings = Object.entries(firstObject) .filter(([key, value]) => value === secondObject[key.toString()]); return equalSettings.length === Object.keys(firstObject).length; }; const processCssValue = (value) => { if (typeof value === 'number') { return `${value}px`; } else if (typeof value === 'string') { const trimmedValue = value.trim(); const numValue = parseInt(trimmedValue, 10); if (!isNaN(numValue) && Number.isFinite(numValue)) { if (numValue.toString() === trimmedValue) { return `${numValue}px`; } else { return value; } } return null; } return null; }; class DraggableDirective { element; ngZone; enableDrag = true; kendoPress = new EventEmitter(); kendoDrag = new EventEmitter(); kendoRelease = new EventEmitter(); draggable; constructor(element, ngZone) { this.element = element; this.ngZone = ngZone; } ngOnInit() { this.toggleDraggable(); } ngOnChanges(changes) { if (isChanged('enableDrag', changes)) { this.toggleDraggable(); } } ngOnDestroy() { this.destroyDraggable(); } toggleDraggable() { if (isDocumentAvailable()) { this.destroyDraggable(); if (this.enableDrag) { this.draggable = new Draggable({ drag: (e) => this.kendoDrag.next(e), press: (e) => this.kendoPress.next(e), release: (e) => this.kendoRelease.next(e) }); this.ngZone.runOutsideAngular(() => this.draggable?.bindTo(this.element.nativeElement)); } } } destroyDraggable() { if (this.draggable) { this.draggable.destroy(); this.draggable = undefined; } } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DraggableDirective, deps: [{ token: i0.ElementRef }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Directive }); static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: DraggableDirective, isStandalone: true, selector: "[kendoDraggable]", inputs: { enableDrag: "enableDrag" }, outputs: { kendoPress: "kendoPress", kendoDrag: "kendoDrag", kendoRelease: "kendoRelease" }, usesOnChanges: true, ngImport: i0 }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DraggableDirective, decorators: [{ type: Directive, args: [{ selector: '[kendoDraggable]', standalone: true }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.NgZone }]; }, propDecorators: { enableDrag: [{ type: Input }], kendoPress: [{ type: Output }], kendoDrag: [{ type: Output }], kendoRelease: [{ type: Output }] } }); const closestInScope = (node, predicate, scope) => { while (node && node !== scope && !predicate(node)) { node = node.parentNode; } if (node !== scope) { return node; } return undefined; }; const closest = (node, predicate) => { while (node && !predicate(node)) { node = node.parentNode; } return node; }; const contains = (parent, node, matchSelf = false) => { const outside = !closest(node, (child) => child === parent); if (outside) { return false; } const el = closest(node, (child) => child === node); return el && (matchSelf || el !== parent); }; const findElement = (node, predicate, matchSelf = true) => { if (!node) { return; } if (matchSelf && predicate(node)) { return node; } node = node.firstChild; while (node) { if (node.nodeType === 1) { const element = findElement(node, predicate); if (element) { return element; } } node = node.nextSibling; } }; const focusableRegex = /^(?:a|input|select|option|textarea|button|object)$/i; const isFocusable = (element) => { if (!element.tagName) { return false; } const tagName = element.tagName.toLowerCase(); const hasTabIndex = Boolean(element.getAttribute('tabIndex')); const focusable = !element.disabled && focusableRegex.test(tagName); return focusable || hasTabIndex; }; const isVisible = (element) => { const rect = element.getBoundingClientRect(); const hasSize = rect.width > 0 && rect.height > 0; const hasPosition = rect.x !== 0 && rect.y !== 0; // Elements can have zero size due to styling, but they will still count as visible. // For example, the selection checkbox has no size, but is made visible through styling. return (hasSize || hasPosition) && window.getComputedStyle(element).visibility !== 'hidden'; }; const isFocusableWithTabKey = (element, checkVisibility = true) => { if (!isFocusable(element)) { return false; } const tabIndex = element.getAttribute('tabIndex'); const visible = !checkVisibility || isVisible(element); return visible && tabIndex !== '-1'; }; const findFocusableChild = (element, checkVisibility = true) => { return findElement(element, (node) => isFocusableWithTabKey(node, checkVisibility), false); }; const findFocusable = (element, checkVisibility = true) => { return findElement(element, (node) => isFocusableWithTabKey(node, checkVisibility)); }; const toClassList = (classNames) => String(classNames).trim().split(' '); const hasClasses = (element, classNames) => { const namesList = toClassList(classNames); return Boolean(toClassList(element.className).find((className) => namesList.indexOf(className) >= 0)); }; const matchesClasses = (classNames) => (element) => hasClasses(element, classNames); const NODE_NAME_PREDICATES = {}; const matchesNodeName = (nodeName) => { if (!NODE_NAME_PREDICATES[nodeName]) { NODE_NAME_PREDICATES[nodeName] = (element) => String(element.nodeName).toLowerCase() === nodeName.toLowerCase(); } return NODE_NAME_PREDICATES[nodeName]; }; /** * Normalizes a scroll position value in RTL mode. */ function rtlScrollPosition(position, element, initial) { let result = position; if (initial < 0) { result = -position; } else if (initial > 0) { result = element.scrollWidth - element.offsetWidth - position; } return result; } function closestBySelector(element, selector) { if (element.closest) { return element.closest(selector); } const matches = Element.prototype.matches ? (el, sel) => el.matches(sel) : (el, sel) => el.msMatchesSelector(sel); let node = element; while (node && !isDocumentNode(node)) { if (matches(node, selector)) { return node; } node = node.parentNode; } } const isDocumentNode = (container) => container.nodeType === 9; /** * @hidden */ class EventsOutsideAngularDirective { element; ngZone; renderer; events = {}; scope; subscriptions; constructor(element, ngZone, renderer) { this.element = element; this.ngZone = ngZone; this.renderer = renderer; } ngOnInit() { if (!this.element || !this.element.nativeElement) { return; } const events = this.events; this.subscriptions = []; this.ngZone.runOutsideAngular(() => { for (const name in events) { if (Object.hasOwnProperty.call(events, name)) { this.subscriptions?.push(this.renderer.listen(this.element.nativeElement, name, this.scope ? events[name].bind(this.scope) : events[name])); } } }); } ngOnDestroy() { if (this.subscriptions) { for (let idx = 0; idx < this.subscriptions.length; idx++) { this.subscriptions[idx](); } this.subscriptions = null; } } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: EventsOutsideAngularDirective, deps: [{ token: i0.ElementRef }, { token: i0.NgZone }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Directive }); static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: EventsOutsideAngularDirective, isStandalone: true, selector: "[kendoEventsOutsideAngular]", inputs: { events: ["kendoEventsOutsideAngular", "events"], scope: "scope" }, ngImport: i0 }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: EventsOutsideAngularDirective, decorators: [{ type: Directive, args: [{ selector: '[kendoEventsOutsideAngular]', standalone: true }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.NgZone }, { type: i0.Renderer2 }]; }, propDecorators: { events: [{ type: Input, args: ['kendoEventsOutsideAngular'] }], scope: [{ type: Input }] } }); class ResizeService { resizeBatchService; resize = new EventEmitter(); acceptedSize = false; lastWidth; lastHeight; state = 0 /* ServiceState.Initial */; parentElement; constructor(resizeBatchService) { this.resizeBatchService = resizeBatchService; } acceptSize(size = this.measure()) { this.lastWidth = size.width; this.lastHeight = size.height; this.acceptedSize = true; } checkChanges() { if (!isDocumentAvailable()) { return; } if (this.state === 0 /* ServiceState.Initial */) { this.state = 1 /* ServiceState.Initializing */; // batch initial measure this.resizeBatchService.schedule(this, this.init); } } destroy() { this.resizeBatchService.cancel(this); } checkSize() { if (!this.parentElement) { return false; } const { width, height } = this.measure(); const sameSize = width === this.lastWidth && height === this.lastHeight; if (sameSize) { return false; } this.lastWidth = width; this.lastHeight = height; this.acceptedSize = false; this.resize.emit({ width, height }); return true; } initSize() { const size = this.measure(); this.lastWidth = size.width; this.lastHeight = size.height; } measure() { let width = 0; let height = 0; if (this.parentElement) { height = this.parentElement.offsetHeight; width = this.parentElement.offsetWidth; } return { height, width }; } } // eslint-disable import/no-deprecated // TODO:NG13 CSP const div = (style) => { const el = document.createElement('div'); el.style.cssText = style; return el; }; const computedProp = (elem, prop) => getComputedStyle(elem, null).getPropertyValue(prop); const WRAP_STYLE = 'position: absolute; display: block; left: 0; top: 0; right: 0; bottom: 0; z-index: -1;' + 'overflow: hidden; visibility: hidden;'; const EXPAND_CHILD_STYLE = 'position: absolute; left: 0; top: 0; transition: 0s;'; const SHRINK_CHILD_STYLE = EXPAND_CHILD_STYLE + 'width: 200%; height: 200%;'; class ResizeCompatService extends ResizeService { element; ngZone; expand; expandChild; shrink; subscription; constructor(resizeBatchService, element, ngZone) { super(resizeBatchService); this.element = element; this.ngZone = ngZone; } checkChanges() { if (this.state === 2 /* ServiceState.Initialized */) { if (!this.resizeBatchService.isScheduled(this)) { this.resizeBatchService.schedule(this, this.checkSize); } return; } super.checkChanges(); } destroy() { super.destroy(); if (this.subscription) { this.subscription.unsubscribe(); } if (this.expand) { const element = this.element?.nativeElement; element.removeChild(this.expand); element.removeChild(this.shrink); this.expand.removeChild(this.expandChild); this.expand = this.expandChild = this.shrink = this.element = null; } } checkSize() { if (super.checkSize()) { this.reset(); return true; } return false; } init() { const parentElement = this.parentElement = this.element?.nativeElement.parentElement; if (computedProp(parentElement, 'position') === 'static') { parentElement.style.position = 'relative'; } this.state = 2 /* ServiceState.Initialized */; this.render(); this.reset(); this.initSize(); this.subscribe(); } render() { const element = this.element?.nativeElement; element.style.cssText = WRAP_STYLE; element.setAttribute('dir', 'ltr'); this.expand = div(WRAP_STYLE); this.expandChild = div(EXPAND_CHILD_STYLE); this.expand.appendChild(this.expandChild); element.appendChild(this.expand); this.shrink = div(WRAP_STYLE); const shrinkChild = div(SHRINK_CHILD_STYLE); this.shrink.appendChild(shrinkChild); element.appendChild(this.shrink); } reset() { const expandChild = this.expandChild; expandChild.style.width = 100000 + 'px'; expandChild.style.height = 100000 + 'px'; const expand = this.expand; expand.scrollLeft = 100000; expand.scrollTop = 100000; const shrink = this.shrink; shrink.scrollLeft = 100000; shrink.scrollTop = 100000; } subscribe() { this.ngZone.runOutsideAngular(() => { this.subscription = merge(fromEvent(this.shrink, 'scroll'), fromEvent(this.expand, 'scroll')) .subscribe(() => { this.checkSize(); }); }); } } const HAS_OBSERVER = typeof ResizeObserver !== 'undefined'; /** * @hidden */ class ResizeObserverService extends ResizeService { element; ngZone; resizeObserver; static supported() { return HAS_OBSERVER; } constructor(resizeBatchService, element, ngZone) { super(resizeBatchService); this.element = element; this.ngZone = ngZone; } destroy() { super.destroy(); if (this.resizeObserver) { this.resizeObserver.disconnect(); this.resizeObserver = null; } this.parentElement = null; } init() { this.parentElement = this.element.nativeElement.parentElement; this.initSize(); this.state = 2 /* ServiceState.Initialized */; this.ngZone.runOutsideAngular(() => { this.resizeObserver = new ResizeObserver(() => { this.checkSize(); }); this.resizeObserver.observe(this.parentElement); }); } } /** * @hidden */ class ResizeBatchService { ngZone; scheduled = []; resolvedPromise = Promise.resolve(null); subscription; constructor(ngZone) { this.ngZone = ngZone; this.flush = this.flush.bind(this); } schedule(instance, method) { this.scheduled.push({ instance, method }); if (!this.subscription) { this.ngZone.runOutsideAngular(() => { this.subscription = from(this.resolvedPromise) .subscribe(this.flush); }); } } isScheduled(instance) { return Boolean(this.scheduled.find(item => item.instance === instance)); } cancel(instance) { const scheduled = this.scheduled; const count = scheduled.length; for (let idx = 0; idx < count; idx++) { if (scheduled[idx].instance === instance) { scheduled.splice(idx, 1); if (!scheduled.length) { this.unsubscribe(); } return; } } } ngOnDestroy() { this.unsubscribe(); } unsubscribe() { if (this.subscription) { this.subscription.unsubscribe(); this.subscription = null; } } flush() { this.scheduled.forEach(item => { item.method.call(item.instance); }); this.scheduled = []; this.unsubscribe(); } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ResizeBatchService, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable }); static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ResizeBatchService, providedIn: 'root' }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ResizeBatchService, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }], ctorParameters: function () { return [{ type: i0.NgZone }]; } }); /** * Emit up to 10 resize events per second by default. * Chosen as a compromise between responsiveness and performance. */ const DEFAULT_RATE_LIMIT = 10; /** * Resize Sensor Component * * Triggers a "resize" event whenever the parent DOM element size changes. */ class ResizeSensorComponent { /** * The maximum number of resize events to emit per second. * * Defaults to 10. */ rateLimit = DEFAULT_RATE_LIMIT; /** * Fires when the parent DOM element has been resized. */ resize = new EventEmitter(); subscription; resizeService; constructor(resizeBatchService, element, ngZone) { const serviceType = ResizeObserverService.supported() ? ResizeObserverService : ResizeCompatService; this.resizeService = new serviceType(resizeBatchService, element, ngZone); const throttleTime = 1000 / (this.rateLimit || DEFAULT_RATE_LIMIT); this.subscription = this.resizeService.resize .pipe(auditTime(throttleTime)) .subscribe(({ width, height }) => { if (!this.resizeService.acceptedSize) { this.resize.emit({ width, height }); } }); } ngAfterViewChecked() { this.resizeService.checkChanges(); } ngOnDestroy() { this.subscription.unsubscribe(); this.resizeService.destroy(); } acceptSize(size) { this.resizeService.acceptSize(size); } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ResizeSensorComponent, deps: [{ token: ResizeBatchService }, { token: i0.ElementRef }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: ResizeSensorComponent, isStandalone: true, selector: "kendo-resize-sensor", inputs: { rateLimit: "rateLimit" }, outputs: { resize: "resize" }, ngImport: i0, template: '', isInline: true }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ResizeSensorComponent, decorators: [{ type: Component, args: [{ selector: 'kendo-resize-sensor', template: '', standalone: true }] }], ctorParameters: function () { return [{ type: ResizeBatchService }, { type: i0.ElementRef }, { type: i0.NgZone }]; }, propDecorators: { rateLimit: [{ type: Input }], resize: [{ type: Output }] } }); class KendoInput { } /** * Enum with key codes. */ var Keys; (function (Keys) { Keys["ArrowDown"] = "ArrowDown"; Keys["ArrowLeft"] = "ArrowLeft"; Keys["ArrowRight"] = "ArrowRight"; Keys["ArrowUp"] = "ArrowUp"; Keys["Backspace"] = "Backspace"; Keys["Delete"] = "Delete"; Keys["Digit0"] = "Digit0"; Keys["Digit1"] = "Digit1"; Keys["Digit2"] = "Digit2"; Keys["Digit3"] = "Digit3"; Keys["Digit4"] = "Digit4"; Keys["Digit5"] = "Digit5"; Keys["Digit6"] = "Digit6"; Keys["Digit7"] = "Digit7"; Keys["Digit8"] = "Digit8"; Keys["Digit9"] = "Digit9"; Keys["End"] = "End"; Keys["Enter"] = "Enter"; Keys["Escape"] = "Escape"; Keys["F1"] = "F1"; Keys["F2"] = "F2"; Keys["F10"] = "F10"; Keys["Home"] = "Home"; Keys["KeyA"] = "KeyA"; Keys["KeyB"] = "KeyB"; Keys["KeyC"] = "KeyC"; Keys["KeyD"] = "KeyD"; Keys["KeyE"] = "KeyE"; Keys["KeyF"] = "KeyF"; Keys["KeyG"] = "KeyG"; Keys["KeyH"] = "KeyH"; Keys["KeyI"] = "KeyI"; Keys["KeyJ"] = "KeyJ"; Keys["KeyK"] = "KeyK"; Keys["KeyL"] = "KeyL"; Keys["KeyM"] = "KeyM"; Keys["KeyN"] = "KeyN"; Keys["KeyO"] = "KeyO"; Keys["KeyP"] = "KeyP"; Keys["KeyQ"] = "KeyQ"; Keys["KeyR"] = "KeyR"; Keys["KeyS"] = "KeyS"; Keys["KeyT"] = "KeyT"; Keys["KeyU"] = "KeyU"; Keys["KeyV"] = "KeyV"; Keys["KeyW"] = "KeyW"; Keys["KeyX"] = "KeyX"; Keys["KeyY"] = "KeyY"; Keys["KeyZ"] = "KeyZ"; Keys["Numpad1"] = "Numpad1"; Keys["Numpad2"] = "Numpad2"; Keys["Numpad3"] = "Numpad3"; Keys["Numpad4"] = "Numpad4"; Keys["Numpad5"] = "Numpad5"; Keys["Numpad6"] = "Numpad6"; Keys["Numpad7"] = "Numpad7"; Keys["Numpad8"] = "Numpad8"; Keys["Numpad9"] = "Numpad9"; Keys["Numpad0"] = "Numpad0"; Keys["NumpadEnter"] = "NumpadEnter"; Keys["NumpadDecimal"] = "NumpadDecimal"; Keys["PageDown"] = "PageDown"; Keys["PageUp"] = "PageUp"; Keys["Space"] = "Space"; Keys["Tab"] = "Tab"; })(Keys || (Keys = {})); /** * @hidden */ const focusableSelector = [ 'a[href]:not([tabindex^="-"]):not([disabled])', 'area[href]:not([tabindex^="-"]):not([disabled])', 'input:not([tabindex^="-"]):not([disabled])', 'select:not([tabindex^="-"]):not([disabled])', 'textarea:not([tabindex^="-"]):not([disabled])', 'button:not([tabindex^="-"]):not([disabled])', 'iframe:not([tabindex^="-"]):not([disabled])', 'object:not([tabindex^="-"]):not([disabled])', 'embed:not([tabindex^="-"]):not([disabled])', '*[tabindex]:not([tabindex^="-"]):not([disabled])', '*[contenteditable]:not([tabindex^="-"]):not([disabled]):not([contenteditable="false"])' ].join(','); /** * @hidden * * On some keyboards, PageUp/Down, Home/End, and arrow keys are mapped to Numpad keys */ const normalizeNumpadKeys = (event) => { if (event.code === Keys.Numpad1 && event.key === Keys.End) { return Keys.End; } if (event.code === Keys.Numpad2 && event.key === Keys.ArrowDown) { return Keys.ArrowDown; } if (event.code === Keys.Numpad3 && event.key === Keys.PageDown) { return Keys.PageDown; } if (event.code === Keys.Numpad4 && event.key === Keys.ArrowLeft) { return Keys.ArrowLeft; } if (event.code === Keys.Numpad6 && event.key === Keys.ArrowRight) { return Keys.ArrowRight; } if (event.code === Keys.Numpad7 && event.key === Keys.Home) { return Keys.Home; } if (event.code === Keys.Numpad8 && event.key === Keys.ArrowUp) { return Keys.ArrowUp; } if (event.code === Keys.Numpad9 && event.key === Keys.PageUp) { return Keys.PageUp; } if (event.code === Keys.NumpadEnter) { return Keys.Enter; } return event.code; }; const FIELD_REGEX$1 = /\[(?:(\d+)|['"](.*?)['"])\]|((?:(?!\[.*?\]|\.).)+)/g; const getterCache = {}; getterCache['undefined'] = () => undefined; /** * @hidden */ function getter(field) { if (getterCache[field]) { return getterCache[field]; } const fields = []; field.replace(FIELD_REGEX$1, function (_match, index, indexAccessor, fieldName) { fields.push(index !== undefined ? index : (indexAccessor || fieldName)); }); getterCache[field] = function (obj) { let result = obj; for (let idx = 0; idx < fields.length && result; idx++) { result = result[fields[idx]]; } return result; }; return getterCache[field]; } const FIELD_REGEX = /\[(?:(\d+)|['"](.*?)['"])\]|((?:(?!\[.*?\]|\.).)+)/g; const setterCache = {}; setterCache['undefined'] = (obj) => obj; /** * @hidden */ function setter(field) { if (setterCache[field]) { return setterCache[field]; } const fields = []; field.replace(FIELD_REGEX, function (_match, index, indexAccessor, fieldName) { fields.push(index !== undefined ? index : (indexAccessor || fieldName)); }); setterCache[field] = function (obj, value) { let root = obj; const depth = fields.length - 1; for (let idx = 0; idx < depth && root; idx++) { root = root[fields[idx]] = root[fields[idx]] || {}; } root[fields[depth]] = value; }; return setterCache[field]; } /** * @hidden */ const watermarkStyles = ` position: absolute; width: 100%; height: 100%; top: 0; left: 0; right: 0; bottom: 0; opacity: 0.2; zIndex: 101; pointerEvents: none; backgroundImage: url(''); `; /** * @hidden */ const bannerStyles = { display: 'flex', justifyContent: 'center', alignItems: 'center', position: 'fixed', top: '16px', right: '16px', padding: '12px', borderRadius: '4px', boxShadow: '0px 4px 5px 0px rgba(0, 0, 0, 0.04), 0px 2px 4px 0px rgba(0, 0, 0, 0.03)', fontSize: '14px', fontWeight: 400, lineHeight: '20px', backgroundColor: '#FFC000', color: '#1E1E1E', zIndex: 20000 }; /** * @hidden */ const buttonStyles = { display: 'inline-flex', position: 'relative', border: 'none', borderRadius: '4px', padding: '5px', backgroundColor: 'transparent', transition: 'color 0.2s ease-in-out', outline: 'none', cursor: 'pointer' }; /** * @hidden */ const licenseKeyUrl = 'https://www.telerik.com/kendo-angular-ui/components/my-license/?utm_medium=product&utm_source=kendoangular&utm_campaign=kendo-ui-angular-purchase-license-keys-banner'; let bannerPresentOnPage = false; /** * @hidden */ class WatermarkOverlayComponent { watermarkStyle = watermarkStyles; licenseMessage; banner; isOpen = true; bannerMounted = false; bannerStyles = bannerStyles; buttonStyles = buttonStyles; licenseKeyUrl = licenseKeyUrl; ngOnInit() { if (!bannerPresentOnPage) { this.bannerMounted = true; bannerPresentOnPage = true; } } ngAfterViewInit() { if (this.isBannerRendered) { document.body.appendChild(this.banner.nativeElement); } } ngOnDestroy() { if (this.isBannerRendered) { document.body.removeChild(this.banner.nativeElement); } } closeBanner() { this.isOpen = false; } get isBannerRendered() { return isDocumentAvailable() && this.banner && this.banner.nativeElement; } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: WatermarkOverlayComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: WatermarkOverlayComponent, isStandalone: true, selector: "div[kendoWatermarkOverlay]", inputs: { licenseMessage: "licenseMessage" }, host: { properties: { "style": "this.watermarkStyle" } }, viewQueries: [{ propertyName: "banner", first: true, predicate: ["banner"], descendants: true }], ngImport: i0, template: ` <div #banner *ngIf="isOpen && bannerMounted" [ngStyle]="bannerStyles"> <span [ngStyle]="{ display: 'flex', alignSelf: 'center', marginRight: '8px' }"> <svg width="16" height="16" viewBox="0 0 16 16" fill="none"> <path fill-rule="evenodd" clip-rule="evenodd" d="M8 1L0 15H16L8 1ZM7 6V11H9V6H7ZM7 14V12H9V14H7Z" fill="#1E1E1E"/> </svg> </span> <span *ngIf="licenseMessage" [innerHtml]="licenseMessage"></span> <span *ngIf="!licenseMessage"> We couldn't verify your <a [href]="licenseKeyUrl">license key</a> for Kendo UI for Angular. Please see the browser console for details and resolution steps. </span> <div [ngStyle]="{ display: 'flex', alignItems: 'center', marginLeft: '24px' }"> <button title='Close' [ngStyle]="buttonStyles" (click)="closeBanner()"> <svg width="16" height="16" viewBox="0 0 16 16" fill="none"> <path d="M13 4.41562L9.41563 8L13 11.5844L11.5844 13L8 9.41563L4.41562 13L3 11.5844L6.58437 8L3 4.41562L4.41562 3L8 6.58437L11.5844 3L13 4.41562Z" fill="#1E1E1E"/> </svg> </button> </div> </div> `, isInline: true, dependencies: [{ kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: WatermarkOverlayComponent, decorators: [{ type: Component, args: [{ selector: 'div[kendoWatermarkOverlay]', template: ` <div #banner *ngIf="isOpen && bannerMounted" [ngStyle]="bannerStyles"> <span [ngStyle]="{ display: 'flex', alignSelf: 'center', marginRight: '8px' }"> <svg width="16" height="16" viewBox="0 0 16 16" fill="none"> <path fill-rule="evenodd" clip-rule="evenodd" d="M8 1L0 15H16L8 1ZM7 6V11H9V6H7ZM7 14V12H9V14H7Z" fill="#1E1E1E"/> </svg> </span> <span *ngIf="licenseMessage" [innerHtml]="licenseMessage"></span> <span *ngIf="!licenseMessage"> We couldn't verify your <a [href]="licenseKeyUrl">license key</a> for Kendo UI for Angular. Please see the browser console for details and resolution steps. </span> <div [ngStyle]="{ display: 'flex', alignItems: 'center', marginLeft: '24px' }"> <button title='Close' [ngStyle]="buttonStyles" (click)="closeBanner()"> <svg width="16" height="16" viewBox="0 0 16 16" fill="none"> <path d="M13 4.41562L9.41563 8L13 11.5844L11.5844 13L8 9.41563L4.41562 13L3 11.5844L6.58437 8L3 4.41562L4.41562 3L8 6.58437L11.5844 3L13 4.41562Z" fill="#1E1E1E"/> </svg> </button> </div> </div> `, standalone: true, imports: [NgIf, NgStyle] }] }], propDecorators: { watermarkStyle: [{ type: HostBinding, args: ['style'] }], licenseMessage: [{ type: Input }], banner: [{ type: ViewChild, args: ['banner'] }] } }); const allowed = ['telerik.com', 'progress.com', 'stackblitz.io', 'csb.app']; /** * @hidden */ function shouldShowValidationUI(isPackageValid) { const skip = allowed.some((hostname) => globalThis.document?.location.hostname.endsWith(hostname)); return !skip && !isPackageValid; } /** * @hidden * * Returns the notification message to display, if any. */ function getLicenseMessage(meta) { const message = getLicenseStatus(meta).message; return message?.notificationMessage; } /** * Specifies the adornments in the prefix container of the [Inputs](slug:adornments_textbox#toc-prefix-adornments) and [DropDowns](slug:adornments_multiselect#toc-prefix-adornments). * ```html * <kendo-textbox> * <ng-template kendoPrefixTemplate> * <button kendoButton look="clear" icon="image"></button> * </ng-template> * </kendo-textbox> * * <kendo-multiselect [data]="data" [(ngModel)]="value"> * <ng-template kendoPrefixTemplate> * <button kendoButton look="clear" icon="image"></button> * </ng-template> * </kendo-multiselect> * ``` */ class PrefixTemplateDirective { templateRef; /** * Sets the `showSeparator` attribute of the `prefixTemplate`. * * @default false */ set showSeparator(value) { this._showSeparator = value; } get showSeparator() { return this._showSeparator; } _showSeparator = false; constructor(templateRef) { this.templateRef = templateRef; } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: PrefixTemplateDirective, deps: [{ token: i0.TemplateRef, optional: true }], target: i0.ɵɵFactoryTarget.Directive }); static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: PrefixTemplateDirective, isStandalone: true, selector: "[kendoPrefixTemplate]", inputs: { showSeparator: "showSeparator" }, ngImport: i0 }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: PrefixTemplateDirective, decorators: [{ type: Directive, args: [{ selector: '[kendoPrefixTemplate]', standalone: true }] }], ctorParameters: function () { return [{ type: i0.TemplateRef, decorators: [{ type: Optional }] }]; }, propDecorators: { showSeparator: [{ type: Input }] } }); /** * Represents the directive for suffix adornments in the Inputs and DropDowns components. * * Use the `kendoSuffixTemplate` directive to add custom content to the suffix container. * * * See [Inputs Suffix Adornments](slug:adornments_textbox#toc-suffix-adornments) * * See [DropDowns Suffix Adornments](slug:adornments_multiselect#toc-suffix-adornments) * * @example * ```html * <kendo-textbox> * <ng-template kendoSuffixTemplate> * <button kendoButton look="clear" icon="image"></button> * </ng-template> * </kendo-textbox> * <kendo-multiselect [data]="data" [(ngModel)]="value"> * <ng-template kendoSuffixTemplate> * <button kendoButton look="clear" icon="image"></button> * </ng-template> * </kendo-multiselect> * ``` */ class SuffixTemplateDirective { templateRef; /** * Sets the `showSeparator` attribute of the `suffixTemplate`. * * @default false */ set showSeparator(value) { this._showSeparator = value; } get showSeparator() { return this._showSeparator; } _showSeparator = false; constructor(templateRef) { this.templateRef = templateRef; } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: SuffixTemplateDirective, deps: [{ token: i0.TemplateRef, optional: true }], target: i0.ɵɵFactoryTarget.Directive }); static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: SuffixTemplateDirective, isStandalone: true, selector: "[kendoSuffixTemplate]", inputs: { showSeparator: "showSeparator" }, ngImport: i0 }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: SuffixTemplateDirective, decorators: [{ type: Directive, args: [{ selector: '[kendoSuffixTemplate]', standalone: true }] }], ctorParameters: function () { return [{ type: i0.TemplateRef, decorators: [{ type: Optional }] }]; }, propDecorators: { showSeparator: [{ type: Input }] } }); /** * Specifies a separator in the content of the [Inputs]({% slug adornments_textbox %}#toc-separator) and [DropDowns]({% slug adornments_multiselect %}#toc-separator). * @example * ```ts-no-run * _@Component({ * selector: 'my-app', * template: ` * <kendo-textbox> * <ng-template kendoSuffixTemplate> * <button kendoB