三大家族

1.offset家族

offset这个单词本身是--偏移,补偿,位移的意思。

js中有一套方便的获取元素尺寸的办法就是offset家族,offsetWidth、offsetHight 以及offsetLeft、offsetTop、offsetParent共同组成了offset家族。

1. offsetWidth和offsetHeight(检测自身盒子宽高+padding+border)

  • offsetWidth = width+padding+border;
  • offsetHeight = Height+padding+border;

2. offsetLeft和offsetTop (检测距离父盒子有定位的左/上面的距离)

  • 返回距离上级盒子(带有定位)左边的位置
  • 如果父级都没有定位则以body为准
  • offsetLeft 从父亲的padding 开始算,父亲的border 不算。
  • 在父盒子有定位的情况下,offsetLeft == style.left(去掉px)

3. offsetParent (检测父系盒子中带有定位的父盒子节点)

  • 返回该对象的父级 (带有定位),如果当前元素的父级元素没有进行CSS定位 (position为absolute或 relative,fixed), offsetParent为body。
  • 如果当前元素的父级元素中有CSS定位(position为absolute或relative,fixed),offsetParent取最近的那个父级元素。

4. offsetLeft和style.left的区别:

  • 最大的区别在于offsetLeft可以返回没有定位盒子的距离左侧的位置,而style.left不行
  • offsetLeft返回的是number型,而style.left返回的是string型,且带有单位px
  • offsetLeft只读,而style.left可读写
  • 还有只有写在行内样式的left,才可以被style.left拿到 ,否则返回的就是空字符串。

5.offsetTop和style.top同上

2. Scroll家族组成

1. scrollWidth和scrollHeight( 不包括border 和 margin)

  • scrollWidth = 盒子本身width+padding
  • 检测盒子的宽高。(调用者:节点元素。属性。)
  • 盒子内容的宽高。(如果有内容超出了,显示内容的宽高度,不超过则为盒子本身的宽高度)
  • IE567可以比盒子小。 IE8+火狐谷歌不能比盒子小

2. scrollTop和scrollLeft

  • 网页,被浏览器遮挡的头部和左边部分。
  • 被卷去的头部和左边部分。
  • 兼容问题:
    一、未声明 DTD(谷歌只认识他、IE9+)、document.body.scrollTop
    二、已经声明DTD(IE678只认识他、IE9+任何时候)、document.documentElement.scrollTop
    三、火狐/谷歌/ie9+以上支持的、window.pageYOffset
说简单点:就是谷歌只识别 document.body, 而IE只识别document.documentElement
火狐才有DTD的声明问题。

兼容写法:
      var top = window.pageYOffset
            || document.documentElement.scrollTop
            || document.body.scrollTop;
或    var top = document.body.scrollTop + document.documentElement.scrollTop;
判断页面是否已声明DTD
document.compatMode === 'BackCompat'
BackCompat    -- 未声明
CSS1Compat    --- 声明


    // scroll方法的封装
    // 需求: 封装一个兼容的scroll(),返回值是json,用scroll().top获取scrollTop的属性
    function scroll() {
        // 如果存在,返回0~无穷大
        // 不存在返回undefined
        // 兼容谷歌、火狐以及IE9+
        if (window.pageYOffset !== undefined) {
            return {
                "top": window.pageYOffset,
                "left": window.pageXOffset
            };
        } else if (document.compatMode === 'CSS1Compat') {
            // IE9以下的  还需要判断是否已声明DTD,CSS1Compat标识已声明
            return {
                "top": document.documentElement.scrollTop,
                "left": document.documentElement.scrollLeft
            };
        } else {
            return {
                "top": document.body.scrollTop,
                "left": document.body.scrollLeft
            }
        }


        // 实际使用的
//       return {
 //           "top": window.pageYOffset || document.body.scrollTop || document.documentElement.scrollTop,
 //           "left": window.pageYOffset || document.body.scrollLeft || document.documentElement.scrollLeft
//        }

    }

