import XEUtils from 'xe-utils' const reClsMap: { [key: string]: any } = {} export const browse = XEUtils.browse() export function getPropClass (property: any, params: any) { return property ? XEUtils.isFunction(property) ? property(params) : property : '' } function getClsRE (cls: any) { if (!reClsMap[cls]) { reClsMap[cls] = new RegExp(`(?:^|\\s)${cls}(?!\\S)`, 'g') } return reClsMap[cls] } function getNodeOffset (elem: any, container: any, rest: any): any { if (elem) { const parentElem = elem.parentNode rest.top += elem.offsetTop rest.left += elem.offsetLeft if (parentElem && parentElem !== document.documentElement && parentElem !== document.body) { rest.top -= parentElem.scrollTop rest.left -= parentElem.scrollLeft } if (container && (elem === container || elem.offsetParent === container) ? 0 : elem.offsetParent) { return getNodeOffset(elem.offsetParent, container, rest) } } return rest } export function isPx (val: any) { return val && /^\d+(px)?$/.test(val) } export function isScale (val: any) { return val && /^\d+%$/.test(val) } export function hasClass (elem: any, cls: any) { return elem && elem.className && elem.className.match && elem.className.match(getClsRE(cls)) } export function removeClass (elem: any, cls: any) { if (elem && hasClass(elem, cls)) { elem.className = elem.className.replace(getClsRE(cls), '') } } export function addClass (elem: any, cls: string) { if (elem && !hasClass(elem, cls)) { removeClass(elem, cls) elem.className = `${elem.className} ${cls}` } } export function getDomNode () { const documentElement = document.documentElement const bodyElem = document.body return { scrollTop: documentElement.scrollTop || bodyElem.scrollTop, scrollLeft: documentElement.scrollLeft || bodyElem.scrollLeft, visibleHeight: documentElement.clientHeight || bodyElem.clientHeight, visibleWidth: documentElement.clientWidth || bodyElem.clientWidth } } export function getOffsetHeight (elem: HTMLElement) { return elem ? elem.offsetHeight : 0 } export function getPaddingTopBottomSize (elem: HTMLElement) { if (elem) { const computedStyle = getComputedStyle(elem) const paddingTop = XEUtils.toNumber(computedStyle.paddingTop) const paddingBottom = XEUtils.toNumber(computedStyle.paddingBottom) return paddingTop + paddingBottom } return 0 } export function setScrollTop (elem: HTMLElement | null, scrollTop: number) { if (elem) { elem.scrollTop = scrollTop } } export function setScrollLeft (elem: HTMLElement | null, scrollLeft: number) { if (elem) { elem.scrollLeft = scrollLeft } } // export function setScrollLeftAndTop (elem: HTMLElement | null, scrollLeft: number, scrollTop: number) { // if (elem) { // elem.scrollLeft = scrollLeft // elem.scrollTop = scrollTop // } // } export function updateCellTitle (overflowElem: any, column: any) { const content = column.type === 'html' ? overflowElem.innerText : overflowElem.textContent if (overflowElem.getAttribute('title') !== content) { overflowElem.setAttribute('title', content) } } /** * 检查触发源是否属于目标节点 */ export function getEventTargetNode (evnt: any, container: any, queryCls?: string, queryMethod?: (target: Element) => boolean) { let targetElem let target = (evnt.target.shadowRoot && evnt.composed) ? (evnt.composedPath()[0] || evnt.target) : evnt.target while (target && target.nodeType && target !== document) { if (queryCls && hasClass(target, queryCls) && (!queryMethod || queryMethod(target))) { targetElem = target } else if (target === container) { return { flag: queryCls ? !!targetElem : true, container, targetElem } } target = target.parentNode } return { flag: false } } /** * 获取元素相对于 document 的位置 */ export function getOffsetPos (elem: any, container: any) { return getNodeOffset(elem, container, { left: 0, top: 0 }) } export function getAbsolutePos (elem: any) { // 当主页面嵌套在iframe时,elem.getBoundingClientRect()计算在当前body内的边界距离,document.body.getBoundingClientRect计算body所在的边界距离 const bodyBounding = document.body.getBoundingClientRect() const bounding = elem.getBoundingClientRect() const boundingBottom = bounding.bottom const boundingTop = bounding.top - bodyBounding.top const boundingLeft = bounding.left - bodyBounding.left const { scrollTop, scrollLeft, visibleHeight, visibleWidth } = getDomNode() return { boundingTop, boundingBottom, top: scrollTop + boundingTop, boundingLeft, left: scrollLeft + boundingLeft, visibleHeight, visibleWidth } } const scrollIntoViewIfNeeded = 'scrollIntoViewIfNeeded' const scrollIntoView = 'scrollIntoView' export function scrollToView (elem: any) { if (elem) { if (elem[scrollIntoViewIfNeeded]) { elem[scrollIntoViewIfNeeded]() } else if (elem[scrollIntoView]) { elem[scrollIntoView]() } } } export function triggerEvent (targetElem: Element, type: string) { if (targetElem) { targetElem.dispatchEvent(new Event(type)) } } export function isNodeElement (elem: any): elem is HTMLElement { return elem && elem.nodeType === 1 } export function isInside (childNode: Element, parentNode: Element): boolean { if (childNode.parentElement) { if (childNode.parentElement === parentNode) { return true } else { return isInside(childNode.parentElement, parentNode) } } else { return false } } /** * 递归使得元素和所有后代元素失去焦点(如果可以的话) */ export const blurRecursive = (elem: HTMLElement) => { if (elem) { elem?.blur() const children = elem.children for (let i = 0; i < children.length; i++) { blurRecursive(children[i] as HTMLElement) } } } export const ignoreWheelList:string[] = [ 'vxe-select', 'vxe-cascader', 'vxe-slider' ]