Skip to content

Latest commit

 

History

History
276 lines (201 loc) · 12.1 KB

Polyfill源码阅读.md

File metadata and controls

276 lines (201 loc) · 12.1 KB
layout title
default
{"site.name" => nil}

Polyfill源码阅读

在勐喆的推荐下,开始看一些比较浅层次的源码,来加深对于规范的理解。看polyfill顺便可以看一下js的兼容,还可以了解一些不常用的API

CSSOM

getBoundingClientRect这个方法在ie8下的修正

if ('TextRectangle' in this && !('width' in TextRectangle.prototype)) {
    Object.defineProperties(TextRectangle.prototype, {
      'width': { get: function() { return this.right - this.left; } },
      'height': { get: function() { return this.bottom - this.top; } }
    });
  }

getBoundingClientRect这个方法得到这个元素的size并且相对于视图的定位,返回的几个值都是只读

这里对getBoundingClientRect方法在ie8下面没有width和height进行了休整。就是对ie8 window下的TextRectangle进行了defineProperties。将width和height进行了定义(这里的定义之所以使用defineProperties是因为这里定义了访问器属性,这个属性不能直接定义)。注意在新的浏览器里面,TextRectangle已经改名了。

注意这里使用的defineProperties在ie 8以下并不支持,所以这个polyfill已经对这个方法进行了修整。就是循环调用了ie8支持的defineProperty而已,至于ie7以下不支持的defineProperty,会在下面进行分析。

Object.prototype.hasOwnProperty.call(properties, name)

这里通过hasOwnProperty这个判断来只遍历properties里面的属性,而不是原型对象上的属性;使用call来调用是为了防止对象上自己申明了这个方法,这样子可以使用原型上的。

var s = {};
s === Object(s) //true

这里通过Object方法来判断传入的是不是一个object~这个写法有点意思。

Dom

querySelectorAll

document.querySelectorAll = function(selectors) {
      var style = document.createElement('style'), elements = [], element;
      document.documentElement.firstChild.appendChild(style);
      document._qsa = [];
      style.styleSheet.cssText = selectors + '{x-qsa:expression(document._qsa && document._qsa.push(this))}';
      window.scrollBy(0, 0);
      style.parentNode.removeChild(style);
      while (document._qsa.length) {
        element = document._qsa.shift();
        element.style.removeAttribute('x-qsa');
        elements.push(element);
      }
      document._qsa = null;
      return elements;
    };

看的好开心,感觉实现很有意思,通过支持率较好的css选择器来做这件事情

就是新建一个style的元素,并且使用传入的字段作为选择器,然后自定义了一个x-qsa的属性。使用到了expression这个东西来调用js,就是应用到的元素自身推入一个准备好的数组中。

然后遍历数组,得到最后的结果了,大赞!!

写法虽然不错,但是css的expression ie8以后不再支持了,比较尴尬,但是用这个来兼容ie7及以下还是很有意思的。

querySelector

就是调用querySelectorAll,然后返回第一个值就行了

getElementsByClassName

类似的,调用上面的querySelectorAll,只要在前面替换个.就行了

Node 类型

因为ie没有公布它的Node Type。所以这里添加了Node.ELEMENT_NODE这些。

DOMException

与上面类似的,定义了DOMException的一些种类

Event

重写了ie8以下的一些事件处理,包括通过attachevent来模拟addEventListener,通过detachEvent来模拟removeEventListener,这些ie9才支持

[Window, HTMLDocument, Element].forEach(function(o) {
      o.prototype.addEventListener = addEventListener;
      o.prototype.removeEventListener = removeEventListener;
    });

重写了这几个方法之后对window和element应用。

DOMTokenList

这个东西就是一个接口,一个格式,元素的classList,rellist就是遵从的这个规范,就是以空格分开的一些字符串而已。这里就是为IE9以下重写了这个东西,提供了一个构造函数,并且使用defineProperty来提供那些toggle,add等等的方法。

es5

getPrototypeOf

这个方法IE8以下不支持。

实现就是在判断了obj !== Object(obj)也就是返回 obj.__proto__ || obj.constructor.prototype || Object.prototype,__proto__这个属性是一些浏览器自己的实现(google,safari,ff)的,不推荐使用。

注意这个方法在ES6下的shim比较复杂,因为在ES6下,一个字符串会返回String.prototype,而es5则会报错,不过这个方法支持的很好。参见ES6支持

getOwnPropertyNames

IE8以下不支持。

这个就是返回所有属于他自己的属性,包括可枚举的和不可枚举的。实现就是在for in中运行一次Object.prototype.hasOwnProperty就可以了,当然这样不可枚举的就不行了。

这个polyfill有一些问题,不能cover下面的,参见mdn文档规范

var arr = ['a', 'b', 'c'];
console.log(Object.getOwnPropertyNames(arr).sort()); // logs '0,1,2,length'

因为他的写法是通过for in循环的,而length在array中是不可枚举的,作者表示除了原生的getOwnPropertyNames,其他没法拿到不可枚举的。

Object.create

Object.create = function (prototype, properties) {
    if (typeof prototype !== "object") { throw TypeError(); }
    function Ctor() {}
    Ctor.prototype = prototype;
    var o = new Ctor();
    if (prototype) { o.constructor = Ctor; }
    if (properties !== undefined) {
      if (properties !== Object(properties)) { throw TypeError(); }
      Object.defineProperties(o, properties);
    }
    return o;
  };

第一个参数是原型,第二个参数是property,才开始还以为这里比mdn文档网站上的polyfill写的好。因为他敢用defineProperties,结果后来才发现他也就支持了get,set和value。