3. 获取title、body、head、html标签

  • document.title --- 文档标题;
  • document.head --- 文档的头标签
  • document.body --- 文档的body标签;
  • document.documentElement --- 这个很重要,它表示文档的html标签, 也就是说,基本结构当中的html标签并不是通过document.html 去访问的,而是document.documentElement

3. client家族

1. clientWidth和clientHeight

  • 调用者不同,意义不同:
    • 盒子调用: 指盒子本身。
    • body/html调用: 可视区域大小。
  • clientX: 鼠标距离可视区域左侧距离(event调用)
  • clientY: 鼠标距离可视区域上侧距离(event调用)
  • clientTop / clientLeft : 盒子的border宽高
兼容写法:
var w=window.innerWidth
|| document.documentElement.clientWidth
|| document.body.clientWidth;

var h=window.innerHeight
|| document.documentElement.clientHeight
|| document.body.clientHeight;

2.三大家族的区别:

1. width和height:
clientWidth  = width  + padding
clientHeight  = height + padding
offsetWidth  = width  + padding + border
offsetHeight  = height + padding + border
scrollWidth   = 内容宽度(不包含border)
scrollHeight  = 内容高度(不包含border)
2. top和left:
offsetTop/offsetLeft :
        调用者:任意元素。(盒子为主)
        作用:距离父系盒子中带有定位的距离。
scrollTop/scrollLeft:(盒子也可以调用,必须有滚动条)
        调用者:document.body.scrollTop/.....(window)
        作用:浏览器无法显示的部分(被卷去的部分)。
clientY/clientX:(clientTop/clientLeft 值的是border)
        调用者:event.clientX(event)
        作用:鼠标距离浏览器可视区域的距离(左、上)。

3.缓动动画原理

leader = leader + (target - leader) /10;
盒子位置 = 盒子本身位置 + (目标位置 - 盒子本身位置)/10
动画原理: 目标位置 = 盒子位置 + 步长
//   需注意的一点是: JS实际运算时会四舍五入取整,然后计算
//   获取盒子距离左侧具有定位的父盒子的距离(没有的body),四舍五入取整。
//   style.left获取的是具体值。


  
/***
 *缓动动画封装
 * @param ele
 * @param target
 * @param type [left || top]
 */
function animate(ele, target, type) {
    // 开始动画 首先要清除定时器
    clearInterval(ele.timer);

     // 开始定时器
    ele.timer = setInterval(function () {

        // 开闭原则
        var CLOSE_INTERVAL = false; // 是否关闭定时器

        //遍历属性和值,分别单独处理json
        //attr == k(键)    target == json[k](值)
        for (var k in json) {
            // 动画原理: 盒子位置 = 盒子位置 + ((目标位置 - 盒子自身位置)/10)
            // 1.获取步长   (目标位置 - 盒子自身位置)/10
            var leader = parseInt(getStyle(ele, k)) || 0;
            var step = (json[k] - leader) / 10;
            // 2. 二次加工, 小于0向下取整  或  大于0向上取整
            step = step > 0 ? Math.ceil(step) : Math.floor(step);
            // 3.移动  盒子位置 = 盒子位置 + 步长
            leader = leader + step;
            ele.style[k] = leader + "px";
            // 4.清除定时器的条件  当步长  大于 (目标位置 - 盒子自身位置)
            // 不考虑小数的情况下,只要目标位置和当前位置不相等,就不能清除清除定时器。
            if (Math.abs(json[k]  - leader) > Math.abs(step)) {
                CLOSE_INTERVAL = true;
            }
        }

        //只有所有的属性都到了指定位置,CLOSE_INTERVAL值才不会变成true;
        if (!CLOSE_INTERVAL) {
            // 将盒子位置直接赋为 目标位置
            // ele.style[type] = target + "px";
            clearInterval(ele.timer);
            callback && callback();
        }
    }, 25);
}

// 获取元素样式的兼容写法
function getStyle(ele, attr) {
    if (window.getComputedStyle) {
        return window.getComputedStyle(ele, null)[attr];
    }
    return ele.currentStyle[attr];
}
元素位置及事件
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容