index.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586
  1. var MutationObserver = window.MutationObserver
  2. || window.WebKitMutationObserver
  3. || window.MozMutationObserver;
  4. /*
  5. * Copyright 2012 The Polymer Authors. All rights reserved.
  6. * Use of this source code is goverened by a BSD-style
  7. * license that can be found in the LICENSE file.
  8. */
  9. var WeakMap = window.WeakMap;
  10. if (typeof WeakMap === 'undefined') {
  11. var defineProperty = Object.defineProperty;
  12. var counter = Date.now() % 1e9;
  13. WeakMap = function() {
  14. this.name = '__st' + (Math.random() * 1e9 >>> 0) + (counter++ + '__');
  15. };
  16. WeakMap.prototype = {
  17. set: function(key, value) {
  18. var entry = key[this.name];
  19. if (entry && entry[0] === key)
  20. entry[1] = value;
  21. else
  22. defineProperty(key, this.name, {value: [key, value], writable: true});
  23. return this;
  24. },
  25. get: function(key) {
  26. var entry;
  27. return (entry = key[this.name]) && entry[0] === key ?
  28. entry[1] : undefined;
  29. },
  30. 'delete': function(key) {
  31. var entry = key[this.name];
  32. if (!entry) return false;
  33. var hasValue = entry[0] === key;
  34. entry[0] = entry[1] = undefined;
  35. return hasValue;
  36. },
  37. has: function(key) {
  38. var entry = key[this.name];
  39. if (!entry) return false;
  40. return entry[0] === key;
  41. }
  42. };
  43. }
  44. var registrationsTable = new WeakMap();
  45. // We use setImmediate or postMessage for our future callback.
  46. var setImmediate = window.msSetImmediate;
  47. // Use post message to emulate setImmediate.
  48. if (!setImmediate) {
  49. var setImmediateQueue = [];
  50. var sentinel = String(Math.random());
  51. window.addEventListener('message', function(e) {
  52. if (e.data === sentinel) {
  53. var queue = setImmediateQueue;
  54. setImmediateQueue = [];
  55. queue.forEach(function(func) {
  56. func();
  57. });
  58. }
  59. });
  60. setImmediate = function(func) {
  61. setImmediateQueue.push(func);
  62. window.postMessage(sentinel, '*');
  63. };
  64. }
  65. // This is used to ensure that we never schedule 2 callas to setImmediate
  66. var isScheduled = false;
  67. // Keep track of observers that needs to be notified next time.
  68. var scheduledObservers = [];
  69. /**
  70. * Schedules |dispatchCallback| to be called in the future.
  71. * @param {MutationObserver} observer
  72. */
  73. function scheduleCallback(observer) {
  74. scheduledObservers.push(observer);
  75. if (!isScheduled) {
  76. isScheduled = true;
  77. setImmediate(dispatchCallbacks);
  78. }
  79. }
  80. function wrapIfNeeded(node) {
  81. return window.ShadowDOMPolyfill &&
  82. window.ShadowDOMPolyfill.wrapIfNeeded(node) ||
  83. node;
  84. }
  85. function dispatchCallbacks() {
  86. // http://dom.spec.whatwg.org/#mutation-observers
  87. isScheduled = false; // Used to allow a new setImmediate call above.
  88. var observers = scheduledObservers;
  89. scheduledObservers = [];
  90. // Sort observers based on their creation UID (incremental).
  91. observers.sort(function(o1, o2) {
  92. return o1.uid_ - o2.uid_;
  93. });
  94. var anyNonEmpty = false;
  95. observers.forEach(function(observer) {
  96. // 2.1, 2.2
  97. var queue = observer.takeRecords();
  98. // 2.3. Remove all transient registered observers whose observer is mo.
  99. removeTransientObserversFor(observer);
  100. // 2.4
  101. if (queue.length) {
  102. observer.callback_(queue, observer);
  103. anyNonEmpty = true;
  104. }
  105. });
  106. // 3.
  107. if (anyNonEmpty)
  108. dispatchCallbacks();
  109. }
  110. function removeTransientObserversFor(observer) {
  111. observer.nodes_.forEach(function(node) {
  112. var registrations = registrationsTable.get(node);
  113. if (!registrations)
  114. return;
  115. registrations.forEach(function(registration) {
  116. if (registration.observer === observer)
  117. registration.removeTransientObservers();
  118. });
  119. });
  120. }
  121. /**
  122. * This function is used for the "For each registered observer observer (with
  123. * observer's options as options) in target's list of registered observers,
  124. * run these substeps:" and the "For each ancestor ancestor of target, and for
  125. * each registered observer observer (with options options) in ancestor's list
  126. * of registered observers, run these substeps:" part of the algorithms. The
  127. * |options.subtree| is checked to ensure that the callback is called
  128. * correctly.
  129. *
  130. * @param {Node} target
  131. * @param {function(MutationObserverInit):MutationRecord} callback
  132. */
  133. function forEachAncestorAndObserverEnqueueRecord(target, callback) {
  134. for (var node = target; node; node = node.parentNode) {
  135. var registrations = registrationsTable.get(node);
  136. if (registrations) {
  137. for (var j = 0; j < registrations.length; j++) {
  138. var registration = registrations[j];
  139. var options = registration.options;
  140. // Only target ignores subtree.
  141. if (node !== target && !options.subtree)
  142. continue;
  143. var record = callback(options);
  144. if (record)
  145. registration.enqueue(record);
  146. }
  147. }
  148. }
  149. }
  150. var uidCounter = 0;
  151. /**
  152. * The class that maps to the DOM MutationObserver interface.
  153. * @param {Function} callback.
  154. * @constructor
  155. */
  156. function JsMutationObserver(callback) {
  157. this.callback_ = callback;
  158. this.nodes_ = [];
  159. this.records_ = [];
  160. this.uid_ = ++uidCounter;
  161. }
  162. JsMutationObserver.prototype = {
  163. observe: function(target, options) {
  164. target = wrapIfNeeded(target);
  165. // 1.1
  166. if (!options.childList && !options.attributes && !options.characterData ||
  167. // 1.2
  168. options.attributeOldValue && !options.attributes ||
  169. // 1.3
  170. options.attributeFilter && options.attributeFilter.length &&
  171. !options.attributes ||
  172. // 1.4
  173. options.characterDataOldValue && !options.characterData) {
  174. throw new SyntaxError();
  175. }
  176. var registrations = registrationsTable.get(target);
  177. if (!registrations)
  178. registrationsTable.set(target, registrations = []);
  179. // 2
  180. // If target's list of registered observers already includes a registered
  181. // observer associated with the context object, replace that registered
  182. // observer's options with options.
  183. var registration;
  184. for (var i = 0; i < registrations.length; i++) {
  185. if (registrations[i].observer === this) {
  186. registration = registrations[i];
  187. registration.removeListeners();
  188. registration.options = options;
  189. break;
  190. }
  191. }
  192. // 3.
  193. // Otherwise, add a new registered observer to target's list of registered
  194. // observers with the context object as the observer and options as the
  195. // options, and add target to context object's list of nodes on which it
  196. // is registered.
  197. if (!registration) {
  198. registration = new Registration(this, target, options);
  199. registrations.push(registration);
  200. this.nodes_.push(target);
  201. }
  202. registration.addListeners();
  203. },
  204. disconnect: function() {
  205. this.nodes_.forEach(function(node) {
  206. var registrations = registrationsTable.get(node);
  207. for (var i = 0; i < registrations.length; i++) {
  208. var registration = registrations[i];
  209. if (registration.observer === this) {
  210. registration.removeListeners();
  211. registrations.splice(i, 1);
  212. // Each node can only have one registered observer associated with
  213. // this observer.
  214. break;
  215. }
  216. }
  217. }, this);
  218. this.records_ = [];
  219. },
  220. takeRecords: function() {
  221. var copyOfRecords = this.records_;
  222. this.records_ = [];
  223. return copyOfRecords;
  224. }
  225. };
  226. /**
  227. * @param {string} type
  228. * @param {Node} target
  229. * @constructor
  230. */
  231. function MutationRecord(type, target) {
  232. this.type = type;
  233. this.target = target;
  234. this.addedNodes = [];
  235. this.removedNodes = [];
  236. this.previousSibling = null;
  237. this.nextSibling = null;
  238. this.attributeName = null;
  239. this.attributeNamespace = null;
  240. this.oldValue = null;
  241. }
  242. function copyMutationRecord(original) {
  243. var record = new MutationRecord(original.type, original.target);
  244. record.addedNodes = original.addedNodes.slice();
  245. record.removedNodes = original.removedNodes.slice();
  246. record.previousSibling = original.previousSibling;
  247. record.nextSibling = original.nextSibling;
  248. record.attributeName = original.attributeName;
  249. record.attributeNamespace = original.attributeNamespace;
  250. record.oldValue = original.oldValue;
  251. return record;
  252. };
  253. // We keep track of the two (possibly one) records used in a single mutation.
  254. var currentRecord, recordWithOldValue;
  255. /**
  256. * Creates a record without |oldValue| and caches it as |currentRecord| for
  257. * later use.
  258. * @param {string} oldValue
  259. * @return {MutationRecord}
  260. */
  261. function getRecord(type, target) {
  262. return currentRecord = new MutationRecord(type, target);
  263. }
  264. /**
  265. * Gets or creates a record with |oldValue| based in the |currentRecord|
  266. * @param {string} oldValue
  267. * @return {MutationRecord}
  268. */
  269. function getRecordWithOldValue(oldValue) {
  270. if (recordWithOldValue)
  271. return recordWithOldValue;
  272. recordWithOldValue = copyMutationRecord(currentRecord);
  273. recordWithOldValue.oldValue = oldValue;
  274. return recordWithOldValue;
  275. }
  276. function clearRecords() {
  277. currentRecord = recordWithOldValue = undefined;
  278. }
  279. /**
  280. * @param {MutationRecord} record
  281. * @return {boolean} Whether the record represents a record from the current
  282. * mutation event.
  283. */
  284. function recordRepresentsCurrentMutation(record) {
  285. return record === recordWithOldValue || record === currentRecord;
  286. }
  287. /**
  288. * Selects which record, if any, to replace the last record in the queue.
  289. * This returns |null| if no record should be replaced.
  290. *
  291. * @param {MutationRecord} lastRecord
  292. * @param {MutationRecord} newRecord
  293. * @param {MutationRecord}
  294. */
  295. function selectRecord(lastRecord, newRecord) {
  296. if (lastRecord === newRecord)
  297. return lastRecord;
  298. // Check if the the record we are adding represents the same record. If
  299. // so, we keep the one with the oldValue in it.
  300. if (recordWithOldValue && recordRepresentsCurrentMutation(lastRecord))
  301. return recordWithOldValue;
  302. return null;
  303. }
  304. /**
  305. * Class used to represent a registered observer.
  306. * @param {MutationObserver} observer
  307. * @param {Node} target
  308. * @param {MutationObserverInit} options
  309. * @constructor
  310. */
  311. function Registration(observer, target, options) {
  312. this.observer = observer;
  313. this.target = target;
  314. this.options = options;
  315. this.transientObservedNodes = [];
  316. }
  317. Registration.prototype = {
  318. enqueue: function(record) {
  319. var records = this.observer.records_;
  320. var length = records.length;
  321. // There are cases where we replace the last record with the new record.
  322. // For example if the record represents the same mutation we need to use
  323. // the one with the oldValue. If we get same record (this can happen as we
  324. // walk up the tree) we ignore the new record.
  325. if (records.length > 0) {
  326. var lastRecord = records[length - 1];
  327. var recordToReplaceLast = selectRecord(lastRecord, record);
  328. if (recordToReplaceLast) {
  329. records[length - 1] = recordToReplaceLast;
  330. return;
  331. }
  332. } else {
  333. scheduleCallback(this.observer);
  334. }
  335. records[length] = record;
  336. },
  337. addListeners: function() {
  338. this.addListeners_(this.target);
  339. },
  340. addListeners_: function(node) {
  341. var options = this.options;
  342. if (options.attributes)
  343. node.addEventListener('DOMAttrModified', this, true);
  344. if (options.characterData)
  345. node.addEventListener('DOMCharacterDataModified', this, true);
  346. if (options.childList)
  347. node.addEventListener('DOMNodeInserted', this, true);
  348. if (options.childList || options.subtree)
  349. node.addEventListener('DOMNodeRemoved', this, true);
  350. },
  351. removeListeners: function() {
  352. this.removeListeners_(this.target);
  353. },
  354. removeListeners_: function(node) {
  355. var options = this.options;
  356. if (options.attributes)
  357. node.removeEventListener('DOMAttrModified', this, true);
  358. if (options.characterData)
  359. node.removeEventListener('DOMCharacterDataModified', this, true);
  360. if (options.childList)
  361. node.removeEventListener('DOMNodeInserted', this, true);
  362. if (options.childList || options.subtree)
  363. node.removeEventListener('DOMNodeRemoved', this, true);
  364. },
  365. /**
  366. * Adds a transient observer on node. The transient observer gets removed
  367. * next time we deliver the change records.
  368. * @param {Node} node
  369. */
  370. addTransientObserver: function(node) {
  371. // Don't add transient observers on the target itself. We already have all
  372. // the required listeners set up on the target.
  373. if (node === this.target)
  374. return;
  375. this.addListeners_(node);
  376. this.transientObservedNodes.push(node);
  377. var registrations = registrationsTable.get(node);
  378. if (!registrations)
  379. registrationsTable.set(node, registrations = []);
  380. // We know that registrations does not contain this because we already
  381. // checked if node === this.target.
  382. registrations.push(this);
  383. },
  384. removeTransientObservers: function() {
  385. var transientObservedNodes = this.transientObservedNodes;
  386. this.transientObservedNodes = [];
  387. transientObservedNodes.forEach(function(node) {
  388. // Transient observers are never added to the target.
  389. this.removeListeners_(node);
  390. var registrations = registrationsTable.get(node);
  391. for (var i = 0; i < registrations.length; i++) {
  392. if (registrations[i] === this) {
  393. registrations.splice(i, 1);
  394. // Each node can only have one registered observer associated with
  395. // this observer.
  396. break;
  397. }
  398. }
  399. }, this);
  400. },
  401. handleEvent: function(e) {
  402. // Stop propagation since we are managing the propagation manually.
  403. // This means that other mutation events on the page will not work
  404. // correctly but that is by design.
  405. e.stopImmediatePropagation();
  406. switch (e.type) {
  407. case 'DOMAttrModified':
  408. // http://dom.spec.whatwg.org/#concept-mo-queue-attributes
  409. var name = e.attrName;
  410. var namespace = e.relatedNode.namespaceURI;
  411. var target = e.target;
  412. // 1.
  413. var record = new getRecord('attributes', target);
  414. record.attributeName = name;
  415. record.attributeNamespace = namespace;
  416. // 2.
  417. var oldValue = null;
  418. if (!(typeof MutationEvent !== 'undefined' && e.attrChange === MutationEvent.ADDITION))
  419. oldValue = e.prevValue;
  420. forEachAncestorAndObserverEnqueueRecord(target, function(options) {
  421. // 3.1, 4.2
  422. if (!options.attributes)
  423. return;
  424. // 3.2, 4.3
  425. if (options.attributeFilter && options.attributeFilter.length &&
  426. options.attributeFilter.indexOf(name) === -1 &&
  427. options.attributeFilter.indexOf(namespace) === -1) {
  428. return;
  429. }
  430. // 3.3, 4.4
  431. if (options.attributeOldValue)
  432. return getRecordWithOldValue(oldValue);
  433. // 3.4, 4.5
  434. return record;
  435. });
  436. break;
  437. case 'DOMCharacterDataModified':
  438. // http://dom.spec.whatwg.org/#concept-mo-queue-characterdata
  439. var target = e.target;
  440. // 1.
  441. var record = getRecord('characterData', target);
  442. // 2.
  443. var oldValue = e.prevValue;
  444. forEachAncestorAndObserverEnqueueRecord(target, function(options) {
  445. // 3.1, 4.2
  446. if (!options.characterData)
  447. return;
  448. // 3.2, 4.3
  449. if (options.characterDataOldValue)
  450. return getRecordWithOldValue(oldValue);
  451. // 3.3, 4.4
  452. return record;
  453. });
  454. break;
  455. case 'DOMNodeRemoved':
  456. this.addTransientObserver(e.target);
  457. // Fall through.
  458. case 'DOMNodeInserted':
  459. // http://dom.spec.whatwg.org/#concept-mo-queue-childlist
  460. var target = e.relatedNode;
  461. var changedNode = e.target;
  462. var addedNodes, removedNodes;
  463. if (e.type === 'DOMNodeInserted') {
  464. addedNodes = [changedNode];
  465. removedNodes = [];
  466. } else {
  467. addedNodes = [];
  468. removedNodes = [changedNode];
  469. }
  470. var previousSibling = changedNode.previousSibling;
  471. var nextSibling = changedNode.nextSibling;
  472. // 1.
  473. var record = getRecord('childList', target);
  474. record.addedNodes = addedNodes;
  475. record.removedNodes = removedNodes;
  476. record.previousSibling = previousSibling;
  477. record.nextSibling = nextSibling;
  478. forEachAncestorAndObserverEnqueueRecord(target, function(options) {
  479. // 2.1, 3.2
  480. if (!options.childList)
  481. return;
  482. // 2.2, 3.3
  483. return record;
  484. });
  485. }
  486. clearRecords();
  487. }
  488. };
  489. if (!MutationObserver) {
  490. MutationObserver = JsMutationObserver;
  491. }
  492. module.exports = MutationObserver;