import { nextTick } from 'vue'; import PropTypes from '../_util/vue-types'; import { alignElement, alignPoint } from 'dom-align'; import addEventListener from '../vc-util/Dom/addEventListener'; import { isWindow, buffer, isSamePoint, isSimilarValue, restoreFocus } from './util'; import { cloneElement } from '../_util/vnode.js'; import clonedeep from 'lodash-es/cloneDeep'; import { getSlot, findDOMNode } from '../_util/props-util'; function getElement(func) { if (typeof func !== 'function' || !func) return null; return func(); } function getPoint(point) { if (typeof point !== 'object' || !point) return null; return point; } export default { props: { childrenProps: PropTypes.object, align: PropTypes.object.isRequired, target: PropTypes.oneOfType([PropTypes.func, PropTypes.object]).def(() => window), monitorBufferTime: PropTypes.number.def(50), monitorWindowResize: PropTypes.bool.def(false), disabled: PropTypes.bool.def(false), }, data() { this.aligned = false; this.prevProps = { ...this.$props }; return {}; }, mounted() { nextTick(() => { const props = this.$props; // if parent ref not attached .... use document.getElementById !this.aligned && this.forceAlign(); if (!props.disabled && props.monitorWindowResize) { this.startMonitorWindowResize(); } }); }, updated() { nextTick(() => { const prevProps = this.prevProps; const props = this.$props; let reAlign = false; if (!props.disabled) { const source = findDOMNode(this); const sourceRect = source ? source.getBoundingClientRect() : null; if (prevProps.disabled) { reAlign = true; } else { const lastElement = getElement(prevProps.target); const currentElement = getElement(props.target); const lastPoint = getPoint(prevProps.target); const currentPoint = getPoint(props.target); if (isWindow(lastElement) && isWindow(currentElement)) { // Skip if is window reAlign = false; } else if ( lastElement !== currentElement || // Element change (lastElement && !currentElement && currentPoint) || // Change from element to point (lastPoint && currentPoint && currentElement) || // Change from point to element (currentPoint && !isSamePoint(lastPoint, currentPoint)) ) { reAlign = true; } // If source element size changed const preRect = this.sourceRect || {}; if ( !reAlign && source && (!isSimilarValue(preRect.width, sourceRect.width) || !isSimilarValue(preRect.height, sourceRect.height)) ) { reAlign = true; } } this.sourceRect = sourceRect; } if (reAlign) { this.forceAlign(); } if (props.monitorWindowResize && !props.disabled) { this.startMonitorWindowResize(); } else { this.stopMonitorWindowResize(); } this.prevProps = { ...this.$props, align: clonedeep(this.$props.align) }; }); }, beforeUnmount() { this.stopMonitorWindowResize(); }, methods: { startMonitorWindowResize() { if (!this.resizeHandler) { this.bufferMonitor = buffer(this.forceAlign, this.$props.monitorBufferTime); this.resizeHandler = addEventListener(window, 'resize', this.bufferMonitor); } }, stopMonitorWindowResize() { if (this.resizeHandler) { this.bufferMonitor.clear(); this.resizeHandler.remove(); this.resizeHandler = null; } }, forceAlign() { const { disabled, target, align } = this.$props; if (!disabled && target) { const source = findDOMNode(this); let result; const element = getElement(target); const point = getPoint(target); // IE lose focus after element realign // We should record activeElement and restore later const activeElement = document.activeElement; if (element) { result = alignElement(source, element, align); } else if (point) { result = alignPoint(source, point, align); } restoreFocus(activeElement, source); this.aligned = true; this.$attrs.onAlign && this.$attrs.onAlign(source, result); } }, }, render() { const { childrenProps } = this.$props; const child = getSlot(this); if (child && childrenProps) { return cloneElement(child[0], childrenProps); } return child && child[0]; }, };