Element 中的 DropDown 定位解析

一、前言

进来事情较少。

今天早上在群里面有看到,对于 iview select 的下拉框定位问题的讨论。

因为主要用的是 Element-UI,就对这一块做了深入的了解。

二、发现问题

如上面说的,在得到这个问题后,想到应该不只是 select 下拉所有的 dropdowm 的定位应该是一样的。

就直接在 Element-UI 官网看下,选择了级联看了下:

在这里看到 dropdown 是直接挂载在 body 下的。

简单分析后得出:

1、dropdown 是通过 absolute 进行绝对定位的;

2、第一次点击前,没有挂载到 dom,点击后才挂载,后面是通过 display: none 控制显示

比较好奇的就是这个绝对定位的 left、top 是怎么计算出来的?

查了一圈,并且看了 Element-UI 源码也没有发现什么(这里是自己了解不够到位)。

在遇到这篇文章后,才有了下面的进一步深入:Element 源码解析系列7-select

三、底层原理

在看了这一篇文章后,才对 popper.js 有深入了解。

在 Element-ui 中对其做了封装在:src\utils\vue-popper.js 这里。这个的作用主要就是计算绝对定位的位置、实时改变

下面的代码都是在源码中的 src\utils\popper.js 文件中

1、初始化

在 Popper 进行初始化的时候,会先对要弹出的 element 进行样式初始化,设置 postion 是 absolute 还是 fixed。

会进行第一次 update、_setupEventListeners

2、计算位置

这个主要是从 update 里面找到的:

    Popper.prototype.update = function() {
        var data = { instance: this, styles: {} };

        // store placement inside the data object, modifiers will be able to edit `placement` if needed
        // and refer to _originalPlacement to know the original value
        data.placement = this._options.placement;
        data._originalPlacement = this._options.placement;

        // compute the popper and reference offsets and put them inside data.offsets
        data.offsets = this._getOffsets(this._popper, this._reference, data.placement);

        // get boundaries
        data.boundaries = this._getBoundaries(data, this._options.boundariesPadding, this._options.boundariesElement);

        data = this.runModifiers(data, this._options.modifiers);

        if (typeof this.state.updateCallback === 'function') {
            this.state.updateCallback(data);
        }
    };

其中 _getOffsets 就是计算对应的偏移的

3、实时更新位置

_setupEventListeners 的作用是设置监听,主要有 resize 和 scroll 两个监听事件:

    Popper.prototype._setupEventListeners = function() {
        // NOTE: 1 DOM access here
        this.state.updateBound = this.update.bind(this);
        root.addEventListener('resize', this.state.updateBound);
        // if the boundariesElement is window we don't need to listen for the scroll event
        if (this._options.boundariesElement !== 'window') {
            var target = getScrollParent(this._reference);
            // here it could be both `body` or `documentElement` thanks to Firefox, we then check both
            if (target === root.document.body || target === root.document.documentElement) {
                target = root;
            }
            target.addEventListener('scroll', this.state.updateBound);
            this.state.scrollTarget = target;
        }
    };

就是当屏幕变化或者滚动时,会再次计算 Popper 的定位位置信息,并更新。

4、底层解析

开始感觉这个定位计算会不会很难什么的,在从 update 的 _getOffsets 步步找下去发现计算的代码:

    function getBoundingClientRect(element) {
        var rect = element.getBoundingClientRect();

        // whether the IE version is lower than 11
        var isIE = navigator.userAgent.indexOf("MSIE") != -1;

        // fix ie document bounding top always 0 bug
        var rectTop = isIE && element.tagName === 'HTML'
            ? -element.scrollTop
            : rect.top;

        return {
            left: rect.left,
            top: rectTop,
            right: rect.right,
            bottom: rect.bottom,
            width: rect.right - rect.left,
            height: rect.bottom - rectTop
        };
    }

其中 element.getBoundingClientRect() 是获取一个元素的大小以及相对视口的位置。

看了这个后真的感觉,所有看似很复杂、牛掰的操作都是源自于最基础最底层的。

在这里用到了 DOM API ,Element.getBoundingClientRect(),这里点击看到 MDN 上面的文档。

posted @ 2020-03-25 11:30  漠里  阅读(7762)  评论(0编辑  收藏  举报