mdn上的直接循环赋值了第二个参数,所以那些getter,setter,writable,enumerable属性没法设定了,这里鸡贼的调用了defineProperties,后面他的实现也只有getter和setter。

这里把constructor赋值回去的做法已经被放弃了,浏览器都已经放弃这一步操作了....

Object.defineProperty

(function() {
  if (!Object.defineProperty ||
      !(function () { try { Object.defineProperty({}, 'x', {}); return true; } catch (e) { return false; } } ())) {
    var orig = Object.defineProperty;
    Object.defineProperty = function (o, prop, desc) {
      // In IE8 try built-in implementation for defining properties on DOM prototypes.
      if (orig) { try { return orig(o, prop, desc); } catch (e) {} }
      if (o !== Object(o)) { throw TypeError("Object.defineProperty called on non-object"); }
      if (Object.prototype.__defineGetter__ && ('get' in desc)) {
        Object.prototype.__defineGetter__.call(o, prop, desc.get);
      }
      if (Object.prototype.__defineSetter__ && ('set' in desc)) {
        Object.prototype.__defineSetter__.call(o, prop, desc.set);
      }
      if ('value' in desc) {
        o[prop] = desc.value;
      }
      return o;
    };
  }
}());

因为IE8部分支持了这个方法(只支持DOM Object),所以这里的检测调用了一次,并用了个try catch来返回值。

这里在能够使用IE8自带的情况下返回自带的。

然后就是使用了两个已经被废弃的函数。__defineGetter____defineSetter__来操作访问器属性,但是那些enumerable,writable就不行了。

Object.defineProperties

这里就是对传入的properties进行for in,然后用hasOwnProperty检测一下。再调用上面的Object.defineProperty。

Object.keys

返回的自己的可枚举的属性。

这个就是标准的使用for in,再使用hasOwnProperty。

for in会返回所有的可枚举的属性,包括原型链上的。

Function.prototype.bind

if (!Function.prototype.bind) {
  Function.prototype.bind = function (o) {
    if (typeof this !== 'function') { throw TypeError("Bind must be called on a function"); }
    var slice = [].slice,
        args = slice.call(arguments, 1),
        self = this,
        bound = function () {
          return self.apply(this instanceof nop ? this : o,
                            args.concat(slice.call(arguments)));
        };

    function nop() {}
    if (self.prototype) {
      // native functions don't have a prototype
      nop.prototype = self.prototype;
    }
    nop.prototype = self.prototype;
    bound.prototype = new nop();
    return bound;
  };
}

bind方法IE9之后才加入的,也就是说一般情况下都不能用咯

这个方法直接抄的mozilla官网的,不过写的相当帅气,把一些边界情况考虑的相当完美了。

这里的三目运算符本来百思不得其解的,最后参考了stackoverflow

  • 判断typeof是因为一般情况下只有function可以执行这个,但是有人强行改变类型调用的话,这里就能阻止住。
  • 判断self.prototype,是因为原生的一些function是没有prototype的,这个很有意思!!!
  • 这里之所以用concat是因为bind返回的还是一个函数,需要支持他调用时的参数

至于这里为什么要绕一个大弯,因为他想要让bind方法适用于new,想让他作为一个constructor来应用,这里参见stack_overflow,才开始百思不得其解,最后发现自己还是太年轻了,这里的写法都还是讲道理的。

Array.isArray

用来判断是不是array的方法

就是通过原生的Object.prototype.toString.call([]) == '[object Array]'来判断的,其实也可以在判断了length是不是数字之后再看下length是不是可枚举的,因为原生的arry的length是不可枚举的。

Array.prototype.indexOf和lastIndexof

ie9以上才能用

这里拿this.length的时候用了this.length >>> 0,这一句是位操作符,是针对数字的,功能是

  • 将所有的非数值转化为0
  • 将数值所有大于等于0的数取整数部分

这样做太安全了,到底有没有价值也不知道。

在进行对比之前,还判断了index in array

其他的就是很正常的实现,for循环进行比较,lastIndexOf类似,只是换成了从后面查找而已

Array.prototype.every;some;foreach;map;filter;reduce;reduceRight

ie9以上才能用

正好复习了一下这几个函数的功能,写法没有什么特别之处,就是for循环进行call的调用

  • every就是对数组的每一项执行传入的函数,如果所有的都返回true,才返回true,否则返回false。
  • some就是数组中只要有一个符合就会返回true,否则返回false
  • foreach就是对每一项进行调用
  • map返回了一个新的数组,将数组每个元素执行方法得到的结果组成一个新数组返回回来
  • filter返回一个执行之后为true的数组,就是将每一个执行,如果为true就添加进结果

Array.prototype.reduce和Array.prototype.reduceRight

函数的功能是从左向右合并通过函数合并,传入的参数除了callback,还有一个初始项。

两个的写法很正常,就是简单的实现而已。

之所以比较长是因为做的很安全,各种检测,包括传入的是不是函数,调用的是不是array等。

String.prototype.trim

这个比较简单,但是这边使用的是+号挺有意思的

  String(this).replace(/^\s+|\s+$/g, '');

Date.now和Date.prototype.toISOString

Date的这两个居然还得IE9

就是Number(new Date())

而toISOString就是返回ISO格式的日期,类似于2011-10-05T14:48:00.000Z

实现就是getUTCFullYear这种都支持的来拼装一下就行了

到此es5的polyfill就看完了

但是没有JSON的实现,JSON对象在IE7以下是没有用的,这里作者推荐使用JSON-js

overline