@progress/kendo-angular-common
Version:
Kendo UI for Angular - Utility Package
1,255 lines (1,207 loc) • 68.3 kB
JavaScript
/**-----------------------------------------------------------------------------------------
* 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