UNPKG

devextreme

Version:

HTML5 JavaScript Component Suite for Responsive Web Development

1,160 lines (1,159 loc) • 43.4 kB
/** * DevExtreme (esm/__internal/ui/overlay/m_overlay.js) * Version: 24.2.6 * Build date: Mon Mar 17 2025 * * Copyright (c) 2012 - 2025 Developer Express Inc. ALL RIGHTS RESERVED * Read about DevExtreme licensing here: https://js.devexpress.com/Licensing/ */ import _extends from "@babel/runtime/helpers/esm/extends"; import { fx } from "../../../common/core/animation"; import { hideCallback as hideTopOverlayCallback } from "../../../common/core/environment/hide_callback"; import eventsEngine from "../../../common/core/events/core/events_engine"; import { move as dragEventMove } from "../../../common/core/events/drag"; import pointerEvents from "../../../common/core/events/pointer"; import { keyboard } from "../../../common/core/events/short"; import { addNamespace, isCommandKeyPressed, normalizeKeyName } from "../../../common/core/events/utils/index"; import { triggerHidingEvent, triggerResizeEvent, triggerShownEvent } from "../../../common/core/events/visibility_change"; import registerComponent from "../../../core/component_registrator"; import devices from "../../../core/devices"; import domAdapter from "../../../core/dom_adapter"; import { getPublicElement } from "../../../core/element"; import errors from "../../../core/errors"; import $ from "../../../core/renderer"; import { EmptyTemplate } from "../../../core/templates/empty_template"; import browser from "../../../core/utils/browser"; import { noop } from "../../../core/utils/common"; import { Deferred } from "../../../core/utils/deferred"; import { extend } from "../../../core/utils/extend"; import { each } from "../../../core/utils/iterator"; import readyCallbacks from "../../../core/utils/ready_callbacks"; import { getOuterHeight, getOuterWidth } from "../../../core/utils/size"; import { isFunction, isObject, isPromise, isWindow } from "../../../core/utils/type"; import { changeCallback } from "../../../core/utils/view_port"; import { tabbable } from "../../../ui/widget/selectors"; import uiErrors from "../../../ui/widget/ui.errors"; import domUtils from "../../core/utils/m_dom"; import Widget from "../../core/widget/widget"; import windowUtils from "../../core/utils/m_window"; import { OVERLAY_POSITION_ALIASES, OverlayPositionController } from "./m_overlay_position_controller"; import * as zIndexPool from "./m_z_index"; const ready = readyCallbacks.add; const window = windowUtils.getWindow(); const viewPortChanged = changeCallback; const OVERLAY_CLASS = "dx-overlay"; const OVERLAY_WRAPPER_CLASS = "dx-overlay-wrapper"; const OVERLAY_CONTENT_CLASS = "dx-overlay-content"; const OVERLAY_SHADER_CLASS = "dx-overlay-shader"; const INNER_OVERLAY_CLASS = "dx-inner-overlay"; const INVISIBLE_STATE_CLASS = "dx-state-invisible"; const ANONYMOUS_TEMPLATE_NAME = "content"; const RTL_DIRECTION_CLASS = "dx-rtl"; const OVERLAY_STACK = []; const PREVENT_SAFARI_SCROLLING_CLASS = "dx-prevent-safari-scrolling"; const TAB_KEY = "tab"; ready((() => { eventsEngine.subscribeGlobal(domAdapter.getDocument(), pointerEvents.down, (e => { for (let i = OVERLAY_STACK.length - 1; i >= 0; i--) { if (!OVERLAY_STACK[i]._proxiedDocumentDownHandler(e)) { return } } })) })); class Overlay extends Widget { _supportedKeys() { return _extends({}, super._supportedKeys(), { escape() { this.hide() } }) } _getDefaultOptions() { return _extends({}, super._getDefaultOptions(), { activeStateEnabled: false, visible: false, deferRendering: true, shading: true, shadingColor: "", wrapperAttr: {}, position: extend({}, OVERLAY_POSITION_ALIASES.center), width: "80vw", minWidth: null, maxWidth: null, height: "80vh", minHeight: null, maxHeight: null, animation: { show: { type: "pop", duration: 300, from: { scale: .55 } }, hide: { type: "pop", duration: 300, from: { opacity: 1, scale: 1 }, to: { opacity: 0, scale: .55 } } }, closeOnOutsideClick: false, hideOnOutsideClick: false, _ignorePreventScrollEventsDeprecation: false, onShowing: null, onShown: null, onHiding: null, onHidden: null, contentTemplate: "content", innerOverlay: false, restorePosition: true, hideTopOverlayHandler: () => { this.hide() }, hideOnParentScroll: false, preventScrollEvents: true, onPositioned: null, propagateOutsideClick: false, ignoreChildEvents: true, _checkParentVisibility: true, _fixWrapperPosition: false, _loopFocus: false }) } _defaultOptionsRules() { return super._defaultOptionsRules().concat([{ device: () => !windowUtils.hasWindow(), options: { width: null, height: null, animation: null, _checkParentVisibility: false } }]) } _setOptionsByReference() { super._setOptionsByReference(); extend(this._optionsByReference, { animation: true }) } $wrapper() { return this._$wrapper } _eventBindingTarget() { return this._$content } _setDeprecatedOptions() { super._setDeprecatedOptions(); extend(this._deprecatedOptions, { closeOnOutsideClick: { since: "22.1", alias: "hideOnOutsideClick" } }) } ctor(element, options) { super.ctor(element, options); if (options) { if ("preventScrollEvents" in options && !options._ignorePreventScrollEventsDeprecation) { this._logDeprecatedPreventScrollEventsInfo() } } } _logDeprecatedPreventScrollEventsInfo() { this._logDeprecatedOptionWarning("preventScrollEvents", { since: "23.1", message: "If you enable this option, end-users may experience scrolling issues." }) } _init() { super._init(); this._initActions(); this._initHideOnOutsideClickHandler(); this._initTabTerminatorHandler(); this._customWrapperClass = null; this._$wrapper = $("<div>").addClass("dx-overlay-wrapper"); this._$content = $("<div>").addClass("dx-overlay-content"); this._initInnerOverlayClass(); const $element = this.$element(); $element.addClass("dx-overlay"); this._$wrapper.attr("data-bind", "dxControlsDescendantBindings: true"); this._toggleViewPortSubscription(true); const { hideTopOverlayHandler: hideTopOverlayHandler } = this.option(); this._initHideTopOverlayHandler(hideTopOverlayHandler); this._parentsScrollSubscriptionInfo = { handler: e => { this._hideOnParentsScrollHandler(e) } }; this.warnPositionAsFunction() } warnPositionAsFunction() { if (isFunction(this.option("position"))) { errors.log("W0018") } } _initInnerOverlayClass() { const { innerOverlay: innerOverlay } = this.option(); this._$content.toggleClass("dx-inner-overlay", innerOverlay) } _initHideTopOverlayHandler(handler) { this._hideTopOverlayHandler = handler } _getActionsList() { return ["onShowing", "onShown", "onHiding", "onHidden", "onPositioned", "onVisualPositionChanged"] } _initActions() { this._actions = {}; const actions = this._getActionsList(); each(actions, ((_, action) => { this._actions[action] = this._createActionByOption(action, { excludeValidators: ["disabled", "readOnly"] }) || noop })) } _initHideOnOutsideClickHandler() { var _this = this; this._proxiedDocumentDownHandler = function() { return _this._documentDownHandler(...arguments) } } _initMarkup() { super._initMarkup(); this._renderWrapperAttributes(); this._initPositionController() } _documentDownHandler(e) { if (this._showAnimationProcessing) { this._stopAnimation() } const isAttachedTarget = $(window.document).is(e.target) || domUtils.contains(window.document, e.target); const isInnerOverlay = $(e.target).closest(".dx-inner-overlay").length; const outsideClick = isAttachedTarget && !isInnerOverlay && !(this._$content.is(e.target) || domUtils.contains(this._$content.get(0), e.target)); if (outsideClick && this._shouldHideOnOutsideClick(e)) { this._outsideClickHandler(e) } const { propagateOutsideClick: propagateOutsideClick } = this.option(); return propagateOutsideClick } _shouldHideOnOutsideClick(e) { const { hideOnOutsideClick: hideOnOutsideClick } = this.option(); if (isFunction(hideOnOutsideClick)) { return hideOnOutsideClick(e) } return hideOnOutsideClick } _outsideClickHandler(e) { if (this.option("shading")) { e.preventDefault() } this.hide() } _getAnonymousTemplateName() { return "content" } _initTemplates() { this._templateManager.addDefaultTemplates({ content: new EmptyTemplate }); super._initTemplates() } _isTopOverlay() { const overlayStack = this._overlayStack(); for (let i = overlayStack.length - 1; i >= 0; i--) { const tabbableElements = overlayStack[i]._findTabbableBounds(); if (tabbableElements.first || tabbableElements.last) { return overlayStack[i] === this } } return false } _overlayStack() { return OVERLAY_STACK } _zIndexInitValue() { return Overlay.baseZIndex() } _toggleViewPortSubscription(toggle) { var _this2 = this; viewPortChanged.remove(this._viewPortChangeHandle); if (toggle) { this._viewPortChangeHandle = function() { _this2._viewPortChangeHandler(...arguments) }; viewPortChanged.add(this._viewPortChangeHandle) } } _viewPortChangeHandler() { this._positionController.updateContainer(this.option("container")); this._refresh() } _renderWrapperAttributes() { const { wrapperAttr: wrapperAttr } = this.option(); const attributes = extend({}, wrapperAttr); const classNames = attributes.class; delete attributes.class; this.$wrapper().attr(attributes).removeClass(this._customWrapperClass).addClass(classNames); this._customWrapperClass = classNames } _renderVisibilityAnimate(visible) { this._stopAnimation(); return visible ? this._show() : this._hide() } _getAnimationConfig() { return this._getOptionValue("animation", this) } _toggleBodyScroll(enabled) {} _animateShowing() { var _this3 = this; const animation = this._getAnimationConfig() ?? {}; const showAnimation = this._normalizeAnimation(animation.show, "to"); const startShowAnimation = (null === showAnimation || void 0 === showAnimation ? void 0 : showAnimation.start) ?? noop; const completeShowAnimation = (null === showAnimation || void 0 === showAnimation ? void 0 : showAnimation.complete) ?? noop; this._animate(showAnimation, (function() { if (_this3._isAnimationPaused) { return } if (_this3.option("focusStateEnabled")) { eventsEngine.trigger(_this3._focusTarget(), "focus") } for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key] } completeShowAnimation.call(_this3, ...args); _this3._showAnimationProcessing = false; _this3._isHidden = false; _this3._actions.onShown(); _this3._toggleSafariScrolling(); _this3._showingDeferred.resolve() }), (function() { if (_this3._isAnimationPaused) { return } for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) { args[_key2] = arguments[_key2] } startShowAnimation.call(_this3, ...args); _this3._showAnimationProcessing = true })) } _processShowingHidingCancel(cancelArg, applyFunction, cancelFunction) { if (isPromise(cancelArg)) { cancelArg.then((shouldCancel => { if (shouldCancel) { cancelFunction() } else { applyFunction() } })).catch((() => applyFunction())) } else { cancelArg ? cancelFunction() : applyFunction() } } _show() { this._showingDeferred = Deferred(); this._parentHidden = this._isParentHidden(); this._showingDeferred.done((() => { delete this._parentHidden })); if (this._parentHidden) { this._isHidden = true; return this._showingDeferred.resolve() } if (this._currentVisible) { return Deferred().resolve().promise() } this._currentVisible = true; if (this._isHidingActionCanceled) { delete this._isHidingActionCanceled; this._showingDeferred.reject() } else { const show = () => { this._stopAnimation(); const { enableBodyScroll: enableBodyScroll } = this.option(); this._toggleBodyScroll(enableBodyScroll); this._toggleVisibility(true); this._$content.css("visibility", "hidden"); this._$content.toggleClass("dx-state-invisible", false); this._updateZIndexStackPosition(true); this._positionController.openingHandled(); this._renderContent(); const showingArgs = { cancel: false }; this._actions.onShowing(showingArgs); this._processShowingHidingCancel(showingArgs.cancel, (() => { this._$content.css("visibility", ""); this._renderVisibility(true); this._animateShowing() }), (() => { this._toggleVisibility(false); this._$content.css("visibility", ""); this._$content.toggleClass("dx-state-invisible", true); this._isShowingActionCanceled = true; this._moveFromContainer(); this._toggleBodyScroll(true); this.option("visible", false); this._showingDeferred.resolve() })) }; if (this.option("templatesRenderAsynchronously")) { this._stopShowTimer(); this._asyncShowTimeout = setTimeout(show) } else { show() } } return this._showingDeferred.promise() } _normalizeAnimation(showHideConfig, direction) { if (showHideConfig) { showHideConfig = extend({ type: "slide", skipElementInitialStyles: true }, showHideConfig); if (isObject(showHideConfig[direction])) { extend(showHideConfig[direction], { position: this._positionController.position }) } } return showHideConfig } _animateHiding() { var _this4 = this; const animation = this._getAnimationConfig() ?? {}; const hideAnimation = this._normalizeAnimation(animation.hide, "from"); const startHideAnimation = (null === hideAnimation || void 0 === hideAnimation ? void 0 : hideAnimation.start) ?? noop; const completeHideAnimation = (null === hideAnimation || void 0 === hideAnimation ? void 0 : hideAnimation.complete) ?? noop; this._animate(hideAnimation, (function() { var _this4$_actions; _this4._$content.css("pointerEvents", ""); _this4._renderVisibility(false); for (var _len3 = arguments.length, args = new Array(_len3), _key3 = 0; _key3 < _len3; _key3++) { args[_key3] = arguments[_key3] } completeHideAnimation.call(_this4, ...args); _this4._hideAnimationProcessing = false; null === (_this4$_actions = _this4._actions) || void 0 === _this4$_actions || _this4$_actions.onHidden(); _this4._hidingDeferred.resolve() }), (function() { _this4._$content.css("pointerEvents", "none"); for (var _len4 = arguments.length, args = new Array(_len4), _key4 = 0; _key4 < _len4; _key4++) { args[_key4] = arguments[_key4] } startHideAnimation.call(_this4, ...args); _this4._hideAnimationProcessing = true })) } _hide() { if (!this._currentVisible) { return Deferred().resolve().promise() } this._currentVisible = false; this._hidingDeferred = Deferred(); const hidingArgs = { cancel: false }; if (this._isShowingActionCanceled) { delete this._isShowingActionCanceled; this._hidingDeferred.reject() } else { this._actions.onHiding(hidingArgs); this._toggleSafariScrolling(); this._toggleBodyScroll(true); const cancelHide = () => { this._isHidingActionCanceled = true; const { enableBodyScroll: enableBodyScroll } = this.option(); this._toggleBodyScroll(enableBodyScroll); this.option("visible", true); this._hidingDeferred.resolve() }; const applyHide = () => { this._forceFocusLost(); this._toggleShading(false); this._toggleSubscriptions(false); this._stopShowTimer(); this._animateHiding() }; this._processShowingHidingCancel(hidingArgs.cancel, applyHide, cancelHide) } return this._hidingDeferred.promise() } _forceFocusLost() { const activeElement = domAdapter.getActiveElement(); const shouldResetActiveElement = !!this._$content.find(activeElement).length; if (shouldResetActiveElement) { domUtils.resetActiveElement() } } _animate(animation, completeCallback, startCallback) { if (animation) { startCallback = startCallback || animation.start || noop; fx.animate(this._$content, extend({}, animation, { start: startCallback, complete: completeCallback })) } else { completeCallback() } } _stopAnimation() { fx.stop(this._$content, true) } _renderVisibility(visible) { if (visible && this._isParentHidden()) { return } this._currentVisible = visible; this._stopAnimation(); if (!visible) { triggerHidingEvent(this._$content) } if (visible) { this._checkContainerExists(); this._moveToContainer(); this._renderGeometry(); triggerShownEvent(this._$content); triggerResizeEvent(this._$content) } else { this._toggleVisibility(visible); this._$content.toggleClass("dx-state-invisible", !visible); this._updateZIndexStackPosition(visible); this._moveFromContainer() } this._toggleShading(visible); this._toggleSubscriptions(visible) } _updateZIndexStackPosition(pushToStack) { const overlayStack = this._overlayStack(); const index = overlayStack.indexOf(this); if (pushToStack) { if (-1 === index) { this._zIndex = zIndexPool.create(this._zIndexInitValue()); overlayStack.push(this) } this._$wrapper.css("zIndex", this._zIndex); this._$content.css("zIndex", this._zIndex) } else if (-1 !== index) { overlayStack.splice(index, 1); zIndexPool.remove(this._zIndex) } } _toggleShading(visible) { const { shading: shading, shadingColor: shadingColor } = this.option(); this._$wrapper.toggleClass("dx-overlay-shader", visible && shading); this._$wrapper.css("backgroundColor", shading ? shadingColor : ""); this._toggleTabTerminator(visible && shading) } _initTabTerminatorHandler() { var _this5 = this; this._proxiedTabTerminatorHandler = function() { _this5._tabKeyHandler(...arguments) } } _toggleTabTerminator(enabled) { const { _loopFocus: _loopFocus } = this.option(); const eventName = addNamespace("keydown", this.NAME); if (_loopFocus || enabled) { eventsEngine.on(domAdapter.getDocument(), eventName, this._proxiedTabTerminatorHandler) } else { eventsEngine.off(domAdapter.getDocument(), eventName, this._proxiedTabTerminatorHandler) } } _findTabbableBounds() { const $elements = this._$wrapper.find("*"); const elementsCount = $elements.length - 1; const result = { first: null, last: null }; for (let i = 0; i <= elementsCount; i++) { if (!result.first && $elements.eq(i).is(tabbable)) { result.first = $elements.eq(i) } if (!result.last && $elements.eq(elementsCount - i).is(tabbable)) { result.last = $elements.eq(elementsCount - i) } if (result.first && result.last) { break } } return result } _tabKeyHandler(e) { if ("tab" !== normalizeKeyName(e) || !this._isTopOverlay()) { return } const tabbableElements = this._findTabbableBounds(); const $firstTabbable = tabbableElements.first; const $lastTabbable = tabbableElements.last; const isTabOnLast = !e.shiftKey && e.target === $lastTabbable.get(0); const isShiftTabOnFirst = e.shiftKey && e.target === $firstTabbable.get(0); const isEmptyTabList = 0 === tabbableElements.length; const isOutsideTarget = !domUtils.contains(this._$wrapper.get(0), e.target); if (isTabOnLast || isShiftTabOnFirst || isEmptyTabList || isOutsideTarget) { e.preventDefault(); const $focusElement = e.shiftKey ? $lastTabbable : $firstTabbable; eventsEngine.trigger($focusElement, "focusin"); eventsEngine.trigger($focusElement, "focus") } } _toggleSubscriptions(enabled) { if (windowUtils.hasWindow()) { this._toggleHideTopOverlayCallback(enabled); this._toggleHideOnParentsScrollSubscription(enabled) } } _toggleHideTopOverlayCallback(subscribe) { if (!this._hideTopOverlayHandler) { return } if (subscribe) { hideTopOverlayCallback.add(this._hideTopOverlayHandler) } else { hideTopOverlayCallback.remove(this._hideTopOverlayHandler) } } _toggleHideOnParentsScrollSubscription(needSubscribe) { const scrollEvent = addNamespace("scroll", this.NAME); const { prevTargets: prevTargets, handler: handler } = this._parentsScrollSubscriptionInfo ?? {}; eventsEngine.off(prevTargets, scrollEvent, handler); const hideOnScroll = this.option("hideOnParentScroll"); if (needSubscribe && hideOnScroll) { let $parents = this._getHideOnParentScrollTarget().parents(); if ("desktop" === devices.real().deviceType) { $parents = $parents.add(window) } eventsEngine.on($parents, scrollEvent, handler); this._parentsScrollSubscriptionInfo.prevTargets = $parents } } _hideOnParentsScrollHandler(e) { let hideHandled = false; const hideOnScroll = this.option("hideOnParentScroll"); if (isFunction(hideOnScroll)) { hideHandled = hideOnScroll(e) } if (!hideHandled && !this._showAnimationProcessing) { this.hide() } } _getHideOnParentScrollTarget() { const { _hideOnParentScrollTarget: _hideOnParentScrollTarget } = this.option(); const $hideOnParentScrollTarget = $(_hideOnParentScrollTarget); if ($hideOnParentScrollTarget.length) { return $hideOnParentScrollTarget } return this._$wrapper } _render() { super._render(); this._appendContentToElement(); this._renderVisibilityAnimate(this.option("visible")) } _appendContentToElement() { if (!this._$content.parent().is(this.$element())) { this._$content.appendTo(this.$element()) } } _renderContent() { const shouldDeferRendering = !this._currentVisible && this.option("deferRendering"); const isParentHidden = this.option("visible") && this._isParentHidden(); if (isParentHidden) { this._isHidden = true; return } if (this._contentAlreadyRendered || shouldDeferRendering) { return } this._contentAlreadyRendered = true; this._appendContentToElement(); super._renderContent() } _isParentHidden() { if (!this.option("_checkParentVisibility")) { return false } if (void 0 !== this._parentHidden) { return this._parentHidden } const $parent = this.$element().parent(); if ($parent.is(":visible")) { return false } let isHidden = false; $parent.add($parent.parents()).each(((index, element) => { const $element = $(element); if ("none" === $element.css("display")) { isHidden = true; return false } })); return isHidden || !domAdapter.getBody().contains($parent.get(0)) } _renderContentImpl() { const whenContentRendered = Deferred(); const contentTemplateOption = this.option("contentTemplate"); const contentTemplate = this._getTemplate(contentTemplateOption); const transclude = this._templateManager.anonymousTemplateName === contentTemplateOption; null === contentTemplate || void 0 === contentTemplate || contentTemplate.render({ container: getPublicElement(this.$content()), noModel: true, transclude: transclude, onRendered: () => { whenContentRendered.resolve(); if (this.option("templatesRenderAsynchronously")) { this._dimensionChanged() } } }); const { preventScrollEvents: preventScrollEvents } = this.option(); this._toggleWrapperScrollEventsSubscription(preventScrollEvents); whenContentRendered.done((() => { if (this.option("visible")) { this._moveToContainer() } })); return whenContentRendered.promise() } _getPositionControllerConfig() { const { container: container, visualContainer: visualContainer, _fixWrapperPosition: _fixWrapperPosition, restorePosition: restorePosition, _skipContentPositioning: _skipContentPositioning } = this.option(); return { container: container, visualContainer: visualContainer, $root: this.$element(), $content: this._$content, $wrapper: this._$wrapper, onPositioned: this._actions.onPositioned, onVisualPositionChanged: this._actions.onVisualPositionChanged, restorePosition: restorePosition, _fixWrapperPosition: _fixWrapperPosition, _skipContentPositioning: _skipContentPositioning } } _initPositionController() { this._positionController = new OverlayPositionController(this._getPositionControllerConfig()) } _toggleWrapperScrollEventsSubscription(enabled) { const eventName = addNamespace(dragEventMove, this.NAME); eventsEngine.off(this._$wrapper, eventName); if (enabled) { eventsEngine.on(this._$wrapper, eventName, { validate: () => true, getDirection: () => "both", _toggleGestureCover(toggle) { if (!toggle) { this._toggleGestureCoverImpl(toggle) } }, _clearSelection: noop, isNative: true }, (e => { const { originalEvent: originalEvent } = e.originalEvent; const { type: type } = originalEvent || {}; const isWheel = "wheel" === type; const isMouseMove = "mousemove" === type; const isScrollByWheel = isWheel && !isCommandKeyPressed(e); e._cancelPreventDefault = true; if (originalEvent && false !== e.cancelable && (!isMouseMove && !isWheel || isScrollByWheel)) { e.preventDefault() } })) } } _moveFromContainer() { this._$content.appendTo(this.$element()); this._$wrapper.detach() } _checkContainerExists() { const $wrapperContainer = this._positionController.$container; if (void 0 === $wrapperContainer) { return } const containerExists = $wrapperContainer.length > 0; if (!containerExists) { uiErrors.log("W1021", this.NAME) } } _moveToContainer() { const $wrapperContainer = this._positionController.$container; this._$wrapper.appendTo($wrapperContainer); this._$content.appendTo(this._$wrapper) } _renderGeometry(options) { const { visible: visible } = this.option(); if (visible && windowUtils.hasWindow()) { this._stopAnimation(); this._renderGeometryImpl() } } _renderGeometryImpl() { this._positionController.updatePosition(this._getOptionValue("position")); this._renderWrapper(); this._renderDimensions(); this._renderPosition() } _renderPosition(state) { this._positionController.positionContent() } _isAllWindowCovered() { const { shading: shading } = this.option(); return isWindow(this._positionController.$visualContainer.get(0)) && shading } _toggleSafariScrolling() { const visible = this.option("visible"); const $body = $(domAdapter.getBody()); const isIosSafari = "ios" === devices.real().platform && browser.safari; const isAllWindowCovered = this._isAllWindowCovered(); const isScrollingPrevented = $body.hasClass("dx-prevent-safari-scrolling"); const shouldPreventScrolling = !isScrollingPrevented && visible && isAllWindowCovered; const shouldEnableScrolling = isScrollingPrevented && (!visible || !isAllWindowCovered || this._disposed); if (isIosSafari) { if (shouldEnableScrolling) { $body.removeClass("dx-prevent-safari-scrolling"); window.scrollTo(0, this._cachedBodyScrollTop); this._cachedBodyScrollTop = void 0 } else if (shouldPreventScrolling) { this._cachedBodyScrollTop = window.pageYOffset; $body.addClass("dx-prevent-safari-scrolling") } } } _renderWrapper() { this._positionController.styleWrapperPosition(); this._renderWrapperDimensions(); this._positionController.positionWrapper() } _renderWrapperDimensions() { const { $visualContainer: $visualContainer } = this._positionController; const documentElement = domAdapter.getDocumentElement(); const isVisualContainerWindow = isWindow($visualContainer.get(0)); const wrapperWidth = isVisualContainerWindow ? documentElement.clientWidth : getOuterWidth($visualContainer); const wrapperHeight = isVisualContainerWindow ? window.innerHeight : getOuterHeight($visualContainer); this._$wrapper.css({ width: wrapperWidth, height: wrapperHeight }) } _renderDimensions() { const content = this._$content.get(0); this._$content.css({ minWidth: this._getOptionValue("minWidth", content), maxWidth: this._getOptionValue("maxWidth", content), minHeight: this._getOptionValue("minHeight", content), maxHeight: this._getOptionValue("maxHeight", content), width: this._getOptionValue("width", content), height: this._getOptionValue("height", content) }) } _focusTarget() { return this._$content } _attachKeyboardEvents() { this._keyboardListenerId = keyboard.on(this._$content, null, (opts => this._keyboardHandler(opts))) } _keyboardHandler(options) { const e = options.originalEvent; const $target = $(e.target); if ($target.is(this._$content) || !this.option("ignoreChildEvents")) { super._keyboardHandler(...arguments) } } _isVisible() { const { visible: visible } = this.option(); return visible } _visibilityChanged(visible) { if (visible) { if (this.option("visible")) { this._renderVisibilityAnimate(visible) } } else { this._renderVisibilityAnimate(visible) } } _dimensionChanged() { this._renderGeometry() } _clean() { const { isRenovated: isRenovated } = this.option(); if (!this._contentAlreadyRendered && !isRenovated) { this.$content().empty() } this._renderVisibility(false); this._stopShowTimer(); this._cleanFocusState() } _stopShowTimer() { if (this._asyncShowTimeout) { clearTimeout(this._asyncShowTimeout) } this._asyncShowTimeout = null } _dispose() { fx.stop(this._$content, false); this._toggleViewPortSubscription(false); this._toggleSubscriptions(false); this._updateZIndexStackPosition(false); this._toggleTabTerminator(false); this._actions = null; this._parentsScrollSubscriptionInfo = null; super._dispose(); this._toggleSafariScrolling(); this.option("visible") && zIndexPool.remove(this._zIndex); this._$wrapper.remove(); this._$content.remove() } _toggleRTLDirection(rtl) { this._$content.toggleClass("dx-rtl", rtl) } _optionChanged(args) { const { value: value, name: name } = args; if (this._getActionsList().includes(name)) { this._initActions(); return } switch (name) { case "animation": case "closeOnOutsideClick": case "hideOnOutsideClick": case "propagateOutsideClick": break; case "_loopFocus": case "shading": { const { visible: visible } = this.option(); this._toggleShading(visible); this._toggleSafariScrolling(); break } case "shadingColor": { const { visible: visible } = this.option(); this._toggleShading(visible); break } case "width": case "height": case "minWidth": case "maxWidth": case "minHeight": case "maxHeight": this._renderGeometry(); break; case "position": this._positionController.updatePosition(this.option("position")); this._positionController.restorePositionOnNextRender(true); this._renderGeometry(); this._toggleSafariScrolling(); break; case "visible": this._renderVisibilityAnimate(value).done((() => { var _this$_animateDeferre; return null === (_this$_animateDeferre = this._animateDeferred) || void 0 === _this$_animateDeferre ? void 0 : _this$_animateDeferre.resolveWith(this) })).fail((() => { var _this$_animateDeferre2; return null === (_this$_animateDeferre2 = this._animateDeferred) || void 0 === _this$_animateDeferre2 ? void 0 : _this$_animateDeferre2.reject() })); break; case "container": this._positionController.updateContainer(value); this._invalidate(); this._toggleSafariScrolling(); break; case "visualContainer": this._positionController.updateVisualContainer(value); this._renderWrapper(); this._toggleSafariScrolling(); break; case "innerOverlay": this._initInnerOverlayClass(); break; case "deferRendering": case "contentTemplate": this._contentAlreadyRendered = false; this._clean(); this._invalidate(); break; case "hideTopOverlayHandler": this._toggleHideTopOverlayCallback(false); this._initHideTopOverlayHandler(value); this._toggleHideTopOverlayCallback(this.option("visible")); break; case "hideOnParentScroll": case "_hideOnParentScrollTarget": { const { visible: visible } = this.option(); this._toggleHideOnParentsScrollSubscription(visible); break } case "rtlEnabled": this._contentAlreadyRendered = false; super._optionChanged(args); break; case "_fixWrapperPosition": this._positionController.fixWrapperPosition = value; break; case "wrapperAttr": this._renderWrapperAttributes(); break; case "restorePosition": this._positionController.restorePosition = value; break; case "preventScrollEvents": this._logDeprecatedPreventScrollEventsInfo(); this._toggleWrapperScrollEventsSubscription(value); break; default: super._optionChanged(args) } } toggle(showing) { showing = void 0 === showing ? !this.option("visible") : showing; const result = Deferred(); if (showing === this.option("visible")) { return result.resolveWith(this, [showing]).promise() } const animateDeferred = Deferred(); this._animateDeferred = animateDeferred; this.option("visible", showing); animateDeferred.promise().done((() => { delete this._animateDeferred; result.resolveWith(this, [this.option("visible")]) })).fail((() => { delete this._animateDeferred; result.reject() })); return result.promise() } $content() { return this._$content } show() { return this.toggle(true) } hide() { return this.toggle(false) } content() { return getPublicElement(this._$content) } repaint() { if (this._contentAlreadyRendered) { this._positionController.restorePositionOnNextRender(true); this._renderGeometry({ forceStopAnimation: true }); triggerResizeEvent(this._$content) } else { super.repaint() } } } Overlay.baseZIndex = zIndex => zIndexPool.base(zIndex); registerComponent("dxOverlay", Overlay); export default Overlay;