element-resize-detector
Advanced tools
Comparing version 0.1.4 to 0.2.2
@@ -1,4 +0,41 @@ | ||
!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var f;"undefined"!=typeof window?f=window:"undefined"!=typeof global?f=global:"undefined"!=typeof self&&(f=self),f.elementResizeDetectorMaker=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){ | ||
(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.elementResizeDetectorMaker = f()}})(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){ | ||
"use strict"; | ||
var detector = module.exports = {}; | ||
detector.isIE = function(version) { | ||
function isAnyIeVersion() { | ||
var agent = navigator.userAgent.toLowerCase(); | ||
return agent.indexOf("msie") !== -1 || agent.indexOf("trident") !== -1; | ||
} | ||
if(!isAnyIeVersion()) { | ||
return false; | ||
} | ||
if(!version) { | ||
return true; | ||
} | ||
//Shamelessly stolen from https://gist.github.com/padolsey/527683 | ||
var ieVersion = (function(){ | ||
var undef, | ||
v = 3, | ||
div = document.createElement("div"), | ||
all = div.getElementsByTagName("i"); | ||
do { | ||
div.innerHTML = "<!--[if gt IE " + (++v) + "]><i></i><![endif]-->"; | ||
} | ||
while (all[0]); | ||
return v > 4 ? v : undef; | ||
}()); | ||
return version === ieVersion; | ||
}; | ||
},{}],2:[function(require,module,exports){ | ||
"use strict"; | ||
var utils = module.exports = {}; | ||
@@ -22,3 +59,3 @@ | ||
},{}],2:[function(require,module,exports){ | ||
},{}],3:[function(require,module,exports){ | ||
//Heavily inspired by http://www.backalleycoder.com/2013/03/18/cross-browser-event-based-element-resize-detection/ | ||
@@ -29,29 +66,100 @@ | ||
var forEach = require("./collection-utils").forEach; | ||
var elementUtils = require("./element-utils"); | ||
var elementUtilsMaker = require("./element-utils"); | ||
var listenerHandlerMaker = require("./listener-handler"); | ||
var idGeneratorMaker = require("./id-generator"); | ||
var listenerHandlerMaker = require("./listener-handler"); | ||
var idHandlerMaker = require("./id-handler"); | ||
var reporterMaker = require("./reporter"); | ||
/** | ||
* @typedef idHandler | ||
* @type {object} | ||
* @property {function} get Gets the resize detector id of the element. | ||
* @property {function} set Generate and sets the resize detector id of the element. | ||
*/ | ||
/** | ||
* @typedef Options | ||
* @type {object} | ||
* @property {boolean} callOnAdd Determines if listeners should be called when they are getting added. | ||
Default is true. If true, the listener is guaranteed to be called when it has been added. | ||
If false, the listener will not be guarenteed to be called when it has been added (does not prevent it from being called). | ||
* @property {idHandler} idHandler A custom id handler that is responsible for generating, setting and retrieving id's for elements. | ||
If not provided, a default id handler will be used. | ||
* @property {reporter} reporter A custom reporter that handles reporting logs, warnings and errors. | ||
If not provided, a default id handler will be used. | ||
If set to false, then nothing will be reported. | ||
*/ | ||
/** | ||
* Creates an element resize detector instance. | ||
* @public | ||
* @param {Options?} options Optional global options object that will decide how this instance will work. | ||
*/ | ||
module.exports = function(options) { | ||
options = options || {}; | ||
var allowMultipleListeners = options.allowMultipleListeners === undefined ? true : false; | ||
var eventListenerHandler = listenerHandlerMaker(); | ||
var idGenerator = idGeneratorMaker(); | ||
//Options to be used as default for the listenTo function. | ||
var globalOptions = {}; | ||
globalOptions.callOnAdd = !!getOption(options, "callOnAdd", true); | ||
//idHandler is currently not an option to the listenTo function, so it should not be added to globalOptions. | ||
var idHandler = options.idHandler; | ||
if(!idHandler) { | ||
var idGenerator = idGeneratorMaker(); | ||
var defaultIdHandler = idHandlerMaker(idGenerator); | ||
idHandler = defaultIdHandler; | ||
} | ||
//reporter is currently not an option to the listenTo function, so it should not be added to globalOptions. | ||
var reporter = options.reporter; | ||
if(!reporter) { | ||
//If options.reporter is false, then the reporter should be quiet. | ||
var quiet = reporter === false; | ||
reporter = reporterMaker(quiet); | ||
} | ||
var eventListenerHandler = listenerHandlerMaker(idHandler); | ||
var elementUtils = elementUtilsMaker(idHandler); | ||
/** | ||
* Makes the given elements resize-detectable and starts listening to resize events on the elements. Calls the event callback for each event for each element. | ||
* @public | ||
* @param {Options?} options Optional options object. These options will override the global options. Some options may not be overriden, such as idHandler. | ||
* @param {element[]|element} elements The given array of elements to detect resize events of. Single element is also valid. | ||
* @param {function} listener The callback to be executed for each resize event for each element. | ||
*/ | ||
function listenTo(elements, listener) { | ||
function isListenedTo(element) { | ||
return elementUtils.isDetectable(element) && eventListenerHandler.get(element).length; | ||
function listenTo(options, elements, listener) { | ||
function onResizeCallback(element) { | ||
var listeners = eventListenerHandler.get(element); | ||
forEach(listeners, function(listener) { | ||
listener(element); | ||
}); | ||
} | ||
function addListener(element, listener) { | ||
elementUtils.addListener(element, listener); | ||
function onElementReadyToAddListener(callOnAdd, element, listener) { | ||
eventListenerHandler.add(element, listener); | ||
if(callOnAdd) { | ||
listener(element); | ||
} | ||
} | ||
//Options object may be omitted. | ||
if(!listener) { | ||
listener = elements; | ||
elements = options; | ||
options = {}; | ||
} | ||
if(!elements) { | ||
throw new Error("At least one element required."); | ||
} | ||
if(!listener) { | ||
throw new Error("Listener required."); | ||
} | ||
if(elements.length === undefined) { | ||
@@ -61,8 +169,27 @@ elements = [elements]; | ||
var callOnAdd = getOption(options, "callOnAdd", globalOptions.callOnAdd); | ||
forEach(elements, function(element) { | ||
//The element may change size directly after the call to listenTo, which would be unable to detect it because | ||
//the async adding of the object. By checking the size before and after, the size change can still be detected | ||
//and the listener can be called accordingly. | ||
var preWidth = element.offsetWidth; | ||
var preHeight = element.offsetHeight; | ||
if(!elementUtils.isDetectable(element)) { | ||
//The element is not prepared to be detectable, so do prepare it and add a listener to it. | ||
var id = idGenerator.newId(); | ||
return elementUtils.makeDetectable(element, id, function(element) { | ||
addListener(element, listener); | ||
return elementUtils.makeDetectable(reporter, element, function(element) { | ||
elementUtils.addListener(element, onResizeCallback); | ||
onElementReadyToAddListener(callOnAdd, element, listener); | ||
//Only here the uncaught resize may occur (since this code is async). | ||
//Check if the size is the same as when adding the listener. | ||
var postWidth = element.offsetWidth; | ||
var postHeight = element.offsetHeight; | ||
//If callOnAdd is true, then the listener will have been called either way, so no need to call the listener manually then. | ||
if(!callOnAdd && (preWidth !== postWidth || preHeight !== postHeight)) { | ||
//The element was changed while the object was being added. Call the listener. | ||
listener(element); | ||
} | ||
}); | ||
@@ -72,10 +199,3 @@ } | ||
//The element has been prepared to be detectable and is ready to be listened to. | ||
if(isListenedTo(element) && !allowMultipleListeners) { | ||
//Since there is a listener and we disallow multiple listeners no listener should be added. | ||
return; | ||
} | ||
//Since multiple listeners is allowed, another listener is added to the element. | ||
return addListener(element, listener); | ||
onElementReadyToAddListener(callOnAdd, element, listener); | ||
}); | ||
@@ -89,105 +209,171 @@ } | ||
},{"./collection-utils":1,"./element-utils":3,"./id-generator":4,"./listener-handler":5}],3:[function(require,module,exports){ | ||
function getOption(options, name, defaultValue) { | ||
var value = options[name]; | ||
if((value === undefined || value === null) && defaultValue !== undefined) { | ||
return defaultValue; | ||
} | ||
return value; | ||
} | ||
},{"./collection-utils":2,"./element-utils":4,"./id-generator":5,"./id-handler":6,"./listener-handler":7,"./reporter":8}],4:[function(require,module,exports){ | ||
"use strict"; | ||
var forEach = require("./collection-utils").forEach; | ||
var browserDetector = require("./browser-detector"); | ||
var utils = module.exports = {}; | ||
module.exports = function(idHandler) { | ||
/** | ||
* Tells if the element has been made detectable and ready to be listened for resize events. | ||
* @public | ||
* @param {element} The element to check. | ||
* @returns {boolean} True or false depending on if the element is detectable or not. | ||
*/ | ||
function isDetectable(element) { | ||
if(browserDetector.isIE(8)) { | ||
//IE 8 does not use the object method. | ||
//Check only if the element has been given an id. | ||
return !!idHandler.get(element); | ||
} | ||
/** | ||
* Gets the elq id of the element. | ||
* @public | ||
* @param {element} The target element to get the id of. | ||
* @returns {string} The id of the element. | ||
*/ | ||
utils.getId = function(element) { | ||
return element.getAttribute("elq-target-id"); | ||
}; | ||
return !!getObject(element); | ||
} | ||
/** | ||
* Tells if the element has been made detectable and ready to be listened for resize events. | ||
* @public | ||
* @param {element} The element to check. | ||
* @returns {boolean} True or false depending on if the element is detectable or not. | ||
*/ | ||
utils.isDetectable = function(element) { | ||
return getObject(element); | ||
}; | ||
/** | ||
* Adds a resize event listener to the element. | ||
* @public | ||
* @param {element} element The element that should have the listener added. | ||
* @param {function} listener The listener callback to be called for each resize event of the element. The element will be given as a parameter to the listener callback. | ||
*/ | ||
function addListener(element, listener) { | ||
if(!isDetectable(element)) { | ||
throw new Error("Element is not detectable."); | ||
} | ||
/** | ||
* Adds a resize event listener to the element. | ||
* @public | ||
* @param {element} element The element that should have the listener added. | ||
* @param {function} listener The listener callback to be called for each resize event of the element. The element will be given as a parameter to the listener callback. | ||
*/ | ||
utils.addListener = function(element, listener) { | ||
if(!utils.isDetectable(element)) { | ||
throw new Error("Element is not detectable."); | ||
function listenerProxy() { | ||
listener(element); | ||
} | ||
if(browserDetector.isIE(8)) { | ||
//IE 8 does not support object, but supports the resize event directly on elements. | ||
element.attachEvent("onresize", listenerProxy); | ||
} else { | ||
var object = getObject(element); | ||
object.contentDocument.defaultView.addEventListener("resize", listenerProxy); | ||
} | ||
} | ||
var object = getObject(element); | ||
object.contentDocument.defaultView.addEventListener("resize", function() { | ||
listener(element); | ||
}); | ||
}; | ||
/** | ||
* Makes an element detectable and ready to be listened for resize events. Will call the callback when the element is ready to be listened for resize changes. | ||
* @private | ||
* @param {element} element The element to make detectable | ||
* @param {function} callback The callback to be called when the element is ready to be listened for resize changes. Will be called with the element as first parameter. | ||
*/ | ||
function makeDetectable(reporter, element, callback) { | ||
function injectObject(id, element, callback) { | ||
var OBJECT_STYLE = "display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; border: none; padding: 0; margin: 0; opacity: 0; z-index: -1000; pointer-events: none;"; | ||
/** | ||
* Makes an element detectable and ready to be listened for resize events. Will call the callback when the element is ready to be listened for resize changes. | ||
* @private | ||
* @param {element} element The element to make detectable | ||
* @param {*} id An unique id in the context of all detectable elements. | ||
* @param {function} callback The callback to be called when the element is ready to be listened for resize changes. Will be called with the element as first parameter. | ||
*/ | ||
utils.makeDetectable = function(element, id, callback) { | ||
function onObjectLoad() { | ||
/*jshint validthis:true */ | ||
function onObjectLoad() { | ||
/*jshint validthis:true */ | ||
//Create the style element to be added to the object. | ||
var objectDocument = this.contentDocument; | ||
var style = objectDocument.createElement("style"); | ||
style.innerHTML = "html, body { margin: 0; padding: 0 } div { -webkit-transition: opacity 0.01s; -ms-transition: opacity 0.01s; -o-transition: opacity 0.01s; transition: opacity 0.01s; opacity: 0; }"; | ||
//Create the style element to be added to the object. | ||
var objectDocument = this.contentDocument; | ||
var style = objectDocument.createElement("style"); | ||
style.innerHTML = "html, body { margin: 0; padding: 0 } div { -webkit-transition: opacity 0.01s; -ms-transition: opacity 0.01s; -o-transition: opacity 0.01s; transition: opacity 0.01s; opacity: 0; }"; | ||
//TODO: Remove any styles that has been set on the object. Only the style above should be styling the object. | ||
//TODO: Remove any styles that has been set on the object. Only the style above should be styling the object. | ||
//Append the style to the object. | ||
objectDocument.head.appendChild(style); | ||
//Append the style to the object. | ||
objectDocument.head.appendChild(style); | ||
this.style.cssText = "display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; border: none; padding: 0; margin: 0; opacity: 0; z-index: -1000; pointer-events: none;"; | ||
//TODO: Is this needed here? | ||
//this.style.cssText = OBJECT_STYLE; | ||
//Notify that the element is ready to be listened to. | ||
callback(element); | ||
} | ||
//Notify that the element is ready to be listened to. | ||
callback(element); | ||
} | ||
//Create an unique elq-target-id for the target element, so that event listeners can be identified to this element. | ||
element.setAttribute("elq-target-id", id); | ||
//The target element needs to be positioned (everything except static) so the absolute positioned object will be positioned relative to the target element. | ||
var style = getComputedStyle(element); | ||
if(style.position === "static") { | ||
element.style.position = "relative"; | ||
//The target element needs to be positioned (everything except static) so the absolute positioned object will be positioned relative to the target element. | ||
if(getComputedStyle(element).position === "static") { | ||
element.style.position = "relative"; | ||
var removeRelativeStyles = function(reporter, element, style, property) { | ||
function getNumericalValue(value) { | ||
return value.replace(/[^-\d\.]/g, ""); | ||
} | ||
var value = style[property]; | ||
if(value !== "auto" && getNumericalValue(value) !== "0") { | ||
reporter.warn("An element that is positioned static has style." + property + "=" + value + " which is ignored due to the static positioning. The element will need to be positioned relateive, so the style." + property + " will be set to 0. Element: ", element); | ||
element.style[property] = 0; | ||
} | ||
}; | ||
//Check so that there are no accidental styles that will make the element styled differently now that is is relative. | ||
//If there are any, set them to 0 (this should be okay with the user since the style properties did nothing before [since the element was positioned static] anyway). | ||
removeRelativeStyles(reporter, element, style, "top"); | ||
removeRelativeStyles(reporter, element, style, "right"); | ||
removeRelativeStyles(reporter, element, style, "bottom"); | ||
removeRelativeStyles(reporter, element, style, "left"); | ||
} | ||
//Add an object element as a child to the target element that will be listened to for resize events. | ||
var object = document.createElement("object"); | ||
object.type = "text/html"; | ||
object.style.cssText = OBJECT_STYLE; | ||
object.onload = onObjectLoad; | ||
object.setAttribute("erd-object-id", id); | ||
//Safari: This must occur before adding the object to the DOM. | ||
//IE: Does not like that this happens before, even if it is also added after. | ||
if(!browserDetector.isIE()) { | ||
object.data = "about:blank"; | ||
} | ||
element.appendChild(object); | ||
//IE: This must occur after adding the object to the DOM. | ||
if(browserDetector.isIE()) { | ||
object.data = "about:blank"; | ||
} | ||
} | ||
//Create an unique erd-target-id for the target element, so that event listeners can be identified to this element. | ||
var id = idHandler.set(element); | ||
if(browserDetector.isIE(8)) { | ||
//IE 8 does not support objects properly. Luckily they do support the resize event. | ||
//So do not inject the object and notify that the element is already ready to be listened to. | ||
//The event handler for the resize event is attached in the utils.addListener instead. | ||
callback(element); | ||
} else { | ||
injectObject(id, element, callback); | ||
} | ||
} | ||
//Add an object element as a child to the target element that will be listened to for resize events. | ||
var object = document.createElement("object"); | ||
object.type = "text/html"; | ||
object.data = "about:blank"; | ||
object.onload = onObjectLoad; | ||
object.setAttribute("elq-object-id", id); | ||
element.appendChild(object); | ||
/** | ||
* Returns the child object of the target element. | ||
* @private | ||
* @param {element} element The target element. | ||
* @returns The object element of the target. | ||
*/ | ||
function getObject(element) { | ||
return forEach(element.children, function(child) { | ||
if(child.hasAttribute("erd-object-id")) { | ||
return child; | ||
} | ||
}); | ||
} | ||
return { | ||
isDetectable: isDetectable, | ||
makeDetectable: makeDetectable, | ||
addListener: addListener, | ||
}; | ||
}; | ||
/** | ||
* Returns the child object of the target element. | ||
* @private | ||
* @param {element} element The target element. | ||
* @returns The object element of the target. | ||
*/ | ||
function getObject(element) { | ||
return forEach(element.children, function(child) { | ||
if(child.hasAttribute("elq-object-id")) { | ||
return child; | ||
} | ||
}); | ||
} | ||
},{"./collection-utils":1}],4:[function(require,module,exports){ | ||
},{"./browser-detector":1,"./collection-utils":2}],5:[function(require,module,exports){ | ||
"use strict"; | ||
@@ -203,3 +389,3 @@ | ||
*/ | ||
function newId() { | ||
function generate() { | ||
return idCount++; | ||
@@ -209,12 +395,49 @@ } | ||
return { | ||
newId: newId | ||
generate: generate | ||
}; | ||
}; | ||
},{}],5:[function(require,module,exports){ | ||
},{}],6:[function(require,module,exports){ | ||
"use strict"; | ||
var elementUtils = require("./element-utils"); | ||
module.exports = function(idGenerator) { | ||
module.exports = function() { | ||
/** | ||
* Gets the resize detector id of the element. | ||
* @public | ||
* @param {element} The target element to get the id of. | ||
* @returns {string|number} The id of the element. | ||
*/ | ||
function getId(element) { | ||
return element.getAttribute("erd-target-id"); | ||
} | ||
/** | ||
* Sets the resize detector id of the element. | ||
* @public | ||
* @param {element} The target element to set the id to. | ||
* @param {string?} An optional id to set to the element. If not specified, an id will be generated. All id's must be unique. | ||
* @returns {string|number} The id of the element. | ||
*/ | ||
function setId(element, id) { | ||
if(!id && id !== 0) { | ||
//Number should be generated. | ||
id = idGenerator.generate(); | ||
} | ||
element.setAttribute("erd-target-id", id); | ||
return id; | ||
} | ||
return { | ||
get: getId, | ||
set: setId | ||
}; | ||
}; | ||
},{}],7:[function(require,module,exports){ | ||
"use strict"; | ||
module.exports = function(idHandler) { | ||
var eventListeners = {}; | ||
@@ -229,3 +452,3 @@ | ||
function getListeners(element) { | ||
return eventListeners[elementUtils.getId(element)]; | ||
return eventListeners[idHandler.get(element)]; | ||
} | ||
@@ -240,3 +463,3 @@ | ||
function addListener(element, listener) { | ||
var id = elementUtils.getId(element); | ||
var id = idHandler.get(element); | ||
@@ -256,4 +479,33 @@ if(!eventListeners[id]) { | ||
},{"./element-utils":3}]},{},[2])(2) | ||
},{}],8:[function(require,module,exports){ | ||
"use strict"; | ||
/* global console: false */ | ||
/** | ||
* Reporter that handles the reporting of logs, warnings and errors. | ||
* @public | ||
* @param {boolean} quiet Tells if the reporter should be quiet or not. | ||
*/ | ||
module.exports = function(quiet) { | ||
var noop = function () { | ||
//Does nothing. | ||
}; | ||
var reporter = { | ||
log: noop, | ||
warn: noop, | ||
error: noop | ||
}; | ||
if(!quiet && window.console) { | ||
reporter.log = console.log; | ||
reporter.warn = console.warn; | ||
reporter.error = console.error; | ||
} | ||
return reporter; | ||
}; | ||
},{}]},{},[3])(3) | ||
}); | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/grunt-browserify/node_modules/browserify/node_modules/browser-pack/_prelude.js","src/collection-utils.js","src/element-resize-detector.js","src/element-utils.js","src/id-generator.js","src/listener-handler.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","\"use strict\";\n\nvar utils = module.exports = {};\n\n/**\n * Loops through the collection and calls the callback for each element. if the callback returns truthy, the loop is broken and returns the same value.\n * @public\n * @param {*} collection The collection to loop through. Needs to have a length property set and have indices set from 0 to length - 1.\n * @param {function} callback The callback to be called for each element. The element will be given as a parameter to the callback. If this callback returns truthy, the loop is broken and the same value is returned.\n * @returns {*} The value that a callback has returned (if truthy). Otherwise nothing.\n */\nutils.forEach = function(collection, callback) {\n    for(var i = 0; i < collection.length; i++) {\n        var result = callback(collection[i]);\n        if(result) {\n            return result;\n        }\n    }\n};\n","//Heavily inspired by http://www.backalleycoder.com/2013/03/18/cross-browser-event-based-element-resize-detection/\n\n\"use strict\";\n\nvar forEach = require(\"./collection-utils\").forEach;\nvar elementUtils = require(\"./element-utils\");\nvar idGeneratorMaker = require(\"./id-generator\");\nvar listenerHandlerMaker = require(\"./listener-handler\");\n\nmodule.exports = function(options) {\n    options = options || {};\n    var allowMultipleListeners = options.allowMultipleListeners === undefined ? true : false;\n\n    var eventListenerHandler = listenerHandlerMaker();\n    var idGenerator = idGeneratorMaker();\n\n    /**\n     * Makes the given elements resize-detectable and starts listening to resize events on the elements. Calls the event callback for each event for each element.\n     * @public\n     * @param {element[]|element} elements The given array of elements to detect resize events of. Single element is also valid.\n     * @param {function} listener The callback to be executed for each resize event for each element.\n     */\n    function listenTo(elements, listener) {\n        function isListenedTo(element) {\n            return elementUtils.isDetectable(element) && eventListenerHandler.get(element).length;\n        }\n\n        function addListener(element, listener) {\n            elementUtils.addListener(element, listener);\n            eventListenerHandler.add(element, listener);\n        }\n\n        if(elements.length === undefined) {\n            elements = [elements];\n        }\n\n        forEach(elements, function(element) {\n            if(!elementUtils.isDetectable(element)) {\n                //The element is not prepared to be detectable, so do prepare it and add a listener to it.\n                var id = idGenerator.newId();\n                return elementUtils.makeDetectable(element, id, function(element) {\n                    addListener(element, listener);\n                });\n            }\n            \n            //The element has been prepared to be detectable and is ready to be listened to.\n            \n            if(isListenedTo(element) && !allowMultipleListeners) {\n                //Since there is a listener and we disallow multiple listeners no listener should be added.\n                return;\n            }\n\n            //Since multiple listeners is allowed, another listener is added to the element.\n            return addListener(element, listener);\n        });\n    }\n\n    return {\n        listenTo: listenTo\n    };\n};\n","\"use strict\";\n\nvar forEach = require(\"./collection-utils\").forEach;\n\nvar utils = module.exports = {};\n\n/**\n * Gets the elq id of the element.\n * @public\n * @param {element} The target element to get the id of.\n * @returns {string} The id of the element.\n */\nutils.getId = function(element) {\n    return element.getAttribute(\"elq-target-id\");\n};\n\n/**\n * Tells if the element has been made detectable and ready to be listened for resize events.\n * @public\n * @param {element} The element to check.\n * @returns {boolean} True or false depending on if the element is detectable or not.\n */\nutils.isDetectable = function(element) {\n    return getObject(element);\n};\n\n/**\n * Adds a resize event listener to the element.\n * @public\n * @param {element} element The element that should have the listener added.\n * @param {function} listener The listener callback to be called for each resize event of the element. The element will be given as a parameter to the listener callback.\n */\nutils.addListener = function(element, listener) {\n    if(!utils.isDetectable(element)) {\n        throw new Error(\"Element is not detectable.\");\n    }\n\n    var object = getObject(element);\n    object.contentDocument.defaultView.addEventListener(\"resize\", function() {\n        listener(element);\n    });\n};\n\n/**\n * Makes an element detectable and ready to be listened for resize events. Will call the callback when the element is ready to be listened for resize changes.\n * @private\n * @param {element} element The element to make detectable\n * @param {*} id An unique id in the context of all detectable elements.\n * @param {function} callback The callback to be called when the element is ready to be listened for resize changes. Will be called with the element as first parameter.\n */\nutils.makeDetectable = function(element, id, callback) {\n    function onObjectLoad() {\n        /*jshint validthis:true */\n\n        //Create the style element to be added to the object.\n        var objectDocument = this.contentDocument;\n        var style = objectDocument.createElement(\"style\");\n        style.innerHTML = \"html, body { margin: 0; padding: 0 } div { -webkit-transition: opacity 0.01s; -ms-transition: opacity 0.01s; -o-transition: opacity 0.01s; transition: opacity 0.01s; opacity: 0; }\";\n\n        //TODO: Remove any styles that has been set on the object. Only the style above should be styling the object.\n\n        //Append the style to the object.\n        objectDocument.head.appendChild(style);\n\n        this.style.cssText = \"display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; border: none; padding: 0; margin: 0; opacity: 0; z-index: -1000; pointer-events: none;\";\n\n        //Notify that the element is ready to be listened to.\n        callback(element);\n    }\n\n    //Create an unique elq-target-id for the target element, so that event listeners can be identified to this element.\n    element.setAttribute(\"elq-target-id\", id);\n\n    //The target element needs to be positioned (everything except static) so the absolute positioned object will be positioned relative to the target element.\n    if(getComputedStyle(element).position === \"static\") {\n        element.style.position = \"relative\";\n    }\n\n    //Add an object element as a child to the target element that will be listened to for resize events.\n    var object = document.createElement(\"object\");\n    object.type = \"text/html\";\n    object.data = \"about:blank\";\n    object.onload = onObjectLoad;\n    object.setAttribute(\"elq-object-id\", id);\n    element.appendChild(object);\n};\n\n/**\n * Returns the child object of the target element.\n * @private\n * @param {element} element The target element.\n * @returns The object element of the target.\n */\nfunction getObject(element) {\n    return forEach(element.children, function(child) {\n        if(child.hasAttribute(\"elq-object-id\")) {\n            return child;\n        }\n    });\n}\n","\"use strict\";\n\nmodule.exports = function() {\n    var idCount = 1;\n\n    /**\n     * Generates a new unique id in the context.\n     * @public\n     * @returns {number} A unique id in the context.\n     */\n    function newId() {\n        return idCount++;\n    }\n\n    return {\n        newId: newId\n    };\n};\n","\"use strict\";\n\nvar elementUtils = require(\"./element-utils\");\n\nmodule.exports = function() {\n    var eventListeners = {};\n\n    /**\n     * Gets all listeners for the given element.\n     * @public\n     * @param {element} element The element to get all listeners for.\n     * @returns All listeners for the given element.\n     */\n    function getListeners(element) {\n        return eventListeners[elementUtils.getId(element)];\n    }\n\n    /**\n     * Stores the given listener for the given element. Will not actually add the listener to the element.\n     * @public\n     * @param {element} element The element that should have the listener added.\n     * @param {function} listener The callback that the element has added.\n     */\n    function addListener(element, listener) {\n        var id = elementUtils.getId(element);\n\n        if(!eventListeners[id]) {\n            eventListeners[id] = [];\n        }\n\n        eventListeners[id].push(listener);\n    }\n\n    return {\n        get: getListeners,\n        add: addListener\n    };\n};\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/grunt-browserify/node_modules/browserify/node_modules/browser-pack/_prelude.js","src/browser-detector.js","src/collection-utils.js","src/element-resize-detector.js","src/element-utils.js","src/id-generator.js","src/id-handler.js","src/listener-handler.js","src/reporter.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","\"use strict\";\n\nvar detector = module.exports = {};\n\ndetector.isIE = function(version) {\n    function isAnyIeVersion() {\n        var agent = navigator.userAgent.toLowerCase();\n        return agent.indexOf(\"msie\") !== -1 || agent.indexOf(\"trident\") !== -1;\n    }\n\n    if(!isAnyIeVersion()) {\n        return false;\n    }\n\n    if(!version) {\n        return true;\n    }\n\n    //Shamelessly stolen from https://gist.github.com/padolsey/527683\n    var ieVersion = (function(){\n        var undef,\n            v = 3,\n            div = document.createElement(\"div\"),\n            all = div.getElementsByTagName(\"i\");\n\n        do {\n            div.innerHTML = \"<!--[if gt IE \" + (++v) + \"]><i></i><![endif]-->\";\n        }\n        while (all[0]);\n\n        return v > 4 ? v : undef;\n    }());\n\n    return version === ieVersion;\n};\n","\"use strict\";\n\nvar utils = module.exports = {};\n\n/**\n * Loops through the collection and calls the callback for each element. if the callback returns truthy, the loop is broken and returns the same value.\n * @public\n * @param {*} collection The collection to loop through. Needs to have a length property set and have indices set from 0 to length - 1.\n * @param {function} callback The callback to be called for each element. The element will be given as a parameter to the callback. If this callback returns truthy, the loop is broken and the same value is returned.\n * @returns {*} The value that a callback has returned (if truthy). Otherwise nothing.\n */\nutils.forEach = function(collection, callback) {\n    for(var i = 0; i < collection.length; i++) {\n        var result = callback(collection[i]);\n        if(result) {\n            return result;\n        }\n    }\n};\n","//Heavily inspired by http://www.backalleycoder.com/2013/03/18/cross-browser-event-based-element-resize-detection/\n\n\"use strict\";\n\nvar forEach = require(\"./collection-utils\").forEach;\nvar elementUtilsMaker = require(\"./element-utils\");\nvar listenerHandlerMaker = require(\"./listener-handler\");\nvar idGeneratorMaker = require(\"./id-generator\");\nvar idHandlerMaker = require(\"./id-handler\");\nvar reporterMaker = require(\"./reporter\");\n\n/**\n * @typedef idHandler\n * @type {object}\n * @property {function} get Gets the resize detector id of the element.\n * @property {function} set Generate and sets the resize detector id of the element.\n */\n\n/**\n * @typedef Options\n * @type {object}\n * @property {boolean} callOnAdd    Determines if listeners should be called when they are getting added. \n                                    Default is true. If true, the listener is guaranteed to be called when it has been added. \n                                    If false, the listener will not be guarenteed to be called when it has been added (does not prevent it from being called).\n * @property {idHandler} idHandler  A custom id handler that is responsible for generating, setting and retrieving id's for elements.\n                                    If not provided, a default id handler will be used.\n * @property {reporter} reporter    A custom reporter that handles reporting logs, warnings and errors. \n                                    If not provided, a default id handler will be used.\n                                    If set to false, then nothing will be reported.\n */\n\n/**\n * Creates an element resize detector instance.\n * @public\n * @param {Options?} options Optional global options object that will decide how this instance will work.\n */\nmodule.exports = function(options) {\n    options = options || {};\n\n    //Options to be used as default for the listenTo function.\n    var globalOptions = {};\n    globalOptions.callOnAdd = !!getOption(options, \"callOnAdd\", true);\n\n    //idHandler is currently not an option to the listenTo function, so it should not be added to globalOptions.\n    var idHandler = options.idHandler;\n\n    if(!idHandler) {\n        var idGenerator = idGeneratorMaker();\n        var defaultIdHandler = idHandlerMaker(idGenerator);\n        idHandler = defaultIdHandler;\n    }\n\n    //reporter is currently not an option to the listenTo function, so it should not be added to globalOptions.\n    var reporter = options.reporter;\n\n    if(!reporter) {\n        //If options.reporter is false, then the reporter should be quiet.\n        var quiet = reporter === false;\n        reporter = reporterMaker(quiet);\n    }\n\n    var eventListenerHandler = listenerHandlerMaker(idHandler);\n    var elementUtils = elementUtilsMaker(idHandler);\n\n    /**\n     * Makes the given elements resize-detectable and starts listening to resize events on the elements. Calls the event callback for each event for each element.\n     * @public\n     * @param {Options?} options Optional options object. These options will override the global options. Some options may not be overriden, such as idHandler.\n     * @param {element[]|element} elements The given array of elements to detect resize events of. Single element is also valid.\n     * @param {function} listener The callback to be executed for each resize event for each element.\n     */\n    function listenTo(options, elements, listener) {\n        function onResizeCallback(element) {\n            var listeners = eventListenerHandler.get(element);\n\n            forEach(listeners, function(listener) {\n                listener(element);\n            });\n        }\n\n        function onElementReadyToAddListener(callOnAdd, element, listener) {\n            eventListenerHandler.add(element, listener);\n            \n            if(callOnAdd) {\n                listener(element);\n            }\n        }\n\n        //Options object may be omitted.\n        if(!listener) {\n            listener = elements;\n            elements = options;\n            options = {};\n        }\n\n        if(!elements) {\n            throw new Error(\"At least one element required.\");\n        }\n\n        if(!listener) {\n            throw new Error(\"Listener required.\");\n        }\n\n        if(elements.length === undefined) {\n            elements = [elements];\n        }\n\n        var callOnAdd = getOption(options, \"callOnAdd\", globalOptions.callOnAdd);\n\n        forEach(elements, function(element) {\n            //The element may change size directly after the call to listenTo, which would be unable to detect it because\n            //the async adding of the object. By checking the size before and after, the size change can still be detected\n            //and the listener can be called accordingly.\n            var preWidth = element.offsetWidth;\n            var preHeight = element.offsetHeight;\n\n            if(!elementUtils.isDetectable(element)) {\n                //The element is not prepared to be detectable, so do prepare it and add a listener to it.\n                return elementUtils.makeDetectable(reporter, element, function(element) {\n                    elementUtils.addListener(element, onResizeCallback);\n                    onElementReadyToAddListener(callOnAdd, element, listener);\n\n                    //Only here the uncaught resize may occur (since this code is async).\n                    //Check if the size is the same as when adding the listener.\n                    var postWidth = element.offsetWidth;\n                    var postHeight = element.offsetHeight;\n\n                    //If callOnAdd is true, then the listener will have been called either way, so no need to call the listener manually then.\n                    if(!callOnAdd && (preWidth !== postWidth || preHeight !== postHeight)) {\n                        //The element was changed while the object was being added. Call the listener.\n                        listener(element);\n                    }\n                });\n            }\n            \n            //The element has been prepared to be detectable and is ready to be listened to.\n            onElementReadyToAddListener(callOnAdd, element, listener);\n        });\n    }\n\n    return {\n        listenTo: listenTo\n    };\n};\n\nfunction getOption(options, name, defaultValue) {\n    var value = options[name];\n\n    if((value === undefined || value === null) && defaultValue !== undefined) {\n        return defaultValue;\n    }\n\n    return value;\n}\n","\"use strict\";\n\nvar forEach = require(\"./collection-utils\").forEach;\nvar browserDetector = require(\"./browser-detector\");\n\nmodule.exports = function(idHandler) {\n    /**\n     * Tells if the element has been made detectable and ready to be listened for resize events.\n     * @public\n     * @param {element} The element to check.\n     * @returns {boolean} True or false depending on if the element is detectable or not.\n     */\n    function isDetectable(element) {\n        if(browserDetector.isIE(8)) {\n            //IE 8 does not use the object method.\n            //Check only if the element has been given an id.\n            return !!idHandler.get(element);\n        }\n\n        return !!getObject(element);\n    }\n\n    /**\n     * Adds a resize event listener to the element.\n     * @public\n     * @param {element} element The element that should have the listener added.\n     * @param {function} listener The listener callback to be called for each resize event of the element. The element will be given as a parameter to the listener callback.\n     */\n    function addListener(element, listener) {\n        if(!isDetectable(element)) {\n            throw new Error(\"Element is not detectable.\");\n        }\n\n        function listenerProxy() {\n            listener(element);\n        }\n\n        if(browserDetector.isIE(8)) {\n            //IE 8 does not support object, but supports the resize event directly on elements.\n            element.attachEvent(\"onresize\", listenerProxy);\n        } else {\n            var object = getObject(element);\n            object.contentDocument.defaultView.addEventListener(\"resize\", listenerProxy);\n        }\n    }\n\n    /**\n     * Makes an element detectable and ready to be listened for resize events. Will call the callback when the element is ready to be listened for resize changes.\n     * @private\n     * @param {element} element The element to make detectable\n     * @param {function} callback The callback to be called when the element is ready to be listened for resize changes. Will be called with the element as first parameter.\n     */\n    function makeDetectable(reporter, element, callback) {\n        function injectObject(id, element, callback) {\n            var OBJECT_STYLE = \"display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; border: none; padding: 0; margin: 0; opacity: 0; z-index: -1000; pointer-events: none;\";\n\n            function onObjectLoad() {\n                /*jshint validthis:true */\n\n                //Create the style element to be added to the object.\n                var objectDocument = this.contentDocument;\n                var style = objectDocument.createElement(\"style\");\n                style.innerHTML = \"html, body { margin: 0; padding: 0 } div { -webkit-transition: opacity 0.01s; -ms-transition: opacity 0.01s; -o-transition: opacity 0.01s; transition: opacity 0.01s; opacity: 0; }\";\n\n                //TODO: Remove any styles that has been set on the object. Only the style above should be styling the object.\n\n                //Append the style to the object.\n                objectDocument.head.appendChild(style);\n\n                //TODO: Is this needed here?\n                //this.style.cssText = OBJECT_STYLE;\n\n                //Notify that the element is ready to be listened to.\n                callback(element);\n            }\n\n            //The target element needs to be positioned (everything except static) so the absolute positioned object will be positioned relative to the target element.\n            var style = getComputedStyle(element);\n            if(style.position === \"static\") {\n                element.style.position = \"relative\";\n\n                var removeRelativeStyles = function(reporter, element, style, property) {\n                    function getNumericalValue(value) {\n                        return value.replace(/[^-\\d\\.]/g, \"\");\n                    }\n\n                    var value = style[property];\n\n                    if(value !== \"auto\" && getNumericalValue(value) !== \"0\") {\n                        reporter.warn(\"An element that is positioned static has style.\" + property + \"=\" + value + \" which is ignored due to the static positioning. The element will need to be positioned relateive, so the style.\" + property + \" will be set to 0. Element: \", element);\n                        element.style[property] = 0;\n                    }\n                };\n\n                //Check so that there are no accidental styles that will make the element styled differently now that is is relative.\n                //If there are any, set them to 0 (this should be okay with the user since the style properties did nothing before [since the element was positioned static] anyway).\n                removeRelativeStyles(reporter, element, style, \"top\");\n                removeRelativeStyles(reporter, element, style, \"right\");\n                removeRelativeStyles(reporter, element, style, \"bottom\");\n                removeRelativeStyles(reporter, element, style, \"left\");\n            }\n\n            //Add an object element as a child to the target element that will be listened to for resize events.\n            var object = document.createElement(\"object\");\n            object.type = \"text/html\";\n            object.style.cssText = OBJECT_STYLE;\n            object.onload = onObjectLoad;\n            object.setAttribute(\"erd-object-id\", id);\n\n            //Safari: This must occur before adding the object to the DOM.\n            //IE: Does not like that this happens before, even if it is also added after.\n            if(!browserDetector.isIE()) {\n                object.data = \"about:blank\";\n            }\n\n            element.appendChild(object);\n\n            //IE: This must occur after adding the object to the DOM.\n            if(browserDetector.isIE()) {\n                object.data = \"about:blank\";\n            }\n        }\n\n        //Create an unique erd-target-id for the target element, so that event listeners can be identified to this element.\n        var id = idHandler.set(element);\n\n        if(browserDetector.isIE(8)) {\n            //IE 8 does not support objects properly. Luckily they do support the resize event.\n            //So do not inject the object and notify that the element is already ready to be listened to.\n            //The event handler for the resize event is attached in the utils.addListener instead.\n            callback(element);\n        } else {\n            injectObject(id, element, callback);\n        }\n    }\n\n    /**\n     * Returns the child object of the target element.\n     * @private\n     * @param {element} element The target element.\n     * @returns The object element of the target.\n     */\n    function getObject(element) {\n        return forEach(element.children, function(child) {\n            if(child.hasAttribute(\"erd-object-id\")) {\n                return child;\n            }\n        });\n    }\n\n    return {\n        isDetectable: isDetectable,\n        makeDetectable: makeDetectable,\n        addListener: addListener,\n    };\n};\n","\"use strict\";\n\nmodule.exports = function() {\n    var idCount = 1;\n\n    /**\n     * Generates a new unique id in the context.\n     * @public\n     * @returns {number} A unique id in the context.\n     */\n    function generate() {\n        return idCount++;\n    }\n\n    return {\n        generate: generate\n    };\n};\n","\"use strict\";\n\nmodule.exports = function(idGenerator) {\n\n    /**\n     * Gets the resize detector id of the element.\n     * @public\n     * @param {element} The target element to get the id of.\n     * @returns {string|number} The id of the element.\n     */\n    function getId(element) {\n        return element.getAttribute(\"erd-target-id\");\n    }\n\n    /**\n     * Sets the resize detector id of the element.\n     * @public\n     * @param {element} The target element to set the id to.\n     * @param {string?} An optional id to set to the element. If not specified, an id will be generated. All id's must be unique.\n     * @returns {string|number} The id of the element.\n     */\n    function setId(element, id) {\n        if(!id && id !== 0) {\n            //Number should be generated.\n            id = idGenerator.generate();\n        }\n\n        element.setAttribute(\"erd-target-id\", id);\n\n        return id;\n    }\n\n    return {\n        get: getId,\n        set: setId\n    };\n};\n","\"use strict\";\n\nmodule.exports = function(idHandler) {\n    var eventListeners = {};\n\n    /**\n     * Gets all listeners for the given element.\n     * @public\n     * @param {element} element The element to get all listeners for.\n     * @returns All listeners for the given element.\n     */\n    function getListeners(element) {\n        return eventListeners[idHandler.get(element)];\n    }\n\n    /**\n     * Stores the given listener for the given element. Will not actually add the listener to the element.\n     * @public\n     * @param {element} element The element that should have the listener added.\n     * @param {function} listener The callback that the element has added.\n     */\n    function addListener(element, listener) {\n        var id = idHandler.get(element);\n\n        if(!eventListeners[id]) {\n            eventListeners[id] = [];\n        }\n\n        eventListeners[id].push(listener);\n    }\n\n    return {\n        get: getListeners,\n        add: addListener\n    };\n};\n","\"use strict\";\n\n/* global console: false */\n\n/**\n * Reporter that handles the reporting of logs, warnings and errors.\n * @public\n * @param {boolean} quiet Tells if the reporter should be quiet or not.\n */\nmodule.exports = function(quiet) {\n    var noop = function () {\n        //Does nothing.\n    };\n\n    var reporter = {\n        log: noop,\n        warn: noop,\n        error: noop\n    };\n\n    if(!quiet && window.console) {\n        reporter.log = console.log;\n        reporter.warn = console.warn;\n        reporter.error = console.error;\n    }\n\n    return reporter;\n};"]} |
/*! | ||
* element-resize-detector 0.1.4 (2015-02-02, 12:04) | ||
* element-resize-detector 0.2.1 (2015-03-23, 19:27) | ||
* https://github.com/wnr/element-resize-detector | ||
@@ -7,5 +7,42 @@ * Licensed under MIT | ||
!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var f;"undefined"!=typeof window?f=window:"undefined"!=typeof global?f=global:"undefined"!=typeof self&&(f=self),f.elementResizeDetectorMaker=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){ | ||
(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.elementResizeDetectorMaker = f()}})(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){ | ||
"use strict"; | ||
var detector = module.exports = {}; | ||
detector.isIE = function(version) { | ||
function isAnyIeVersion() { | ||
var agent = navigator.userAgent.toLowerCase(); | ||
return agent.indexOf("msie") !== -1 || agent.indexOf("trident") !== -1; | ||
} | ||
if(!isAnyIeVersion()) { | ||
return false; | ||
} | ||
if(!version) { | ||
return true; | ||
} | ||
//Shamelessly stolen from https://gist.github.com/padolsey/527683 | ||
var ieVersion = (function(){ | ||
var undef, | ||
v = 3, | ||
div = document.createElement("div"), | ||
all = div.getElementsByTagName("i"); | ||
do { | ||
div.innerHTML = "<!--[if gt IE " + (++v) + "]><i></i><![endif]-->"; | ||
} | ||
while (all[0]); | ||
return v > 4 ? v : undef; | ||
}()); | ||
return version === ieVersion; | ||
}; | ||
},{}],2:[function(require,module,exports){ | ||
"use strict"; | ||
var utils = module.exports = {}; | ||
@@ -29,3 +66,3 @@ | ||
},{}],2:[function(require,module,exports){ | ||
},{}],3:[function(require,module,exports){ | ||
//Heavily inspired by http://www.backalleycoder.com/2013/03/18/cross-browser-event-based-element-resize-detection/ | ||
@@ -36,29 +73,100 @@ | ||
var forEach = require("./collection-utils").forEach; | ||
var elementUtils = require("./element-utils"); | ||
var elementUtilsMaker = require("./element-utils"); | ||
var listenerHandlerMaker = require("./listener-handler"); | ||
var idGeneratorMaker = require("./id-generator"); | ||
var listenerHandlerMaker = require("./listener-handler"); | ||
var idHandlerMaker = require("./id-handler"); | ||
var reporterMaker = require("./reporter"); | ||
/** | ||
* @typedef idHandler | ||
* @type {object} | ||
* @property {function} get Gets the resize detector id of the element. | ||
* @property {function} set Generate and sets the resize detector id of the element. | ||
*/ | ||
/** | ||
* @typedef Options | ||
* @type {object} | ||
* @property {boolean} callOnAdd Determines if listeners should be called when they are getting added. | ||
Default is true. If true, the listener is guaranteed to be called when it has been added. | ||
If false, the listener will not be guarenteed to be called when it has been added (does not prevent it from being called). | ||
* @property {idHandler} idHandler A custom id handler that is responsible for generating, setting and retrieving id's for elements. | ||
If not provided, a default id handler will be used. | ||
* @property {reporter} reporter A custom reporter that handles reporting logs, warnings and errors. | ||
If not provided, a default id handler will be used. | ||
If set to false, then nothing will be reported. | ||
*/ | ||
/** | ||
* Creates an element resize detector instance. | ||
* @public | ||
* @param {Options?} options Optional global options object that will decide how this instance will work. | ||
*/ | ||
module.exports = function(options) { | ||
options = options || {}; | ||
var allowMultipleListeners = options.allowMultipleListeners === undefined ? true : false; | ||
var eventListenerHandler = listenerHandlerMaker(); | ||
var idGenerator = idGeneratorMaker(); | ||
//Options to be used as default for the listenTo function. | ||
var globalOptions = {}; | ||
globalOptions.callOnAdd = !!getOption(options, "callOnAdd", true); | ||
//idHandler is currently not an option to the listenTo function, so it should not be added to globalOptions. | ||
var idHandler = options.idHandler; | ||
if(!idHandler) { | ||
var idGenerator = idGeneratorMaker(); | ||
var defaultIdHandler = idHandlerMaker(idGenerator); | ||
idHandler = defaultIdHandler; | ||
} | ||
//reporter is currently not an option to the listenTo function, so it should not be added to globalOptions. | ||
var reporter = options.reporter; | ||
if(!reporter) { | ||
//If options.reporter is false, then the reporter should be quiet. | ||
var quiet = reporter === false; | ||
reporter = reporterMaker(quiet); | ||
} | ||
var eventListenerHandler = listenerHandlerMaker(idHandler); | ||
var elementUtils = elementUtilsMaker(idHandler); | ||
/** | ||
* Makes the given elements resize-detectable and starts listening to resize events on the elements. Calls the event callback for each event for each element. | ||
* @public | ||
* @param {Options?} options Optional options object. These options will override the global options. Some options may not be overriden, such as idHandler. | ||
* @param {element[]|element} elements The given array of elements to detect resize events of. Single element is also valid. | ||
* @param {function} listener The callback to be executed for each resize event for each element. | ||
*/ | ||
function listenTo(elements, listener) { | ||
function isListenedTo(element) { | ||
return elementUtils.isDetectable(element) && eventListenerHandler.get(element).length; | ||
function listenTo(options, elements, listener) { | ||
function onResizeCallback(element) { | ||
var listeners = eventListenerHandler.get(element); | ||
forEach(listeners, function(listener) { | ||
listener(element); | ||
}); | ||
} | ||
function addListener(element, listener) { | ||
elementUtils.addListener(element, listener); | ||
function onElementReadyToAddListener(callOnAdd, element, listener) { | ||
eventListenerHandler.add(element, listener); | ||
if(callOnAdd) { | ||
listener(element); | ||
} | ||
} | ||
//Options object may be omitted. | ||
if(!listener) { | ||
listener = elements; | ||
elements = options; | ||
options = {}; | ||
} | ||
if(!elements) { | ||
throw new Error("At least one element required."); | ||
} | ||
if(!listener) { | ||
throw new Error("Listener required."); | ||
} | ||
if(elements.length === undefined) { | ||
@@ -68,8 +176,27 @@ elements = [elements]; | ||
var callOnAdd = getOption(options, "callOnAdd", globalOptions.callOnAdd); | ||
forEach(elements, function(element) { | ||
//The element may change size directly after the call to listenTo, which would be unable to detect it because | ||
//the async adding of the object. By checking the size before and after, the size change can still be detected | ||
//and the listener can be called accordingly. | ||
var preWidth = element.offsetWidth; | ||
var preHeight = element.offsetHeight; | ||
if(!elementUtils.isDetectable(element)) { | ||
//The element is not prepared to be detectable, so do prepare it and add a listener to it. | ||
var id = idGenerator.newId(); | ||
return elementUtils.makeDetectable(element, id, function(element) { | ||
addListener(element, listener); | ||
return elementUtils.makeDetectable(reporter, element, function(element) { | ||
elementUtils.addListener(element, onResizeCallback); | ||
onElementReadyToAddListener(callOnAdd, element, listener); | ||
//Only here the uncaught resize may occur (since this code is async). | ||
//Check if the size is the same as when adding the listener. | ||
var postWidth = element.offsetWidth; | ||
var postHeight = element.offsetHeight; | ||
//If callOnAdd is true, then the listener will have been called either way, so no need to call the listener manually then. | ||
if(!callOnAdd && (preWidth !== postWidth || preHeight !== postHeight)) { | ||
//The element was changed while the object was being added. Call the listener. | ||
listener(element); | ||
} | ||
}); | ||
@@ -79,10 +206,3 @@ } | ||
//The element has been prepared to be detectable and is ready to be listened to. | ||
if(isListenedTo(element) && !allowMultipleListeners) { | ||
//Since there is a listener and we disallow multiple listeners no listener should be added. | ||
return; | ||
} | ||
//Since multiple listeners is allowed, another listener is added to the element. | ||
return addListener(element, listener); | ||
onElementReadyToAddListener(callOnAdd, element, listener); | ||
}); | ||
@@ -96,105 +216,171 @@ } | ||
},{"./collection-utils":1,"./element-utils":3,"./id-generator":4,"./listener-handler":5}],3:[function(require,module,exports){ | ||
function getOption(options, name, defaultValue) { | ||
var value = options[name]; | ||
if((value === undefined || value === null) && defaultValue !== undefined) { | ||
return defaultValue; | ||
} | ||
return value; | ||
} | ||
},{"./collection-utils":2,"./element-utils":4,"./id-generator":5,"./id-handler":6,"./listener-handler":7,"./reporter":8}],4:[function(require,module,exports){ | ||
"use strict"; | ||
var forEach = require("./collection-utils").forEach; | ||
var browserDetector = require("./browser-detector"); | ||
var utils = module.exports = {}; | ||
module.exports = function(idHandler) { | ||
/** | ||
* Tells if the element has been made detectable and ready to be listened for resize events. | ||
* @public | ||
* @param {element} The element to check. | ||
* @returns {boolean} True or false depending on if the element is detectable or not. | ||
*/ | ||
function isDetectable(element) { | ||
if(browserDetector.isIE(8)) { | ||
//IE 8 does not use the object method. | ||
//Check only if the element has been given an id. | ||
return !!idHandler.get(element); | ||
} | ||
/** | ||
* Gets the elq id of the element. | ||
* @public | ||
* @param {element} The target element to get the id of. | ||
* @returns {string} The id of the element. | ||
*/ | ||
utils.getId = function(element) { | ||
return element.getAttribute("elq-target-id"); | ||
}; | ||
return !!getObject(element); | ||
} | ||
/** | ||
* Tells if the element has been made detectable and ready to be listened for resize events. | ||
* @public | ||
* @param {element} The element to check. | ||
* @returns {boolean} True or false depending on if the element is detectable or not. | ||
*/ | ||
utils.isDetectable = function(element) { | ||
return getObject(element); | ||
}; | ||
/** | ||
* Adds a resize event listener to the element. | ||
* @public | ||
* @param {element} element The element that should have the listener added. | ||
* @param {function} listener The listener callback to be called for each resize event of the element. The element will be given as a parameter to the listener callback. | ||
*/ | ||
function addListener(element, listener) { | ||
if(!isDetectable(element)) { | ||
throw new Error("Element is not detectable."); | ||
} | ||
/** | ||
* Adds a resize event listener to the element. | ||
* @public | ||
* @param {element} element The element that should have the listener added. | ||
* @param {function} listener The listener callback to be called for each resize event of the element. The element will be given as a parameter to the listener callback. | ||
*/ | ||
utils.addListener = function(element, listener) { | ||
if(!utils.isDetectable(element)) { | ||
throw new Error("Element is not detectable."); | ||
function listenerProxy() { | ||
listener(element); | ||
} | ||
if(browserDetector.isIE(8)) { | ||
//IE 8 does not support object, but supports the resize event directly on elements. | ||
element.attachEvent("onresize", listenerProxy); | ||
} else { | ||
var object = getObject(element); | ||
object.contentDocument.defaultView.addEventListener("resize", listenerProxy); | ||
} | ||
} | ||
var object = getObject(element); | ||
object.contentDocument.defaultView.addEventListener("resize", function() { | ||
listener(element); | ||
}); | ||
}; | ||
/** | ||
* Makes an element detectable and ready to be listened for resize events. Will call the callback when the element is ready to be listened for resize changes. | ||
* @private | ||
* @param {element} element The element to make detectable | ||
* @param {function} callback The callback to be called when the element is ready to be listened for resize changes. Will be called with the element as first parameter. | ||
*/ | ||
function makeDetectable(reporter, element, callback) { | ||
function injectObject(id, element, callback) { | ||
var OBJECT_STYLE = "display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; border: none; padding: 0; margin: 0; opacity: 0; z-index: -1000; pointer-events: none;"; | ||
/** | ||
* Makes an element detectable and ready to be listened for resize events. Will call the callback when the element is ready to be listened for resize changes. | ||
* @private | ||
* @param {element} element The element to make detectable | ||
* @param {*} id An unique id in the context of all detectable elements. | ||
* @param {function} callback The callback to be called when the element is ready to be listened for resize changes. Will be called with the element as first parameter. | ||
*/ | ||
utils.makeDetectable = function(element, id, callback) { | ||
function onObjectLoad() { | ||
/*jshint validthis:true */ | ||
function onObjectLoad() { | ||
/*jshint validthis:true */ | ||
//Create the style element to be added to the object. | ||
var objectDocument = this.contentDocument; | ||
var style = objectDocument.createElement("style"); | ||
style.innerHTML = "html, body { margin: 0; padding: 0 } div { -webkit-transition: opacity 0.01s; -ms-transition: opacity 0.01s; -o-transition: opacity 0.01s; transition: opacity 0.01s; opacity: 0; }"; | ||
//Create the style element to be added to the object. | ||
var objectDocument = this.contentDocument; | ||
var style = objectDocument.createElement("style"); | ||
style.innerHTML = "html, body { margin: 0; padding: 0 } div { -webkit-transition: opacity 0.01s; -ms-transition: opacity 0.01s; -o-transition: opacity 0.01s; transition: opacity 0.01s; opacity: 0; }"; | ||
//TODO: Remove any styles that has been set on the object. Only the style above should be styling the object. | ||
//TODO: Remove any styles that has been set on the object. Only the style above should be styling the object. | ||
//Append the style to the object. | ||
objectDocument.head.appendChild(style); | ||
//Append the style to the object. | ||
objectDocument.head.appendChild(style); | ||
this.style.cssText = "display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; border: none; padding: 0; margin: 0; opacity: 0; z-index: -1000; pointer-events: none;"; | ||
//TODO: Is this needed here? | ||
//this.style.cssText = OBJECT_STYLE; | ||
//Notify that the element is ready to be listened to. | ||
callback(element); | ||
} | ||
//Notify that the element is ready to be listened to. | ||
callback(element); | ||
} | ||
//Create an unique elq-target-id for the target element, so that event listeners can be identified to this element. | ||
element.setAttribute("elq-target-id", id); | ||
//The target element needs to be positioned (everything except static) so the absolute positioned object will be positioned relative to the target element. | ||
var style = getComputedStyle(element); | ||
if(style.position === "static") { | ||
element.style.position = "relative"; | ||
//The target element needs to be positioned (everything except static) so the absolute positioned object will be positioned relative to the target element. | ||
if(getComputedStyle(element).position === "static") { | ||
element.style.position = "relative"; | ||
var removeRelativeStyles = function(reporter, element, style, property) { | ||
function getNumericalValue(value) { | ||
return value.replace(/[^-\d\.]/g, ""); | ||
} | ||
var value = style[property]; | ||
if(value !== "auto" && getNumericalValue(value) !== "0") { | ||
reporter.warn("An element that is positioned static has style." + property + "=" + value + " which is ignored due to the static positioning. The element will need to be positioned relateive, so the style." + property + " will be set to 0. Element: ", element); | ||
element.style[property] = 0; | ||
} | ||
}; | ||
//Check so that there are no accidental styles that will make the element styled differently now that is is relative. | ||
//If there are any, set them to 0 (this should be okay with the user since the style properties did nothing before [since the element was positioned static] anyway). | ||
removeRelativeStyles(reporter, element, style, "top"); | ||
removeRelativeStyles(reporter, element, style, "right"); | ||
removeRelativeStyles(reporter, element, style, "bottom"); | ||
removeRelativeStyles(reporter, element, style, "left"); | ||
} | ||
//Add an object element as a child to the target element that will be listened to for resize events. | ||
var object = document.createElement("object"); | ||
object.type = "text/html"; | ||
object.style.cssText = OBJECT_STYLE; | ||
object.onload = onObjectLoad; | ||
object.setAttribute("erd-object-id", id); | ||
//Safari: This must occur before adding the object to the DOM. | ||
//IE: Does not like that this happens before, even if it is also added after. | ||
if(!browserDetector.isIE()) { | ||
object.data = "about:blank"; | ||
} | ||
element.appendChild(object); | ||
//IE: This must occur after adding the object to the DOM. | ||
if(browserDetector.isIE()) { | ||
object.data = "about:blank"; | ||
} | ||
} | ||
//Create an unique erd-target-id for the target element, so that event listeners can be identified to this element. | ||
var id = idHandler.set(element); | ||
if(browserDetector.isIE(8)) { | ||
//IE 8 does not support objects properly. Luckily they do support the resize event. | ||
//So do not inject the object and notify that the element is already ready to be listened to. | ||
//The event handler for the resize event is attached in the utils.addListener instead. | ||
callback(element); | ||
} else { | ||
injectObject(id, element, callback); | ||
} | ||
} | ||
//Add an object element as a child to the target element that will be listened to for resize events. | ||
var object = document.createElement("object"); | ||
object.type = "text/html"; | ||
object.data = "about:blank"; | ||
object.onload = onObjectLoad; | ||
object.setAttribute("elq-object-id", id); | ||
element.appendChild(object); | ||
/** | ||
* Returns the child object of the target element. | ||
* @private | ||
* @param {element} element The target element. | ||
* @returns The object element of the target. | ||
*/ | ||
function getObject(element) { | ||
return forEach(element.children, function(child) { | ||
if(child.hasAttribute("erd-object-id")) { | ||
return child; | ||
} | ||
}); | ||
} | ||
return { | ||
isDetectable: isDetectable, | ||
makeDetectable: makeDetectable, | ||
addListener: addListener, | ||
}; | ||
}; | ||
/** | ||
* Returns the child object of the target element. | ||
* @private | ||
* @param {element} element The target element. | ||
* @returns The object element of the target. | ||
*/ | ||
function getObject(element) { | ||
return forEach(element.children, function(child) { | ||
if(child.hasAttribute("elq-object-id")) { | ||
return child; | ||
} | ||
}); | ||
} | ||
},{"./collection-utils":1}],4:[function(require,module,exports){ | ||
},{"./browser-detector":1,"./collection-utils":2}],5:[function(require,module,exports){ | ||
"use strict"; | ||
@@ -210,3 +396,3 @@ | ||
*/ | ||
function newId() { | ||
function generate() { | ||
return idCount++; | ||
@@ -216,12 +402,49 @@ } | ||
return { | ||
newId: newId | ||
generate: generate | ||
}; | ||
}; | ||
},{}],5:[function(require,module,exports){ | ||
},{}],6:[function(require,module,exports){ | ||
"use strict"; | ||
var elementUtils = require("./element-utils"); | ||
module.exports = function(idGenerator) { | ||
module.exports = function() { | ||
/** | ||
* Gets the resize detector id of the element. | ||
* @public | ||
* @param {element} The target element to get the id of. | ||
* @returns {string|number} The id of the element. | ||
*/ | ||
function getId(element) { | ||
return element.getAttribute("erd-target-id"); | ||
} | ||
/** | ||
* Sets the resize detector id of the element. | ||
* @public | ||
* @param {element} The target element to set the id to. | ||
* @param {string?} An optional id to set to the element. If not specified, an id will be generated. All id's must be unique. | ||
* @returns {string|number} The id of the element. | ||
*/ | ||
function setId(element, id) { | ||
if(!id && id !== 0) { | ||
//Number should be generated. | ||
id = idGenerator.generate(); | ||
} | ||
element.setAttribute("erd-target-id", id); | ||
return id; | ||
} | ||
return { | ||
get: getId, | ||
set: setId | ||
}; | ||
}; | ||
},{}],7:[function(require,module,exports){ | ||
"use strict"; | ||
module.exports = function(idHandler) { | ||
var eventListeners = {}; | ||
@@ -236,3 +459,3 @@ | ||
function getListeners(element) { | ||
return eventListeners[elementUtils.getId(element)]; | ||
return eventListeners[idHandler.get(element)]; | ||
} | ||
@@ -247,3 +470,3 @@ | ||
function addListener(element, listener) { | ||
var id = elementUtils.getId(element); | ||
var id = idHandler.get(element); | ||
@@ -263,3 +486,32 @@ if(!eventListeners[id]) { | ||
},{"./element-utils":3}]},{},[2])(2) | ||
},{}],8:[function(require,module,exports){ | ||
"use strict"; | ||
/* global console: false */ | ||
/** | ||
* Reporter that handles the reporting of logs, warnings and errors. | ||
* @public | ||
* @param {boolean} quiet Tells if the reporter should be quiet or not. | ||
*/ | ||
module.exports = function(quiet) { | ||
var noop = function () { | ||
//Does nothing. | ||
}; | ||
var reporter = { | ||
log: noop, | ||
warn: noop, | ||
error: noop | ||
}; | ||
if(!quiet && window.console) { | ||
reporter.log = console.log; | ||
reporter.warn = console.warn; | ||
reporter.error = console.error; | ||
} | ||
return reporter; | ||
}; | ||
},{}]},{},[3])(3) | ||
}); |
/*! | ||
* element-resize-detector 0.1.4 (2015-02-02, 12:04) | ||
* element-resize-detector 0.2.1 (2015-03-23, 19:27) | ||
* https://github.com/wnr/element-resize-detector | ||
@@ -7,2 +7,2 @@ * Licensed under MIT | ||
!function(a){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=a();else if("function"==typeof define&&define.amd)define([],a);else{var b;"undefined"!=typeof window?b=window:"undefined"!=typeof global?b=global:"undefined"!=typeof self&&(b=self),b.elementResizeDetectorMaker=a()}}(function(){return function a(b,c,d){function e(g,h){if(!c[g]){if(!b[g]){var i="function"==typeof require&&require;if(!h&&i)return i(g,!0);if(f)return f(g,!0);var j=new Error("Cannot find module '"+g+"'");throw j.code="MODULE_NOT_FOUND",j}var k=c[g]={exports:{}};b[g][0].call(k.exports,function(a){var c=b[g][1][a];return e(c?c:a)},k,k.exports,a,b,c,d)}return c[g].exports}for(var f="function"==typeof require&&require,g=0;g<d.length;g++)e(d[g]);return e}({1:[function(a,b){"use strict";var c=b.exports={};c.forEach=function(a,b){for(var c=0;c<a.length;c++){var d=b(a[c]);if(d)return d}}},{}],2:[function(a,b){"use strict";var c=a("./collection-utils").forEach,d=a("./element-utils"),e=a("./id-generator"),f=a("./listener-handler");b.exports=function(a){function b(a,b){function e(a){return d.isDetectable(a)&&h.get(a).length}function f(a,b){d.addListener(a,b),h.add(a,b)}void 0===a.length&&(a=[a]),c(a,function(a){if(!d.isDetectable(a)){var c=i.newId();return d.makeDetectable(a,c,function(a){f(a,b)})}if(!e(a)||g)return f(a,b)})}a=a||{};var g=void 0===a.allowMultipleListeners?!0:!1,h=f(),i=e();return{listenTo:b}}},{"./collection-utils":1,"./element-utils":3,"./id-generator":4,"./listener-handler":5}],3:[function(a,b){"use strict";function c(a){return d(a.children,function(a){return a.hasAttribute("elq-object-id")?a:void 0})}var d=a("./collection-utils").forEach,e=b.exports={};e.getId=function(a){return a.getAttribute("elq-target-id")},e.isDetectable=function(a){return c(a)},e.addListener=function(a,b){if(!e.isDetectable(a))throw new Error("Element is not detectable.");var d=c(a);d.contentDocument.defaultView.addEventListener("resize",function(){b(a)})},e.makeDetectable=function(a,b,c){function d(){var b=this.contentDocument,d=b.createElement("style");d.innerHTML="html, body { margin: 0; padding: 0 } div { -webkit-transition: opacity 0.01s; -ms-transition: opacity 0.01s; -o-transition: opacity 0.01s; transition: opacity 0.01s; opacity: 0; }",b.head.appendChild(d),this.style.cssText="display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; border: none; padding: 0; margin: 0; opacity: 0; z-index: -1000; pointer-events: none;",c(a)}a.setAttribute("elq-target-id",b),"static"===getComputedStyle(a).position&&(a.style.position="relative");var e=document.createElement("object");e.type="text/html",e.data="about:blank",e.onload=d,e.setAttribute("elq-object-id",b),a.appendChild(e)}},{"./collection-utils":1}],4:[function(a,b){"use strict";b.exports=function(){function a(){return b++}var b=1;return{newId:a}}},{}],5:[function(a,b){"use strict";var c=a("./element-utils");b.exports=function(){function a(a){return d[c.getId(a)]}function b(a,b){var e=c.getId(a);d[e]||(d[e]=[]),d[e].push(b)}var d={};return{get:a,add:b}}},{"./element-utils":3}]},{},[2])(2)}); | ||
!function(a){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=a();else if("function"==typeof define&&define.amd)define([],a);else{var b;b="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:this,b.elementResizeDetectorMaker=a()}}(function(){return function a(b,c,d){function e(g,h){if(!c[g]){if(!b[g]){var i="function"==typeof require&&require;if(!h&&i)return i(g,!0);if(f)return f(g,!0);var j=new Error("Cannot find module '"+g+"'");throw j.code="MODULE_NOT_FOUND",j}var k=c[g]={exports:{}};b[g][0].call(k.exports,function(a){var c=b[g][1][a];return e(c?c:a)},k,k.exports,a,b,c,d)}return c[g].exports}for(var f="function"==typeof require&&require,g=0;g<d.length;g++)e(d[g]);return e}({1:[function(a,b){"use strict";var c=b.exports={};c.isIE=function(a){function b(){var a=navigator.userAgent.toLowerCase();return-1!==a.indexOf("msie")||-1!==a.indexOf("trident")}if(!b())return!1;if(!a)return!0;var c=function(){var a,b=3,c=document.createElement("div"),d=c.getElementsByTagName("i");do c.innerHTML="<!--[if gt IE "+ ++b+"]><i></i><![endif]-->";while(d[0]);return b>4?b:a}();return a===c}},{}],2:[function(a,b){"use strict";var c=b.exports={};c.forEach=function(a,b){for(var c=0;c<a.length;c++){var d=b(a[c]);if(d)return d}}},{}],3:[function(a,b){"use strict";function c(a,b,c){var d=a[b];return void 0!==d&&null!==d||void 0===c?d:c}var d=a("./collection-utils").forEach,e=a("./element-utils"),f=a("./listener-handler"),g=a("./id-generator"),h=a("./id-handler"),i=a("./reporter");b.exports=function(a){function b(a,b,e){function f(a){var b=p.get(a);d(b,function(b){b(a)})}function g(a,b,c){p.add(b,c),a&&c(b)}if(e||(e=b,b=a,a={}),!b)throw new Error("At least one element required.");if(!e)throw new Error("Listener required.");void 0===b.length&&(b=[b]);var h=c(a,"callOnAdd",j.callOnAdd);d(b,function(a){var b=a.offsetWidth,c=a.offsetHeight;return q.isDetectable(a)?void g(h,a,e):q.makeDetectable(n,a,function(a){q.addListener(a,f),g(h,a,e);var d=a.offsetWidth,i=a.offsetHeight;h||b===d&&c===i||e(a)})})}a=a||{};var j={};j.callOnAdd=!!c(a,"callOnAdd",!0);var k=a.idHandler;if(!k){var l=g(),m=h(l);k=m}var n=a.reporter;if(!n){var o=n===!1;n=i(o)}var p=f(k),q=e(k);return{listenTo:b}}},{"./collection-utils":2,"./element-utils":4,"./id-generator":5,"./id-handler":6,"./listener-handler":7,"./reporter":8}],4:[function(a,b){"use strict";var c=a("./collection-utils").forEach,d=a("./browser-detector");b.exports=function(a){function b(b){return d.isIE(8)?!!a.get(b):!!g(b)}function e(a,c){function e(){c(a)}if(!b(a))throw new Error("Element is not detectable.");if(d.isIE(8))a.attachEvent("onresize",e);else{var f=g(a);f.contentDocument.defaultView.addEventListener("resize",e)}}function f(b,c,e){function f(a,c,e){function f(){var a=this.contentDocument,b=a.createElement("style");b.innerHTML="html, body { margin: 0; padding: 0 } div { -webkit-transition: opacity 0.01s; -ms-transition: opacity 0.01s; -o-transition: opacity 0.01s; transition: opacity 0.01s; opacity: 0; }",a.head.appendChild(b),e(c)}var g="display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; border: none; padding: 0; margin: 0; opacity: 0; z-index: -1000; pointer-events: none;",h=getComputedStyle(c);if("static"===h.position){c.style.position="relative";var i=function(a,b,c,d){function e(a){return a.replace(/[^-\d\.]/g,"")}var f=c[d];"auto"!==f&&"0"!==e(f)&&(a.warn("An element that is positioned static has style."+d+"="+f+" which is ignored due to the static positioning. The element will need to be positioned relateive, so the style."+d+" will be set to 0. Element: ",b),b.style[d]=0)};i(b,c,h,"top"),i(b,c,h,"right"),i(b,c,h,"bottom"),i(b,c,h,"left")}var j=document.createElement("object");j.type="text/html",j.style.cssText=g,j.onload=f,j.setAttribute("erd-object-id",a),d.isIE()||(j.data="about:blank"),c.appendChild(j),d.isIE()&&(j.data="about:blank")}var g=a.set(c);d.isIE(8)?e(c):f(g,c,e)}function g(a){return c(a.children,function(a){return a.hasAttribute("erd-object-id")?a:void 0})}return{isDetectable:b,makeDetectable:f,addListener:e}}},{"./browser-detector":1,"./collection-utils":2}],5:[function(a,b){"use strict";b.exports=function(){function a(){return b++}var b=1;return{generate:a}}},{}],6:[function(a,b){"use strict";b.exports=function(a){function b(a){return a.getAttribute("erd-target-id")}function c(b,c){return c||0===c||(c=a.generate()),b.setAttribute("erd-target-id",c),c}return{get:b,set:c}}},{}],7:[function(a,b){"use strict";b.exports=function(a){function b(b){return d[a.get(b)]}function c(b,c){var e=a.get(b);d[e]||(d[e]=[]),d[e].push(c)}var d={};return{get:b,add:c}}},{}],8:[function(a,b){"use strict";b.exports=function(a){var b=function(){},c={log:b,warn:b,error:b};return!a&&window.console&&(c.log=console.log,c.warn=console.warn,c.error=console.error),c}},{}]},{},[3])(3)}); |
151
Gruntfile.js
@@ -0,3 +1,46 @@ | ||
/* global process:false */ | ||
"use strict"; | ||
var _ = require("lodash"); | ||
var sauceConnectLauncher = require("sauce-connect-launcher"); | ||
function registerSauceBrowsers(config, sauceBrowsers, configFile) { | ||
function capitalize(string) { | ||
if(!string.charAt) { | ||
return string; | ||
} | ||
return string.charAt(0).toUpperCase() + string.slice(1); | ||
} | ||
var karma = config.karma; | ||
var tasks = []; | ||
var formatName = function(result, part) { | ||
return result + capitalize(part); | ||
}; | ||
for(var key in sauceBrowsers) { | ||
if(sauceBrowsers.hasOwnProperty(key)) { | ||
var parts = key.toLowerCase().split("_"); | ||
var name = _.reduce(parts, formatName, "sauce"); | ||
var configObject = { | ||
configFile: configFile, | ||
options: { | ||
browsers: sauceBrowsers[key] | ||
} | ||
}; | ||
karma[name] = configObject; | ||
tasks.push("karma:" + name); | ||
} | ||
} | ||
return tasks; | ||
} | ||
module.exports = function(grunt) { | ||
@@ -62,5 +105,53 @@ require("load-grunt-tasks")(grunt); | ||
} | ||
} | ||
}, | ||
karma: { | ||
local: { | ||
configFile: "karma.conf.js", | ||
options: { | ||
browsers: [ | ||
"Chrome", | ||
"Safari", | ||
"Firefox", | ||
//"IE8 - Win7", | ||
//"IE10 - Win7", | ||
//"IE11 - Win8.1" | ||
], | ||
singleRun: true | ||
} | ||
} | ||
}, | ||
"sauce_connect": { | ||
options: { | ||
username: process.env.SAUCE_USERNAME, | ||
accessKey: process.env.SAUCE_ACCESS_KEY, | ||
verbose: true, | ||
build: process.env.TRAVIS_BUILD_NUMBER || process.env.BUILD_NUMBER, | ||
testName: "element-resize-detector" | ||
}, | ||
tunnel: {} | ||
} | ||
}; | ||
var sauceBrowsers = [ | ||
"SL_CHROME_LATEST_OSX", "SL_CHROME_LATEST_WINDOWS", "SL_CHROME_LATEST_LINUX", | ||
"SL_FIREFOX_LATEST_OSX", "SL_FIREFOX_LATEST_WINDOWS", "SL_FIREFOX_LATEST_LINUX", | ||
"SL_SAFARI_LATEST_OSX", "SL_SAFARI_LATEST_WINDOWS", | ||
"SL_OPERA_LATEST_WINDOWS", "SL_OPERA_LATEST_LINUX", | ||
"SL_IE_LATEST_WINDOWS", "SL_IE_10_WINDOWS", "SL_IE_9_WINDOWS", "SL_IE_8_WINDOWS", | ||
"SL_IOS_LATEST_IPHONE", "SL_IOS_LATEST_IPAD", | ||
"SL_IOS_7_IPHONE", "SL_IOS_7_IPAD" | ||
]; | ||
function batchSauceBrowsers(browsers, batchSize) { | ||
var number = 1; | ||
var batchMap = {}; | ||
_.forEach(_.chunk(browsers, batchSize), function(chunk) { | ||
batchMap["sauceBrowserChunk" + number++] = chunk; | ||
}); | ||
return batchMap; | ||
} | ||
var NUM_PARALLEL_BROWSERS = 3; | ||
var sauceBrowserTasks = registerSauceBrowsers(config, batchSauceBrowsers(sauceBrowsers, NUM_PARALLEL_BROWSERS), "karma.sauce.conf.js"); | ||
grunt.initConfig(config); | ||
@@ -75,5 +166,61 @@ | ||
grunt.registerTask("test:style", ["jshint"]); | ||
grunt.registerTask("test", ["test:style", "build:dev"]); //TODO | ||
grunt.registerTask("test:sauce", ["build"].concat(sauceBrowserTasks)); | ||
grunt.registerTask("test", ["test:style", "build:dev", "karma:local"]); | ||
grunt.registerTask("ci", ["test:style", "sauceConnectTunnel", "test:sauce"]); | ||
grunt.registerTask("default", ["test"]); | ||
var sauceConnectTunnel = {}; | ||
grunt.registerTask("sauceConnectTunnel", "Starts a sauce connect tunnel", function(keepAlive) { | ||
if(!process.env.SAUCE_USERNAME) { | ||
grunt.log.error("env SAUCE_USERNAME needs to be set."); | ||
return false; | ||
} | ||
if(!process.env.SAUCE_ACCESS_KEY) { | ||
grunt.log.error("env SAUCE_ACCESS_KEY needs to be set."); | ||
return false; | ||
} | ||
var done = this.async(); | ||
sauceConnectLauncher({ | ||
username: process.env.SAUCE_USERNAME, | ||
accessKey: process.env.SAUCE_ACCESS_KEY, | ||
logger: grunt.log.writeln, | ||
verbose: true, | ||
logfile: "sauce-connect.log" | ||
}, function (err, sauceConnectProcess) { | ||
function stop() { | ||
grunt.log.writeln("Stopping..."); | ||
sauceConnectTunnel.process.close(function() { | ||
grunt.log.writeln("Closed Sauce Connect process"); | ||
done(); | ||
}); | ||
} | ||
if (err) { | ||
grunt.log.error(err.message); | ||
done(false); | ||
} | ||
sauceConnectTunnel.process = sauceConnectProcess; | ||
grunt.log.success("Sauce Connect ready!"); | ||
if(keepAlive) { | ||
grunt.log.writeln("The tunnel will be kept alive. Stop it by terminating this process with SIGINT (Ctrl-C)."); | ||
process.on("SIGINT", function() { | ||
grunt.log.writeln(); | ||
stop(); | ||
}); | ||
} else { | ||
grunt.log.writeln("Closing tunnel since the :keepAlive argument is not present..."); | ||
done(); | ||
} | ||
}); | ||
}); | ||
}; |
{ | ||
"name": "element-resize-detector", | ||
"version": "0.1.4", | ||
"version": "0.2.2", | ||
"description": "resize event emitter for elements.", | ||
@@ -17,6 +17,20 @@ "homepage": "https://github.com/wnr/element-resize-detector", | ||
"grunt-browserify": "^3.3.0", | ||
"grunt-cli": "^0.1.13", | ||
"grunt-contrib-jshint": "^0.11.0", | ||
"grunt-contrib-uglify": "^0.7.0", | ||
"jquery": "^2.1.3", | ||
"load-grunt-tasks": "^3.0.0" | ||
"grunt-karma": "^0.10.1", | ||
"jasmine-core": "^2.2.0", | ||
"jasmine-expect": "^2.0.0-beta1", | ||
"jasmine-jquery": "^2.0.6", | ||
"jquery": "^1.11.2", | ||
"karma": "^0.12.31", | ||
"karma-chrome-launcher": "^0.1.7", | ||
"karma-firefox-launcher": "^0.1.4", | ||
"karma-ievms": "0.0.4", | ||
"karma-jasmine": "^0.3.5", | ||
"karma-safari-launcher": "^0.1.1", | ||
"karma-sauce-launcher": "^0.2.10", | ||
"load-grunt-tasks": "^3.0.0", | ||
"lodash": "^3.3.1", | ||
"sauce-connect-launcher": "^0.10.1" | ||
}, | ||
@@ -26,4 +40,5 @@ "scripts": { | ||
"dist": "grunt dist", | ||
"test": "grunt test" | ||
"test": "grunt test", | ||
"test-ci": "grunt ci" | ||
} | ||
} |
@@ -28,7 +28,2 @@ # element-resize-detector | ||
var erdDefault = elementResizeDetectorMaker(); | ||
//With custom options. | ||
var erdCustom = elementResizeDetectorMaker({ | ||
allowMultipleListeners: true //Default is false, which only allows one listener function per element. | ||
}); | ||
``` | ||
@@ -35,0 +30,0 @@ |
@@ -6,29 +6,100 @@ //Heavily inspired by http://www.backalleycoder.com/2013/03/18/cross-browser-event-based-element-resize-detection/ | ||
var forEach = require("./collection-utils").forEach; | ||
var elementUtils = require("./element-utils"); | ||
var elementUtilsMaker = require("./element-utils"); | ||
var listenerHandlerMaker = require("./listener-handler"); | ||
var idGeneratorMaker = require("./id-generator"); | ||
var listenerHandlerMaker = require("./listener-handler"); | ||
var idHandlerMaker = require("./id-handler"); | ||
var reporterMaker = require("./reporter"); | ||
/** | ||
* @typedef idHandler | ||
* @type {object} | ||
* @property {function} get Gets the resize detector id of the element. | ||
* @property {function} set Generate and sets the resize detector id of the element. | ||
*/ | ||
/** | ||
* @typedef Options | ||
* @type {object} | ||
* @property {boolean} callOnAdd Determines if listeners should be called when they are getting added. | ||
Default is true. If true, the listener is guaranteed to be called when it has been added. | ||
If false, the listener will not be guarenteed to be called when it has been added (does not prevent it from being called). | ||
* @property {idHandler} idHandler A custom id handler that is responsible for generating, setting and retrieving id's for elements. | ||
If not provided, a default id handler will be used. | ||
* @property {reporter} reporter A custom reporter that handles reporting logs, warnings and errors. | ||
If not provided, a default id handler will be used. | ||
If set to false, then nothing will be reported. | ||
*/ | ||
/** | ||
* Creates an element resize detector instance. | ||
* @public | ||
* @param {Options?} options Optional global options object that will decide how this instance will work. | ||
*/ | ||
module.exports = function(options) { | ||
options = options || {}; | ||
var allowMultipleListeners = options.allowMultipleListeners === undefined ? true : false; | ||
var eventListenerHandler = listenerHandlerMaker(); | ||
var idGenerator = idGeneratorMaker(); | ||
//Options to be used as default for the listenTo function. | ||
var globalOptions = {}; | ||
globalOptions.callOnAdd = !!getOption(options, "callOnAdd", true); | ||
//idHandler is currently not an option to the listenTo function, so it should not be added to globalOptions. | ||
var idHandler = options.idHandler; | ||
if(!idHandler) { | ||
var idGenerator = idGeneratorMaker(); | ||
var defaultIdHandler = idHandlerMaker(idGenerator); | ||
idHandler = defaultIdHandler; | ||
} | ||
//reporter is currently not an option to the listenTo function, so it should not be added to globalOptions. | ||
var reporter = options.reporter; | ||
if(!reporter) { | ||
//If options.reporter is false, then the reporter should be quiet. | ||
var quiet = reporter === false; | ||
reporter = reporterMaker(quiet); | ||
} | ||
var eventListenerHandler = listenerHandlerMaker(idHandler); | ||
var elementUtils = elementUtilsMaker(idHandler); | ||
/** | ||
* Makes the given elements resize-detectable and starts listening to resize events on the elements. Calls the event callback for each event for each element. | ||
* @public | ||
* @param {Options?} options Optional options object. These options will override the global options. Some options may not be overriden, such as idHandler. | ||
* @param {element[]|element} elements The given array of elements to detect resize events of. Single element is also valid. | ||
* @param {function} listener The callback to be executed for each resize event for each element. | ||
*/ | ||
function listenTo(elements, listener) { | ||
function isListenedTo(element) { | ||
return elementUtils.isDetectable(element) && eventListenerHandler.get(element).length; | ||
function listenTo(options, elements, listener) { | ||
function onResizeCallback(element) { | ||
var listeners = eventListenerHandler.get(element); | ||
forEach(listeners, function(listener) { | ||
listener(element); | ||
}); | ||
} | ||
function addListener(element, listener) { | ||
elementUtils.addListener(element, listener); | ||
function onElementReadyToAddListener(callOnAdd, element, listener) { | ||
eventListenerHandler.add(element, listener); | ||
if(callOnAdd) { | ||
listener(element); | ||
} | ||
} | ||
//Options object may be omitted. | ||
if(!listener) { | ||
listener = elements; | ||
elements = options; | ||
options = {}; | ||
} | ||
if(!elements) { | ||
throw new Error("At least one element required."); | ||
} | ||
if(!listener) { | ||
throw new Error("Listener required."); | ||
} | ||
if(elements.length === undefined) { | ||
@@ -38,8 +109,27 @@ elements = [elements]; | ||
var callOnAdd = getOption(options, "callOnAdd", globalOptions.callOnAdd); | ||
forEach(elements, function(element) { | ||
//The element may change size directly after the call to listenTo, which would be unable to detect it because | ||
//the async adding of the object. By checking the size before and after, the size change can still be detected | ||
//and the listener can be called accordingly. | ||
var preWidth = element.offsetWidth; | ||
var preHeight = element.offsetHeight; | ||
if(!elementUtils.isDetectable(element)) { | ||
//The element is not prepared to be detectable, so do prepare it and add a listener to it. | ||
var id = idGenerator.newId(); | ||
return elementUtils.makeDetectable(element, id, function(element) { | ||
addListener(element, listener); | ||
return elementUtils.makeDetectable(reporter, element, function(element) { | ||
elementUtils.addListener(element, onResizeCallback); | ||
onElementReadyToAddListener(callOnAdd, element, listener); | ||
//Only here the uncaught resize may occur (since this code is async). | ||
//Check if the size is the same as when adding the listener. | ||
var postWidth = element.offsetWidth; | ||
var postHeight = element.offsetHeight; | ||
//If callOnAdd is true, then the listener will have been called either way, so no need to call the listener manually then. | ||
if(!callOnAdd && (preWidth !== postWidth || preHeight !== postHeight)) { | ||
//The element was changed while the object was being added. Call the listener. | ||
listener(element); | ||
} | ||
}); | ||
@@ -49,10 +139,3 @@ } | ||
//The element has been prepared to be detectable and is ready to be listened to. | ||
if(isListenedTo(element) && !allowMultipleListeners) { | ||
//Since there is a listener and we disallow multiple listeners no listener should be added. | ||
return; | ||
} | ||
//Since multiple listeners is allowed, another listener is added to the element. | ||
return addListener(element, listener); | ||
onElementReadyToAddListener(callOnAdd, element, listener); | ||
}); | ||
@@ -65,1 +148,11 @@ } | ||
}; | ||
function getOption(options, name, defaultValue) { | ||
var value = options[name]; | ||
if((value === undefined || value === null) && defaultValue !== undefined) { | ||
return defaultValue; | ||
} | ||
return value; | ||
} |
"use strict"; | ||
var forEach = require("./collection-utils").forEach; | ||
var browserDetector = require("./browser-detector"); | ||
var utils = module.exports = {}; | ||
module.exports = function(idHandler) { | ||
/** | ||
* Tells if the element has been made detectable and ready to be listened for resize events. | ||
* @public | ||
* @param {element} The element to check. | ||
* @returns {boolean} True or false depending on if the element is detectable or not. | ||
*/ | ||
function isDetectable(element) { | ||
if(browserDetector.isIE(8)) { | ||
//IE 8 does not use the object method. | ||
//Check only if the element has been given an id. | ||
return !!idHandler.get(element); | ||
} | ||
/** | ||
* Gets the elq id of the element. | ||
* @public | ||
* @param {element} The target element to get the id of. | ||
* @returns {string} The id of the element. | ||
*/ | ||
utils.getId = function(element) { | ||
return element.getAttribute("elq-target-id"); | ||
}; | ||
return !!getObject(element); | ||
} | ||
/** | ||
* Tells if the element has been made detectable and ready to be listened for resize events. | ||
* @public | ||
* @param {element} The element to check. | ||
* @returns {boolean} True or false depending on if the element is detectable or not. | ||
*/ | ||
utils.isDetectable = function(element) { | ||
return getObject(element); | ||
}; | ||
/** | ||
* Adds a resize event listener to the element. | ||
* @public | ||
* @param {element} element The element that should have the listener added. | ||
* @param {function} listener The listener callback to be called for each resize event of the element. The element will be given as a parameter to the listener callback. | ||
*/ | ||
function addListener(element, listener) { | ||
if(!isDetectable(element)) { | ||
throw new Error("Element is not detectable."); | ||
} | ||
/** | ||
* Adds a resize event listener to the element. | ||
* @public | ||
* @param {element} element The element that should have the listener added. | ||
* @param {function} listener The listener callback to be called for each resize event of the element. The element will be given as a parameter to the listener callback. | ||
*/ | ||
utils.addListener = function(element, listener) { | ||
if(!utils.isDetectable(element)) { | ||
throw new Error("Element is not detectable."); | ||
function listenerProxy() { | ||
listener(element); | ||
} | ||
if(browserDetector.isIE(8)) { | ||
//IE 8 does not support object, but supports the resize event directly on elements. | ||
element.attachEvent("onresize", listenerProxy); | ||
} else { | ||
var object = getObject(element); | ||
object.contentDocument.defaultView.addEventListener("resize", listenerProxy); | ||
} | ||
} | ||
var object = getObject(element); | ||
object.contentDocument.defaultView.addEventListener("resize", function() { | ||
listener(element); | ||
}); | ||
}; | ||
/** | ||
* Makes an element detectable and ready to be listened for resize events. Will call the callback when the element is ready to be listened for resize changes. | ||
* @private | ||
* @param {element} element The element to make detectable | ||
* @param {function} callback The callback to be called when the element is ready to be listened for resize changes. Will be called with the element as first parameter. | ||
*/ | ||
function makeDetectable(reporter, element, callback) { | ||
function injectObject(id, element, callback) { | ||
var OBJECT_STYLE = "display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; border: none; padding: 0; margin: 0; opacity: 0; z-index: -1000; pointer-events: none;"; | ||
/** | ||
* Makes an element detectable and ready to be listened for resize events. Will call the callback when the element is ready to be listened for resize changes. | ||
* @private | ||
* @param {element} element The element to make detectable | ||
* @param {*} id An unique id in the context of all detectable elements. | ||
* @param {function} callback The callback to be called when the element is ready to be listened for resize changes. Will be called with the element as first parameter. | ||
*/ | ||
utils.makeDetectable = function(element, id, callback) { | ||
function onObjectLoad() { | ||
/*jshint validthis:true */ | ||
function onObjectLoad() { | ||
/*jshint validthis:true */ | ||
//Create the style element to be added to the object. | ||
var objectDocument = this.contentDocument; | ||
var style = objectDocument.createElement("style"); | ||
style.innerHTML = "html, body { margin: 0; padding: 0 } div { -webkit-transition: opacity 0.01s; -ms-transition: opacity 0.01s; -o-transition: opacity 0.01s; transition: opacity 0.01s; opacity: 0; }"; | ||
//Create the style element to be added to the object. | ||
var objectDocument = this.contentDocument; | ||
var style = objectDocument.createElement("style"); | ||
style.innerHTML = "html, body { margin: 0; padding: 0 } div { -webkit-transition: opacity 0.01s; -ms-transition: opacity 0.01s; -o-transition: opacity 0.01s; transition: opacity 0.01s; opacity: 0; }"; | ||
//TODO: Remove any styles that has been set on the object. Only the style above should be styling the object. | ||
//TODO: Remove any styles that has been set on the object. Only the style above should be styling the object. | ||
//Append the style to the object. | ||
objectDocument.head.appendChild(style); | ||
//Append the style to the object. | ||
objectDocument.head.appendChild(style); | ||
this.style.cssText = "display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; border: none; padding: 0; margin: 0; opacity: 0; z-index: -1000; pointer-events: none;"; | ||
//TODO: Is this needed here? | ||
//this.style.cssText = OBJECT_STYLE; | ||
//Notify that the element is ready to be listened to. | ||
callback(element); | ||
} | ||
//Notify that the element is ready to be listened to. | ||
callback(element); | ||
} | ||
//Create an unique elq-target-id for the target element, so that event listeners can be identified to this element. | ||
element.setAttribute("elq-target-id", id); | ||
//The target element needs to be positioned (everything except static) so the absolute positioned object will be positioned relative to the target element. | ||
var style = getComputedStyle(element); | ||
if(style.position === "static") { | ||
element.style.position = "relative"; | ||
//The target element needs to be positioned (everything except static) so the absolute positioned object will be positioned relative to the target element. | ||
if(getComputedStyle(element).position === "static") { | ||
element.style.position = "relative"; | ||
var removeRelativeStyles = function(reporter, element, style, property) { | ||
function getNumericalValue(value) { | ||
return value.replace(/[^-\d\.]/g, ""); | ||
} | ||
var value = style[property]; | ||
if(value !== "auto" && getNumericalValue(value) !== "0") { | ||
reporter.warn("An element that is positioned static has style." + property + "=" + value + " which is ignored due to the static positioning. The element will need to be positioned relateive, so the style." + property + " will be set to 0. Element: ", element); | ||
element.style[property] = 0; | ||
} | ||
}; | ||
//Check so that there are no accidental styles that will make the element styled differently now that is is relative. | ||
//If there are any, set them to 0 (this should be okay with the user since the style properties did nothing before [since the element was positioned static] anyway). | ||
removeRelativeStyles(reporter, element, style, "top"); | ||
removeRelativeStyles(reporter, element, style, "right"); | ||
removeRelativeStyles(reporter, element, style, "bottom"); | ||
removeRelativeStyles(reporter, element, style, "left"); | ||
} | ||
//Add an object element as a child to the target element that will be listened to for resize events. | ||
var object = document.createElement("object"); | ||
object.type = "text/html"; | ||
object.style.cssText = OBJECT_STYLE; | ||
object.onload = onObjectLoad; | ||
object.setAttribute("erd-object-id", id); | ||
//Safari: This must occur before adding the object to the DOM. | ||
//IE: Does not like that this happens before, even if it is also added after. | ||
if(!browserDetector.isIE()) { | ||
object.data = "about:blank"; | ||
} | ||
element.appendChild(object); | ||
//IE: This must occur after adding the object to the DOM. | ||
if(browserDetector.isIE()) { | ||
object.data = "about:blank"; | ||
} | ||
} | ||
//Create an unique erd-target-id for the target element, so that event listeners can be identified to this element. | ||
var id = idHandler.set(element); | ||
if(browserDetector.isIE(8)) { | ||
//IE 8 does not support objects properly. Luckily they do support the resize event. | ||
//So do not inject the object and notify that the element is already ready to be listened to. | ||
//The event handler for the resize event is attached in the utils.addListener instead. | ||
callback(element); | ||
} else { | ||
injectObject(id, element, callback); | ||
} | ||
} | ||
//Add an object element as a child to the target element that will be listened to for resize events. | ||
var object = document.createElement("object"); | ||
object.type = "text/html"; | ||
object.data = "about:blank"; | ||
object.onload = onObjectLoad; | ||
object.setAttribute("elq-object-id", id); | ||
element.appendChild(object); | ||
/** | ||
* Returns the child object of the target element. | ||
* @private | ||
* @param {element} element The target element. | ||
* @returns The object element of the target. | ||
*/ | ||
function getObject(element) { | ||
return forEach(element.children, function(child) { | ||
if(child.hasAttribute("erd-object-id")) { | ||
return child; | ||
} | ||
}); | ||
} | ||
return { | ||
isDetectable: isDetectable, | ||
makeDetectable: makeDetectable, | ||
addListener: addListener, | ||
}; | ||
}; | ||
/** | ||
* Returns the child object of the target element. | ||
* @private | ||
* @param {element} element The target element. | ||
* @returns The object element of the target. | ||
*/ | ||
function getObject(element) { | ||
return forEach(element.children, function(child) { | ||
if(child.hasAttribute("elq-object-id")) { | ||
return child; | ||
} | ||
}); | ||
} |
@@ -11,3 +11,3 @@ "use strict"; | ||
*/ | ||
function newId() { | ||
function generate() { | ||
return idCount++; | ||
@@ -17,4 +17,4 @@ } | ||
return { | ||
newId: newId | ||
generate: generate | ||
}; | ||
}; |
"use strict"; | ||
var elementUtils = require("./element-utils"); | ||
module.exports = function() { | ||
module.exports = function(idHandler) { | ||
var eventListeners = {}; | ||
@@ -15,3 +13,3 @@ | ||
function getListeners(element) { | ||
return eventListeners[elementUtils.getId(element)]; | ||
return eventListeners[idHandler.get(element)]; | ||
} | ||
@@ -26,3 +24,3 @@ | ||
function addListener(element, listener) { | ||
var id = elementUtils.getId(element); | ||
var id = idHandler.get(element); | ||
@@ -29,0 +27,0 @@ if(!eventListeners[id]) { |
Sorry, the diff of this file is not supported yet
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
New author
Supply chain riskA new npm collaborator published a version of the package for the first time. New collaborators are usually benign additions to a project, but do indicate a change to the security surface area of a package.
Found 1 instance in 1 package
Environment variable access
Supply chain riskPackage accesses environment variables, which may be a sign of credential stuffing or data theft.
Found 4 instances in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
122320
23
1976
21
48
14
2