{"version":3,"file":"nixps-nebulapagebuilderframework.js","sources":["nixps-nebulapagebuilderframework.js"],"sourcesContent":["(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c=\"function\"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error(\"Cannot find module '\"+i+\"'\");throw a.code=\"MODULE_NOT_FOUND\",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u=\"function\"==typeof require&&require,i=0;i= frameOSVersionInfo.compatibility.cloudflow.trueMinor);\n } else {\n checkCallback(false, false);\n }\n });\n },\n isFrameHostCompatibleWithCloudflow: function isFrameHostCompatibleWithCloudflow(pCallback) {\n var that = this;\n var checkCallback = pCallback;\n this._getVersions(function () {\n if (cloudflowVersionInfo !== null && cloudflowVersionInfo.trueMinor !== undefined && frameHostVersionInfo !== null && frameHostVersionInfo.compatibility !== undefined && frameHostVersionInfo.compatibility.cloudflow !== undefined) {\n checkCallback(true, cloudflowVersionInfo.trueMinor >= frameHostVersionInfo.compatibility.cloudflow.trueMinor);\n } else {\n checkCallback(false, false);\n }\n });\n },\n isCloudflowCompatibleWithFrameOS: function isCloudflowCompatibleWithFrameOS(pCallback) {\n var that = this;\n var checkCallback = pCallback;\n this._getVersions(function () {\n if (frameOSVersionInfo !== null && frameOSVersionInfo.version !== undefined && cloudflowVersionInfo !== null && cloudflowVersionInfo.compatibility !== undefined && cloudflowVersionInfo.compatibility.frame !== undefined) {\n checkCallback(true, that._compareMajorAndMinor(frameOSVersionInfo.version.major, frameOSVersionInfo.version.minor, cloudflowVersionInfo.compatibility.frame.major, cloudflowVersionInfo.compatibility.frame.minor));\n } else {\n checkCallback(false, false);\n }\n });\n },\n isCloudflowCompatibleWithFrameHost: function isCloudflowCompatibleWithFrameHost(pCallback) {\n var that = this;\n var checkCallback = pCallback;\n this._getVersions(function () {\n if (frameHostVersionInfo !== null && frameHostVersionInfo.version !== undefined && cloudflowVersionInfo !== null && cloudflowVersionInfo.compatibility !== undefined && cloudflowVersionInfo.compatibility.frame !== undefined) {\n checkCallback(true, that._compareMajorAndMinor(frameHostVersionInfo.version.major, frameHostVersionInfo.version.minor, cloudflowVersionInfo.compatibility.frame.major, cloudflowVersionInfo.compatibility.frame.minor));\n } else {\n checkCallback(false, false);\n }\n });\n },\n _getVersions: function _getVersions(pCallBack) {\n var that = this;\n if (cloudflowVersionInfo === null) {\n api_async.portal.version(function (pResult) {\n cloudflowVersionInfo = pResult;\n if (cloudflowVersionInfo.major !== undefined) {\n cloudflowVersionInfo.trueMinor = cloudflowVersionInfo.major * 100 + cloudflowVersionInfo.minor;\n }\n // try again after loading\n that._getVersions(pCallBack);\n });\n return;\n }\n if (frameOSVersionInfo === null) {\n api_async.frame.os.get().done(function () {\n api_async.frame.os.get_version(function (pResult) {\n frameOSVersionInfo = pResult;\n if (frameOSVersionInfo.compatibility !== undefined && frameOSVersionInfo.compatibility.cloudflow !== undefined) {\n frameOSVersionInfo.compatibility.cloudflow.trueMinor = frameOSVersionInfo.compatibility.cloudflow.major * 100 + frameOSVersionInfo.compatibility.cloudflow.minor;\n }\n // try again after loading\n that._getVersions(pCallBack);\n });\n return;\n });\n }\n if (frameHostVersionInfo === null) {\n api_async.frame.host.get().done(function () {\n api_async.frame.host.get_version(function (pResult) {\n frameHostVersionInfo = pResult;\n if (frameHostVersionInfo.compatibility !== undefined && frameHostVersionInfo.compatibility.cloudflow !== undefined) {\n frameHostVersionInfo.compatibility.cloudflow.trueMinor = frameHostVersionInfo.compatibility.cloudflow.major * 100 + frameHostVersionInfo.compatibility.cloudflow.minor;\n }\n // try again after loading\n that._getVersions(pCallBack);\n });\n return;\n });\n }\n\n // if we come here we have loaded whatever we can load (or we simply have\n // all the versions\n pCallBack();\n },\n _compareMajorAndMinor: function _compareMajorAndMinor(pLeftMajor, pLeftMinor, pRightMajor, pRightMinor) {\n if (pLeftMajor === pRightMajor) {\n if (pLeftMinor >= pRightMinor) {\n return true;\n } else {\n return false;\n }\n }\n return false;\n }\n};\nmodule.exports = FrameCompatibility;\n\n},{}],2:[function(require,module,exports){\n\"use strict\";\n\n/**\n * known events\n * - hostAPILoaded\n * - when : after the host api has been succesfully loaded into the api objects.\n * - data: none\n * - loadedFilesChanged\n * - when : after a document has been opened/closed/saved etc. That is, each time the result\n * of calling list_loaded_files() will return different results\n * - data: none\n * - setCloudflowSession\n * - when : after a session expired, and a new session could be created automatically\n * - data:\n * - session: the new session string\n **/\nfunction FrameHostEventManager() {}\nvar eventRegistry = {};\nvar entryCount = 1;\nvar hostCallback = undefined;\nFrameHostEventManager.prototype = {\n constructor: FrameHostEventManager,\n on: function on(pEvent, pCallback) {\n var eventEntry = [];\n if (pEvent in eventRegistry === false) {\n eventRegistry[pEvent] = [];\n }\n eventEntry = eventRegistry[pEvent];\n var entryID = entryCount++;\n eventEntry.push({\n id: entryID,\n callback: pCallback\n });\n return function () {\n api.frame.host.eventManager.remove(entryID);\n };\n },\n fire: function fire(pEvent, pData) {\n if (pEvent === \"setCloudflowSession\") {\n api.set_session(pData.session);\n }\n if (pEvent in eventRegistry === true) {\n var eventEntry = eventRegistry[pEvent];\n for (var countEntrys = 0; countEntrys < eventEntry.length; ++countEntrys) {\n eventEntry[countEntrys].callback(pData);\n }\n }\n if (hostCallback !== undefined) {\n hostCallback(pEvent, pData);\n }\n },\n remove: function remove(pEntryID) {\n for (var keyName in eventRegistry) {\n var eventEntry = eventRegistry[keyName];\n for (var countEntrys = 0; countEntrys < eventEntry.length; ++countEntrys) {\n if (eventEntry[countEntrys].id === pEntryID) {\n eventEntry.splice(countEntrys, 1);\n return;\n }\n }\n }\n },\n registerHostCallback: function registerHostCallback(pCallback) {\n hostCallback = pCallback;\n }\n};\nmodule.exports = FrameHostEventManager;\n\n},{}],3:[function(require,module,exports){\n\"use strict\";\n\n/**\n * Returns a promise that resolves in case that the Cloudflow version is compatible with Frame\n */\nfunction isCompatible(pCompatibleCloudflowVersion) {\n return $.Deferred(function (pDefer) {\n api_async.portal.version(function (pResult) {\n var cloudflowMajor = pResult.major;\n var cloudflowMinor = pResult.minor;\n var cloudflowVersion = cloudflowMajor * 100 + cloudflowMinor;\n if (cloudflowVersion >= pCompatibleCloudflowVersion) {\n pDefer.resolve();\n } else {\n var code = \"Cloudflow not compatible\";\n var description = \"The running Cloudflow is not compatible with Frame, need at least \" + pCompatibleCloudflowVersion;\n pDefer.reject({\n error_code: code,\n error: description,\n messages: [{\n type: code,\n severity: \"error\",\n description: description\n }]\n });\n }\n }, function (pError) {\n pDefer.reject(pError);\n });\n });\n}\nfunction FrameHostLoader() {\n this.m_promise = {};\n}\nFrameHostLoader.prototype = {\n constructor: FrameHostLoader,\n /**\n * Resolves the promise in case the page is running inside a host\n * @param {*} pHost name of the host the page is running in (\"illustrator\", \"packz\", \"ic3d\")\n * @param {*} pParent parent api that to which the promise will resolve (eg api, api_async)\n * @param {*} pForceVersion set to true if you want to override the compatibility check\n */\n get: function get(pHost, pParent, pForceVersion) {\n var that = this;\n if (this.m_promise[pHost] === undefined) {\n if (pHost === \"illustrator\") {\n this.m_promise[\"illustrator\"] = $.Deferred(function (pDefer) {\n that._loadIllustrator(function () {\n pDefer.resolve(pParent);\n }, function (pError) {\n pDefer.reject(pError);\n }, pForceVersion);\n });\n } else if (pHost === \"packz\") {\n this.m_promise[\"packz\"] = $.Deferred(function (pDefer) {\n that._loadPackz(function () {\n pDefer.resolve(pParent);\n }, function (pError) {\n pDefer.reject(pError);\n }, pForceVersion);\n });\n } else if (pHost === \"ic3d\") {\n this.m_promise[\"ic3d\"] = $.Deferred(function (pDefer) {\n that._loadiC3D(function () {\n pDefer.resolve(pParent);\n }, function (pError) {\n pDefer.reject(pError);\n }, pForceVersion);\n });\n } else {\n return $.Deferred().reject(\"no such host: \" + pHost);\n }\n }\n return this.m_promise[pHost];\n },\n load: function load(pHost, pCallback, pCallbackError) {\n if (pHost === \"illustrator\") {\n this._loadIllustrator(pCallback, pCallbackError);\n } else if (pHost === \"packz\") {\n this._loadPackz(pCallback, pCallbackError);\n } else if (pHost === \"ic3d\") {\n this._loadiC3D(pCallback, pCallbackError);\n }\n },\n _loadIllustrator: function _loadIllustrator(pCallback, pCallbackError, pForceVersion) {\n try {\n // add the listener to react on the bootstrap string (and install the API)\n window.__adobe_cep__.addEventListener(\"com.nixps.csxs.events.callAPIFunctionResult\", function (pEvent) {\n if (pEvent.appId === \"frame.host.get_api_bootstrap\") {\n if (pEvent.data.javascript !== undefined) {\n var install = new Function(pEvent.data.javascript);\n // Always install the api calls as we need to report an error eventually with show_status_messages\n install();\n\n // #29894: Redefine \"get\" as it gets lost after the install\n window.api_defer.frame.host.get = function (pHost, pOptions) {\n var parent = this.m_parent;\n return $.Deferred(function (pDefer) {\n pDefer.resolve(window.frameHostLoader.get(pHost, parent, pForceVersion));\n }).then(function (pApi) {\n return pApi;\n });\n };\n if (pForceVersion === true) {\n api.frame.host.eventManager.fire(\"hostAPILoaded\", {});\n if (pCallback !== undefined) {\n pCallback();\n }\n return;\n }\n var compatibleCloudflowVersion = pEvent.data.compatibleCloudflowVersion;\n if (compatibleCloudflowVersion !== undefined) {\n isCompatible(compatibleCloudflowVersion).then(function () {\n api.frame.host.eventManager.fire(\"hostAPILoaded\", {});\n if (pCallback !== undefined) {\n pCallback();\n }\n }).fail(function (error) {\n try {\n api_defer.frame.host.show_status_messages(error.messages);\n } catch (error) {\n console.error(error);\n }\n if (pCallbackError !== undefined) {\n pCallbackError();\n }\n });\n } else {\n api.frame.host.eventManager.fire(\"hostAPILoaded\", {});\n if (pCallback !== undefined) {\n pCallback();\n }\n }\n }\n }\n }, this);\n window.__adobe_cep__.addEventListener(\"com.nixps.csxs.events.frameHostEvent\", function (pEvent) {\n if (pEvent.appId === \"Illustrator-Frame\" && (pEvent.extensionId === null || pEvent.extensionId === \"\")) {\n api.frame.host.eventManager.fire(pEvent.data.event, pEvent.data.data);\n }\n }, this);\n\n // and now trigger the event that should return us the bootstrap string\n var getAPIBootstrap = {\n type: \"com.nixps.csxs.events.callAPIFunction\",\n scope: \"APPLICATION\",\n appId: \"ILST\",\n extensionID: \"com.nixps.illustrator.HybridFrameClient.api\"\n };\n getAPIBootstrap.data = JSON.stringify({\n method: \"frame.host.get_api_bootstrap\"\n });\n window.__adobe_cep__.dispatchEvent(getAPIBootstrap);\n api.frame.host.eventManager.registerHostCallback(this._illustratorEventManagerCallback);\n } catch (err) {\n // not running inside illustrator...\n pCallbackError(err);\n }\n },\n _illustratorEventManagerCallback: function _illustratorEventManagerCallback(pEvent, pData) {\n var frameHostEvent = {\n type: \"com.nixps.csxs.events.frameHostedPageEvent\",\n scope: \"APPLICATION\",\n appId: \"ILST\",\n extensionID: \"com.nixps.illustrator.frame.hostEvents\"\n };\n frameHostEvent.data = JSON.stringify({\n event: pEvent,\n data: pData\n });\n window.__adobe_cep__.dispatchEvent(frameHostEvent);\n },\n _loadPackz: function _loadPackz(pCallback, pCallbackError, pForceVersion) {\n try {\n var doInstall = function doInstall() {\n window.packz_init.install(function (bootstrapCode) {\n var install = new Function(bootstrapCode);\n install();\n\n // #29894: Redefine \"get\" as it gets lost after the install\n window.api_defer.frame.host.get = function (pHost, pOptions) {\n var parent = this.m_parent;\n return $.Deferred(function (pDefer) {\n pDefer.resolve(window.frameHostLoader.get(pHost, parent, pForceVersion));\n }).then(function (pApi) {\n return pApi;\n });\n };\n api.frame.host.eventManager.fire(\"hostAPILoaded\", {});\n if (pCallback !== undefined) {\n pCallback();\n }\n });\n };\n if (pForceVersion === true) {\n doInstall();\n } else {\n window.packz_init.getCompatibleCloudflowVersion(function (compatibleCloudflowVersion) {\n if (compatibleCloudflowVersion !== undefined) {\n isCompatible(compatibleCloudflowVersion).then(function () {\n doInstall();\n }).fail(function (error) {\n if (pCallbackError !== undefined) {\n pCallbackError();\n }\n });\n } else {\n doInstall();\n }\n });\n }\n api.frame.host.eventManager.registerHostCallback(this._packzEventManagerCallback);\n } catch (err) {\n // not running inside packz...\n pCallbackError(err);\n }\n },\n _packzEventManagerCallback: function _packzEventManagerCallback(pEvent, pData) {\n var frameHostEventdata = JSON.stringify({\n event: pEvent,\n data: pData\n });\n window.packz_init.dispatchEvent(frameHostEventdata);\n },\n _ic3dEventManagerCallback: function _ic3dEventManagerCallback(pEvent, pData) {\n var frameHostEventdata = JSON.stringify({\n event: pEvent,\n data: pData\n });\n window.ic3diface.dispatchEvent(frameHostEventdata);\n },\n _loadiC3D: function _loadiC3D(pCallback, pCallbackError, pForceVersion) {\n try {\n var doInstall = function doInstall() {\n window.ic3diface.install(function (bootstrapCode) {\n var install = new Function(bootstrapCode);\n install();\n\n // #29894: Redefine \"get\" as it gets lost after the install\n window.api_defer.frame.host.get = function (pHost, pOptions) {\n var parent = this.m_parent;\n return $.Deferred(function (pDefer) {\n pDefer.resolve(window.frameHostLoader.get(pHost, parent, pForceVersion));\n }).then(function (pApi) {\n return pApi;\n });\n };\n api.frame.host.eventManager.fire(\"hostAPILoaded\", {});\n if (pCallback !== undefined) {\n pCallback();\n }\n });\n };\n new QWebChannel(qt.webChannelTransport, function (channel) {\n window.ic3diface = channel.objects.ic3diface;\n window.ic3diface.getCompatibleCloudflowVersion(function (compatibleCloudflowVersion) {\n if (compatibleCloudflowVersion !== undefined) {\n isCompatible(compatibleCloudflowVersion).then(function () {\n // Start install of bootstrap\n doInstall();\n }).fail(function (error) {\n if (pCallbackError !== undefined) {\n pCallbackError();\n }\n });\n }\n });\n });\n\n // Register event manager\n api.frame.host.eventManager.registerHostCallback(this._ic3dEventManagerCallback);\n } catch (err) {\n // not running inside ic3d...\n pCallbackError(err);\n }\n }\n};\nmodule.exports = FrameHostLoader;\n\n},{}],4:[function(require,module,exports){\n\"use strict\";\n\nvar FrameHostEventManager = require(\"./FrameHostEventManager.js\");\nvar FrameHostLoader = require(\"./FrameHostLoader.js\");\nvar FrameCompatibility = require(\"./FrameCompatibility.js\");\nif (window.frameHostLoader !== undefined) {\n console.error('frame.js is already loaded, this is not needed as it is included framework');\n}\nwindow.FrameHostLoader = FrameHostLoader;\nwindow.FrameHostEventManager = FrameHostEventManager;\nwindow.FrameCompatibility = FrameCompatibility;\nwindow.frameHostLoader = new FrameHostLoader();\nif (window.api === undefined || window.api.frame === undefined) {\n console.error('frame.js: cloudflow api should be included first');\n} else {\n api.frame.host.eventManager = new FrameHostEventManager();\n\n // Check the frame features\n $(function () {\n var setFrameFeatures = require(\"./setFrameFeatures.js\");\n setFrameFeatures();\n });\n}\n\n},{\"./FrameCompatibility.js\":1,\"./FrameHostEventManager.js\":2,\"./FrameHostLoader.js\":3,\"./setFrameFeatures.js\":5}],5:[function(require,module,exports){\n\"use strict\";\n\nfunction hasMappings(filestoreSetups) {\n return Array.isArray(filestoreSetups) === true && filestoreSetups.length > 0;\n}\nfunction hasPackz(apps) {\n if (Array.isArray(apps) === false) {\n return false;\n }\n for (var i = 0; i < apps.length; i++) {\n var current = apps[i];\n if (current.name === \"PACKZ\" && current.path_is_valid === true) {\n return true;\n }\n }\n return false;\n}\n\n/**\n * Sets the frame features when frame is installed\n */\nfunction setFrameFeatures() {\n api_defer.frame.os.get().then(function () {\n $(\"body\").addClass(\"frame-os-installed\");\n }).fail(function () {\n $(\"body\").removeClass(\"frame-os-installed\");\n });\n api_defer.frame.os.get().then(function () {\n return api_defer.frame.os.is_active_cloudflow_server(window.location.href);\n }).then(function (matches) {\n if (matches.match === false) {\n return $.Deferred().reject(\"Frame server does not match\");\n }\n return api_defer.frame.os.get_workstation_setup();\n }).then(function (setup) {\n if (hasPackz(setup.setup.applications)) {\n $(\"body\").addClass(\"frame-packz-app-enabled\");\n } else {\n $(\"body\").removeClass(\"frame-packz-app-enabled\");\n }\n if (hasMappings(setup.setup.filestore_setups)) {\n $(\"body\").addClass(\"frame-os-enabled\");\n } else {\n $(\"body\").removeClass(\"frame-os-enabled\");\n }\n }).fail(function () {\n $(\"body\").removeClass(\"frame-os-enabled\");\n });\n var matches = window.location.href.match(/host=([^&]*)/);\n if (Array.isArray(matches) && matches.length >= 2) {\n var matchGroup = matches[1];\n api_defer.frame.host.get(matchGroup).then(function () {\n $(\"body\").addClass(\"frame-host-enabled\");\n\n // Show a relevant error message when in Illustrator and frame os did not initialize correctly\n api_defer.frame.os.get().then(function () {\n return api_defer.frame.os.is_active_cloudflow_server(window.location.href).then(function (matches) {\n if (matches.match === false) {\n return $.Deferred().reject({\n messages: [{\n type: \"frame configuration server does not match\",\n description: \"This page is not served from the Frame-Cloudflow server in the settings\",\n severity: \"error\"\n }]\n });\n }\n });\n }).fail(function (error) {\n api_defer.frame.host.show_status_messages(error.messages);\n });\n }).fail(function () {\n $(\"body\").removeClass(\"frame-host-enabled\");\n });\n }\n}\nmodule.exports = setFrameFeatures;\n\n},{}],6:[function(require,module,exports){\n\"use strict\";\n\n/**\n * Detect Element Resize Plugin for jQuery\n *\n * https://github.com/sdecima/javascript-detect-element-resize\n * Sebastian Decima\n *\n * sdecima/javascript-detect-element-resize is licensed under the MIT License\n * version: 0.5.3\n **/\n\n(function ($) {\n var attachEvent = document.attachEvent,\n stylesCreated = false;\n var jQuery_onresize = $.fn.onresize;\n $.fn.onresize = function (callback) {\n return this.each(function () {\n if (this == window) jQuery_onresize.call(jQuery(this), callback);else addResizeListener(this, callback);\n });\n };\n $.fn.removeOnResize = function (callback) {\n return this.each(function () {\n removeResizeListener(this, callback);\n });\n };\n if (!attachEvent) {\n var _resetTriggers = function _resetTriggers(element) {\n var triggers = element.__resizeTriggers__,\n expand = triggers.firstElementChild,\n contract = triggers.lastElementChild,\n expandChild = expand.firstElementChild;\n contract.scrollLeft = contract.scrollWidth;\n contract.scrollTop = contract.scrollHeight;\n expandChild.style.width = expand.offsetWidth + 1 + 'px';\n expandChild.style.height = expand.offsetHeight + 1 + 'px';\n expand.scrollLeft = expand.scrollWidth;\n expand.scrollTop = expand.scrollHeight;\n };\n var checkTriggers = function checkTriggers(element) {\n return element.offsetWidth != element.__resizeLast__.width || element.offsetHeight != element.__resizeLast__.height;\n };\n var _scrollListener = function _scrollListener(e) {\n var element = this;\n _resetTriggers(this);\n if (this.__resizeRAF__) cancelFrame(this.__resizeRAF__);\n this.__resizeRAF__ = requestFrame(function () {\n if (checkTriggers(element)) {\n element.__resizeLast__.width = element.offsetWidth;\n element.__resizeLast__.height = element.offsetHeight;\n element.__resizeListeners__.forEach(function (fn) {\n fn.call(element, e);\n });\n }\n });\n };\n var requestFrame = function () {\n var raf = window.requestAnimationFrame || window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || function (fn) {\n return window.setTimeout(fn, 20);\n };\n return function (fn) {\n return raf(fn);\n };\n }();\n var cancelFrame = function () {\n var cancel = window.cancelAnimationFrame || window.mozCancelAnimationFrame || window.webkitCancelAnimationFrame || window.clearTimeout;\n return function (id) {\n return cancel(id);\n };\n }();\n ;\n ;\n\n /* Detect CSS Animations support to detect element display/re-attach */\n var animation = false,\n animationstring = 'animation',\n keyframeprefix = '',\n animationstartevent = 'animationstart',\n domPrefixes = 'Webkit Moz O ms'.split(' '),\n startEvents = 'webkitAnimationStart animationstart oAnimationStart MSAnimationStart'.split(' '),\n pfx = '';\n {\n var elm = document.createElement('fakeelement');\n if (elm.style.animationName !== undefined) {\n animation = true;\n }\n if (animation === false) {\n for (var i = 0; i < domPrefixes.length; i++) {\n if (elm.style[domPrefixes[i] + 'AnimationName'] !== undefined) {\n pfx = domPrefixes[i];\n animationstring = pfx + 'Animation';\n keyframeprefix = '-' + pfx.toLowerCase() + '-';\n animationstartevent = startEvents[i];\n animation = true;\n break;\n }\n }\n }\n }\n var animationName = 'resizeanim';\n var animationKeyframes = '@' + keyframeprefix + 'keyframes ' + animationName + ' { from { opacity: 0; } to { opacity: 0; } } ';\n var animationStyle = keyframeprefix + 'animation: 1ms ' + animationName + '; ';\n }\n function createStyles() {\n if (!stylesCreated) {\n //opacity:0 works around a chrome bug https://code.google.com/p/chromium/issues/detail?id=286360\n var css = (animationKeyframes ? animationKeyframes : '') + '.resize-triggers { ' + (animationStyle ? animationStyle : '') + 'visibility: hidden; opacity: 0; } ' + '.resize-triggers, .resize-triggers > div, .contract-trigger:before { content: \\\" \\\"; display: block; position: absolute; top: 0; left: 0; height: 100%; width: 100%; overflow: hidden; } .resize-triggers > div { background: #eee; overflow: auto; } .contract-trigger:before { width: 200%; height: 200%; }',\n head = document.head || document.getElementsByTagName('head')[0],\n style = document.createElement('style');\n style.type = 'text/css';\n if (style.styleSheet) {\n style.styleSheet.cssText = css;\n } else {\n style.appendChild(document.createTextNode(css));\n }\n head.appendChild(style);\n stylesCreated = true;\n }\n }\n window.addResizeListener = function (element, fn) {\n if (attachEvent) element.attachEvent('onresize', fn);else {\n if (!element.__resizeTriggers__) {\n if (getComputedStyle(element).position == 'static') element.style.position = 'relative';\n createStyles();\n element.__resizeLast__ = {};\n element.__resizeListeners__ = [];\n (element.__resizeTriggers__ = document.createElement('div')).className = 'resize-triggers';\n element.__resizeTriggers__.innerHTML = '
' + '
';\n element.appendChild(element.__resizeTriggers__);\n resetTriggers(element);\n element.addEventListener('scroll', scrollListener, true);\n\n /* Listen for a css animation to detect element display/re-attach */\n animationstartevent && element.__resizeTriggers__.addEventListener(animationstartevent, function (e) {\n if (e.animationName == animationName) resetTriggers(element);\n });\n }\n element.__resizeListeners__.push(fn);\n }\n };\n window.removeResizeListener = function (element, fn) {\n if (attachEvent) element.detachEvent('onresize', fn);else {\n element.__resizeListeners__.splice(element.__resizeListeners__.indexOf(fn), 1);\n if (!element.__resizeListeners__.length) {\n element.removeEventListener('scroll', scrollListener);\n element.__resizeTriggers__ = !element.removeChild(element.__resizeTriggers__);\n }\n }\n };\n})(jQuery);\n\n},{}],7:[function(require,module,exports){\n\"use strict\";\n\nfunction _typeof(obj) { \"@babel/helpers - typeof\"; return _typeof = \"function\" == typeof Symbol && \"symbol\" == typeof Symbol.iterator ? function (obj) { return typeof obj; } : function (obj) { return obj && \"function\" == typeof Symbol && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; }, _typeof(obj); }\n/* Port of strftime() by T. H. Doan (https://thdoan.github.io/strftime/)\n *\n * Day of year (%j) code based on Joe Orost's answer:\n * http://stackoverflow.com/questions/8619879/javascript-calculate-the-day-of-the-year-1-366\n *\n * Week number (%V) code based on Taco van den Broek's prototype:\n * http://techblog.procurios.nl/k/news/view/33796/14863/calculate-iso-8601-week-and-year-in-javascript.html\n */\n(function () {\n function strftime(sFormat, date) {\n if (!(date instanceof Date)) date = new Date();\n var nDay = date.getDay(),\n nDate = date.getDate(),\n nMonth = date.getMonth(),\n nYear = date.getFullYear(),\n nHour = date.getHours(),\n aDays = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'],\n aMonths = ['January', 'February', 'March', 'April', 'May', 'June', 'July', 'August', 'September', 'October', 'November', 'December'],\n aDayCount = [0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334],\n isLeapYear = function isLeapYear() {\n return nYear % 4 === 0 && nYear % 100 !== 0 || nYear % 400 === 0;\n },\n getThursday = function getThursday() {\n var target = new Date(date);\n target.setDate(nDate - (nDay + 6) % 7 + 3);\n return target;\n },\n zeroPad = function zeroPad(nNum, nPad) {\n return (Math.pow(10, nPad) + nNum + '').slice(1);\n };\n return sFormat.replace(/%[a-z]/gi, function (sMatch) {\n return ({\n '%a': aDays[nDay].slice(0, 3),\n '%A': aDays[nDay],\n '%b': aMonths[nMonth].slice(0, 3),\n '%B': aMonths[nMonth],\n '%c': date.toUTCString(),\n '%C': Math.floor(nYear / 100),\n '%d': zeroPad(nDate, 2),\n '%e': nDate,\n '%F': date.toISOString().slice(0, 10),\n '%G': getThursday().getFullYear(),\n '%g': (getThursday().getFullYear() + '').slice(2),\n '%H': zeroPad(nHour, 2),\n '%I': zeroPad((nHour + 11) % 12 + 1, 2),\n '%j': zeroPad(aDayCount[nMonth] + nDate + (nMonth > 1 && isLeapYear() ? 1 : 0), 3),\n '%k': nHour,\n '%l': (nHour + 11) % 12 + 1,\n '%m': zeroPad(nMonth + 1, 2),\n '%n': nMonth + 1,\n '%M': zeroPad(date.getMinutes(), 2),\n '%p': nHour < 12 ? 'AM' : 'PM',\n '%P': nHour < 12 ? 'am' : 'pm',\n '%s': Math.round(date.getTime() / 1000),\n '%S': zeroPad(date.getSeconds(), 2),\n '%u': nDay || 7,\n '%V': function () {\n var target = getThursday(),\n n1stThu = target.valueOf();\n target.setMonth(0, 1);\n var nJan1 = target.getDay();\n if (nJan1 !== 4) target.setMonth(0, 1 + (4 - nJan1 + 7) % 7);\n return zeroPad(1 + Math.ceil((n1stThu - target) / 604800000), 2);\n }(),\n '%w': nDay,\n '%x': date.toLocaleDateString(),\n '%X': date.toLocaleTimeString(),\n '%y': (nYear + '').slice(2),\n '%Y': nYear,\n '%z': date.toTimeString().replace(/.+GMT([+-]\\d+).+/, '$1'),\n '%Z': date.toTimeString().replace(/.+\\((.+?)\\)$/, '$1')\n }[sMatch] || '') + '' || sMatch;\n });\n }\n if ((typeof module === \"undefined\" ? \"undefined\" : _typeof(module)) === \"object\" && module && _typeof(module.exports) === \"object\") {\n //loaders that implement the Node module pattern (including browserify)\n module.exports = strftime;\n } else {\n // Otherwise expose juration\n window.strftime = strftime;\n }\n})();\n\n},{}],8:[function(require,module,exports){\n\"use strict\";\n\n/**\n * @file timeline.js\n *\n * @brief\n * The Timeline is an interactive visualization chart to visualize events in\n * time, having a start and end date.\n * You can freely move and zoom in the timeline by dragging\n * and scrolling in the Timeline. Items are optionally dragable. The time\n * scale on the axis is adjusted automatically, and supports scales ranging\n * from milliseconds to years.\n *\n * Timeline is part of the CHAP Links library.\n *\n * Timeline is tested on Firefox 3.6, Safari 5.0, Chrome 6.0, Opera 10.6, and\n * Internet Explorer 6+.\n *\n * @license\n * Licensed under the Apache License, Version 2.0 (the \"License\"); you may not\n * use this file except in compliance with the License. You may obtain a copy\n * of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS, WITHOUT\n * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the\n * License for the specific language governing permissions and limitations under\n * the License.\n *\n * Copyright (c) 2011-2015 Almende B.V.\n *\n * @author Jos de Jong, \n * @date 2015-03-04\n * @version 2.9.1\n */\n\n/*\n * i18n mods by github user iktuz (https://gist.github.com/iktuz/3749287/)\n * added to v2.4.1 with da_DK language by @bjarkebech\n */\n\n/*\n * TODO\n *\n * Add zooming with pinching on Android\n *\n * Bug: when an item contains a javascript onclick or a link, this does not work\n * when the item is not selected (when the item is being selected,\n * it is redrawn, which cancels any onclick or link action)\n * Bug: when an item contains an image without size, or a css max-width, it is not sized correctly\n * Bug: neglect items when they have no valid start/end, instead of throwing an error\n * Bug: Pinching on ipad does not work very well, sometimes the page will zoom when pinching vertically\n * Bug: cannot set max width for an item, like div.timeline-event-content {white-space: normal; max-width: 100px;}\n * Bug on IE in Quirks mode. When you have groups, and delete an item, the groups become invisible\n */\n\n/**\n * Declare a unique namespace for CHAP's Common Hybrid Visualisation Library,\n * \"links\"\n */\nif (typeof links === 'undefined') {\n window.links = {};\n // important: do not use var, as \"var links = {};\" will overwrite\n // the existing links variable value with undefined in IE8, IE7.\n}\n\n/**\n * Ensure the variable google exists\n */\nif (typeof google === 'undefined') {\n window.google = undefined;\n // important: do not use var, as \"var google = undefined;\" will overwrite\n // the existing google variable value with undefined in IE8, IE7.\n}\n\n// Internet Explorer 8 and older does not support Array.indexOf,\n// so we define it here in that case\n// http://soledadpenades.com/2007/05/17/arrayindexof-in-internet-explorer/\nif (!Array.prototype.indexOf) {\n Array.prototype.indexOf = function (obj) {\n for (var i = 0; i < this.length; i++) {\n if (this[i] == obj) {\n return i;\n }\n }\n return -1;\n };\n}\n\n// Internet Explorer 8 and older does not support Array.forEach,\n// so we define it here in that case\n// https://developer.mozilla.org/en-US/docs/JavaScript/Reference/Global_Objects/Array/forEach\nif (!Array.prototype.forEach) {\n Array.prototype.forEach = function (fn, scope) {\n for (var i = 0, len = this.length; i < len; ++i) {\n fn.call(scope || this, this[i], i, this);\n }\n };\n}\n\n/**\n * @constructor links.Timeline\n * The timeline is a visualization chart to visualize events in time.\n *\n * The timeline is developed in javascript as a Google Visualization Chart.\n *\n * @param {Element} container The DOM element in which the Timeline will\n * be created. Normally a div element.\n * @param {Object} options A name/value map containing settings for the\n * timeline. Optional.\n */\nlinks.Timeline = function (container, options) {\n if (!container) {\n // this call was probably only for inheritance, no constructor-code is required\n return;\n }\n\n // create variables and set default values\n this.dom = {};\n this.conversion = {};\n this.eventParams = {}; // stores parameters for mouse events\n this.groups = [];\n this.groupIndexes = {};\n this.items = [];\n this.renderQueue = {\n show: [],\n // Items made visible but not yet added to DOM\n hide: [],\n // Items currently visible but not yet removed from DOM\n update: [] // Items with changed data but not yet adjusted DOM\n };\n\n this.renderedItems = []; // Items currently rendered in the DOM\n this.clusterGenerator = new links.Timeline.ClusterGenerator(this);\n this.currentClusters = [];\n this.selection = undefined; // stores index and item which is currently selected\n\n this.listeners = {}; // event listener callbacks\n\n // Initialize sizes.\n // Needed for IE (which gives an error when you try to set an undefined\n // value in a style)\n this.size = {\n 'actualHeight': 0,\n 'axis': {\n 'characterMajorHeight': 0,\n 'characterMajorWidth': 0,\n 'characterMinorHeight': 0,\n 'characterMinorWidth': 0,\n 'height': 0,\n 'labelMajorTop': 0,\n 'labelMinorTop': 0,\n 'line': 0,\n 'lineMajorWidth': 0,\n 'lineMinorHeight': 0,\n 'lineMinorTop': 0,\n 'lineMinorWidth': 0,\n 'top': 0\n },\n 'contentHeight': 0,\n 'contentLeft': 0,\n 'contentWidth': 0,\n 'frameHeight': 0,\n 'frameWidth': 0,\n 'groupsLeft': 0,\n 'groupsWidth': 0,\n 'items': {\n 'top': 0\n }\n };\n this.dom.container = container;\n\n //\n // Let's set the default options first\n //\n this.options = {\n 'width': \"100%\",\n 'height': \"auto\",\n 'minHeight': 0,\n // minimal height in pixels\n 'groupMinHeight': 0,\n 'autoHeight': true,\n 'eventMargin': 10,\n // minimal margin between events\n 'eventMarginAxis': 20,\n // minimal margin between events and the axis\n 'dragAreaWidth': 10,\n // pixels\n\n 'min': undefined,\n 'max': undefined,\n 'zoomMin': 10,\n // milliseconds\n 'zoomMax': 1000 * 60 * 60 * 24 * 365 * 10000,\n // milliseconds\n\n 'moveable': true,\n 'zoomable': true,\n 'selectable': true,\n 'unselectable': true,\n 'editable': false,\n 'snapEvents': true,\n 'groupsChangeable': true,\n 'timeChangeable': true,\n 'showCurrentTime': true,\n // show a red bar displaying the current time\n 'showCustomTime': false,\n // show a blue, draggable bar displaying a custom time\n 'showMajorLabels': true,\n 'showMinorLabels': true,\n 'showNavigation': false,\n 'showButtonNew': false,\n 'groupsOnRight': false,\n 'groupsOrder': true,\n 'axisOnTop': false,\n 'stackEvents': true,\n 'animate': true,\n 'animateZoom': true,\n 'cluster': false,\n 'clusterMaxItems': 5,\n 'style': 'box',\n 'customStackOrder': false,\n //a function(a,b) for determining stackorder amongst a group of items. Essentially a comparator, -ve value for \"a before b\" and vice versa\n\n // i18n: Timeline only has built-in English text per default. Include timeline-locales.js to support more localized text.\n 'locale': 'en',\n 'MONTHS': [\"January\", \"February\", \"March\", \"April\", \"May\", \"June\", \"July\", \"August\", \"September\", \"October\", \"November\", \"December\"],\n 'MONTHS_SHORT': [\"Jan\", \"Feb\", \"Mar\", \"Apr\", \"May\", \"Jun\", \"Jul\", \"Aug\", \"Sep\", \"Oct\", \"Nov\", \"Dec\"],\n 'DAYS': [\"Sunday\", \"Monday\", \"Tuesday\", \"Wednesday\", \"Thursday\", \"Friday\", \"Saturday\"],\n 'DAYS_SHORT': [\"Sun\", \"Mon\", \"Tue\", \"Wed\", \"Thu\", \"Fri\", \"Sat\"],\n 'ZOOM_IN': \"Zoom in\",\n 'ZOOM_OUT': \"Zoom out\",\n 'MOVE_LEFT': \"Move left\",\n 'MOVE_RIGHT': \"Move right\",\n 'NEW': \"New\",\n 'CREATE_NEW_EVENT': \"Create new event\"\n };\n\n //\n // Now we can set the givenproperties\n //\n this.setOptions(options);\n this.clientTimeOffset = 0; // difference between client time and the time\n // set via Timeline.setCurrentTime()\n var dom = this.dom;\n\n // remove all elements from the container element.\n while (dom.container.hasChildNodes()) {\n dom.container.removeChild(dom.container.firstChild);\n }\n\n // create a step for drawing the axis\n this.step = new links.Timeline.StepDate();\n\n // add standard item types\n this.itemTypes = {\n box: links.Timeline.ItemBox,\n range: links.Timeline.ItemRange,\n floatingRange: links.Timeline.ItemFloatingRange,\n dot: links.Timeline.ItemDot\n };\n\n // initialize data\n this.data = [];\n this.firstDraw = true;\n\n // date interval must be initialized\n this.setVisibleChartRange(undefined, undefined, false);\n\n // render for the first time\n this.render();\n\n // fire the ready event\n var me = this;\n setTimeout(function () {\n me.trigger('ready');\n }, 0);\n};\n\n/**\n * Main drawing logic. This is the function that needs to be called\n * in the html page, to draw the timeline.\n *\n * A data table with the events must be provided, and an options table.\n *\n * @param {google.visualization.DataTable} data\n * The data containing the events for the timeline.\n * Object DataTable is defined in\n * google.visualization.DataTable\n * @param {Object} options A name/value map containing settings for the\n * timeline. Optional. The use of options here\n * is deprecated. Pass timeline options in the\n * constructor or use setOptions()\n */\nlinks.Timeline.prototype.draw = function (data, options) {\n if (options) {\n console.log(\"WARNING: Passing options in draw() is deprecated. Pass options to the constructur or use setOptions() instead!\");\n this.setOptions(options);\n }\n if (this.options.selectable) {\n links.Timeline.addClassName(this.dom.frame, \"timeline-selectable\");\n }\n\n // read the data\n this.setData(data);\n if (this.firstDraw) {\n this.setVisibleChartRangeAuto();\n }\n this.firstDraw = false;\n};\n\n/**\n * Set options for the timeline.\n * Timeline must be redrawn afterwards\n * @param {Object} options A name/value map containing settings for the\n * timeline. Optional.\n */\nlinks.Timeline.prototype.setOptions = function (options) {\n if (options) {\n // retrieve parameter values\n for (var i in options) {\n if (options.hasOwnProperty(i)) {\n this.options[i] = options[i];\n }\n }\n\n // prepare i18n dependent on set locale\n if (typeof links.locales !== 'undefined' && this.options.locale !== 'en') {\n var localeOpts = links.locales[this.options.locale];\n if (localeOpts) {\n for (var l in localeOpts) {\n if (localeOpts.hasOwnProperty(l)) {\n this.options[l] = localeOpts[l];\n }\n }\n }\n }\n\n // check for deprecated options\n if (options.showButtonAdd != undefined) {\n this.options.showButtonNew = options.showButtonAdd;\n console.log('WARNING: Option showButtonAdd is deprecated. Use showButtonNew instead');\n }\n if (options.intervalMin != undefined) {\n this.options.zoomMin = options.intervalMin;\n console.log('WARNING: Option intervalMin is deprecated. Use zoomMin instead');\n }\n if (options.intervalMax != undefined) {\n this.options.zoomMax = options.intervalMax;\n console.log('WARNING: Option intervalMax is deprecated. Use zoomMax instead');\n }\n if (options.scale && options.step) {\n this.step.setScale(options.scale, options.step);\n }\n }\n\n // validate options\n this.options.autoHeight = this.options.height === \"auto\";\n};\n\n/**\n * Get options for the timeline.\n *\n * @return the options object\n */\nlinks.Timeline.prototype.getOptions = function () {\n return this.options;\n};\n\n/**\n * Add new type of items\n * @param {String} typeName Name of new type\n * @param {links.Timeline.Item} typeFactory Constructor of items\n */\nlinks.Timeline.prototype.addItemType = function (typeName, typeFactory) {\n this.itemTypes[typeName] = typeFactory;\n};\n\n/**\n * Retrieve a map with the column indexes of the columns by column name.\n * For example, the method returns the map\n * {\n * start: 0,\n * end: 1,\n * content: 2,\n * group: undefined,\n * className: undefined\n * editable: undefined\n * type: undefined\n * }\n * @param {google.visualization.DataTable} dataTable\n * @type {Object} map\n */\nlinks.Timeline.mapColumnIds = function (dataTable) {\n var cols = {},\n colCount = dataTable.getNumberOfColumns(),\n allUndefined = true;\n\n // loop over the columns, and map the column id's to the column indexes\n for (var col = 0; col < colCount; col++) {\n var id = dataTable.getColumnId(col) || dataTable.getColumnLabel(col);\n cols[id] = col;\n if (id == 'start' || id == 'end' || id == 'content' || id == 'group' || id == 'className' || id == 'editable' || id == 'type') {\n allUndefined = false;\n }\n }\n\n // if no labels or ids are defined, use the default mapping\n // for start, end, content, group, className, editable, type\n if (allUndefined) {\n cols.start = 0;\n cols.end = 1;\n cols.content = 2;\n if (colCount > 3) {\n cols.group = 3;\n }\n if (colCount > 4) {\n cols.className = 4;\n }\n if (colCount > 5) {\n cols.editable = 5;\n }\n if (colCount > 6) {\n cols.type = 6;\n }\n }\n return cols;\n};\n\n/**\n * Set data for the timeline\n * @param {google.visualization.DataTable | Array} data\n */\nlinks.Timeline.prototype.setData = function (data) {\n // unselect any previously selected item\n this.unselectItem();\n if (!data) {\n data = [];\n }\n\n // clear all data\n this.stackCancelAnimation();\n this.clearItems();\n this.data = data;\n var items = this.items;\n this.deleteGroups();\n if (google && google.visualization && data instanceof google.visualization.DataTable) {\n // map the datatable columns\n var cols = links.Timeline.mapColumnIds(data);\n\n // read DataTable\n for (var row = 0, rows = data.getNumberOfRows(); row < rows; row++) {\n items.push(this.createItem({\n 'start': cols.start != undefined ? data.getValue(row, cols.start) : undefined,\n 'end': cols.end != undefined ? data.getValue(row, cols.end) : undefined,\n 'content': cols.content != undefined ? data.getValue(row, cols.content) : undefined,\n 'group': cols.group != undefined ? data.getValue(row, cols.group) : undefined,\n 'className': cols.className != undefined ? data.getValue(row, cols.className) : undefined,\n 'editable': cols.editable != undefined ? data.getValue(row, cols.editable) : undefined,\n 'type': cols.type != undefined ? data.getValue(row, cols.type) : undefined\n }));\n }\n } else if (links.Timeline.isArray(data)) {\n // read JSON array\n for (var row = 0, rows = data.length; row < rows; row++) {\n var itemData = data[row];\n var item = this.createItem(itemData);\n items.push(item);\n }\n } else {\n throw \"Unknown data type. DataTable or Array expected.\";\n }\n\n // prepare data for clustering, by filtering and sorting by type\n if (this.options.cluster) {\n this.clusterGenerator.setData(this.items);\n }\n this.render({\n animate: false\n });\n};\n\n/**\n * Return the original data table.\n * @return {google.visualization.DataTable | Array} data\n */\nlinks.Timeline.prototype.getData = function () {\n return this.data;\n};\n\n/**\n * Update the original data with changed start, end or group.\n *\n * @param {Number} index\n * @param {Object} values An object containing some of the following parameters:\n * {Date} start,\n * {Date} end,\n * {String} content,\n * {String} group\n */\nlinks.Timeline.prototype.updateData = function (index, values) {\n var data = this.data,\n prop;\n if (google && google.visualization && data instanceof google.visualization.DataTable) {\n // update the original google DataTable\n var missingRows = index + 1 - data.getNumberOfRows();\n if (missingRows > 0) {\n data.addRows(missingRows);\n }\n\n // map the column id's by name\n var cols = links.Timeline.mapColumnIds(data);\n\n // merge all fields from the provided data into the current data\n for (prop in values) {\n if (values.hasOwnProperty(prop)) {\n var col = cols[prop];\n if (col == undefined) {\n // create new column\n var value = values[prop];\n var valueType = 'string';\n if (typeof value == 'number') {\n valueType = 'number';\n } else if (typeof value == 'boolean') {\n valueType = 'boolean';\n } else if (value instanceof Date) {\n valueType = 'datetime';\n }\n col = data.addColumn(valueType, prop);\n }\n data.setValue(index, col, values[prop]);\n\n // TODO: correctly serialize the start and end Date to the desired type (Date, String, or Number)\n }\n }\n } else if (links.Timeline.isArray(data)) {\n // update the original JSON table\n var row = data[index];\n if (row == undefined) {\n row = {};\n data[index] = row;\n }\n\n // merge all fields from the provided data into the current data\n for (prop in values) {\n if (values.hasOwnProperty(prop)) {\n row[prop] = values[prop];\n\n // TODO: correctly serialize the start and end Date to the desired type (Date, String, or Number)\n }\n }\n } else {\n throw \"Cannot update data, unknown type of data\";\n }\n};\n\n/**\n * Find the item index from a given HTML element\n * If no item index is found, undefined is returned\n * @param {Element} element\n * @return {Number | undefined} index\n */\nlinks.Timeline.prototype.getItemIndex = function (element) {\n var e = element,\n dom = this.dom,\n frame = dom.items.frame,\n items = this.items,\n index = undefined;\n\n // try to find the frame where the items are located in\n while (e.parentNode && e.parentNode !== frame) {\n e = e.parentNode;\n }\n if (e.parentNode === frame) {\n // yes! we have found the parent element of all items\n // retrieve its id from the array with items\n for (var i = 0, iMax = items.length; i < iMax; i++) {\n if (items[i].dom === e) {\n index = i;\n break;\n }\n }\n }\n return index;\n};\n\n/**\n * Find the cluster index from a given HTML element\n * If no cluster index is found, undefined is returned\n * @param {Element} element\n * @return {Number | undefined} index\n */\nlinks.Timeline.prototype.getClusterIndex = function (element) {\n var e = element,\n dom = this.dom,\n frame = dom.items.frame,\n clusters = this.clusters,\n index = undefined;\n if (this.clusters) {\n // try to find the frame where the clusters are located in\n while (e.parentNode && e.parentNode !== frame) {\n e = e.parentNode;\n }\n if (e.parentNode === frame) {\n // yes! we have found the parent element of all clusters\n // retrieve its id from the array with clusters\n for (var i = 0, iMax = clusters.length; i < iMax; i++) {\n if (clusters[i].dom === e) {\n index = i;\n break;\n }\n }\n }\n }\n return index;\n};\n\n/**\n * Find all elements within the start and end range\n * If no element is found, returns an empty array\n * @param start time\n * @param end time\n * @return Array itemsInRange\n */\nlinks.Timeline.prototype.getVisibleItems = function (start, end) {\n var items = this.items;\n var itemsInRange = [];\n if (items) {\n for (var i = 0, iMax = items.length; i < iMax; i++) {\n var item = items[i];\n if (item.end) {\n // Time range object // NH use getLeft and getRight here\n if (start <= item.start && item.end <= end) {\n itemsInRange.push({\n \"row\": i\n });\n }\n } else {\n // Point object\n if (start <= item.start && item.start <= end) {\n itemsInRange.push({\n \"row\": i\n });\n }\n }\n }\n }\n\n // var sel = [];\n // if (this.selection) {\n // sel.push({\"row\": this.selection.index});\n // }\n // return sel;\n\n return itemsInRange;\n};\n\n/**\n * Set a new size for the timeline\n * @param {string} width Width in pixels or percentage (for example \"800px\"\n * or \"50%\")\n * @param {string} height Height in pixels or percentage (for example \"400px\"\n * or \"30%\")\n */\nlinks.Timeline.prototype.setSize = function (width, height) {\n if (width) {\n this.options.width = width;\n this.dom.frame.style.width = width;\n }\n if (height) {\n this.options.height = height;\n this.options.autoHeight = this.options.height === \"auto\";\n if (height !== \"auto\") {\n this.dom.frame.style.height = height;\n }\n }\n this.render({\n animate: false\n });\n};\n\n/**\n * Set a new value for the visible range int the timeline.\n * Set start undefined to include everything from the earliest date to end.\n * Set end undefined to include everything from start to the last date.\n * Example usage:\n * myTimeline.setVisibleChartRange(new Date(\"2010-08-22\"),\n * new Date(\"2010-09-13\"));\n * @param {Date} start The start date for the timeline. optional\n * @param {Date} end The end date for the timeline. optional\n * @param {boolean} redraw Optional. If true (default) the Timeline is\n * directly redrawn\n */\nlinks.Timeline.prototype.setVisibleChartRange = function (start, end, redraw) {\n var range = {};\n if (!start || !end) {\n // retrieve the date range of the items\n range = this.getDataRange(true);\n }\n if (!start) {\n if (end) {\n if (range.min && range.min.valueOf() < end.valueOf()) {\n // start of the data\n start = range.min;\n } else {\n // 7 days before the end\n start = new Date(end.valueOf());\n start.setDate(start.getDate() - 7);\n }\n } else {\n // default of 3 days ago\n start = new Date();\n start.setDate(start.getDate() - 3);\n }\n }\n if (!end) {\n if (range.max) {\n // end of the data\n end = range.max;\n } else {\n // 7 days after start\n end = new Date(start.valueOf());\n end.setDate(end.getDate() + 7);\n }\n }\n\n // prevent start Date <= end Date\n if (end <= start) {\n end = new Date(start.valueOf());\n end.setDate(end.getDate() + 7);\n }\n\n // limit to the allowed range (don't let this do by applyRange,\n // because that method will try to maintain the interval (end-start)\n var min = this.options.min ? this.options.min : undefined; // date\n if (min != undefined && start.valueOf() < min.valueOf()) {\n start = new Date(min.valueOf()); // date\n }\n\n var max = this.options.max ? this.options.max : undefined; // date\n if (max != undefined && end.valueOf() > max.valueOf()) {\n end = new Date(max.valueOf()); // date\n }\n\n this.applyRange(start, end);\n if (redraw == undefined || redraw == true) {\n this.render({\n animate: false\n }); // TODO: optimize, no reflow needed\n } else {\n this.recalcConversion();\n }\n};\n\n/**\n * Change the visible chart range such that all items become visible\n */\nlinks.Timeline.prototype.setVisibleChartRangeAuto = function () {\n var range = this.getDataRange(true);\n this.setVisibleChartRange(range.min, range.max);\n};\n\n/**\n * Adjust the visible range such that the current time is located in the center\n * of the timeline\n */\nlinks.Timeline.prototype.setVisibleChartRangeNow = function () {\n var now = new Date();\n var diff = this.end.valueOf() - this.start.valueOf();\n var startNew = new Date(now.valueOf() - diff / 2);\n var endNew = new Date(startNew.valueOf() + diff);\n this.setVisibleChartRange(startNew, endNew);\n};\n\n/**\n * Retrieve the current visible range in the timeline.\n * @return {Object} An object with start and end properties\n */\nlinks.Timeline.prototype.getVisibleChartRange = function () {\n return {\n 'start': new Date(this.start.valueOf()),\n 'end': new Date(this.end.valueOf())\n };\n};\n\n/**\n * Get the date range of the items.\n * @param {boolean} [withMargin] If true, 5% of whitespace is added to the\n * left and right of the range. Default is false.\n * @return {Object} range An object with parameters min and max.\n * - {Date} min is the lowest start date of the items\n * - {Date} max is the highest start or end date of the items\n * If no data is available, the values of min and max\n * will be undefined\n */\nlinks.Timeline.prototype.getDataRange = function (withMargin) {\n var items = this.items,\n min = undefined,\n // number\n max = undefined; // number\n\n if (items) {\n for (var i = 0, iMax = items.length; i < iMax; i++) {\n var item = items[i],\n start = item.start != undefined ? item.start.valueOf() : undefined,\n end = item.end != undefined ? item.end.valueOf() : start;\n if (start != undefined) {\n min = min != undefined ? Math.min(min.valueOf(), start.valueOf()) : start;\n }\n if (end != undefined) {\n max = max != undefined ? Math.max(max.valueOf(), end.valueOf()) : end;\n }\n }\n }\n if (min && max && withMargin) {\n // zoom out 5% such that you have a little white space on the left and right\n var diff = max - min;\n min = min - diff * 0.05;\n max = max + diff * 0.05;\n }\n return {\n 'min': min != undefined ? new Date(min) : undefined,\n 'max': max != undefined ? new Date(max) : undefined\n };\n};\n\n/**\n * Re-render (reflow and repaint) all components of the Timeline: frame, axis,\n * items, ...\n * @param {Object} [options] Available options:\n * {boolean} renderTimesLeft Number of times the\n * render may be repeated\n * 5 times by default.\n * {boolean} animate takes options.animate\n * as default value\n */\nlinks.Timeline.prototype.render = function (options) {\n var frameResized = this.reflowFrame();\n var axisResized = this.reflowAxis();\n var groupsResized = this.reflowGroups();\n var itemsResized = this.reflowItems();\n var resized = frameResized || axisResized || groupsResized || itemsResized;\n\n // TODO: only stackEvents/filterItems when resized or changed. (gives a bootstrap issue).\n // if (resized) {\n var animate = this.options.animate;\n if (options && options.animate != undefined) {\n animate = options.animate;\n }\n this.recalcConversion();\n this.clusterItems();\n this.filterItems();\n this.stackItems(animate);\n this.recalcItems();\n\n // TODO: only repaint when resized or when filterItems or stackItems gave a change?\n var needsReflow = this.repaint();\n\n // re-render once when needed (prevent endless re-render loop)\n if (needsReflow) {\n var renderTimesLeft = options ? options.renderTimesLeft : undefined;\n if (renderTimesLeft == undefined) {\n renderTimesLeft = 5;\n }\n if (renderTimesLeft > 0) {\n this.render({\n 'animate': options ? options.animate : undefined,\n 'renderTimesLeft': renderTimesLeft - 1\n });\n }\n }\n};\n\n/**\n * Repaint all components of the Timeline\n * @return {boolean} needsReflow Returns true if the DOM is changed such that\n * a reflow is needed.\n */\nlinks.Timeline.prototype.repaint = function () {\n var frameNeedsReflow = this.repaintFrame();\n var axisNeedsReflow = this.repaintAxis();\n var groupsNeedsReflow = this.repaintGroups();\n var itemsNeedsReflow = this.repaintItems();\n this.repaintCurrentTime();\n this.repaintCustomTime();\n return frameNeedsReflow || axisNeedsReflow || groupsNeedsReflow || itemsNeedsReflow;\n};\n\n/**\n * Reflow the timeline frame\n * @return {boolean} resized Returns true if any of the frame elements\n * have been resized.\n */\nlinks.Timeline.prototype.reflowFrame = function () {\n var dom = this.dom,\n options = this.options,\n size = this.size,\n resized = false;\n\n // Note: IE7 has issues with giving frame.clientWidth, therefore I use offsetWidth instead\n var frameWidth = dom.frame ? dom.frame.offsetWidth : 0,\n frameHeight = dom.frame ? dom.frame.clientHeight : 0;\n resized = resized || size.frameWidth !== frameWidth;\n resized = resized || size.frameHeight !== frameHeight;\n size.frameWidth = frameWidth;\n size.frameHeight = frameHeight;\n return resized;\n};\n\n/**\n * repaint the Timeline frame\n * @return {boolean} needsReflow Returns true if the DOM is changed such that\n * a reflow is needed.\n */\nlinks.Timeline.prototype.repaintFrame = function () {\n var needsReflow = false,\n dom = this.dom,\n options = this.options,\n size = this.size;\n\n // main frame\n if (!dom.frame) {\n dom.frame = document.createElement(\"DIV\");\n dom.frame.className = \"timeline-frame ui-widget ui-widget-content ui-corner-all\";\n dom.container.appendChild(dom.frame);\n needsReflow = true;\n }\n var height = options.autoHeight ? size.actualHeight + \"px\" : options.height || \"100%\";\n var width = options.width || \"100%\";\n needsReflow = needsReflow || dom.frame.style.height != height;\n needsReflow = needsReflow || dom.frame.style.width != width;\n dom.frame.style.height = height;\n dom.frame.style.width = width;\n\n // contents\n if (!dom.content) {\n // create content box where the axis and items will be created\n dom.content = document.createElement(\"DIV\");\n dom.content.className = \"timeline-content\";\n dom.frame.appendChild(dom.content);\n var timelines = document.createElement(\"DIV\");\n timelines.style.position = \"absolute\";\n timelines.style.left = \"0px\";\n timelines.style.top = \"0px\";\n timelines.style.height = \"100%\";\n timelines.style.width = \"0px\";\n dom.content.appendChild(timelines);\n dom.contentTimelines = timelines;\n var params = this.eventParams,\n me = this;\n if (!params.onMouseDown) {\n params.onMouseDown = function (event) {\n me.onMouseDown(event);\n };\n links.Timeline.addEventListener(dom.content, \"mousedown\", params.onMouseDown);\n }\n if (!params.onTouchStart) {\n params.onTouchStart = function (event) {\n me.onTouchStart(event);\n };\n links.Timeline.addEventListener(dom.content, \"touchstart\", params.onTouchStart);\n }\n if (!params.onMouseWheel) {\n params.onMouseWheel = function (event) {\n me.onMouseWheel(event);\n };\n links.Timeline.addEventListener(dom.content, \"mousewheel\", params.onMouseWheel);\n }\n if (!params.onDblClick) {\n params.onDblClick = function (event) {\n me.onDblClick(event);\n };\n links.Timeline.addEventListener(dom.content, \"dblclick\", params.onDblClick);\n }\n needsReflow = true;\n }\n dom.content.style.left = size.contentLeft + \"px\";\n dom.content.style.top = \"0px\";\n dom.content.style.width = size.contentWidth + \"px\";\n dom.content.style.height = size.frameHeight + \"px\";\n this.repaintNavigation();\n return needsReflow;\n};\n\n/**\n * Reflow the timeline axis. Calculate its height, width, positioning, etc...\n * @return {boolean} resized returns true if the axis is resized\n */\nlinks.Timeline.prototype.reflowAxis = function () {\n var resized = false,\n dom = this.dom,\n options = this.options,\n size = this.size,\n axisDom = dom.axis;\n var characterMinorWidth = axisDom && axisDom.characterMinor ? axisDom.characterMinor.clientWidth : 0,\n characterMinorHeight = axisDom && axisDom.characterMinor ? axisDom.characterMinor.clientHeight : 0,\n characterMajorWidth = axisDom && axisDom.characterMajor ? axisDom.characterMajor.clientWidth : 0,\n characterMajorHeight = axisDom && axisDom.characterMajor ? axisDom.characterMajor.clientHeight : 0,\n axisHeight = (options.showMinorLabels ? characterMinorHeight : 0) + (options.showMajorLabels ? characterMajorHeight : 0);\n var axisTop = options.axisOnTop ? 0 : size.frameHeight - axisHeight,\n axisLine = options.axisOnTop ? axisHeight : axisTop;\n resized = resized || size.axis.top !== axisTop;\n resized = resized || size.axis.line !== axisLine;\n resized = resized || size.axis.height !== axisHeight;\n size.axis.top = axisTop;\n size.axis.line = axisLine;\n size.axis.height = axisHeight;\n size.axis.labelMajorTop = options.axisOnTop ? 0 : axisLine + (options.showMinorLabels ? characterMinorHeight : 0);\n size.axis.labelMinorTop = options.axisOnTop ? options.showMajorLabels ? characterMajorHeight : 0 : axisLine;\n size.axis.lineMinorTop = options.axisOnTop ? size.axis.labelMinorTop : 0;\n size.axis.lineMinorHeight = options.showMajorLabels ? size.frameHeight - characterMajorHeight : size.frameHeight;\n if (axisDom && axisDom.minorLines && axisDom.minorLines.length) {\n size.axis.lineMinorWidth = axisDom.minorLines[0].offsetWidth;\n } else {\n size.axis.lineMinorWidth = 1;\n }\n if (axisDom && axisDom.majorLines && axisDom.majorLines.length) {\n size.axis.lineMajorWidth = axisDom.majorLines[0].offsetWidth;\n } else {\n size.axis.lineMajorWidth = 1;\n }\n resized = resized || size.axis.characterMinorWidth !== characterMinorWidth;\n resized = resized || size.axis.characterMinorHeight !== characterMinorHeight;\n resized = resized || size.axis.characterMajorWidth !== characterMajorWidth;\n resized = resized || size.axis.characterMajorHeight !== characterMajorHeight;\n size.axis.characterMinorWidth = characterMinorWidth;\n size.axis.characterMinorHeight = characterMinorHeight;\n size.axis.characterMajorWidth = characterMajorWidth;\n size.axis.characterMajorHeight = characterMajorHeight;\n var contentHeight = Math.max(size.frameHeight - axisHeight, 0);\n size.contentLeft = options.groupsOnRight ? 0 : size.groupsWidth;\n size.contentWidth = Math.max(size.frameWidth - size.groupsWidth, 0);\n size.contentHeight = contentHeight;\n return resized;\n};\n\n/**\n * Redraw the timeline axis with minor and major labels\n * @return {boolean} needsReflow Returns true if the DOM is changed such\n * that a reflow is needed.\n */\nlinks.Timeline.prototype.repaintAxis = function () {\n var needsReflow = false,\n dom = this.dom,\n options = this.options,\n size = this.size,\n step = this.step;\n var axis = dom.axis;\n if (!axis) {\n axis = {};\n dom.axis = axis;\n }\n if (!size.axis.properties) {\n size.axis.properties = {};\n }\n if (!axis.minorTexts) {\n axis.minorTexts = [];\n }\n if (!axis.minorLines) {\n axis.minorLines = [];\n }\n if (!axis.majorTexts) {\n axis.majorTexts = [];\n }\n if (!axis.majorLines) {\n axis.majorLines = [];\n }\n if (!axis.frame) {\n axis.frame = document.createElement(\"DIV\");\n axis.frame.style.position = \"absolute\";\n axis.frame.style.left = \"0px\";\n axis.frame.style.top = \"0px\";\n dom.content.appendChild(axis.frame);\n }\n\n // take axis offline\n dom.content.removeChild(axis.frame);\n axis.frame.style.width = size.contentWidth + \"px\";\n axis.frame.style.height = size.axis.height + \"px\";\n\n // the drawn axis is more wide than the actual visual part, such that\n // the axis can be dragged without having to redraw it each time again.\n var start = this.screenToTime(0);\n var end = this.screenToTime(size.contentWidth);\n\n // calculate minimum step (in milliseconds) based on character size\n if (size.axis.characterMinorWidth) {\n this.minimumStep = this.screenToTime(size.axis.characterMinorWidth * 6) - this.screenToTime(0);\n step.setRange(start, end, this.minimumStep);\n }\n var charsNeedsReflow = this.repaintAxisCharacters();\n needsReflow = needsReflow || charsNeedsReflow;\n\n // The current labels on the axis will be re-used (much better performance),\n // therefore, the repaintAxis method uses the mechanism with\n // repaintAxisStartOverwriting, repaintAxisEndOverwriting, and\n // this.size.axis.properties is used.\n this.repaintAxisStartOverwriting();\n step.start();\n var xFirstMajorLabel = undefined;\n var max = 0;\n while (!step.end() && max < 1000) {\n max++;\n var cur = step.getCurrent(),\n x = this.timeToScreen(cur),\n isMajor = step.isMajor();\n if (options.showMinorLabels) {\n this.repaintAxisMinorText(x, step.getLabelMinor(options));\n }\n if (isMajor && options.showMajorLabels) {\n if (x > 0) {\n if (xFirstMajorLabel == undefined) {\n xFirstMajorLabel = x;\n }\n this.repaintAxisMajorText(x, step.getLabelMajor(options));\n }\n this.repaintAxisMajorLine(x);\n } else {\n this.repaintAxisMinorLine(x);\n }\n step.next();\n }\n\n // create a major label on the left when needed\n if (options.showMajorLabels) {\n var leftTime = this.screenToTime(0),\n leftText = this.step.getLabelMajor(options, leftTime),\n width = leftText.length * size.axis.characterMajorWidth + 10; // upper bound estimation\n\n if (xFirstMajorLabel == undefined || width < xFirstMajorLabel) {\n this.repaintAxisMajorText(0, leftText, leftTime);\n }\n }\n\n // cleanup left over labels\n this.repaintAxisEndOverwriting();\n this.repaintAxisHorizontal();\n\n // put axis online\n dom.content.insertBefore(axis.frame, dom.content.firstChild);\n return needsReflow;\n};\n\n/**\n * Create characters used to determine the size of text on the axis\n * @return {boolean} needsReflow Returns true if the DOM is changed such that\n * a reflow is needed.\n */\nlinks.Timeline.prototype.repaintAxisCharacters = function () {\n // calculate the width and height of a single character\n // this is used to calculate the step size, and also the positioning of the\n // axis\n var needsReflow = false,\n dom = this.dom,\n axis = dom.axis,\n text;\n if (!axis.characterMinor) {\n text = document.createTextNode(\"0\");\n var characterMinor = document.createElement(\"DIV\");\n characterMinor.className = \"timeline-axis-text timeline-axis-text-minor\";\n characterMinor.appendChild(text);\n characterMinor.style.position = \"absolute\";\n characterMinor.style.visibility = \"hidden\";\n characterMinor.style.paddingLeft = \"0px\";\n characterMinor.style.paddingRight = \"0px\";\n axis.frame.appendChild(characterMinor);\n axis.characterMinor = characterMinor;\n needsReflow = true;\n }\n if (!axis.characterMajor) {\n text = document.createTextNode(\"0\");\n var characterMajor = document.createElement(\"DIV\");\n characterMajor.className = \"timeline-axis-text timeline-axis-text-major\";\n characterMajor.appendChild(text);\n characterMajor.style.position = \"absolute\";\n characterMajor.style.visibility = \"hidden\";\n characterMajor.style.paddingLeft = \"0px\";\n characterMajor.style.paddingRight = \"0px\";\n axis.frame.appendChild(characterMajor);\n axis.characterMajor = characterMajor;\n needsReflow = true;\n }\n return needsReflow;\n};\n\n/**\n * Initialize redraw of the axis. All existing labels and lines will be\n * overwritten and reused.\n */\nlinks.Timeline.prototype.repaintAxisStartOverwriting = function () {\n var properties = this.size.axis.properties;\n properties.minorTextNum = 0;\n properties.minorLineNum = 0;\n properties.majorTextNum = 0;\n properties.majorLineNum = 0;\n};\n\n/**\n * End of overwriting HTML DOM elements of the axis.\n * remaining elements will be removed\n */\nlinks.Timeline.prototype.repaintAxisEndOverwriting = function () {\n var dom = this.dom,\n props = this.size.axis.properties,\n frame = this.dom.axis.frame,\n num;\n\n // remove leftovers\n var minorTexts = dom.axis.minorTexts;\n num = props.minorTextNum;\n while (minorTexts.length > num) {\n var minorText = minorTexts[num];\n frame.removeChild(minorText);\n minorTexts.splice(num, 1);\n }\n var minorLines = dom.axis.minorLines;\n num = props.minorLineNum;\n while (minorLines.length > num) {\n var minorLine = minorLines[num];\n frame.removeChild(minorLine);\n minorLines.splice(num, 1);\n }\n var majorTexts = dom.axis.majorTexts;\n num = props.majorTextNum;\n while (majorTexts.length > num) {\n var majorText = majorTexts[num];\n frame.removeChild(majorText);\n majorTexts.splice(num, 1);\n }\n var majorLines = dom.axis.majorLines;\n num = props.majorLineNum;\n while (majorLines.length > num) {\n var majorLine = majorLines[num];\n frame.removeChild(majorLine);\n majorLines.splice(num, 1);\n }\n};\n\n/**\n * Repaint the horizontal line and background of the axis\n */\nlinks.Timeline.prototype.repaintAxisHorizontal = function () {\n var axis = this.dom.axis,\n size = this.size,\n options = this.options;\n\n // line behind all axis elements (possibly having a background color)\n var hasAxis = options.showMinorLabels || options.showMajorLabels;\n if (hasAxis) {\n if (!axis.backgroundLine) {\n // create the axis line background (for a background color or so)\n var backgroundLine = document.createElement(\"DIV\");\n backgroundLine.className = \"timeline-axis\";\n backgroundLine.style.position = \"absolute\";\n backgroundLine.style.left = \"0px\";\n backgroundLine.style.width = \"100%\";\n backgroundLine.style.border = \"none\";\n axis.frame.insertBefore(backgroundLine, axis.frame.firstChild);\n axis.backgroundLine = backgroundLine;\n }\n if (axis.backgroundLine) {\n axis.backgroundLine.style.top = size.axis.top + \"px\";\n axis.backgroundLine.style.height = size.axis.height + \"px\";\n }\n } else {\n if (axis.backgroundLine) {\n axis.frame.removeChild(axis.backgroundLine);\n delete axis.backgroundLine;\n }\n }\n\n // line before all axis elements\n if (hasAxis) {\n if (axis.line) {\n // put this line at the end of all childs\n var line = axis.frame.removeChild(axis.line);\n axis.frame.appendChild(line);\n } else {\n // make the axis line\n var line = document.createElement(\"DIV\");\n line.className = \"timeline-axis\";\n line.style.position = \"absolute\";\n line.style.left = \"0px\";\n line.style.width = \"100%\";\n line.style.height = \"0px\";\n axis.frame.appendChild(line);\n axis.line = line;\n }\n axis.line.style.top = size.axis.line + \"px\";\n } else {\n if (axis.line && axis.line.parentElement) {\n axis.frame.removeChild(axis.line);\n delete axis.line;\n }\n }\n};\n\n/**\n * Create a minor label for the axis at position x\n * @param {Number} x\n * @param {String} text\n */\nlinks.Timeline.prototype.repaintAxisMinorText = function (x, text) {\n var size = this.size,\n dom = this.dom,\n props = size.axis.properties,\n frame = dom.axis.frame,\n minorTexts = dom.axis.minorTexts,\n index = props.minorTextNum,\n label;\n if (index < minorTexts.length) {\n label = minorTexts[index];\n } else {\n // create new label\n var content = document.createTextNode(\"\");\n label = document.createElement(\"DIV\");\n label.appendChild(content);\n label.className = \"timeline-axis-text timeline-axis-text-minor\";\n label.style.position = \"absolute\";\n frame.appendChild(label);\n minorTexts.push(label);\n }\n label.childNodes[0].nodeValue = text;\n label.style.left = x + \"px\";\n label.style.top = size.axis.labelMinorTop + \"px\";\n //label.title = title; // TODO: this is a heavy operation\n\n props.minorTextNum++;\n};\n\n/**\n * Create a minor line for the axis at position x\n * @param {Number} x\n */\nlinks.Timeline.prototype.repaintAxisMinorLine = function (x) {\n var axis = this.size.axis,\n dom = this.dom,\n props = axis.properties,\n frame = dom.axis.frame,\n minorLines = dom.axis.minorLines,\n index = props.minorLineNum,\n line;\n if (index < minorLines.length) {\n line = minorLines[index];\n } else {\n // create vertical line\n line = document.createElement(\"DIV\");\n line.className = \"timeline-axis-grid timeline-axis-grid-minor\";\n line.style.position = \"absolute\";\n line.style.width = \"0px\";\n frame.appendChild(line);\n minorLines.push(line);\n }\n line.style.top = axis.lineMinorTop + \"px\";\n line.style.height = axis.lineMinorHeight + \"px\";\n line.style.left = x - axis.lineMinorWidth / 2 + \"px\";\n props.minorLineNum++;\n};\n\n/**\n * Create a Major label for the axis at position x\n * @param {Number} x\n * @param {String} text\n */\nlinks.Timeline.prototype.repaintAxisMajorText = function (x, text) {\n var size = this.size,\n props = size.axis.properties,\n frame = this.dom.axis.frame,\n majorTexts = this.dom.axis.majorTexts,\n index = props.majorTextNum,\n label;\n if (index < majorTexts.length) {\n label = majorTexts[index];\n } else {\n // create label\n var content = document.createTextNode(text);\n label = document.createElement(\"DIV\");\n label.className = \"timeline-axis-text timeline-axis-text-major\";\n label.appendChild(content);\n label.style.position = \"absolute\";\n label.style.top = \"0px\";\n frame.appendChild(label);\n majorTexts.push(label);\n }\n label.childNodes[0].nodeValue = text;\n label.style.top = size.axis.labelMajorTop + \"px\";\n label.style.left = x + \"px\";\n //label.title = title; // TODO: this is a heavy operation\n\n props.majorTextNum++;\n};\n\n/**\n * Create a Major line for the axis at position x\n * @param {Number} x\n */\nlinks.Timeline.prototype.repaintAxisMajorLine = function (x) {\n var size = this.size,\n props = size.axis.properties,\n axis = this.size.axis,\n frame = this.dom.axis.frame,\n majorLines = this.dom.axis.majorLines,\n index = props.majorLineNum,\n line;\n if (index < majorLines.length) {\n line = majorLines[index];\n } else {\n // create vertical line\n line = document.createElement(\"DIV\");\n line.className = \"timeline-axis-grid timeline-axis-grid-major\";\n line.style.position = \"absolute\";\n line.style.top = \"0px\";\n line.style.width = \"0px\";\n frame.appendChild(line);\n majorLines.push(line);\n }\n line.style.left = x - axis.lineMajorWidth / 2 + \"px\";\n line.style.height = size.frameHeight + \"px\";\n props.majorLineNum++;\n};\n\n/**\n * Reflow all items, retrieve their actual size\n * @return {boolean} resized returns true if any of the items is resized\n */\nlinks.Timeline.prototype.reflowItems = function () {\n var resized = false,\n i,\n iMax,\n group,\n groups = this.groups,\n renderedItems = this.renderedItems;\n if (groups) {\n // TODO: need to check if labels exists?\n // loop through all groups to reset the items height\n groups.forEach(function (group) {\n group.itemsHeight = group.labelHeight || 0;\n });\n }\n\n // loop through the width and height of all visible items\n for (i = 0, iMax = renderedItems.length; i < iMax; i++) {\n var item = renderedItems[i],\n domItem = item.dom;\n group = item.group;\n if (domItem) {\n // TODO: move updating width and height into item.reflow\n var width = domItem ? domItem.clientWidth : 0;\n var height = domItem ? domItem.clientHeight : 0;\n resized = resized || item.width != width;\n resized = resized || item.height != height;\n item.width = width;\n item.height = height;\n //item.borderWidth = (domItem.offsetWidth - domItem.clientWidth - 2) / 2; // TODO: borderWidth\n item.reflow();\n }\n if (group) {\n group.itemsHeight = Math.max(this.options.groupMinHeight, group.itemsHeight ? Math.max(group.itemsHeight, item.height) : item.height);\n }\n }\n return resized;\n};\n\n/**\n * Recalculate item properties:\n * - the height of each group.\n * - the actualHeight, from the stacked items or the sum of the group heights\n * @return {boolean} resized returns true if any of the items properties is\n * changed\n */\nlinks.Timeline.prototype.recalcItems = function () {\n var resized = false,\n i,\n iMax,\n item,\n finalItem,\n finalItems,\n group,\n groups = this.groups,\n size = this.size,\n options = this.options,\n renderedItems = this.renderedItems;\n var actualHeight = 0;\n if (groups.length == 0) {\n // calculate actual height of the timeline when there are no groups\n // but stacked items\n if (options.autoHeight || options.cluster) {\n var min = 0,\n max = 0;\n if (this.stack && this.stack.finalItems) {\n // adjust the offset of all finalItems when the actualHeight has been changed\n finalItems = this.stack.finalItems;\n finalItem = finalItems[0];\n if (finalItem && finalItem.top) {\n min = finalItem.top;\n max = finalItem.top + finalItem.height;\n }\n for (i = 1, iMax = finalItems.length; i < iMax; i++) {\n finalItem = finalItems[i];\n min = Math.min(min, finalItem.top);\n max = Math.max(max, finalItem.top + finalItem.height);\n }\n } else {\n item = renderedItems[0];\n if (item && item.top) {\n min = item.top;\n max = item.top + item.height;\n }\n for (i = 1, iMax = renderedItems.length; i < iMax; i++) {\n item = renderedItems[i];\n if (item.top) {\n min = Math.min(min, item.top);\n max = Math.max(max, item.top + item.height);\n }\n }\n }\n actualHeight = max - min + 2 * options.eventMarginAxis + size.axis.height;\n if (actualHeight < options.minHeight) {\n actualHeight = options.minHeight;\n }\n if (size.actualHeight != actualHeight && options.autoHeight && !options.axisOnTop) {\n // adjust the offset of all items when the actualHeight has been changed\n var diff = actualHeight - size.actualHeight;\n if (this.stack && this.stack.finalItems) {\n finalItems = this.stack.finalItems;\n for (i = 0, iMax = finalItems.length; i < iMax; i++) {\n finalItems[i].top += diff;\n finalItems[i].item.top += diff;\n }\n } else {\n for (i = 0, iMax = renderedItems.length; i < iMax; i++) {\n renderedItems[i].top += diff;\n }\n }\n }\n }\n } else {\n // loop through all groups to get the height of each group, and the\n // total height\n actualHeight = size.axis.height + 2 * options.eventMarginAxis;\n for (i = 0, iMax = groups.length; i < iMax; i++) {\n group = groups[i];\n\n //\n // TODO: Do we want to apply a max height? how ?\n //\n var groupHeight = group.itemsHeight;\n resized = resized || groupHeight != group.height;\n group.height = Math.max(groupHeight, options.groupMinHeight);\n actualHeight += groups[i].height + options.eventMargin;\n }\n\n // calculate top positions of the group labels and lines\n var eventMargin = options.eventMargin,\n top = options.axisOnTop ? options.eventMarginAxis + eventMargin / 2 : size.contentHeight - options.eventMarginAxis + eventMargin / 2,\n axisHeight = size.axis.height;\n for (i = 0, iMax = groups.length; i < iMax; i++) {\n group = groups[i];\n if (options.axisOnTop) {\n group.top = top + axisHeight;\n group.labelTop = top + axisHeight + (group.height - group.labelHeight) / 2;\n group.lineTop = top + axisHeight + group.height + eventMargin / 2;\n top += group.height + eventMargin;\n } else {\n top -= group.height + eventMargin;\n group.top = top;\n group.labelTop = top + (group.height - group.labelHeight) / 2;\n group.lineTop = top - eventMargin / 2;\n }\n }\n resized = true;\n }\n if (actualHeight < options.minHeight) {\n actualHeight = options.minHeight;\n }\n resized = resized || actualHeight != size.actualHeight;\n size.actualHeight = actualHeight;\n return resized;\n};\n\n/**\n * This method clears the (internal) array this.items in a safe way: neatly\n * cleaning up the DOM, and accompanying arrays this.renderedItems and\n * the created clusters.\n */\nlinks.Timeline.prototype.clearItems = function () {\n // add all visible items to the list to be hidden\n var hideItems = this.renderQueue.hide;\n this.renderedItems.forEach(function (item) {\n hideItems.push(item);\n });\n\n // clear the cluster generator\n this.clusterGenerator.clear();\n\n // actually clear the items\n this.items = [];\n};\n\n/**\n * Repaint all items\n * @return {boolean} needsReflow Returns true if the DOM is changed such that\n * a reflow is needed.\n */\nlinks.Timeline.prototype.repaintItems = function () {\n var i, iMax, item, index;\n var needsReflow = false,\n dom = this.dom,\n size = this.size,\n timeline = this,\n renderedItems = this.renderedItems;\n if (!dom.items) {\n dom.items = {};\n }\n\n // draw the frame containing the items\n var frame = dom.items.frame;\n if (!frame) {\n frame = document.createElement(\"DIV\");\n frame.style.position = \"relative\";\n dom.content.appendChild(frame);\n dom.items.frame = frame;\n }\n frame.style.left = \"0px\";\n frame.style.top = size.items.top + \"px\";\n frame.style.height = \"0px\";\n\n // Take frame offline (for faster manipulation of the DOM)\n dom.content.removeChild(frame);\n\n // process the render queue with changes\n var queue = this.renderQueue;\n var newImageUrls = [];\n needsReflow = needsReflow || queue.show.length > 0 || queue.update.length > 0 || queue.hide.length > 0; // TODO: reflow needed on hide of items?\n\n while (item = queue.show.shift()) {\n item.showDOM(frame);\n item.getImageUrls(newImageUrls);\n renderedItems.push(item);\n }\n while (item = queue.update.shift()) {\n item.updateDOM(frame);\n item.getImageUrls(newImageUrls);\n index = this.renderedItems.indexOf(item);\n if (index == -1) {\n renderedItems.push(item);\n }\n }\n while (item = queue.hide.shift()) {\n item.hideDOM(frame);\n index = this.renderedItems.indexOf(item);\n if (index != -1) {\n renderedItems.splice(index, 1);\n }\n }\n\n // reposition all visible items\n renderedItems.forEach(function (item) {\n item.updatePosition(timeline);\n });\n\n // redraw the delete button and dragareas of the selected item (if any)\n this.repaintDeleteButton();\n this.repaintDragAreas();\n\n // put frame online again\n dom.content.appendChild(frame);\n if (newImageUrls.length) {\n // retrieve all image sources from the items, and set a callback once\n // all images are retrieved\n var callback = function callback() {\n timeline.render();\n };\n var sendCallbackWhenAlreadyLoaded = false;\n links.imageloader.loadAll(newImageUrls, callback, sendCallbackWhenAlreadyLoaded);\n }\n return needsReflow;\n};\n\n/**\n * Reflow the size of the groups\n * @return {boolean} resized Returns true if any of the frame elements\n * have been resized.\n */\nlinks.Timeline.prototype.reflowGroups = function () {\n var resized = false,\n options = this.options,\n size = this.size,\n dom = this.dom;\n\n // calculate the groups width and height\n // TODO: only update when data is changed! -> use an updateSeq\n var groupsWidth = 0;\n\n // loop through all groups to get the labels width and height\n var groups = this.groups;\n var labels = this.dom.groups ? this.dom.groups.labels : [];\n for (var i = 0, iMax = groups.length; i < iMax; i++) {\n var group = groups[i];\n var label = labels[i];\n group.labelWidth = label ? label.clientWidth : 0;\n group.labelHeight = label ? label.clientHeight : 0;\n group.width = group.labelWidth; // TODO: group.width is redundant with labelWidth\n\n groupsWidth = Math.max(groupsWidth, group.width);\n }\n\n // limit groupsWidth to the groups width in the options\n if (options.groupsWidth !== undefined) {\n groupsWidth = dom.groups && dom.groups.frame ? dom.groups.frame.clientWidth : 0;\n }\n\n // compensate for the border width. TODO: calculate the real border width\n groupsWidth += 1;\n var groupsLeft = options.groupsOnRight ? size.frameWidth - groupsWidth : 0;\n resized = resized || size.groupsWidth !== groupsWidth;\n resized = resized || size.groupsLeft !== groupsLeft;\n size.groupsWidth = groupsWidth;\n size.groupsLeft = groupsLeft;\n return resized;\n};\n\n/**\n * Redraw the group labels\n */\nlinks.Timeline.prototype.repaintGroups = function () {\n var dom = this.dom,\n timeline = this,\n options = this.options,\n size = this.size,\n groups = this.groups;\n if (dom.groups === undefined) {\n dom.groups = {};\n }\n var labels = dom.groups.labels;\n if (!labels) {\n labels = [];\n dom.groups.labels = labels;\n }\n var labelLines = dom.groups.labelLines;\n if (!labelLines) {\n labelLines = [];\n dom.groups.labelLines = labelLines;\n }\n var itemLines = dom.groups.itemLines;\n if (!itemLines) {\n itemLines = [];\n dom.groups.itemLines = itemLines;\n }\n\n // create the frame for holding the groups\n var frame = dom.groups.frame;\n if (!frame) {\n frame = document.createElement(\"DIV\");\n frame.className = \"timeline-groups-axis\";\n frame.style.position = \"absolute\";\n frame.style.overflow = \"hidden\";\n frame.style.top = \"0px\";\n frame.style.height = \"100%\";\n dom.frame.appendChild(frame);\n dom.groups.frame = frame;\n }\n frame.style.left = size.groupsLeft + \"px\";\n frame.style.width = options.groupsWidth !== undefined ? options.groupsWidth : size.groupsWidth + \"px\";\n\n // hide groups axis when there are no groups\n if (groups.length == 0) {\n frame.style.display = 'none';\n } else {\n frame.style.display = '';\n }\n\n // TODO: only create/update groups when data is changed.\n\n // create the items\n var current = labels.length,\n needed = groups.length;\n\n // overwrite existing group labels\n for (var i = 0, iMax = Math.min(current, needed); i < iMax; i++) {\n var group = groups[i];\n var label = labels[i];\n label.innerHTML = this.getGroupName(group);\n label.style.display = '';\n }\n\n // append new items when needed\n for (var i = current; i < needed; i++) {\n var group = groups[i];\n\n // create text label\n var label = document.createElement(\"DIV\");\n label.className = \"timeline-groups-text\";\n label.style.position = \"absolute\";\n if (options.groupsWidth === undefined) {\n label.style.whiteSpace = \"nowrap\";\n }\n label.innerHTML = this.getGroupName(group);\n frame.appendChild(label);\n labels[i] = label;\n\n // create the grid line between the group labels\n var labelLine = document.createElement(\"DIV\");\n labelLine.className = \"timeline-axis-grid timeline-axis-grid-minor\";\n labelLine.style.position = \"absolute\";\n labelLine.style.left = \"0px\";\n labelLine.style.width = \"100%\";\n labelLine.style.height = \"0px\";\n labelLine.style.borderTopStyle = \"solid\";\n frame.appendChild(labelLine);\n labelLines[i] = labelLine;\n\n // create the grid line between the items\n var itemLine = document.createElement(\"DIV\");\n itemLine.className = \"timeline-axis-grid timeline-axis-grid-minor\";\n itemLine.style.position = \"absolute\";\n itemLine.style.left = \"0px\";\n itemLine.style.width = \"100%\";\n itemLine.style.height = \"0px\";\n itemLine.style.borderTopStyle = \"solid\";\n dom.content.insertBefore(itemLine, dom.content.firstChild);\n itemLines[i] = itemLine;\n }\n\n // remove redundant items from the DOM when needed\n for (var i = needed; i < current; i++) {\n var label = labels[i],\n labelLine = labelLines[i],\n itemLine = itemLines[i];\n frame.removeChild(label);\n frame.removeChild(labelLine);\n dom.content.removeChild(itemLine);\n }\n labels.splice(needed, current - needed);\n labelLines.splice(needed, current - needed);\n itemLines.splice(needed, current - needed);\n links.Timeline.addClassName(frame, options.groupsOnRight ? 'timeline-groups-axis-onright' : 'timeline-groups-axis-onleft');\n\n // position the groups\n for (var i = 0, iMax = groups.length; i < iMax; i++) {\n var group = groups[i],\n label = labels[i],\n labelLine = labelLines[i],\n itemLine = itemLines[i];\n label.style.top = group.labelTop + \"px\";\n labelLine.style.top = group.lineTop + \"px\";\n itemLine.style.top = group.lineTop + \"px\";\n itemLine.style.width = size.contentWidth + \"px\";\n }\n if (!dom.groups.background) {\n // create the axis grid line background\n var background = document.createElement(\"DIV\");\n background.className = \"timeline-axis\";\n background.style.position = \"absolute\";\n background.style.left = \"0px\";\n background.style.width = \"100%\";\n background.style.border = \"none\";\n frame.appendChild(background);\n dom.groups.background = background;\n }\n dom.groups.background.style.top = size.axis.top + 'px';\n dom.groups.background.style.height = size.axis.height + 'px';\n if (!dom.groups.line) {\n // create the axis grid line\n var line = document.createElement(\"DIV\");\n line.className = \"timeline-axis\";\n line.style.position = \"absolute\";\n line.style.left = \"0px\";\n line.style.width = \"100%\";\n line.style.height = \"0px\";\n frame.appendChild(line);\n dom.groups.line = line;\n }\n dom.groups.line.style.top = size.axis.line + 'px';\n\n // create a callback when there are images which are not yet loaded\n // TODO: more efficiently load images in the groups\n if (dom.groups.frame && groups.length) {\n var imageUrls = [];\n links.imageloader.filterImageUrls(dom.groups.frame, imageUrls);\n if (imageUrls.length) {\n // retrieve all image sources from the items, and set a callback once\n // all images are retrieved\n var callback = function callback() {\n timeline.render();\n };\n var sendCallbackWhenAlreadyLoaded = false;\n links.imageloader.loadAll(imageUrls, callback, sendCallbackWhenAlreadyLoaded);\n }\n }\n};\n\n/**\n * Redraw the current time bar\n */\nlinks.Timeline.prototype.repaintCurrentTime = function () {\n var options = this.options,\n dom = this.dom,\n size = this.size;\n if (!options.showCurrentTime) {\n if (dom.currentTime) {\n dom.contentTimelines.removeChild(dom.currentTime);\n delete dom.currentTime;\n }\n return;\n }\n if (!dom.currentTime) {\n // create the current time bar\n var currentTime = document.createElement(\"DIV\");\n currentTime.className = \"timeline-currenttime\";\n currentTime.style.position = \"absolute\";\n currentTime.style.top = \"0px\";\n currentTime.style.height = \"100%\";\n dom.contentTimelines.appendChild(currentTime);\n dom.currentTime = currentTime;\n }\n var now = new Date();\n var nowOffset = new Date(now.valueOf() + this.clientTimeOffset);\n var x = this.timeToScreen(nowOffset);\n var visible = x > -size.contentWidth && x < 2 * size.contentWidth;\n dom.currentTime.style.display = visible ? '' : 'none';\n dom.currentTime.style.left = x + \"px\";\n dom.currentTime.title = \"Current time: \" + nowOffset;\n\n // start a timer to adjust for the new time\n if (this.currentTimeTimer != undefined) {\n clearTimeout(this.currentTimeTimer);\n delete this.currentTimeTimer;\n }\n var timeline = this;\n var onTimeout = function onTimeout() {\n timeline.repaintCurrentTime();\n };\n // the time equal to the width of one pixel, divided by 2 for more smoothness\n var interval = 1 / this.conversion.factor / 2;\n if (interval < 30) interval = 30;\n this.currentTimeTimer = setTimeout(onTimeout, interval);\n};\n\n/**\n * Redraw the custom time bar\n */\nlinks.Timeline.prototype.repaintCustomTime = function () {\n var options = this.options,\n dom = this.dom,\n size = this.size;\n if (!options.showCustomTime) {\n if (dom.customTime) {\n dom.contentTimelines.removeChild(dom.customTime);\n delete dom.customTime;\n }\n return;\n }\n if (!dom.customTime) {\n var customTime = document.createElement(\"DIV\");\n customTime.className = \"timeline-customtime\";\n customTime.style.position = \"absolute\";\n customTime.style.top = \"0px\";\n customTime.style.height = \"100%\";\n var drag = document.createElement(\"DIV\");\n drag.style.position = \"relative\";\n drag.style.top = \"0px\";\n drag.style.left = \"-10px\";\n drag.style.height = \"100%\";\n drag.style.width = \"20px\";\n customTime.appendChild(drag);\n dom.contentTimelines.appendChild(customTime);\n dom.customTime = customTime;\n\n // initialize parameter\n this.customTime = new Date();\n }\n var x = this.timeToScreen(this.customTime),\n visible = x > -size.contentWidth && x < 2 * size.contentWidth;\n dom.customTime.style.display = visible ? '' : 'none';\n dom.customTime.style.left = x + \"px\";\n dom.customTime.title = \"Time: \" + this.customTime;\n};\n\n/**\n * Redraw the delete button, on the top right of the currently selected item\n * if there is no item selected, the button is hidden.\n */\nlinks.Timeline.prototype.repaintDeleteButton = function () {\n var timeline = this,\n dom = this.dom,\n frame = dom.items.frame;\n var deleteButton = dom.items.deleteButton;\n if (!deleteButton) {\n // create a delete button\n deleteButton = document.createElement(\"DIV\");\n deleteButton.className = \"timeline-navigation-delete\";\n deleteButton.style.position = \"absolute\";\n frame.appendChild(deleteButton);\n dom.items.deleteButton = deleteButton;\n }\n var index = this.selection && this.selection.index !== undefined ? this.selection.index : -1,\n item = this.selection && this.selection.index !== undefined ? this.items[index] : undefined;\n if (item && item.rendered && this.isEditable(item)) {\n var right = item.getRight(this),\n top = item.top;\n deleteButton.style.left = right + 'px';\n deleteButton.style.top = top + 'px';\n deleteButton.style.display = '';\n frame.removeChild(deleteButton);\n frame.appendChild(deleteButton);\n } else {\n deleteButton.style.display = 'none';\n }\n};\n\n/**\n * Redraw the drag areas. When an item (ranges only) is selected,\n * it gets a drag area on the left and right side, to change its width\n */\nlinks.Timeline.prototype.repaintDragAreas = function () {\n var timeline = this,\n options = this.options,\n dom = this.dom,\n frame = this.dom.items.frame;\n\n // create left drag area\n var dragLeft = dom.items.dragLeft;\n if (!dragLeft) {\n dragLeft = document.createElement(\"DIV\");\n dragLeft.className = \"timeline-event-range-drag-left\";\n dragLeft.style.position = \"absolute\";\n frame.appendChild(dragLeft);\n dom.items.dragLeft = dragLeft;\n }\n\n // create right drag area\n var dragRight = dom.items.dragRight;\n if (!dragRight) {\n dragRight = document.createElement(\"DIV\");\n dragRight.className = \"timeline-event-range-drag-right\";\n dragRight.style.position = \"absolute\";\n frame.appendChild(dragRight);\n dom.items.dragRight = dragRight;\n }\n\n // reposition left and right drag area\n var index = this.selection && this.selection.index !== undefined ? this.selection.index : -1,\n item = this.selection && this.selection.index !== undefined ? this.items[index] : undefined;\n if (item && item.rendered && this.isEditable(item) && (item instanceof links.Timeline.ItemRange || item instanceof links.Timeline.ItemFloatingRange)) {\n var left = item.getLeft(this),\n // NH change to getLeft\n right = item.getRight(this),\n // NH change to getRight\n top = item.top,\n height = item.height;\n dragLeft.style.left = left + 'px';\n dragLeft.style.top = top + 'px';\n dragLeft.style.width = options.dragAreaWidth + \"px\";\n dragLeft.style.height = height + 'px';\n dragLeft.style.display = '';\n frame.removeChild(dragLeft);\n frame.appendChild(dragLeft);\n dragRight.style.left = right - options.dragAreaWidth + 'px';\n dragRight.style.top = top + 'px';\n dragRight.style.width = options.dragAreaWidth + \"px\";\n dragRight.style.height = height + 'px';\n dragRight.style.display = '';\n frame.removeChild(dragRight);\n frame.appendChild(dragRight);\n } else {\n dragLeft.style.display = 'none';\n dragRight.style.display = 'none';\n }\n};\n\n/**\n * Create the navigation buttons for zooming and moving\n */\nlinks.Timeline.prototype.repaintNavigation = function () {\n var timeline = this,\n options = this.options,\n dom = this.dom,\n frame = dom.frame,\n navBar = dom.navBar;\n if (!navBar) {\n var showButtonNew = options.showButtonNew && options.editable;\n var showNavigation = options.showNavigation && (options.zoomable || options.moveable);\n if (showNavigation || showButtonNew) {\n // create a navigation bar containing the navigation buttons\n navBar = document.createElement(\"DIV\");\n navBar.style.position = \"absolute\";\n navBar.className = \"timeline-navigation ui-widget ui-state-highlight ui-corner-all\";\n if (options.groupsOnRight) {\n navBar.style.left = '10px';\n } else {\n navBar.style.right = '10px';\n }\n if (options.axisOnTop) {\n navBar.style.bottom = '10px';\n } else {\n navBar.style.top = '10px';\n }\n dom.navBar = navBar;\n frame.appendChild(navBar);\n }\n if (showButtonNew) {\n // create a new in button\n navBar.addButton = document.createElement(\"DIV\");\n navBar.addButton.className = \"timeline-navigation-new\";\n navBar.addButton.title = options.CREATE_NEW_EVENT;\n var addIconSpan = document.createElement(\"SPAN\");\n addIconSpan.className = \"ui-icon ui-icon-circle-plus\";\n navBar.addButton.appendChild(addIconSpan);\n var onAdd = function onAdd(event) {\n links.Timeline.preventDefault(event);\n links.Timeline.stopPropagation(event);\n\n // create a new event at the center of the frame\n var w = timeline.size.contentWidth;\n var x = w / 2;\n var xstart = timeline.screenToTime(x);\n if (options.snapEvents) {\n timeline.step.snap(xstart);\n }\n var content = options.NEW;\n var group = timeline.groups.length ? timeline.groups[0].content : undefined;\n var preventRender = true;\n timeline.addItem({\n 'start': xstart,\n 'content': content,\n 'group': group\n }, preventRender);\n var index = timeline.items.length - 1;\n timeline.selectItem(index);\n timeline.applyAdd = true;\n\n // fire an add event.\n // Note that the change can be canceled from within an event listener if\n // this listener calls the method cancelAdd().\n timeline.trigger('add');\n if (timeline.applyAdd) {\n // render and select the item\n timeline.render({\n animate: false\n });\n timeline.selectItem(index);\n } else {\n // undo an add\n timeline.deleteItem(index);\n }\n };\n links.Timeline.addEventListener(navBar.addButton, \"mousedown\", onAdd);\n navBar.appendChild(navBar.addButton);\n }\n if (showButtonNew && showNavigation) {\n // create a separator line\n links.Timeline.addClassName(navBar.addButton, 'timeline-navigation-new-line');\n }\n if (showNavigation) {\n if (options.zoomable) {\n // create a zoom in button\n navBar.zoomInButton = document.createElement(\"DIV\");\n navBar.zoomInButton.className = \"timeline-navigation-zoom-in\";\n navBar.zoomInButton.title = this.options.ZOOM_IN;\n var ziIconSpan = document.createElement(\"SPAN\");\n ziIconSpan.className = \"ui-icon ui-icon-circle-zoomin\";\n navBar.zoomInButton.appendChild(ziIconSpan);\n var onZoomIn = function onZoomIn(event) {\n links.Timeline.preventDefault(event);\n links.Timeline.stopPropagation(event);\n timeline.zoom(0.4);\n timeline.trigger(\"rangechange\");\n timeline.trigger(\"rangechanged\");\n };\n links.Timeline.addEventListener(navBar.zoomInButton, \"mousedown\", onZoomIn);\n navBar.appendChild(navBar.zoomInButton);\n\n // create a zoom out button\n navBar.zoomOutButton = document.createElement(\"DIV\");\n navBar.zoomOutButton.className = \"timeline-navigation-zoom-out\";\n navBar.zoomOutButton.title = this.options.ZOOM_OUT;\n var zoIconSpan = document.createElement(\"SPAN\");\n zoIconSpan.className = \"ui-icon ui-icon-circle-zoomout\";\n navBar.zoomOutButton.appendChild(zoIconSpan);\n var onZoomOut = function onZoomOut(event) {\n links.Timeline.preventDefault(event);\n links.Timeline.stopPropagation(event);\n timeline.zoom(-0.4);\n timeline.trigger(\"rangechange\");\n timeline.trigger(\"rangechanged\");\n };\n links.Timeline.addEventListener(navBar.zoomOutButton, \"mousedown\", onZoomOut);\n navBar.appendChild(navBar.zoomOutButton);\n }\n if (options.moveable) {\n // create a move left button\n navBar.moveLeftButton = document.createElement(\"DIV\");\n navBar.moveLeftButton.className = \"timeline-navigation-move-left\";\n navBar.moveLeftButton.title = this.options.MOVE_LEFT;\n var mlIconSpan = document.createElement(\"SPAN\");\n mlIconSpan.className = \"ui-icon ui-icon-circle-arrow-w\";\n navBar.moveLeftButton.appendChild(mlIconSpan);\n var onMoveLeft = function onMoveLeft(event) {\n links.Timeline.preventDefault(event);\n links.Timeline.stopPropagation(event);\n timeline.move(-0.2);\n timeline.trigger(\"rangechange\");\n timeline.trigger(\"rangechanged\");\n };\n links.Timeline.addEventListener(navBar.moveLeftButton, \"mousedown\", onMoveLeft);\n navBar.appendChild(navBar.moveLeftButton);\n\n // create a move right button\n navBar.moveRightButton = document.createElement(\"DIV\");\n navBar.moveRightButton.className = \"timeline-navigation-move-right\";\n navBar.moveRightButton.title = this.options.MOVE_RIGHT;\n var mrIconSpan = document.createElement(\"SPAN\");\n mrIconSpan.className = \"ui-icon ui-icon-circle-arrow-e\";\n navBar.moveRightButton.appendChild(mrIconSpan);\n var onMoveRight = function onMoveRight(event) {\n links.Timeline.preventDefault(event);\n links.Timeline.stopPropagation(event);\n timeline.move(0.2);\n timeline.trigger(\"rangechange\");\n timeline.trigger(\"rangechanged\");\n };\n links.Timeline.addEventListener(navBar.moveRightButton, \"mousedown\", onMoveRight);\n navBar.appendChild(navBar.moveRightButton);\n }\n }\n }\n};\n\n/**\n * Set current time. This function can be used to set the time in the client\n * timeline equal with the time on a server.\n * @param {Date} time\n */\nlinks.Timeline.prototype.setCurrentTime = function (time) {\n var now = new Date();\n this.clientTimeOffset = time.valueOf() - now.valueOf();\n this.repaintCurrentTime();\n};\n\n/**\n * Get current time. The time can have an offset from the real time, when\n * the current time has been changed via the method setCurrentTime.\n * @return {Date} time\n */\nlinks.Timeline.prototype.getCurrentTime = function () {\n var now = new Date();\n return new Date(now.valueOf() + this.clientTimeOffset);\n};\n\n/**\n * Set custom time.\n * The custom time bar can be used to display events in past or future.\n * @param {Date} time\n */\nlinks.Timeline.prototype.setCustomTime = function (time) {\n this.customTime = new Date(time.valueOf());\n this.repaintCustomTime();\n};\n\n/**\n * Retrieve the current custom time.\n * @return {Date} customTime\n */\nlinks.Timeline.prototype.getCustomTime = function () {\n return new Date(this.customTime.valueOf());\n};\n\n/**\n * Set a custom scale. Autoscaling will be disabled.\n * For example setScale(SCALE.MINUTES, 5) will result\n * in minor steps of 5 minutes, and major steps of an hour.\n *\n * @param {links.Timeline.StepDate.SCALE} scale\n * A scale. Choose from SCALE.MILLISECOND,\n * SCALE.SECOND, SCALE.MINUTE, SCALE.HOUR,\n * SCALE.WEEKDAY, SCALE.DAY, SCALE.MONTH,\n * SCALE.YEAR.\n * @param {int} step A step size, by default 1. Choose for\n * example 1, 2, 5, or 10.\n */\nlinks.Timeline.prototype.setScale = function (scale, step) {\n this.step.setScale(scale, step);\n this.render(); // TODO: optimize: only reflow/repaint axis\n};\n\n/**\n * Enable or disable autoscaling\n * @param {boolean} enable If true or not defined, autoscaling is enabled.\n * If false, autoscaling is disabled.\n */\nlinks.Timeline.prototype.setAutoScale = function (enable) {\n this.step.setAutoScale(enable);\n this.render(); // TODO: optimize: only reflow/repaint axis\n};\n\n/**\n * Redraw the timeline\n * Reloads the (linked) data table and redraws the timeline when resized.\n * See also the method checkResize\n */\nlinks.Timeline.prototype.redraw = function () {\n this.setData(this.data);\n};\n\n/**\n * Check if the timeline is resized, and if so, redraw the timeline.\n * Useful when the webpage is resized.\n */\nlinks.Timeline.prototype.checkResize = function () {\n // TODO: re-implement the method checkResize, or better, make it redundant as this.render will be smarter\n this.render();\n};\n\n/**\n * Check whether a given item is editable\n * @param {links.Timeline.Item} item\n * @return {boolean} editable\n */\nlinks.Timeline.prototype.isEditable = function (item) {\n if (item) {\n if (item.editable != undefined) {\n return item.editable;\n } else {\n return this.options.editable;\n }\n }\n return false;\n};\n\n/**\n * Calculate the factor and offset to convert a position on screen to the\n * corresponding date and vice versa.\n * After the method calcConversionFactor is executed once, the methods screenToTime and\n * timeToScreen can be used.\n */\nlinks.Timeline.prototype.recalcConversion = function () {\n this.conversion.offset = this.start.valueOf();\n this.conversion.factor = this.size.contentWidth / (this.end.valueOf() - this.start.valueOf());\n};\n\n/**\n * Convert a position on screen (pixels) to a datetime\n * Before this method can be used, the method calcConversionFactor must be\n * executed once.\n * @param {int} x Position on the screen in pixels\n * @return {Date} time The datetime the corresponds with given position x\n */\nlinks.Timeline.prototype.screenToTime = function (x) {\n var conversion = this.conversion;\n return new Date(x / conversion.factor + conversion.offset);\n};\n\n/**\n * Convert a datetime (Date object) into a position on the screen\n * Before this method can be used, the method calcConversionFactor must be\n * executed once.\n * @param {Date} time A date\n * @return {int} x The position on the screen in pixels which corresponds\n * with the given date.\n */\nlinks.Timeline.prototype.timeToScreen = function (time) {\n var conversion = this.conversion;\n return (time.valueOf() - conversion.offset) * conversion.factor;\n};\n\n/**\n * Event handler for touchstart event on mobile devices\n */\nlinks.Timeline.prototype.onTouchStart = function (event) {\n var params = this.eventParams,\n me = this;\n if (params.touchDown) {\n // if already moving, return\n return;\n }\n params.touchDown = true;\n params.zoomed = false;\n this.onMouseDown(event);\n if (!params.onTouchMove) {\n params.onTouchMove = function (event) {\n me.onTouchMove(event);\n };\n links.Timeline.addEventListener(document, \"touchmove\", params.onTouchMove);\n }\n if (!params.onTouchEnd) {\n params.onTouchEnd = function (event) {\n me.onTouchEnd(event);\n };\n links.Timeline.addEventListener(document, \"touchend\", params.onTouchEnd);\n }\n\n /* TODO\n // check for double tap event\n var delta = 500; // ms\n var doubleTapStart = (new Date()).valueOf();\n var target = links.Timeline.getTarget(event);\n var doubleTapItem = this.getItemIndex(target);\n if (params.doubleTapStart &&\n (doubleTapStart - params.doubleTapStart) < delta &&\n doubleTapItem == params.doubleTapItem) {\n delete params.doubleTapStart;\n delete params.doubleTapItem;\n me.onDblClick(event);\n params.touchDown = false;\n }\n params.doubleTapStart = doubleTapStart;\n params.doubleTapItem = doubleTapItem;\n */\n // store timing for double taps\n var target = links.Timeline.getTarget(event);\n var item = this.getItemIndex(target);\n params.doubleTapStartPrev = params.doubleTapStart;\n params.doubleTapStart = new Date().valueOf();\n params.doubleTapItemPrev = params.doubleTapItem;\n params.doubleTapItem = item;\n links.Timeline.preventDefault(event);\n};\n\n/**\n * Event handler for touchmove event on mobile devices\n */\nlinks.Timeline.prototype.onTouchMove = function (event) {\n var params = this.eventParams;\n if (event.scale && event.scale !== 1) {\n params.zoomed = true;\n }\n if (!params.zoomed) {\n // move\n this.onMouseMove(event);\n } else {\n if (this.options.zoomable) {\n // pinch\n // TODO: pinch only supported on iPhone/iPad. Create something manually for Android?\n params.zoomed = true;\n var scale = event.scale,\n oldWidth = params.end.valueOf() - params.start.valueOf(),\n newWidth = oldWidth / scale,\n diff = newWidth - oldWidth,\n start = new Date(parseInt(params.start.valueOf() - diff / 2)),\n end = new Date(parseInt(params.end.valueOf() + diff / 2));\n\n // TODO: determine zoom-around-date from touch positions?\n\n this.setVisibleChartRange(start, end);\n this.trigger(\"rangechange\");\n }\n }\n links.Timeline.preventDefault(event);\n};\n\n/**\n * Event handler for touchend event on mobile devices\n */\nlinks.Timeline.prototype.onTouchEnd = function (event) {\n var params = this.eventParams;\n var me = this;\n params.touchDown = false;\n if (params.zoomed) {\n this.trigger(\"rangechanged\");\n }\n if (params.onTouchMove) {\n links.Timeline.removeEventListener(document, \"touchmove\", params.onTouchMove);\n delete params.onTouchMove;\n }\n if (params.onTouchEnd) {\n links.Timeline.removeEventListener(document, \"touchend\", params.onTouchEnd);\n delete params.onTouchEnd;\n }\n this.onMouseUp(event);\n\n // check for double tap event\n var delta = 500; // ms\n var doubleTapEnd = new Date().valueOf();\n var target = links.Timeline.getTarget(event);\n var doubleTapItem = this.getItemIndex(target);\n if (params.doubleTapStartPrev && doubleTapEnd - params.doubleTapStartPrev < delta && params.doubleTapItem == params.doubleTapItemPrev) {\n params.touchDown = true;\n me.onDblClick(event);\n params.touchDown = false;\n }\n links.Timeline.preventDefault(event);\n};\n\n/**\n * Start a moving operation inside the provided parent element\n * @param {Event} event The event that occurred (required for\n * retrieving the mouse position)\n */\nlinks.Timeline.prototype.onMouseDown = function (event) {\n event = event || window.event;\n var params = this.eventParams,\n options = this.options,\n dom = this.dom;\n\n // only react on left mouse button down\n var leftButtonDown = event.which ? event.which == 1 : event.button == 1;\n if (!leftButtonDown && !params.touchDown) {\n return;\n }\n\n // get mouse position\n params.mouseX = links.Timeline.getPageX(event);\n params.mouseY = links.Timeline.getPageY(event);\n params.frameLeft = links.Timeline.getAbsoluteLeft(this.dom.content);\n params.frameTop = links.Timeline.getAbsoluteTop(this.dom.content);\n params.previousLeft = 0;\n params.previousOffset = 0;\n params.moved = false;\n params.start = new Date(this.start.valueOf());\n params.end = new Date(this.end.valueOf());\n params.target = links.Timeline.getTarget(event);\n var dragLeft = dom.items && dom.items.dragLeft ? dom.items.dragLeft : undefined;\n var dragRight = dom.items && dom.items.dragRight ? dom.items.dragRight : undefined;\n params.itemDragLeft = params.target === dragLeft;\n params.itemDragRight = params.target === dragRight;\n if (params.itemDragLeft || params.itemDragRight) {\n params.itemIndex = this.selection && this.selection.index !== undefined ? this.selection.index : undefined;\n delete params.clusterIndex;\n } else {\n params.itemIndex = this.getItemIndex(params.target);\n params.clusterIndex = this.getClusterIndex(params.target);\n }\n params.customTime = params.target === dom.customTime || params.target.parentNode === dom.customTime ? this.customTime : undefined;\n params.addItem = options.editable && event.ctrlKey;\n if (params.addItem) {\n // create a new event at the current mouse position\n var x = params.mouseX - params.frameLeft;\n var y = params.mouseY - params.frameTop;\n var xstart = this.screenToTime(x);\n if (options.snapEvents) {\n this.step.snap(xstart);\n }\n var xend = new Date(xstart.valueOf());\n var content = options.NEW;\n var group = this.getGroupFromHeight(y);\n this.addItem({\n 'start': xstart,\n 'end': xend,\n 'content': content,\n 'group': this.getGroupName(group)\n });\n params.itemIndex = this.items.length - 1;\n delete params.clusterIndex;\n this.selectItem(params.itemIndex);\n params.itemDragRight = true;\n }\n var item = this.items[params.itemIndex];\n var isSelected = this.isSelected(params.itemIndex);\n params.editItem = isSelected && this.isEditable(item);\n if (params.editItem) {\n params.itemStart = item.start;\n params.itemEnd = item.end;\n params.itemGroup = item.group;\n params.itemLeft = item.getLeft(this); // NH Use item.getLeft here\n params.itemRight = item.getRight(this); // NH Use item.getRight here\n } else {\n this.dom.frame.style.cursor = 'move';\n }\n if (!params.touchDown) {\n // add event listeners to handle moving the contents\n // we store the function onmousemove and onmouseup in the timeline, so we can\n // remove the eventlisteners lateron in the function mouseUp()\n var me = this;\n if (!params.onMouseMove) {\n params.onMouseMove = function (event) {\n me.onMouseMove(event);\n };\n links.Timeline.addEventListener(document, \"mousemove\", params.onMouseMove);\n }\n if (!params.onMouseUp) {\n params.onMouseUp = function (event) {\n me.onMouseUp(event);\n };\n links.Timeline.addEventListener(document, \"mouseup\", params.onMouseUp);\n }\n links.Timeline.preventDefault(event);\n }\n};\n\n/**\n * Perform moving operating.\n * This function activated from within the funcion links.Timeline.onMouseDown().\n * @param {Event} event Well, eehh, the event\n */\nlinks.Timeline.prototype.onMouseMove = function (event) {\n event = event || window.event;\n var params = this.eventParams,\n size = this.size,\n dom = this.dom,\n options = this.options;\n\n // calculate change in mouse position\n var mouseX = links.Timeline.getPageX(event);\n var mouseY = links.Timeline.getPageY(event);\n if (params.mouseX == undefined) {\n params.mouseX = mouseX;\n }\n if (params.mouseY == undefined) {\n params.mouseY = mouseY;\n }\n var diffX = mouseX - params.mouseX;\n var diffY = mouseY - params.mouseY;\n\n // if mouse movement is big enough, register it as a \"moved\" event\n if (Math.abs(diffX) >= 1) {\n params.moved = true;\n }\n if (params.customTime) {\n var x = this.timeToScreen(params.customTime);\n var xnew = x + diffX;\n this.customTime = this.screenToTime(xnew);\n this.repaintCustomTime();\n\n // fire a timechange event\n this.trigger('timechange');\n } else if (params.editItem) {\n var item = this.items[params.itemIndex],\n left,\n right;\n if (params.itemDragLeft && options.timeChangeable) {\n // move the start of the item\n left = params.itemLeft + diffX;\n right = params.itemRight;\n item.start = this.screenToTime(left);\n if (options.snapEvents) {\n this.step.snap(item.start);\n left = this.timeToScreen(item.start);\n }\n if (left > right) {\n left = right;\n item.start = this.screenToTime(left);\n }\n this.trigger('change');\n } else if (params.itemDragRight && options.timeChangeable) {\n // move the end of the item\n left = params.itemLeft;\n right = params.itemRight + diffX;\n item.end = this.screenToTime(right);\n if (options.snapEvents) {\n this.step.snap(item.end);\n right = this.timeToScreen(item.end);\n }\n if (right < left) {\n right = left;\n item.end = this.screenToTime(right);\n }\n this.trigger('change');\n } else if (options.timeChangeable) {\n // move the item\n left = params.itemLeft + diffX;\n item.start = this.screenToTime(left);\n if (options.snapEvents) {\n this.step.snap(item.start);\n left = this.timeToScreen(item.start);\n }\n if (item.end) {\n right = left + (params.itemRight - params.itemLeft);\n item.end = this.screenToTime(right);\n }\n this.trigger('change');\n }\n item.setPosition(left, right);\n var dragging = params.itemDragLeft || params.itemDragRight;\n if (this.groups.length && !dragging) {\n // move item from one group to another when needed\n var y = mouseY - params.frameTop;\n var group = this.getGroupFromHeight(y);\n if (options.groupsChangeable && item.group !== group) {\n // move item to the other group\n var index = this.items.indexOf(item);\n this.changeItem(index, {\n 'group': this.getGroupName(group)\n });\n } else {\n this.repaintDeleteButton();\n this.repaintDragAreas();\n }\n } else {\n // TODO: does not work well in FF, forces redraw with every mouse move it seems\n this.render(); // TODO: optimize, only redraw the items?\n // Note: when animate==true, no redraw is needed here, its done by stackItems animation\n }\n } else if (options.moveable) {\n var interval = params.end.valueOf() - params.start.valueOf();\n var diffMillisecs = Math.round(-diffX / size.contentWidth * interval);\n var newStart = new Date(params.start.valueOf() + diffMillisecs);\n var newEnd = new Date(params.end.valueOf() + diffMillisecs);\n this.applyRange(newStart, newEnd);\n // if the applied range is moved due to a fixed min or max,\n // change the diffMillisecs accordingly\n var appliedDiff = this.start.valueOf() - newStart.valueOf();\n if (appliedDiff) {\n diffMillisecs += appliedDiff;\n }\n this.recalcConversion();\n\n // move the items by changing the left position of their frame.\n // this is much faster than repositioning all elements individually via the\n // repaintFrame() function (which is done once at mouseup)\n // note that we round diffX to prevent wrong positioning on millisecond scale\n var previousLeft = params.previousLeft || 0;\n var currentLeft = parseFloat(dom.items.frame.style.left) || 0;\n var previousOffset = params.previousOffset || 0;\n var frameOffset = previousOffset + (currentLeft - previousLeft);\n var frameLeft = -diffMillisecs / interval * size.contentWidth + frameOffset;\n dom.items.frame.style.left = frameLeft + \"px\";\n\n // read the left again from DOM (IE8- rounds the value)\n params.previousOffset = frameOffset;\n params.previousLeft = parseFloat(dom.items.frame.style.left) || frameLeft;\n this.repaintCurrentTime();\n this.repaintCustomTime();\n this.repaintAxis();\n\n // fire a rangechange event\n this.trigger('rangechange');\n }\n links.Timeline.preventDefault(event);\n};\n\n/**\n * Stop moving operating.\n * This function activated from within the funcion links.Timeline.onMouseDown().\n * @param {event} event The event\n */\nlinks.Timeline.prototype.onMouseUp = function (event) {\n var params = this.eventParams,\n options = this.options;\n event = event || window.event;\n this.dom.frame.style.cursor = 'auto';\n\n // remove event listeners here, important for Safari\n if (params.onMouseMove) {\n links.Timeline.removeEventListener(document, \"mousemove\", params.onMouseMove);\n delete params.onMouseMove;\n }\n if (params.onMouseUp) {\n links.Timeline.removeEventListener(document, \"mouseup\", params.onMouseUp);\n delete params.onMouseUp;\n }\n //links.Timeline.preventDefault(event);\n\n if (params.customTime) {\n // fire a timechanged event\n this.trigger('timechanged');\n } else if (params.editItem) {\n var item = this.items[params.itemIndex];\n if (params.moved || params.addItem) {\n this.applyChange = true;\n this.applyAdd = true;\n this.updateData(params.itemIndex, {\n 'start': item.start,\n 'end': item.end\n });\n\n // fire an add or changed event.\n // Note that the change can be canceled from within an event listener if\n // this listener calls the method cancelChange().\n this.trigger(params.addItem ? 'add' : 'changed');\n\n //retrieve item data again to include changes made to it in the triggered event handlers\n item = this.items[params.itemIndex];\n if (params.addItem) {\n if (this.applyAdd) {\n this.updateData(params.itemIndex, {\n 'start': item.start,\n 'end': item.end,\n 'content': item.content,\n 'group': this.getGroupName(item.group)\n });\n } else {\n // undo an add\n this.deleteItem(params.itemIndex);\n }\n } else {\n if (this.applyChange) {\n this.updateData(params.itemIndex, {\n 'start': item.start,\n 'end': item.end\n });\n } else {\n // undo a change\n delete this.applyChange;\n delete this.applyAdd;\n var item = this.items[params.itemIndex],\n domItem = item.dom;\n item.start = params.itemStart;\n item.end = params.itemEnd;\n item.group = params.itemGroup;\n // TODO: original group should be restored too\n item.setPosition(params.itemLeft, params.itemRight);\n this.updateData(params.itemIndex, {\n 'start': params.itemStart,\n 'end': params.itemEnd\n });\n }\n }\n\n // prepare data for clustering, by filtering and sorting by type\n if (this.options.cluster) {\n this.clusterGenerator.updateData();\n }\n this.render();\n }\n } else {\n if (!params.moved && !params.zoomed) {\n // mouse did not move -> user has selected an item\n\n if (params.target === this.dom.items.deleteButton) {\n // delete item\n if (this.selection && this.selection.index !== undefined) {\n this.confirmDeleteItem(this.selection.index);\n }\n } else if (options.selectable) {\n // select/unselect item\n if (params.itemIndex != undefined) {\n if (!this.isSelected(params.itemIndex)) {\n this.selectItem(params.itemIndex);\n this.trigger('select');\n }\n } else if (params.clusterIndex != undefined) {\n this.selectCluster(params.clusterIndex);\n this.trigger('select');\n } else {\n if (options.unselectable) {\n this.unselectItem();\n this.trigger('select');\n }\n }\n }\n } else {\n // timeline is moved\n // TODO: optimize: no need to reflow and cluster again?\n this.render();\n if (params.moved && options.moveable || params.zoomed && options.zoomable) {\n // fire a rangechanged event\n this.trigger('rangechanged');\n }\n }\n }\n};\n\n/**\n * Double click event occurred for an item\n * @param {Event} event\n */\nlinks.Timeline.prototype.onDblClick = function (event) {\n var params = this.eventParams,\n options = this.options,\n dom = this.dom,\n size = this.size;\n event = event || window.event;\n if (params.itemIndex != undefined) {\n var item = this.items[params.itemIndex];\n if (item && this.isEditable(item)) {\n // fire the edit event\n this.trigger('edit');\n }\n } else {\n if (options.editable) {\n // create a new item\n\n // get mouse position\n params.mouseX = links.Timeline.getPageX(event);\n params.mouseY = links.Timeline.getPageY(event);\n var x = params.mouseX - links.Timeline.getAbsoluteLeft(dom.content);\n var y = params.mouseY - links.Timeline.getAbsoluteTop(dom.content);\n\n // create a new event at the current mouse position\n var xstart = this.screenToTime(x);\n if (options.snapEvents) {\n this.step.snap(xstart);\n }\n var content = options.NEW;\n var group = this.getGroupFromHeight(y); // (group may be undefined)\n var preventRender = true;\n this.addItem({\n 'start': xstart,\n 'content': content,\n 'group': this.getGroupName(group)\n }, preventRender);\n params.itemIndex = this.items.length - 1;\n this.selectItem(params.itemIndex);\n this.applyAdd = true;\n\n // fire an add event.\n // Note that the change can be canceled from within an event listener if\n // this listener calls the method cancelAdd().\n this.trigger('add');\n if (this.applyAdd) {\n // render and select the item\n this.render({\n animate: false\n });\n this.selectItem(params.itemIndex);\n } else {\n // undo an add\n this.deleteItem(params.itemIndex);\n }\n }\n }\n links.Timeline.preventDefault(event);\n};\n\n/**\n * Event handler for mouse wheel event, used to zoom the timeline\n * Code from http://adomas.org/javascript-mouse-wheel/\n * @param {Event} event The event\n */\nlinks.Timeline.prototype.onMouseWheel = function (event) {\n if (!this.options.zoomable) return;\n if (!event) {\n /* For IE. */\n event = window.event;\n }\n\n // retrieve delta\n var delta = 0;\n if (event.wheelDelta) {\n /* IE/Opera. */\n delta = event.wheelDelta / 120;\n } else if (event.detail) {\n /* Mozilla case. */\n // In Mozilla, sign of delta is different than in IE.\n // Also, delta is multiple of 3.\n delta = -event.detail / 3;\n }\n\n // If delta is nonzero, handle it.\n // Basically, delta is now positive if wheel was scrolled up,\n // and negative, if wheel was scrolled down.\n if (delta) {\n // TODO: on FireFox, the window is not redrawn within repeated scroll-events\n // -> use a delayed redraw? Make a zoom queue?\n\n var timeline = this;\n var zoom = function zoom() {\n // perform the zoom action. Delta is normally 1 or -1\n var zoomFactor = delta / 5.0;\n var frameLeft = links.Timeline.getAbsoluteLeft(timeline.dom.content);\n var mouseX = links.Timeline.getPageX(event);\n var zoomAroundDate = mouseX != undefined && frameLeft != undefined ? timeline.screenToTime(mouseX - frameLeft) : undefined;\n timeline.zoom(zoomFactor, zoomAroundDate);\n\n // fire a rangechange and a rangechanged event\n timeline.trigger(\"rangechange\");\n timeline.trigger(\"rangechanged\");\n };\n var scroll = function scroll() {\n // Scroll the timeline\n timeline.move(delta * -0.2);\n timeline.trigger(\"rangechange\");\n timeline.trigger(\"rangechanged\");\n };\n if (event.shiftKey) {\n scroll();\n } else {\n zoom();\n }\n }\n\n // Prevent default actions caused by mouse wheel.\n // That might be ugly, but we handle scrolls somehow\n // anyway, so don't bother here...\n links.Timeline.preventDefault(event);\n};\n\n/**\n * Zoom the timeline the given zoomfactor in or out. Start and end date will\n * be adjusted, and the timeline will be redrawn. You can optionally give a\n * date around which to zoom.\n * For example, try zoomfactor = 0.1 or -0.1\n * @param {Number} zoomFactor Zooming amount. Positive value will zoom in,\n * negative value will zoom out\n * @param {Date} zoomAroundDate Date around which will be zoomed. Optional\n */\nlinks.Timeline.prototype.zoom = function (zoomFactor, zoomAroundDate) {\n // if zoomAroundDate is not provided, take it half between start Date and end Date\n if (zoomAroundDate == undefined) {\n zoomAroundDate = new Date((this.start.valueOf() + this.end.valueOf()) / 2);\n }\n\n // prevent zoom factor larger than 1 or smaller than -1 (larger than 1 will\n // result in a start>=end )\n if (zoomFactor >= 1) {\n zoomFactor = 0.9;\n }\n if (zoomFactor <= -1) {\n zoomFactor = -0.9;\n }\n\n // adjust a negative factor such that zooming in with 0.1 equals zooming\n // out with a factor -0.1\n if (zoomFactor < 0) {\n zoomFactor = zoomFactor / (1 + zoomFactor);\n }\n\n // zoom start Date and end Date relative to the zoomAroundDate\n var startDiff = this.start.valueOf() - zoomAroundDate;\n var endDiff = this.end.valueOf() - zoomAroundDate;\n\n // calculate new dates\n var newStart = new Date(this.start.valueOf() - startDiff * zoomFactor);\n var newEnd = new Date(this.end.valueOf() - endDiff * zoomFactor);\n\n // only zoom in when interval is larger than minimum interval (to prevent\n // sliding to left/right when having reached the minimum zoom level)\n var interval = newEnd.valueOf() - newStart.valueOf();\n var zoomMin = Number(this.options.zoomMin) || 10;\n if (zoomMin < 10) {\n zoomMin = 10;\n }\n if (interval >= zoomMin) {\n this.applyRange(newStart, newEnd, zoomAroundDate);\n this.render({\n animate: this.options.animate && this.options.animateZoom\n });\n }\n};\n\n/**\n * Move the timeline the given movefactor to the left or right. Start and end\n * date will be adjusted, and the timeline will be redrawn.\n * For example, try moveFactor = 0.1 or -0.1\n * @param {Number} moveFactor Moving amount. Positive value will move right,\n * negative value will move left\n */\nlinks.Timeline.prototype.move = function (moveFactor) {\n // zoom start Date and end Date relative to the zoomAroundDate\n var diff = this.end.valueOf() - this.start.valueOf();\n\n // apply new dates\n var newStart = new Date(this.start.valueOf() + diff * moveFactor);\n var newEnd = new Date(this.end.valueOf() + diff * moveFactor);\n this.applyRange(newStart, newEnd);\n this.render(); // TODO: optimize, no need to reflow, only to recalc conversion and repaint\n};\n\n/**\n * Apply a visible range. The range is limited to feasible maximum and minimum\n * range.\n * @param {Date} start\n * @param {Date} end\n * @param {Date} zoomAroundDate Optional. Date around which will be zoomed.\n */\nlinks.Timeline.prototype.applyRange = function (start, end, zoomAroundDate) {\n // calculate new start and end value\n var startValue = start.valueOf(); // number\n var endValue = end.valueOf(); // number\n var interval = endValue - startValue;\n\n // determine maximum and minimum interval\n var options = this.options;\n var year = 1000 * 60 * 60 * 24 * 365;\n var zoomMin = Number(options.zoomMin) || 10;\n if (zoomMin < 10) {\n zoomMin = 10;\n }\n var zoomMax = Number(options.zoomMax) || 10000 * year;\n if (zoomMax > 10000 * year) {\n zoomMax = 10000 * year;\n }\n if (zoomMax < zoomMin) {\n zoomMax = zoomMin;\n }\n\n // determine min and max date value\n var min = options.min ? options.min.valueOf() : undefined; // number\n var max = options.max ? options.max.valueOf() : undefined; // number\n if (min != undefined && max != undefined) {\n if (min >= max) {\n // empty range\n var day = 1000 * 60 * 60 * 24;\n max = min + day;\n }\n if (zoomMax > max - min) {\n zoomMax = max - min;\n }\n if (zoomMin > max - min) {\n zoomMin = max - min;\n }\n }\n\n // prevent empty interval\n if (startValue >= endValue) {\n endValue += 1000 * 60 * 60 * 24;\n }\n\n // prevent too small scale\n // TODO: IE has problems with milliseconds\n if (interval < zoomMin) {\n var diff = zoomMin - interval;\n var f = zoomAroundDate ? (zoomAroundDate.valueOf() - startValue) / interval : 0.5;\n startValue -= Math.round(diff * f);\n endValue += Math.round(diff * (1 - f));\n }\n\n // prevent too large scale\n if (interval > zoomMax) {\n var diff = interval - zoomMax;\n var f = zoomAroundDate ? (zoomAroundDate.valueOf() - startValue) / interval : 0.5;\n startValue += Math.round(diff * f);\n endValue -= Math.round(diff * (1 - f));\n }\n\n // prevent to small start date\n if (min != undefined) {\n var diff = startValue - min;\n if (diff < 0) {\n startValue -= diff;\n endValue -= diff;\n }\n }\n\n // prevent to large end date\n if (max != undefined) {\n var diff = max - endValue;\n if (diff < 0) {\n startValue += diff;\n endValue += diff;\n }\n }\n\n // apply new dates\n this.start = new Date(startValue);\n this.end = new Date(endValue);\n};\n\n/**\n * Delete an item after a confirmation.\n * The deletion can be cancelled by executing .cancelDelete() during the\n * triggered event 'delete'.\n * @param {int} index Index of the item to be deleted\n */\nlinks.Timeline.prototype.confirmDeleteItem = function (index) {\n this.applyDelete = true;\n\n // select the event to be deleted\n if (!this.isSelected(index)) {\n this.selectItem(index);\n }\n\n // fire a delete event trigger.\n // Note that the delete event can be canceled from within an event listener if\n // this listener calls the method cancelChange().\n this.trigger('delete');\n if (this.applyDelete) {\n this.deleteItem(index);\n }\n delete this.applyDelete;\n};\n\n/**\n * Delete an item\n * @param {int} index Index of the item to be deleted\n * @param {boolean} [preventRender=false] Do not re-render timeline if true\n * (optimization for multiple delete)\n */\nlinks.Timeline.prototype.deleteItem = function (index, preventRender) {\n if (index >= this.items.length) {\n throw \"Cannot delete row, index out of range\";\n }\n if (this.selection && this.selection.index !== undefined) {\n // adjust the selection\n if (this.selection.index == index) {\n // item to be deleted is selected\n this.unselectItem();\n } else if (this.selection.index > index) {\n // update selection index\n this.selection.index--;\n }\n }\n\n // actually delete the item and remove it from the DOM\n var item = this.items.splice(index, 1)[0];\n this.renderQueue.hide.push(item);\n\n // delete the row in the original data table\n if (this.data) {\n if (google && google.visualization && this.data instanceof google.visualization.DataTable) {\n this.data.removeRow(index);\n } else if (links.Timeline.isArray(this.data)) {\n this.data.splice(index, 1);\n } else {\n throw \"Cannot delete row from data, unknown data type\";\n }\n }\n\n // prepare data for clustering, by filtering and sorting by type\n if (this.options.cluster) {\n this.clusterGenerator.updateData();\n }\n if (!preventRender) {\n this.render();\n }\n};\n\n/**\n * Delete all items\n */\nlinks.Timeline.prototype.deleteAllItems = function () {\n this.unselectItem();\n\n // delete the loaded items\n this.clearItems();\n\n // delete the groups\n this.deleteGroups();\n\n // empty original data table\n if (this.data) {\n if (google && google.visualization && this.data instanceof google.visualization.DataTable) {\n this.data.removeRows(0, this.data.getNumberOfRows());\n } else if (links.Timeline.isArray(this.data)) {\n this.data.splice(0, this.data.length);\n } else {\n throw \"Cannot delete row from data, unknown data type\";\n }\n }\n\n // prepare data for clustering, by filtering and sorting by type\n if (this.options.cluster) {\n this.clusterGenerator.updateData();\n }\n this.render();\n};\n\n/**\n * Find the group from a given height in the timeline\n * @param {Number} height Height in the timeline\n * @return {Object | undefined} group The group object, or undefined if out\n * of range\n */\nlinks.Timeline.prototype.getGroupFromHeight = function (height) {\n var i,\n group,\n groups = this.groups;\n if (groups.length) {\n if (this.options.axisOnTop) {\n for (i = groups.length - 1; i >= 0; i--) {\n group = groups[i];\n if (height > group.top) {\n return group;\n }\n }\n } else {\n for (i = 0; i < groups.length; i++) {\n group = groups[i];\n if (height > group.top) {\n return group;\n }\n }\n }\n return group; // return the last group\n }\n\n return undefined;\n};\n\n/**\n * @constructor links.Timeline.Item\n * @param {Object} data Object containing parameters start, end\n * content, group, type, editable.\n * @param {Object} [options] Options to set initial property values\n * {Number} top\n * {Number} left\n * {Number} width\n * {Number} height\n */\nlinks.Timeline.Item = function (data, options) {\n if (data) {\n /* TODO: use parseJSONDate as soon as it is tested and working (in two directions)\n this.start = links.Timeline.parseJSONDate(data.start);\n this.end = links.Timeline.parseJSONDate(data.end);\n */\n this.start = data.start;\n this.end = data.end;\n this.content = data.content;\n this.className = data.className;\n this.editable = data.editable;\n this.group = data.group;\n this.type = data.type;\n }\n this.top = 0;\n this.left = 0;\n this.width = 0;\n this.height = 0;\n this.lineWidth = 0;\n this.dotWidth = 0;\n this.dotHeight = 0;\n this.rendered = false; // true when the item is draw in the Timeline DOM\n\n if (options) {\n // override the default properties\n for (var option in options) {\n if (options.hasOwnProperty(option)) {\n this[option] = options[option];\n }\n }\n }\n};\n\n/**\n * Reflow the Item: retrieve its actual size from the DOM\n * @return {boolean} resized returns true if the axis is resized\n */\nlinks.Timeline.Item.prototype.reflow = function () {\n // Should be implemented by sub-prototype\n return false;\n};\n\n/**\n * Append all image urls present in the items DOM to the provided array\n * @param {String[]} imageUrls\n */\nlinks.Timeline.Item.prototype.getImageUrls = function (imageUrls) {\n if (this.dom) {\n links.imageloader.filterImageUrls(this.dom, imageUrls);\n }\n};\n\n/**\n * Select the item\n */\nlinks.Timeline.Item.prototype.select = function () {\n // Should be implemented by sub-prototype\n};\n\n/**\n * Unselect the item\n */\nlinks.Timeline.Item.prototype.unselect = function () {\n // Should be implemented by sub-prototype\n};\n\n/**\n * Creates the DOM for the item, depending on its type\n * @return {Element | undefined}\n */\nlinks.Timeline.Item.prototype.createDOM = function () {\n // Should be implemented by sub-prototype\n};\n\n/**\n * Append the items DOM to the given HTML container. If items DOM does not yet\n * exist, it will be created first.\n * @param {Element} container\n */\nlinks.Timeline.Item.prototype.showDOM = function (container) {\n // Should be implemented by sub-prototype\n};\n\n/**\n * Remove the items DOM from the current HTML container\n * @param {Element} container\n */\nlinks.Timeline.Item.prototype.hideDOM = function (container) {\n // Should be implemented by sub-prototype\n};\n\n/**\n * Update the DOM of the item. This will update the content and the classes\n * of the item\n */\nlinks.Timeline.Item.prototype.updateDOM = function () {\n // Should be implemented by sub-prototype\n};\n\n/**\n * Reposition the item, recalculate its left, top, and width, using the current\n * range of the timeline and the timeline options.\n * @param {links.Timeline} timeline\n */\nlinks.Timeline.Item.prototype.updatePosition = function (timeline) {\n // Should be implemented by sub-prototype\n};\n\n/**\n * Check if the item is drawn in the timeline (i.e. the DOM of the item is\n * attached to the frame. You may also just request the parameter item.rendered\n * @return {boolean} rendered\n */\nlinks.Timeline.Item.prototype.isRendered = function () {\n return this.rendered;\n};\n\n/**\n * Check if the item is located in the visible area of the timeline, and\n * not part of a cluster\n * @param {Date} start\n * @param {Date} end\n * @return {boolean} visible\n */\nlinks.Timeline.Item.prototype.isVisible = function (start, end) {\n // Should be implemented by sub-prototype\n return false;\n};\n\n/**\n * Reposition the item\n * @param {Number} left\n * @param {Number} right\n */\nlinks.Timeline.Item.prototype.setPosition = function (left, right) {\n // Should be implemented by sub-prototype\n};\n\n/**\n * Calculate the left position of the item\n * @param {links.Timeline} timeline\n * @return {Number} left\n */\nlinks.Timeline.Item.prototype.getLeft = function (timeline) {\n // Should be implemented by sub-prototype\n return 0;\n};\n\n/**\n * Calculate the right position of the item\n * @param {links.Timeline} timeline\n * @return {Number} right\n */\nlinks.Timeline.Item.prototype.getRight = function (timeline) {\n // Should be implemented by sub-prototype\n return 0;\n};\n\n/**\n * Calculate the width of the item\n * @param {links.Timeline} timeline\n * @return {Number} width\n */\nlinks.Timeline.Item.prototype.getWidth = function (timeline) {\n // Should be implemented by sub-prototype\n return this.width || 0; // last rendered width\n};\n\n/**\n * @constructor links.Timeline.ItemBox\n * @extends links.Timeline.Item\n * @param {Object} data Object containing parameters start, end\n * content, group, type, className, editable.\n * @param {Object} [options] Options to set initial property values\n * {Number} top\n * {Number} left\n * {Number} width\n * {Number} height\n */\nlinks.Timeline.ItemBox = function (data, options) {\n links.Timeline.Item.call(this, data, options);\n};\nlinks.Timeline.ItemBox.prototype = new links.Timeline.Item();\n\n/**\n * Reflow the Item: retrieve its actual size from the DOM\n * @return {boolean} resized returns true if the axis is resized\n * @override\n */\nlinks.Timeline.ItemBox.prototype.reflow = function () {\n var dom = this.dom,\n dotHeight = dom.dot.offsetHeight,\n dotWidth = dom.dot.offsetWidth,\n lineWidth = dom.line.offsetWidth,\n resized = this.dotHeight != dotHeight || this.dotWidth != dotWidth || this.lineWidth != lineWidth;\n this.dotHeight = dotHeight;\n this.dotWidth = dotWidth;\n this.lineWidth = lineWidth;\n return resized;\n};\n\n/**\n * Select the item\n * @override\n */\nlinks.Timeline.ItemBox.prototype.select = function () {\n var dom = this.dom;\n links.Timeline.addClassName(dom, 'timeline-event-selected ui-state-active');\n links.Timeline.addClassName(dom.line, 'timeline-event-selected ui-state-active');\n links.Timeline.addClassName(dom.dot, 'timeline-event-selected ui-state-active');\n};\n\n/**\n * Unselect the item\n * @override\n */\nlinks.Timeline.ItemBox.prototype.unselect = function () {\n var dom = this.dom;\n links.Timeline.removeClassName(dom, 'timeline-event-selected ui-state-active');\n links.Timeline.removeClassName(dom.line, 'timeline-event-selected ui-state-active');\n links.Timeline.removeClassName(dom.dot, 'timeline-event-selected ui-state-active');\n};\n\n/**\n * Creates the DOM for the item, depending on its type\n * @return {Element | undefined}\n * @override\n */\nlinks.Timeline.ItemBox.prototype.createDOM = function () {\n // background box\n var divBox = document.createElement(\"DIV\");\n divBox.style.position = \"absolute\";\n divBox.style.left = this.left + \"px\";\n divBox.style.top = this.top + \"px\";\n\n // contents box (inside the background box). used for making margins\n var divContent = document.createElement(\"DIV\");\n divContent.className = \"timeline-event-content\";\n divContent.innerHTML = this.content;\n divBox.appendChild(divContent);\n\n // line to axis\n var divLine = document.createElement(\"DIV\");\n divLine.style.position = \"absolute\";\n divLine.style.width = \"0px\";\n // important: the vertical line is added at the front of the list of elements,\n // so it will be drawn behind all boxes and ranges\n divBox.line = divLine;\n\n // dot on axis\n var divDot = document.createElement(\"DIV\");\n divDot.style.position = \"absolute\";\n divDot.style.width = \"0px\";\n divDot.style.height = \"0px\";\n divBox.dot = divDot;\n this.dom = divBox;\n this.updateDOM();\n return divBox;\n};\n\n/**\n * Append the items DOM to the given HTML container. If items DOM does not yet\n * exist, it will be created first.\n * @param {Element} container\n * @override\n */\nlinks.Timeline.ItemBox.prototype.showDOM = function (container) {\n var dom = this.dom;\n if (!dom) {\n dom = this.createDOM();\n }\n if (dom.parentNode != container) {\n if (dom.parentNode) {\n // container is changed. remove from old container\n this.hideDOM();\n }\n\n // append to this container\n container.appendChild(dom);\n container.insertBefore(dom.line, container.firstChild);\n // Note: line must be added in front of the this,\n // such that it stays below all this\n container.appendChild(dom.dot);\n this.rendered = true;\n }\n};\n\n/**\n * Remove the items DOM from the current HTML container, but keep the DOM in\n * memory\n * @override\n */\nlinks.Timeline.ItemBox.prototype.hideDOM = function () {\n var dom = this.dom;\n if (dom) {\n if (dom.parentNode) {\n dom.parentNode.removeChild(dom);\n }\n if (dom.line && dom.line.parentNode) {\n dom.line.parentNode.removeChild(dom.line);\n }\n if (dom.dot && dom.dot.parentNode) {\n dom.dot.parentNode.removeChild(dom.dot);\n }\n this.rendered = false;\n }\n};\n\n/**\n * Update the DOM of the item. This will update the content and the classes\n * of the item\n * @override\n */\nlinks.Timeline.ItemBox.prototype.updateDOM = function () {\n var divBox = this.dom;\n if (divBox) {\n var divLine = divBox.line;\n var divDot = divBox.dot;\n\n // update contents\n divBox.firstChild.innerHTML = this.content;\n\n // update class\n divBox.className = \"timeline-event timeline-event-box ui-widget ui-state-default\";\n divLine.className = \"timeline-event timeline-event-line ui-widget ui-state-default\";\n divDot.className = \"timeline-event timeline-event-dot ui-widget ui-state-default\";\n if (this.isCluster) {\n links.Timeline.addClassName(divBox, 'timeline-event-cluster ui-widget-header');\n links.Timeline.addClassName(divLine, 'timeline-event-cluster ui-widget-header');\n links.Timeline.addClassName(divDot, 'timeline-event-cluster ui-widget-header');\n }\n\n // add item specific class name when provided\n if (this.className) {\n links.Timeline.addClassName(divBox, this.className);\n links.Timeline.addClassName(divLine, this.className);\n links.Timeline.addClassName(divDot, this.className);\n }\n\n // TODO: apply selected className?\n }\n};\n\n/**\n * Reposition the item, recalculate its left, top, and width, using the current\n * range of the timeline and the timeline options.\n * @param {links.Timeline} timeline\n * @override\n */\nlinks.Timeline.ItemBox.prototype.updatePosition = function (timeline) {\n var dom = this.dom;\n if (dom) {\n var left = timeline.timeToScreen(this.start),\n axisOnTop = timeline.options.axisOnTop,\n axisTop = timeline.size.axis.top,\n axisHeight = timeline.size.axis.height,\n boxAlign = timeline.options.box && timeline.options.box.align ? timeline.options.box.align : undefined;\n dom.style.top = this.top + \"px\";\n if (boxAlign == 'right') {\n dom.style.left = left - this.width + \"px\";\n } else if (boxAlign == 'left') {\n dom.style.left = left + \"px\";\n } else {\n // default or 'center'\n dom.style.left = left - this.width / 2 + \"px\";\n }\n var line = dom.line;\n var dot = dom.dot;\n line.style.left = left - this.lineWidth / 2 + \"px\";\n dot.style.left = left - this.dotWidth / 2 + \"px\";\n if (axisOnTop) {\n line.style.top = axisHeight + \"px\";\n line.style.height = Math.max(this.top - axisHeight, 0) + \"px\";\n dot.style.top = axisHeight - this.dotHeight / 2 + \"px\";\n } else {\n line.style.top = this.top + this.height + \"px\";\n line.style.height = Math.max(axisTop - this.top - this.height, 0) + \"px\";\n dot.style.top = axisTop - this.dotHeight / 2 + \"px\";\n }\n }\n};\n\n/**\n * Check if the item is visible in the timeline, and not part of a cluster\n * @param {Date} start\n * @param {Date} end\n * @return {Boolean} visible\n * @override\n */\nlinks.Timeline.ItemBox.prototype.isVisible = function (start, end) {\n if (this.cluster) {\n return false;\n }\n return this.start > start && this.start < end;\n};\n\n/**\n * Reposition the item\n * @param {Number} left\n * @param {Number} right\n * @override\n */\nlinks.Timeline.ItemBox.prototype.setPosition = function (left, right) {\n var dom = this.dom;\n dom.style.left = left - this.width / 2 + \"px\";\n dom.line.style.left = left - this.lineWidth / 2 + \"px\";\n dom.dot.style.left = left - this.dotWidth / 2 + \"px\";\n if (this.group) {\n this.top = this.group.top;\n dom.style.top = this.top + 'px';\n }\n};\n\n/**\n * Calculate the left position of the item\n * @param {links.Timeline} timeline\n * @return {Number} left\n * @override\n */\nlinks.Timeline.ItemBox.prototype.getLeft = function (timeline) {\n var boxAlign = timeline.options.box && timeline.options.box.align ? timeline.options.box.align : undefined;\n var left = timeline.timeToScreen(this.start);\n if (boxAlign == 'right') {\n left = left - width;\n } else {\n // default or 'center'\n left = left - this.width / 2;\n }\n return left;\n};\n\n/**\n * Calculate the right position of the item\n * @param {links.Timeline} timeline\n * @return {Number} right\n * @override\n */\nlinks.Timeline.ItemBox.prototype.getRight = function (timeline) {\n var boxAlign = timeline.options.box && timeline.options.box.align ? timeline.options.box.align : undefined;\n var left = timeline.timeToScreen(this.start);\n var right;\n if (boxAlign == 'right') {\n right = left;\n } else if (boxAlign == 'left') {\n right = left + this.width;\n } else {\n // default or 'center'\n right = left + this.width / 2;\n }\n return right;\n};\n\n/**\n * @constructor links.Timeline.ItemRange\n * @extends links.Timeline.Item\n * @param {Object} data Object containing parameters start, end\n * content, group, type, className, editable.\n * @param {Object} [options] Options to set initial property values\n * {Number} top\n * {Number} left\n * {Number} width\n * {Number} height\n */\nlinks.Timeline.ItemRange = function (data, options) {\n links.Timeline.Item.call(this, data, options);\n};\nlinks.Timeline.ItemRange.prototype = new links.Timeline.Item();\n\n/**\n * Select the item\n * @override\n */\nlinks.Timeline.ItemRange.prototype.select = function () {\n var dom = this.dom;\n links.Timeline.addClassName(dom, 'timeline-event-selected ui-state-active');\n};\n\n/**\n * Unselect the item\n * @override\n */\nlinks.Timeline.ItemRange.prototype.unselect = function () {\n var dom = this.dom;\n links.Timeline.removeClassName(dom, 'timeline-event-selected ui-state-active');\n};\n\n/**\n * Creates the DOM for the item, depending on its type\n * @return {Element | undefined}\n * @override\n */\nlinks.Timeline.ItemRange.prototype.createDOM = function () {\n // background box\n var divBox = document.createElement(\"DIV\");\n divBox.style.position = \"absolute\";\n\n // contents box\n var divContent = document.createElement(\"DIV\");\n divContent.className = \"timeline-event-content\";\n divBox.appendChild(divContent);\n this.dom = divBox;\n this.updateDOM();\n return divBox;\n};\n\n/**\n * Append the items DOM to the given HTML container. If items DOM does not yet\n * exist, it will be created first.\n * @param {Element} container\n * @override\n */\nlinks.Timeline.ItemRange.prototype.showDOM = function (container) {\n var dom = this.dom;\n if (!dom) {\n dom = this.createDOM();\n }\n if (dom.parentNode != container) {\n if (dom.parentNode) {\n // container changed. remove the item from the old container\n this.hideDOM();\n }\n\n // append to the new container\n container.appendChild(dom);\n this.rendered = true;\n }\n};\n\n/**\n * Remove the items DOM from the current HTML container\n * The DOM will be kept in memory\n * @override\n */\nlinks.Timeline.ItemRange.prototype.hideDOM = function () {\n var dom = this.dom;\n if (dom) {\n if (dom.parentNode) {\n dom.parentNode.removeChild(dom);\n }\n this.rendered = false;\n }\n};\n\n/**\n * Update the DOM of the item. This will update the content and the classes\n * of the item\n * @override\n */\nlinks.Timeline.ItemRange.prototype.updateDOM = function () {\n var divBox = this.dom;\n if (divBox) {\n // update contents\n divBox.firstChild.innerHTML = this.content;\n\n // update class\n divBox.className = \"timeline-event timeline-event-range ui-widget ui-state-default\";\n if (this.isCluster) {\n links.Timeline.addClassName(divBox, 'timeline-event-cluster ui-widget-header');\n }\n\n // add item specific class name when provided\n if (this.className) {\n links.Timeline.addClassName(divBox, this.className);\n }\n\n // TODO: apply selected className?\n }\n};\n\n/**\n * Reposition the item, recalculate its left, top, and width, using the current\n * range of the timeline and the timeline options. *\n * @param {links.Timeline} timeline\n * @override\n */\nlinks.Timeline.ItemRange.prototype.updatePosition = function (timeline) {\n var dom = this.dom;\n if (dom) {\n var contentWidth = timeline.size.contentWidth,\n left = timeline.timeToScreen(this.start),\n right = timeline.timeToScreen(this.end);\n\n // limit the width of the this, as browsers cannot draw very wide divs\n if (left < -contentWidth) {\n left = -contentWidth;\n }\n if (right > 2 * contentWidth) {\n right = 2 * contentWidth;\n }\n dom.style.top = this.top + \"px\";\n dom.style.left = left + \"px\";\n //dom.style.width = Math.max(right - left - 2 * this.borderWidth, 1) + \"px\"; // TODO: borderWidth\n dom.style.width = Math.max(right - left, 1) + \"px\";\n }\n};\n\n/**\n * Check if the item is visible in the timeline, and not part of a cluster\n * @param {Number} start\n * @param {Number} end\n * @return {boolean} visible\n * @override\n */\nlinks.Timeline.ItemRange.prototype.isVisible = function (start, end) {\n if (this.cluster) {\n return false;\n }\n return this.end > start && this.start < end;\n};\n\n/**\n * Reposition the item\n * @param {Number} left\n * @param {Number} right\n * @override\n */\nlinks.Timeline.ItemRange.prototype.setPosition = function (left, right) {\n var dom = this.dom;\n dom.style.left = left + 'px';\n dom.style.width = right - left + 'px';\n if (this.group) {\n this.top = this.group.top;\n dom.style.top = this.top + 'px';\n }\n};\n\n/**\n * Calculate the left position of the item\n * @param {links.Timeline} timeline\n * @return {Number} left\n * @override\n */\nlinks.Timeline.ItemRange.prototype.getLeft = function (timeline) {\n return timeline.timeToScreen(this.start);\n};\n\n/**\n * Calculate the right position of the item\n * @param {links.Timeline} timeline\n * @return {Number} right\n * @override\n */\nlinks.Timeline.ItemRange.prototype.getRight = function (timeline) {\n return timeline.timeToScreen(this.end);\n};\n\n/**\n * Calculate the width of the item\n * @param {links.Timeline} timeline\n * @return {Number} width\n * @override\n */\nlinks.Timeline.ItemRange.prototype.getWidth = function (timeline) {\n return timeline.timeToScreen(this.end) - timeline.timeToScreen(this.start);\n};\n\n/**\n * @constructor links.Timeline.ItemFloatingRange\n * @extends links.Timeline.Item\n * @param {Object} data Object containing parameters start, end\n * content, group, type, className, editable.\n * @param {Object} [options] Options to set initial property values\n * {Number} top\n * {Number} left\n * {Number} width\n * {Number} height\n */\nlinks.Timeline.ItemFloatingRange = function (data, options) {\n links.Timeline.Item.call(this, data, options);\n};\nlinks.Timeline.ItemFloatingRange.prototype = new links.Timeline.Item();\n\n/**\n * Select the item\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.select = function () {\n var dom = this.dom;\n links.Timeline.addClassName(dom, 'timeline-event-selected ui-state-active');\n};\n\n/**\n * Unselect the item\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.unselect = function () {\n var dom = this.dom;\n links.Timeline.removeClassName(dom, 'timeline-event-selected ui-state-active');\n};\n\n/**\n * Creates the DOM for the item, depending on its type\n * @return {Element | undefined}\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.createDOM = function () {\n // background box\n var divBox = document.createElement(\"DIV\");\n divBox.style.position = \"absolute\";\n\n // contents box\n var divContent = document.createElement(\"DIV\");\n divContent.className = \"timeline-event-content\";\n divBox.appendChild(divContent);\n this.dom = divBox;\n this.updateDOM();\n return divBox;\n};\n\n/**\n * Append the items DOM to the given HTML container. If items DOM does not yet\n * exist, it will be created first.\n * @param {Element} container\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.showDOM = function (container) {\n var dom = this.dom;\n if (!dom) {\n dom = this.createDOM();\n }\n if (dom.parentNode != container) {\n if (dom.parentNode) {\n // container changed. remove the item from the old container\n this.hideDOM();\n }\n\n // append to the new container\n container.appendChild(dom);\n this.rendered = true;\n }\n};\n\n/**\n * Remove the items DOM from the current HTML container\n * The DOM will be kept in memory\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.hideDOM = function () {\n var dom = this.dom;\n if (dom) {\n if (dom.parentNode) {\n dom.parentNode.removeChild(dom);\n }\n this.rendered = false;\n }\n};\n\n/**\n * Update the DOM of the item. This will update the content and the classes\n * of the item\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.updateDOM = function () {\n var divBox = this.dom;\n if (divBox) {\n // update contents\n divBox.firstChild.innerHTML = this.content;\n\n // update class\n divBox.className = \"timeline-event timeline-event-range ui-widget ui-state-default\";\n if (this.isCluster) {\n links.Timeline.addClassName(divBox, 'timeline-event-cluster ui-widget-header');\n }\n\n // add item specific class name when provided\n if (this.className) {\n links.Timeline.addClassName(divBox, this.className);\n }\n\n // TODO: apply selected className?\n }\n};\n\n/**\n * Reposition the item, recalculate its left, top, and width, using the current\n * range of the timeline and the timeline options. *\n * @param {links.Timeline} timeline\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.updatePosition = function (timeline) {\n var dom = this.dom;\n if (dom) {\n var contentWidth = timeline.size.contentWidth,\n left = this.getLeft(timeline),\n // NH use getLeft\n right = this.getRight(timeline); // NH use getRight;\n\n // limit the width of the this, as browsers cannot draw very wide divs\n if (left < -contentWidth) {\n left = -contentWidth;\n }\n if (right > 2 * contentWidth) {\n right = 2 * contentWidth;\n }\n dom.style.top = this.top + \"px\";\n dom.style.left = left + \"px\";\n //dom.style.width = Math.max(right - left - 2 * this.borderWidth, 1) + \"px\"; // TODO: borderWidth\n dom.style.width = Math.max(right - left, 1) + \"px\";\n }\n};\n\n/**\n * Check if the item is visible in the timeline, and not part of a cluster\n * @param {Number} start\n * @param {Number} end\n * @return {boolean} visible\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.isVisible = function (start, end) {\n if (this.cluster) {\n return false;\n }\n\n // NH check for no end value\n if (this.end && this.start) {\n return this.end > start && this.start < end;\n } else if (this.start) {\n return this.start < end;\n } else if (this.end) {\n return this.end > start;\n } else {\n return true;\n }\n};\n\n/**\n * Reposition the item\n * @param {Number} left\n * @param {Number} right\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.setPosition = function (left, right) {\n var dom = this.dom;\n dom.style.left = left + 'px';\n dom.style.width = right - left + 'px';\n if (this.group) {\n this.top = this.group.top;\n dom.style.top = this.top + 'px';\n }\n};\n\n/**\n * Calculate the left position of the item\n * @param {links.Timeline} timeline\n * @return {Number} left\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.getLeft = function (timeline) {\n // NH check for no start value\n if (this.start) {\n return timeline.timeToScreen(this.start);\n } else {\n return 0;\n }\n};\n\n/**\n * Calculate the right position of the item\n * @param {links.Timeline} timeline\n * @return {Number} right\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.getRight = function (timeline) {\n // NH check for no end value\n if (this.end) {\n return timeline.timeToScreen(this.end);\n } else {\n return timeline.size.contentWidth;\n }\n};\n\n/**\n * Calculate the width of the item\n * @param {links.Timeline} timeline\n * @return {Number} width\n * @override\n */\nlinks.Timeline.ItemFloatingRange.prototype.getWidth = function (timeline) {\n return this.getRight(timeline) - this.getLeft(timeline);\n};\n\n/**\n * @constructor links.Timeline.ItemDot\n * @extends links.Timeline.Item\n * @param {Object} data Object containing parameters start, end\n * content, group, type, className, editable.\n * @param {Object} [options] Options to set initial property values\n * {Number} top\n * {Number} left\n * {Number} width\n * {Number} height\n */\nlinks.Timeline.ItemDot = function (data, options) {\n links.Timeline.Item.call(this, data, options);\n};\nlinks.Timeline.ItemDot.prototype = new links.Timeline.Item();\n\n/**\n * Reflow the Item: retrieve its actual size from the DOM\n * @return {boolean} resized returns true if the axis is resized\n * @override\n */\nlinks.Timeline.ItemDot.prototype.reflow = function () {\n var dom = this.dom,\n dotHeight = dom.dot.offsetHeight,\n dotWidth = dom.dot.offsetWidth,\n contentHeight = dom.content.offsetHeight,\n resized = this.dotHeight != dotHeight || this.dotWidth != dotWidth || this.contentHeight != contentHeight;\n this.dotHeight = dotHeight;\n this.dotWidth = dotWidth;\n this.contentHeight = contentHeight;\n return resized;\n};\n\n/**\n * Select the item\n * @override\n */\nlinks.Timeline.ItemDot.prototype.select = function () {\n var dom = this.dom;\n links.Timeline.addClassName(dom, 'timeline-event-selected ui-state-active');\n};\n\n/**\n * Unselect the item\n * @override\n */\nlinks.Timeline.ItemDot.prototype.unselect = function () {\n var dom = this.dom;\n links.Timeline.removeClassName(dom, 'timeline-event-selected ui-state-active');\n};\n\n/**\n * Creates the DOM for the item, depending on its type\n * @return {Element | undefined}\n * @override\n */\nlinks.Timeline.ItemDot.prototype.createDOM = function () {\n // background box\n var divBox = document.createElement(\"DIV\");\n divBox.style.position = \"absolute\";\n\n // contents box, right from the dot\n var divContent = document.createElement(\"DIV\");\n divContent.className = \"timeline-event-content\";\n divBox.appendChild(divContent);\n\n // dot at start\n var divDot = document.createElement(\"DIV\");\n divDot.style.position = \"absolute\";\n divDot.style.width = \"0px\";\n divDot.style.height = \"0px\";\n divBox.appendChild(divDot);\n divBox.content = divContent;\n divBox.dot = divDot;\n this.dom = divBox;\n this.updateDOM();\n return divBox;\n};\n\n/**\n * Append the items DOM to the given HTML container. If items DOM does not yet\n * exist, it will be created first.\n * @param {Element} container\n * @override\n */\nlinks.Timeline.ItemDot.prototype.showDOM = function (container) {\n var dom = this.dom;\n if (!dom) {\n dom = this.createDOM();\n }\n if (dom.parentNode != container) {\n if (dom.parentNode) {\n // container changed. remove it from old container first\n this.hideDOM();\n }\n\n // append to container\n container.appendChild(dom);\n this.rendered = true;\n }\n};\n\n/**\n * Remove the items DOM from the current HTML container\n * @override\n */\nlinks.Timeline.ItemDot.prototype.hideDOM = function () {\n var dom = this.dom;\n if (dom) {\n if (dom.parentNode) {\n dom.parentNode.removeChild(dom);\n }\n this.rendered = false;\n }\n};\n\n/**\n * Update the DOM of the item. This will update the content and the classes\n * of the item\n * @override\n */\nlinks.Timeline.ItemDot.prototype.updateDOM = function () {\n if (this.dom) {\n var divBox = this.dom;\n var divDot = divBox.dot;\n\n // update contents\n divBox.firstChild.innerHTML = this.content;\n\n // update classes\n divBox.className = \"timeline-event-dot-container\";\n divDot.className = \"timeline-event timeline-event-dot ui-widget ui-state-default\";\n if (this.isCluster) {\n links.Timeline.addClassName(divBox, 'timeline-event-cluster ui-widget-header');\n links.Timeline.addClassName(divDot, 'timeline-event-cluster ui-widget-header');\n }\n\n // add item specific class name when provided\n if (this.className) {\n links.Timeline.addClassName(divBox, this.className);\n links.Timeline.addClassName(divDot, this.className);\n }\n\n // TODO: apply selected className?\n }\n};\n\n/**\n * Reposition the item, recalculate its left, top, and width, using the current\n * range of the timeline and the timeline options. *\n * @param {links.Timeline} timeline\n * @override\n */\nlinks.Timeline.ItemDot.prototype.updatePosition = function (timeline) {\n var dom = this.dom;\n if (dom) {\n var left = timeline.timeToScreen(this.start);\n dom.style.top = this.top + \"px\";\n dom.style.left = left - this.dotWidth / 2 + \"px\";\n dom.content.style.marginLeft = 1.5 * this.dotWidth + \"px\";\n //dom.content.style.marginRight = (0.5 * this.dotWidth) + \"px\"; // TODO\n dom.dot.style.top = (this.height - this.dotHeight) / 2 + \"px\";\n }\n};\n\n/**\n * Check if the item is visible in the timeline, and not part of a cluster.\n * @param {Date} start\n * @param {Date} end\n * @return {boolean} visible\n * @override\n */\nlinks.Timeline.ItemDot.prototype.isVisible = function (start, end) {\n if (this.cluster) {\n return false;\n }\n return this.start > start && this.start < end;\n};\n\n/**\n * Reposition the item\n * @param {Number} left\n * @param {Number} right\n * @override\n */\nlinks.Timeline.ItemDot.prototype.setPosition = function (left, right) {\n var dom = this.dom;\n dom.style.left = left - this.dotWidth / 2 + \"px\";\n if (this.group) {\n this.top = this.group.top;\n dom.style.top = this.top + 'px';\n }\n};\n\n/**\n * Calculate the left position of the item\n * @param {links.Timeline} timeline\n * @return {Number} left\n * @override\n */\nlinks.Timeline.ItemDot.prototype.getLeft = function (timeline) {\n return timeline.timeToScreen(this.start);\n};\n\n/**\n * Calculate the right position of the item\n * @param {links.Timeline} timeline\n * @return {Number} right\n * @override\n */\nlinks.Timeline.ItemDot.prototype.getRight = function (timeline) {\n return timeline.timeToScreen(this.start) + this.width;\n};\n\n/**\n * Retrieve the properties of an item.\n * @param {Number} index\n * @return {Object} itemData Object containing item properties:
\n * {Date} start (required),\n * {Date} end (optional),\n * {String} content (required),\n * {String} group (optional),\n * {String} className (optional)\n * {boolean} editable (optional)\n * {String} type (optional)\n */\nlinks.Timeline.prototype.getItem = function (index) {\n if (index >= this.items.length) {\n throw \"Cannot get item, index out of range\";\n }\n\n // take the original data as start, includes foreign fields\n var data = this.data,\n itemData;\n if (google && google.visualization && data instanceof google.visualization.DataTable) {\n // map the datatable columns\n var cols = links.Timeline.mapColumnIds(data);\n itemData = {};\n for (var col in cols) {\n if (cols.hasOwnProperty(col)) {\n itemData[col] = this.data.getValue(index, cols[col]);\n }\n }\n } else if (links.Timeline.isArray(this.data)) {\n // read JSON array\n itemData = links.Timeline.clone(this.data[index]);\n } else {\n throw \"Unknown data type. DataTable or Array expected.\";\n }\n\n // override the data with current settings of the item (should be the same)\n var item = this.items[index];\n itemData.start = new Date(item.start.valueOf());\n if (item.end) {\n itemData.end = new Date(item.end.valueOf());\n }\n itemData.content = item.content;\n if (item.group) {\n itemData.group = this.getGroupName(item.group);\n }\n if (item.className) {\n itemData.className = item.className;\n }\n if (typeof item.editable !== 'undefined') {\n itemData.editable = item.editable;\n }\n if (item.type) {\n itemData.type = item.type;\n }\n return itemData;\n};\n\n/**\n * Retrieve the properties of a cluster.\n * @param {Number} index\n * @return {Object} clusterdata Object containing cluster properties:
\n * {Date} start (required),\n * {String} type (optional)\n * {Array} array with item data as is in getItem()\n */\nlinks.Timeline.prototype.getCluster = function (index) {\n if (index >= this.clusters.length) {\n throw \"Cannot get cluster, index out of range\";\n }\n var clusterData = {},\n cluster = this.clusters[index],\n clusterItems = cluster.items;\n clusterData.start = new Date(cluster.start.valueOf());\n if (cluster.type) {\n clusterData.type = cluster.type;\n }\n\n // push cluster item data\n clusterData.items = [];\n for (var i = 0; i < clusterItems.length; i++) {\n for (var j = 0; j < this.items.length; j++) {\n // TODO could be nicer to be able to have the item index into the cluster\n if (this.items[j] == clusterItems[i]) {\n clusterData.items.push(this.getItem(j));\n break;\n }\n }\n }\n return clusterData;\n};\n\n/**\n * Add a new item.\n * @param {Object} itemData Object containing item properties:
\n * {Date} start (required),\n * {Date} end (optional),\n * {String} content (required),\n * {String} group (optional)\n * {String} className (optional)\n * {Boolean} editable (optional)\n * {String} type (optional)\n * @param {boolean} [preventRender=false] Do not re-render timeline if true\n */\nlinks.Timeline.prototype.addItem = function (itemData, preventRender) {\n var itemsData = [itemData];\n this.addItems(itemsData, preventRender);\n};\n\n/**\n * Add new items.\n * @param {Array} itemsData An array containing Objects.\n * The objects must have the following parameters:\n * {Date} start,\n * {Date} end,\n * {String} content with text or HTML code,\n * {String} group (optional)\n * {String} className (optional)\n * {String} editable (optional)\n * {String} type (optional)\n * @param {boolean} [preventRender=false] Do not re-render timeline if true\n */\nlinks.Timeline.prototype.addItems = function (itemsData, preventRender) {\n var timeline = this,\n items = this.items;\n\n // append the items\n itemsData.forEach(function (itemData) {\n var index = items.length;\n items.push(timeline.createItem(itemData));\n timeline.updateData(index, itemData);\n\n // note: there is no need to add the item to the renderQueue, that\n // will be done when this.render() is executed and all items are\n // filtered again.\n });\n\n // prepare data for clustering, by filtering and sorting by type\n if (this.options.cluster) {\n this.clusterGenerator.updateData();\n }\n if (!preventRender) {\n this.render({\n animate: false\n });\n }\n};\n\n/**\n * Create an item object, containing all needed parameters\n * @param {Object} itemData Object containing parameters start, end\n * content, group.\n * @return {Object} item\n */\nlinks.Timeline.prototype.createItem = function (itemData) {\n var type = itemData.type || (itemData.end ? 'range' : this.options.style);\n var data = links.Timeline.clone(itemData);\n data.type = type;\n data.group = this.getGroup(itemData.group);\n // TODO: optimize this, when creating an item, all data is copied twice...\n\n // TODO: is initialTop needed?\n var initialTop,\n options = this.options;\n if (options.axisOnTop) {\n initialTop = this.size.axis.height + options.eventMarginAxis + options.eventMargin / 2;\n } else {\n initialTop = this.size.contentHeight - options.eventMarginAxis - options.eventMargin / 2;\n }\n if (type in this.itemTypes) {\n return new this.itemTypes[type](data, {\n 'top': initialTop\n });\n }\n console.log('ERROR: Unknown event type \"' + type + '\"');\n return new links.Timeline.Item(data, {\n 'top': initialTop\n });\n};\n\n/**\n * Edit an item\n * @param {Number} index\n * @param {Object} itemData Object containing item properties:
\n * {Date} start (required),\n * {Date} end (optional),\n * {String} content (required),\n * {String} group (optional)\n * @param {boolean} [preventRender=false] Do not re-render timeline if true\n */\nlinks.Timeline.prototype.changeItem = function (index, itemData, preventRender) {\n var oldItem = this.items[index];\n if (!oldItem) {\n throw \"Cannot change item, index out of range\";\n }\n\n // replace item, merge the changes\n var newItem = this.createItem({\n 'start': itemData.hasOwnProperty('start') ? itemData.start : oldItem.start,\n 'end': itemData.hasOwnProperty('end') ? itemData.end : oldItem.end,\n 'content': itemData.hasOwnProperty('content') ? itemData.content : oldItem.content,\n 'group': itemData.hasOwnProperty('group') ? itemData.group : this.getGroupName(oldItem.group),\n 'className': itemData.hasOwnProperty('className') ? itemData.className : oldItem.className,\n 'editable': itemData.hasOwnProperty('editable') ? itemData.editable : oldItem.editable,\n 'type': itemData.hasOwnProperty('type') ? itemData.type : oldItem.type\n });\n this.items[index] = newItem;\n\n // append the changes to the render queue\n this.renderQueue.hide.push(oldItem);\n this.renderQueue.show.push(newItem);\n\n // update the original data table\n this.updateData(index, itemData);\n\n // prepare data for clustering, by filtering and sorting by type\n if (this.options.cluster) {\n this.clusterGenerator.updateData();\n }\n if (!preventRender) {\n // redraw timeline\n this.render({\n animate: false\n });\n if (this.selection && this.selection.index == index) {\n newItem.select();\n }\n }\n};\n\n/**\n * Delete all groups\n */\nlinks.Timeline.prototype.deleteGroups = function () {\n this.groups = [];\n this.groupIndexes = {};\n};\n\n/**\n * Get a group by the group name. When the group does not exist,\n * it will be created.\n * @param {String} groupName the name of the group\n * @return {Object} groupObject\n */\nlinks.Timeline.prototype.getGroup = function (groupName) {\n var groups = this.groups,\n groupIndexes = this.groupIndexes,\n groupObj = undefined;\n var groupIndex = groupIndexes[groupName];\n if (groupIndex == undefined && groupName != undefined) {\n // not null or undefined\n groupObj = {\n 'content': groupName,\n 'labelTop': 0,\n 'lineTop': 0\n // note: this object will lateron get addition information,\n // such as height and width of the group\n };\n\n groups.push(groupObj);\n // sort the groups\n if (this.options.groupsOrder == true) {\n groups = groups.sort(function (a, b) {\n if (a.content > b.content) {\n return 1;\n }\n if (a.content < b.content) {\n return -1;\n }\n return 0;\n });\n } else if (typeof this.options.groupsOrder == \"function\") {\n groups = groups.sort(this.options.groupsOrder);\n }\n\n // rebuilt the groupIndexes\n for (var i = 0, iMax = groups.length; i < iMax; i++) {\n groupIndexes[groups[i].content] = i;\n }\n } else {\n groupObj = groups[groupIndex];\n }\n return groupObj;\n};\n\n/**\n * Get the group name from a group object.\n * @param {Object} groupObj\n * @return {String} groupName the name of the group, or undefined when group\n * was not provided\n */\nlinks.Timeline.prototype.getGroupName = function (groupObj) {\n return groupObj ? groupObj.content : undefined;\n};\n\n/**\n * Cancel a change item\n * This method can be called insed an event listener which catches the \"change\"\n * event. The changed event position will be undone.\n */\nlinks.Timeline.prototype.cancelChange = function () {\n this.applyChange = false;\n};\n\n/**\n * Cancel deletion of an item\n * This method can be called insed an event listener which catches the \"delete\"\n * event. Deletion of the event will be undone.\n */\nlinks.Timeline.prototype.cancelDelete = function () {\n this.applyDelete = false;\n};\n\n/**\n * Cancel creation of a new item\n * This method can be called insed an event listener which catches the \"new\"\n * event. Creation of the new the event will be undone.\n */\nlinks.Timeline.prototype.cancelAdd = function () {\n this.applyAdd = false;\n};\n\n/**\n * Select an event. The visible chart range will be moved such that the selected\n * event is placed in the middle.\n * For example selection = [{row: 5}];\n * @param {Array} selection An array with a column row, containing the row\n * number (the id) of the event to be selected.\n * @return {boolean} true if selection is succesfully set, else false.\n */\nlinks.Timeline.prototype.setSelection = function (selection) {\n if (selection != undefined && selection.length > 0) {\n if (selection[0].row != undefined) {\n var index = selection[0].row;\n if (this.items[index]) {\n var item = this.items[index];\n this.selectItem(index);\n\n // move the visible chart range to the selected event.\n var start = item.start;\n var end = item.end;\n var middle; // number\n if (end != undefined) {\n middle = (end.valueOf() + start.valueOf()) / 2;\n } else {\n middle = start.valueOf();\n }\n var diff = this.end.valueOf() - this.start.valueOf(),\n newStart = new Date(middle - diff / 2),\n newEnd = new Date(middle + diff / 2);\n this.setVisibleChartRange(newStart, newEnd);\n return true;\n }\n }\n } else {\n // unselect current selection\n this.unselectItem();\n }\n return false;\n};\n\n/**\n * Retrieve the currently selected event\n * @return {Array} sel An array with a column row, containing the row number\n * of the selected event. If there is no selection, an\n * empty array is returned.\n */\nlinks.Timeline.prototype.getSelection = function () {\n var sel = [];\n if (this.selection) {\n if (this.selection.index !== undefined) {\n sel.push({\n \"row\": this.selection.index\n });\n } else {\n sel.push({\n \"cluster\": this.selection.cluster\n });\n }\n }\n return sel;\n};\n\n/**\n * Select an item by its index\n * @param {Number} index\n */\nlinks.Timeline.prototype.selectItem = function (index) {\n this.unselectItem();\n this.selection = undefined;\n if (this.items[index] != undefined) {\n var item = this.items[index],\n domItem = item.dom;\n this.selection = {\n 'index': index\n };\n if (item && item.dom) {\n // TODO: move adjusting the domItem to the item itself\n if (this.isEditable(item)) {\n item.dom.style.cursor = 'move';\n }\n item.select();\n }\n this.repaintDeleteButton();\n this.repaintDragAreas();\n }\n};\n\n/**\n * Select an cluster by its index\n * @param {Number} index\n */\nlinks.Timeline.prototype.selectCluster = function (index) {\n this.unselectItem();\n this.selection = undefined;\n if (this.clusters[index] != undefined) {\n this.selection = {\n 'cluster': index\n };\n this.repaintDeleteButton();\n this.repaintDragAreas();\n }\n};\n\n/**\n * Check if an item is currently selected\n * @param {Number} index\n * @return {boolean} true if row is selected, else false\n */\nlinks.Timeline.prototype.isSelected = function (index) {\n return this.selection && this.selection.index == index;\n};\n\n/**\n * Unselect the currently selected event (if any)\n */\nlinks.Timeline.prototype.unselectItem = function () {\n if (this.selection && this.selection.index !== undefined) {\n var item = this.items[this.selection.index];\n if (item && item.dom) {\n var domItem = item.dom;\n domItem.style.cursor = '';\n item.unselect();\n }\n this.selection = undefined;\n this.repaintDeleteButton();\n this.repaintDragAreas();\n }\n};\n\n/**\n * Stack the items such that they don't overlap. The items will have a minimal\n * distance equal to options.eventMargin.\n * @param {boolean | undefined} animate if animate is true, the items are\n * moved to their new position animated\n * defaults to false.\n */\nlinks.Timeline.prototype.stackItems = function (animate) {\n if (animate == undefined) {\n animate = false;\n }\n\n // calculate the order and final stack position of the items\n var stack = this.stack;\n if (!stack) {\n stack = {};\n this.stack = stack;\n }\n stack.sortedItems = this.stackOrder(this.renderedItems);\n stack.finalItems = this.stackCalculateFinal(stack.sortedItems);\n if (animate || stack.timer) {\n // move animated to the final positions\n var timeline = this;\n var step = function step() {\n var arrived = timeline.stackMoveOneStep(stack.sortedItems, stack.finalItems);\n timeline.repaint();\n if (!arrived) {\n stack.timer = setTimeout(step, 30);\n } else {\n delete stack.timer;\n }\n };\n if (!stack.timer) {\n stack.timer = setTimeout(step, 30);\n }\n } else {\n // move immediately to the final positions\n this.stackMoveToFinal(stack.sortedItems, stack.finalItems);\n }\n};\n\n/**\n * Cancel any running animation\n */\nlinks.Timeline.prototype.stackCancelAnimation = function () {\n if (this.stack && this.stack.timer) {\n clearTimeout(this.stack.timer);\n delete this.stack.timer;\n }\n};\nlinks.Timeline.prototype.getItemsByGroup = function (items) {\n var itemsByGroup = {};\n for (var i = 0; i < items.length; ++i) {\n var item = items[i];\n var group = \"undefined\";\n if (item.group) {\n if (item.group.content) {\n group = item.group.content;\n } else {\n group = item.group;\n }\n }\n if (!itemsByGroup[group]) {\n itemsByGroup[group] = [];\n }\n itemsByGroup[group].push(item);\n }\n return itemsByGroup;\n};\n\n/**\n * Order the items in the array this.items. The default order is determined via:\n * - Ranges go before boxes and dots.\n * - The item with the oldest start time goes first\n * If a custom function has been provided via the stackorder option, then this will be used.\n * @param {Array} items Array with items\n * @return {Array} sortedItems Array with sorted items\n */\nlinks.Timeline.prototype.stackOrder = function (items) {\n // TODO: store the sorted items, to have less work later on\n var sortedItems = items.concat([]);\n\n //if a customer stack order function exists, use it.\n var f = this.options.customStackOrder && typeof this.options.customStackOrder === 'function' ? this.options.customStackOrder : function (a, b) {\n if ((a instanceof links.Timeline.ItemRange || a instanceof links.Timeline.ItemFloatingRange) && !(b instanceof links.Timeline.ItemRange || b instanceof links.Timeline.ItemFloatingRange)) {\n return -1;\n }\n if (!(a instanceof links.Timeline.ItemRange || a instanceof links.Timeline.ItemFloatingRange) && (b instanceof links.Timeline.ItemRange || b instanceof links.Timeline.ItemFloatingRange)) {\n return 1;\n }\n return a.left - b.left;\n };\n sortedItems.sort(f);\n return sortedItems;\n};\n\n/**\n * Adjust vertical positions of the events such that they don't overlap each\n * other.\n * @param {timeline.Item[]} items\n * @return {Object[]} finalItems\n */\nlinks.Timeline.prototype.stackCalculateFinal = function (items) {\n var size = this.size,\n options = this.options,\n axisOnTop = options.axisOnTop,\n eventMargin = options.eventMargin,\n eventMarginAxis = options.eventMarginAxis,\n groupBase = axisOnTop ? size.axis.height + eventMarginAxis + eventMargin / 2 : size.contentHeight - eventMarginAxis - eventMargin / 2,\n groupedItems,\n groupFinalItems,\n finalItems = [];\n groupedItems = this.getItemsByGroup(items);\n\n //\n // groupedItems contains all items by group, plus it may contain an\n // additional \"undefined\" group which contains all items with no group. We\n // first process the grouped items, and then the ungrouped\n //\n for (var j = 0; j < this.groups.length; ++j) {\n var group = this.groups[j];\n if (!groupedItems[group.content]) {\n if (axisOnTop) {\n groupBase += options.groupMinHeight + eventMargin;\n } else {\n groupBase -= options.groupMinHeight + eventMargin;\n }\n continue;\n }\n\n // initialize final positions and fill finalItems\n groupFinalItems = this.finalItemsPosition(groupedItems[group.content], groupBase, group);\n groupFinalItems.forEach(function (item) {\n finalItems.push(item);\n });\n if (axisOnTop) {\n groupBase += group.itemsHeight + eventMargin;\n } else {\n groupBase -= group.itemsHeight + eventMargin;\n }\n }\n\n //\n // Ungrouped items' turn now!\n //\n if (groupedItems[\"undefined\"]) {\n // initialize final positions and fill finalItems\n groupFinalItems = this.finalItemsPosition(groupedItems[\"undefined\"], groupBase);\n groupFinalItems.forEach(function (item) {\n finalItems.push(item);\n });\n }\n return finalItems;\n};\nlinks.Timeline.prototype.finalItemsPosition = function (items, groupBase, group) {\n var i,\n iMax,\n options = this.options,\n axisOnTop = options.axisOnTop,\n eventMargin = options.eventMargin,\n groupFinalItems;\n\n // initialize final positions and fill finalItems\n groupFinalItems = this.initialItemsPosition(items, groupBase);\n\n // calculate new, non-overlapping positions\n for (i = 0, iMax = groupFinalItems.length; i < iMax; i++) {\n var finalItem = groupFinalItems[i];\n var collidingItem = null;\n if (this.options.stackEvents) {\n do {\n // TODO: optimize checking for overlap. when there is a gap without items,\n // you only need to check for items from the next item on, not from zero\n collidingItem = this.stackItemsCheckOverlap(groupFinalItems, i, 0, i - 1);\n if (collidingItem != null) {\n // There is a collision. Reposition the event above the colliding element\n if (axisOnTop) {\n finalItem.top = collidingItem.top + collidingItem.height + eventMargin;\n } else {\n finalItem.top = collidingItem.top - finalItem.height - eventMargin;\n }\n finalItem.bottom = finalItem.top + finalItem.height;\n }\n } while (collidingItem);\n }\n if (group) {\n if (axisOnTop) {\n group.itemsHeight = group.itemsHeight ? Math.max(group.itemsHeight, finalItem.bottom - groupBase) : finalItem.height + eventMargin;\n } else {\n group.itemsHeight = group.itemsHeight ? Math.max(group.itemsHeight, groupBase - finalItem.top) : finalItem.height + eventMargin;\n }\n }\n }\n return groupFinalItems;\n};\nlinks.Timeline.prototype.initialItemsPosition = function (items, groupBase) {\n var options = this.options,\n axisOnTop = options.axisOnTop,\n finalItems = [];\n for (var i = 0, iMax = items.length; i < iMax; ++i) {\n var item = items[i],\n top,\n bottom,\n height = item.height,\n width = item.getWidth(this),\n right = item.getRight(this),\n left = right - width;\n top = axisOnTop ? groupBase : groupBase - height;\n bottom = top + height;\n finalItems.push({\n 'left': left,\n 'top': top,\n 'right': right,\n 'bottom': bottom,\n 'height': height,\n 'item': item\n });\n }\n return finalItems;\n};\n\n/**\n * Move the events one step in the direction of their final positions\n * @param {Array} currentItems Array with the real items and their current\n * positions\n * @param {Array} finalItems Array with objects containing the final\n * positions of the items\n * @return {boolean} arrived True if all items have reached their final\n * location, else false\n */\nlinks.Timeline.prototype.stackMoveOneStep = function (currentItems, finalItems) {\n var arrived = true;\n\n // apply new positions animated\n for (var i = 0, iMax = finalItems.length; i < iMax; i++) {\n var finalItem = finalItems[i],\n item = finalItem.item;\n var topNow = parseInt(item.top);\n var topFinal = parseInt(finalItem.top);\n var diff = topFinal - topNow;\n if (diff) {\n var step = topFinal == topNow ? 0 : topFinal > topNow ? 1 : -1;\n if (Math.abs(diff) > 4) step = diff / 4;\n var topNew = parseInt(topNow + step);\n if (topNew != topFinal) {\n arrived = false;\n }\n item.top = topNew;\n item.bottom = item.top + item.height;\n } else {\n item.top = finalItem.top;\n item.bottom = finalItem.bottom;\n }\n item.left = finalItem.left;\n item.right = finalItem.right;\n }\n return arrived;\n};\n\n/**\n * Move the events from their current position to the final position\n * @param {Array} currentItems Array with the real items and their current\n * positions\n * @param {Array} finalItems Array with objects containing the final\n * positions of the items\n */\nlinks.Timeline.prototype.stackMoveToFinal = function (currentItems, finalItems) {\n // Put the events directly at there final position\n for (var i = 0, iMax = finalItems.length; i < iMax; i++) {\n var finalItem = finalItems[i],\n current = finalItem.item;\n current.left = finalItem.left;\n current.top = finalItem.top;\n current.right = finalItem.right;\n current.bottom = finalItem.bottom;\n }\n};\n\n/**\n * Check if the destiny position of given item overlaps with any\n * of the other items from index itemStart to itemEnd.\n * @param {Array} items Array with items\n * @param {int} itemIndex Number of the item to be checked for overlap\n * @param {int} itemStart First item to be checked.\n * @param {int} itemEnd Last item to be checked.\n * @return {Object} colliding item, or undefined when no collisions\n */\nlinks.Timeline.prototype.stackItemsCheckOverlap = function (items, itemIndex, itemStart, itemEnd) {\n var eventMargin = this.options.eventMargin,\n collision = this.collision;\n\n // we loop from end to start, as we suppose that the chance of a\n // collision is larger for items at the end, so check these first.\n var item1 = items[itemIndex];\n for (var i = itemEnd; i >= itemStart; i--) {\n var item2 = items[i];\n if (collision(item1, item2, eventMargin)) {\n if (i != itemIndex) {\n return item2;\n }\n }\n }\n return undefined;\n};\n\n/**\n * Test if the two provided items collide\n * The items must have parameters left, right, top, and bottom.\n * @param {Element} item1 The first item\n * @param {Element} item2 The second item\n * @param {Number} margin A minimum required margin. Optional.\n * If margin is provided, the two items will be\n * marked colliding when they overlap or\n * when the margin between the two is smaller than\n * the requested margin.\n * @return {boolean} true if item1 and item2 collide, else false\n */\nlinks.Timeline.prototype.collision = function (item1, item2, margin) {\n // set margin if not specified\n if (margin == undefined) {\n margin = 0;\n }\n\n // calculate if there is overlap (collision)\n return item1.left - margin < item2.right && item1.right + margin > item2.left && item1.top - margin < item2.bottom && item1.bottom + margin > item2.top;\n};\n\n/**\n * fire an event\n * @param {String} event The name of an event, for example \"rangechange\" or \"edit\"\n */\nlinks.Timeline.prototype.trigger = function (event) {\n // built up properties\n var properties = null;\n switch (event) {\n case 'rangechange':\n case 'rangechanged':\n properties = {\n 'start': new Date(this.start.valueOf()),\n 'end': new Date(this.end.valueOf())\n };\n break;\n case 'timechange':\n case 'timechanged':\n properties = {\n 'time': new Date(this.customTime.valueOf())\n };\n break;\n }\n\n // trigger the links event bus\n links.events.trigger(this, event, properties);\n\n // trigger the google event bus\n if (google && google.visualization) {\n google.visualization.events.trigger(this, event, properties);\n }\n};\n\n/**\n * Cluster the events\n */\nlinks.Timeline.prototype.clusterItems = function () {\n if (!this.options.cluster) {\n return;\n }\n var clusters = this.clusterGenerator.getClusters(this.conversion.factor, this.options.clusterMaxItems);\n if (this.clusters != clusters) {\n // cluster level changed\n var queue = this.renderQueue;\n\n // remove the old clusters from the scene\n if (this.clusters) {\n this.clusters.forEach(function (cluster) {\n queue.hide.push(cluster);\n\n // unlink the items\n cluster.items.forEach(function (item) {\n item.cluster = undefined;\n });\n });\n }\n\n // append the new clusters\n clusters.forEach(function (cluster) {\n // don't add to the queue.show here, will be done in .filterItems()\n\n // link all items to the cluster\n cluster.items.forEach(function (item) {\n item.cluster = cluster;\n });\n });\n this.clusters = clusters;\n }\n};\n\n/**\n * Filter the visible events\n */\nlinks.Timeline.prototype.filterItems = function () {\n var queue = this.renderQueue,\n window = this.end - this.start,\n start = new Date(this.start.valueOf() - window),\n end = new Date(this.end.valueOf() + window);\n function filter(arr) {\n arr.forEach(function (item) {\n var rendered = item.rendered;\n var visible = item.isVisible(start, end);\n if (rendered != visible) {\n if (rendered) {\n queue.hide.push(item); // item is rendered but no longer visible\n }\n\n if (visible && queue.show.indexOf(item) == -1) {\n queue.show.push(item); // item is visible but neither rendered nor queued up to be rendered\n }\n }\n });\n }\n\n // filter all items and all clusters\n filter(this.items);\n if (this.clusters) {\n filter(this.clusters);\n }\n};\n\n/** ------------------------------------------------------------------------ **/\n\n/**\n * @constructor links.Timeline.ClusterGenerator\n * Generator which creates clusters of items, based on the visible range in\n * the Timeline. There is a set of cluster levels which is cached.\n * @param {links.Timeline} timeline\n */\nlinks.Timeline.ClusterGenerator = function (timeline) {\n this.timeline = timeline;\n this.clear();\n};\n\n/**\n * Clear all cached clusters and data, and initialize all variables\n */\nlinks.Timeline.ClusterGenerator.prototype.clear = function () {\n // cache containing created clusters for each cluster level\n this.items = [];\n this.groups = {};\n this.clearCache();\n};\n\n/**\n * Clear the cached clusters\n */\nlinks.Timeline.ClusterGenerator.prototype.clearCache = function () {\n // cache containing created clusters for each cluster level\n this.cache = {};\n this.cacheLevel = -1;\n this.cache[this.cacheLevel] = [];\n};\n\n/**\n * Set the items to be clustered.\n * This will clear cached clusters.\n * @param {Item[]} items\n * @param {Object} [options] Available options:\n * {boolean} applyOnChangedLevel\n * If true (default), the changed data is applied\n * as soon the cluster level changes. If false,\n * The changed data is applied immediately\n */\nlinks.Timeline.ClusterGenerator.prototype.setData = function (items, options) {\n this.items = items || [];\n this.dataChanged = true;\n this.applyOnChangedLevel = true;\n if (options && options.applyOnChangedLevel) {\n this.applyOnChangedLevel = options.applyOnChangedLevel;\n }\n // console.log('clustergenerator setData applyOnChangedLevel=' + this.applyOnChangedLevel); // TODO: cleanup\n};\n\n/**\n * Update the current data set: clear cache, and recalculate the clustering for\n * the current level\n */\nlinks.Timeline.ClusterGenerator.prototype.updateData = function () {\n this.dataChanged = true;\n this.applyOnChangedLevel = false;\n};\n\n/**\n * Filter the items per group.\n * @private\n */\nlinks.Timeline.ClusterGenerator.prototype.filterData = function () {\n // filter per group\n var items = this.items || [];\n var groups = {};\n this.groups = groups;\n\n // split the items per group\n items.forEach(function (item) {\n // put the item in the correct group\n var groupName = item.group ? item.group.content : '';\n var group = groups[groupName];\n if (!group) {\n group = [];\n groups[groupName] = group;\n }\n group.push(item);\n\n // calculate the center of the item\n if (item.start) {\n if (item.end) {\n // range\n item.center = (item.start.valueOf() + item.end.valueOf()) / 2;\n } else {\n // box, dot\n item.center = item.start.valueOf();\n }\n }\n });\n\n // sort the items per group\n for (var groupName in groups) {\n if (groups.hasOwnProperty(groupName)) {\n groups[groupName].sort(function (a, b) {\n return a.center - b.center;\n });\n }\n }\n this.dataChanged = false;\n};\n\n/**\n * Cluster the events which are too close together\n * @param {Number} scale The scale of the current window,\n * defined as (windowWidth / (endDate - startDate))\n * @return {Item[]} clusters\n */\nlinks.Timeline.ClusterGenerator.prototype.getClusters = function (scale, maxItems) {\n var level = -1,\n granularity = 2,\n // TODO: what granularity is needed for the cluster levels?\n timeWindow = 0; // milliseconds\n\n if (scale > 0) {\n level = Math.round(Math.log(100 / scale) / Math.log(granularity));\n timeWindow = Math.pow(granularity, level);\n }\n\n // clear the cache when and re-filter the data when needed.\n if (this.dataChanged) {\n var levelChanged = level != this.cacheLevel;\n var applyDataNow = this.applyOnChangedLevel ? levelChanged : true;\n if (applyDataNow) {\n // TODO: currently drawn clusters should be removed! mark them as invisible?\n this.clearCache();\n this.filterData();\n // console.log('clustergenerator: cache cleared...'); // TODO: cleanup\n }\n }\n\n this.cacheLevel = level;\n var clusters = this.cache[level];\n if (!clusters) {\n // console.log('clustergenerator: create cluster level ' + level); // TODO: cleanup\n clusters = [];\n\n // TODO: spit this method, it is too large\n for (var groupName in this.groups) {\n if (this.groups.hasOwnProperty(groupName)) {\n var items = this.groups[groupName];\n var iMax = items.length;\n var i = 0;\n while (i < iMax) {\n // find all items around current item, within the timeWindow\n var item = items[i];\n var neighbors = 1; // start at 1, to include itself)\n\n // loop through items left from the current item\n var j = i - 1;\n while (j >= 0 && item.center - items[j].center < timeWindow / 2) {\n if (!items[j].cluster) {\n neighbors++;\n }\n j--;\n }\n\n // loop through items right from the current item\n var k = i + 1;\n while (k < items.length && items[k].center - item.center < timeWindow / 2) {\n neighbors++;\n k++;\n }\n\n // loop through the created clusters\n var l = clusters.length - 1;\n while (l >= 0 && item.center - clusters[l].center < timeWindow / 2) {\n if (item.group == clusters[l].group) {\n neighbors++;\n }\n l--;\n }\n\n // aggregate until the number of items is within maxItems\n if (neighbors > maxItems) {\n // too busy in this window.\n var num = neighbors - maxItems + 1;\n var clusterItems = [];\n\n // append the items to the cluster,\n // and calculate the average start for the cluster\n var avg = undefined; // number. average of all start dates\n var min = undefined; // number. minimum of all start dates\n var max = undefined; // number. maximum of all start and end dates\n var containsRanges = false;\n var count = 0;\n var m = i;\n while (clusterItems.length < num && m < items.length) {\n var p = items[m];\n var start = p.start.valueOf();\n var end = p.end ? p.end.valueOf() : p.start.valueOf();\n clusterItems.push(p);\n if (count) {\n // calculate new average (use fractions to prevent overflow)\n avg = count / (count + 1) * avg + 1 / (count + 1) * p.center;\n } else {\n avg = p.center;\n }\n min = min != undefined ? Math.min(min, start) : start;\n max = max != undefined ? Math.max(max, end) : end;\n containsRanges = containsRanges || p instanceof links.Timeline.ItemRange || p instanceof links.Timeline.ItemFloatingRange;\n count++;\n m++;\n }\n var cluster;\n var title = 'Cluster containing ' + count + ' events. Zoom in to see the individual events.';\n var content = '
' + count + ' events
';\n var group = item.group ? item.group.content : undefined;\n if (containsRanges) {\n // boxes and/or ranges\n cluster = this.timeline.createItem({\n 'start': new Date(min),\n 'end': new Date(max),\n 'content': content,\n 'group': group\n });\n } else {\n // boxes only\n cluster = this.timeline.createItem({\n 'start': new Date(avg),\n 'content': content,\n 'group': group\n });\n }\n cluster.isCluster = true;\n cluster.items = clusterItems;\n cluster.items.forEach(function (item) {\n item.cluster = cluster;\n });\n clusters.push(cluster);\n i += num;\n } else {\n delete item.cluster;\n i += 1;\n }\n }\n }\n }\n this.cache[level] = clusters;\n }\n return clusters;\n};\n\n/** ------------------------------------------------------------------------ **/\n\n/**\n * Event listener (singleton)\n */\nlinks.events = links.events || {\n 'listeners': [],\n /**\n * Find a single listener by its object\n * @param {Object} object\n * @return {Number} index -1 when not found\n */\n 'indexOf': function indexOf(object) {\n var listeners = this.listeners;\n for (var i = 0, iMax = this.listeners.length; i < iMax; i++) {\n var listener = listeners[i];\n if (listener && listener.object == object) {\n return i;\n }\n }\n return -1;\n },\n /**\n * Add an event listener\n * @param {Object} object\n * @param {String} event The name of an event, for example 'select'\n * @param {function} callback The callback method, called when the\n * event takes place\n */\n 'addListener': function addListener(object, event, callback) {\n var index = this.indexOf(object);\n var listener = this.listeners[index];\n if (!listener) {\n listener = {\n 'object': object,\n 'events': {}\n };\n this.listeners.push(listener);\n }\n var callbacks = listener.events[event];\n if (!callbacks) {\n callbacks = [];\n listener.events[event] = callbacks;\n }\n\n // add the callback if it does not yet exist\n if (callbacks.indexOf(callback) == -1) {\n callbacks.push(callback);\n }\n },\n /**\n * Remove an event listener\n * @param {Object} object\n * @param {String} event The name of an event, for example 'select'\n * @param {function} callback The registered callback method\n */\n 'removeListener': function removeListener(object, event, callback) {\n var index = this.indexOf(object);\n var listener = this.listeners[index];\n if (listener) {\n var callbacks = listener.events[event];\n if (callbacks) {\n var index = callbacks.indexOf(callback);\n if (index != -1) {\n callbacks.splice(index, 1);\n }\n\n // remove the array when empty\n if (callbacks.length == 0) {\n delete listener.events[event];\n }\n }\n\n // count the number of registered events. remove listener when empty\n var count = 0;\n var events = listener.events;\n for (var e in events) {\n if (events.hasOwnProperty(e)) {\n count++;\n }\n }\n if (count == 0) {\n delete this.listeners[index];\n }\n }\n },\n /**\n * Remove all registered event listeners\n */\n 'removeAllListeners': function removeAllListeners() {\n this.listeners = [];\n },\n /**\n * Trigger an event. All registered event handlers will be called\n * @param {Object} object\n * @param {String} event\n * @param {Object} properties (optional)\n */\n 'trigger': function trigger(object, event, properties) {\n var index = this.indexOf(object);\n var listener = this.listeners[index];\n if (listener) {\n var callbacks = listener.events[event];\n if (callbacks) {\n for (var i = 0, iMax = callbacks.length; i < iMax; i++) {\n callbacks[i](properties);\n }\n }\n }\n }\n};\n\n/** ------------------------------------------------------------------------ **/\n\n/**\n * @constructor links.Timeline.StepDate\n * The class StepDate is an iterator for dates. You provide a start date and an\n * end date. The class itself determines the best scale (step size) based on the\n * provided start Date, end Date, and minimumStep.\n *\n * If minimumStep is provided, the step size is chosen as close as possible\n * to the minimumStep but larger than minimumStep. If minimumStep is not\n * provided, the scale is set to 1 DAY.\n * The minimumStep should correspond with the onscreen size of about 6 characters\n *\n * Alternatively, you can set a scale by hand.\n * After creation, you can initialize the class by executing start(). Then you\n * can iterate from the start date to the end date via next(). You can check if\n * the end date is reached with the function end(). After each step, you can\n * retrieve the current date via get().\n * The class step has scales ranging from milliseconds, seconds, minutes, hours,\n * days, to years.\n *\n * Version: 1.2\n *\n * @param {Date} start The start date, for example new Date(2010, 9, 21)\n * or new Date(2010, 9, 21, 23, 45, 00)\n * @param {Date} end The end date\n * @param {Number} minimumStep Optional. Minimum step size in milliseconds\n */\nlinks.Timeline.StepDate = function (start, end, minimumStep) {\n // variables\n this.current = new Date();\n this._start = new Date();\n this._end = new Date();\n this.autoScale = true;\n this.scale = links.Timeline.StepDate.SCALE.DAY;\n this.step = 1;\n\n // initialize the range\n this.setRange(start, end, minimumStep);\n};\n\n/// enum scale\nlinks.Timeline.StepDate.SCALE = {\n MILLISECOND: 1,\n SECOND: 2,\n MINUTE: 3,\n HOUR: 4,\n DAY: 5,\n WEEKDAY: 6,\n MONTH: 7,\n YEAR: 8\n};\n\n/**\n * Set a new range\n * If minimumStep is provided, the step size is chosen as close as possible\n * to the minimumStep but larger than minimumStep. If minimumStep is not\n * provided, the scale is set to 1 DAY.\n * The minimumStep should correspond with the onscreen size of about 6 characters\n * @param {Date} start The start date and time.\n * @param {Date} end The end date and time.\n * @param {int} minimumStep Optional. Minimum step size in milliseconds\n */\nlinks.Timeline.StepDate.prototype.setRange = function (start, end, minimumStep) {\n if (!(start instanceof Date) || !(end instanceof Date)) {\n //throw \"No legal start or end date in method setRange\";\n return;\n }\n this._start = start != undefined ? new Date(start.valueOf()) : new Date();\n this._end = end != undefined ? new Date(end.valueOf()) : new Date();\n if (this.autoScale) {\n this.setMinimumStep(minimumStep);\n }\n};\n\n/**\n * Set the step iterator to the start date.\n */\nlinks.Timeline.StepDate.prototype.start = function () {\n this.current = new Date(this._start.valueOf());\n this.roundToMinor();\n};\n\n/**\n * Round the current date to the first minor date value\n * This must be executed once when the current date is set to start Date\n */\nlinks.Timeline.StepDate.prototype.roundToMinor = function () {\n // round to floor\n // IMPORTANT: we have no breaks in this switch! (this is no bug)\n //noinspection FallthroughInSwitchStatementJS\n switch (this.scale) {\n case links.Timeline.StepDate.SCALE.YEAR:\n this.current.setFullYear(this.step * Math.floor(this.current.getFullYear() / this.step));\n this.current.setMonth(0);\n case links.Timeline.StepDate.SCALE.MONTH:\n this.current.setDate(1);\n case links.Timeline.StepDate.SCALE.DAY: // intentional fall through\n case links.Timeline.StepDate.SCALE.WEEKDAY:\n this.current.setHours(0);\n case links.Timeline.StepDate.SCALE.HOUR:\n this.current.setMinutes(0);\n case links.Timeline.StepDate.SCALE.MINUTE:\n this.current.setSeconds(0);\n case links.Timeline.StepDate.SCALE.SECOND:\n this.current.setMilliseconds(0);\n //case links.Timeline.StepDate.SCALE.MILLISECOND: // nothing to do for milliseconds\n }\n\n if (this.step != 1) {\n // round down to the first minor value that is a multiple of the current step size\n switch (this.scale) {\n case links.Timeline.StepDate.SCALE.MILLISECOND:\n this.current.setMilliseconds(this.current.getMilliseconds() - this.current.getMilliseconds() % this.step);\n break;\n case links.Timeline.StepDate.SCALE.SECOND:\n this.current.setSeconds(this.current.getSeconds() - this.current.getSeconds() % this.step);\n break;\n case links.Timeline.StepDate.SCALE.MINUTE:\n this.current.setMinutes(this.current.getMinutes() - this.current.getMinutes() % this.step);\n break;\n case links.Timeline.StepDate.SCALE.HOUR:\n this.current.setHours(this.current.getHours() - this.current.getHours() % this.step);\n break;\n case links.Timeline.StepDate.SCALE.WEEKDAY: // intentional fall through\n case links.Timeline.StepDate.SCALE.DAY:\n this.current.setDate(this.current.getDate() - 1 - (this.current.getDate() - 1) % this.step + 1);\n break;\n case links.Timeline.StepDate.SCALE.MONTH:\n this.current.setMonth(this.current.getMonth() - this.current.getMonth() % this.step);\n break;\n case links.Timeline.StepDate.SCALE.YEAR:\n this.current.setFullYear(this.current.getFullYear() - this.current.getFullYear() % this.step);\n break;\n default:\n break;\n }\n }\n};\n\n/**\n * Check if the end date is reached\n * @return {boolean} true if the current date has passed the end date\n */\nlinks.Timeline.StepDate.prototype.end = function () {\n return this.current.valueOf() > this._end.valueOf();\n};\n\n/**\n * Do the next step\n */\nlinks.Timeline.StepDate.prototype.next = function () {\n var prev = this.current.valueOf();\n\n // Two cases, needed to prevent issues with switching daylight savings\n // (end of March and end of October)\n if (this.current.getMonth() < 6) {\n switch (this.scale) {\n case links.Timeline.StepDate.SCALE.MILLISECOND:\n this.current = new Date(this.current.valueOf() + this.step);\n break;\n case links.Timeline.StepDate.SCALE.SECOND:\n this.current = new Date(this.current.valueOf() + this.step * 1000);\n break;\n case links.Timeline.StepDate.SCALE.MINUTE:\n this.current = new Date(this.current.valueOf() + this.step * 1000 * 60);\n break;\n case links.Timeline.StepDate.SCALE.HOUR:\n this.current = new Date(this.current.valueOf() + this.step * 1000 * 60 * 60);\n // in case of skipping an hour for daylight savings, adjust the hour again (else you get: 0h 5h 9h ... instead of 0h 4h 8h ...)\n var h = this.current.getHours();\n this.current.setHours(h - h % this.step);\n break;\n case links.Timeline.StepDate.SCALE.WEEKDAY: // intentional fall through\n case links.Timeline.StepDate.SCALE.DAY:\n this.current.setDate(this.current.getDate() + this.step);\n break;\n case links.Timeline.StepDate.SCALE.MONTH:\n this.current.setMonth(this.current.getMonth() + this.step);\n break;\n case links.Timeline.StepDate.SCALE.YEAR:\n this.current.setFullYear(this.current.getFullYear() + this.step);\n break;\n default:\n break;\n }\n } else {\n switch (this.scale) {\n case links.Timeline.StepDate.SCALE.MILLISECOND:\n this.current = new Date(this.current.valueOf() + this.step);\n break;\n case links.Timeline.StepDate.SCALE.SECOND:\n this.current.setSeconds(this.current.getSeconds() + this.step);\n break;\n case links.Timeline.StepDate.SCALE.MINUTE:\n this.current.setMinutes(this.current.getMinutes() + this.step);\n break;\n case links.Timeline.StepDate.SCALE.HOUR:\n this.current.setHours(this.current.getHours() + this.step);\n break;\n case links.Timeline.StepDate.SCALE.WEEKDAY: // intentional fall through\n case links.Timeline.StepDate.SCALE.DAY:\n this.current.setDate(this.current.getDate() + this.step);\n break;\n case links.Timeline.StepDate.SCALE.MONTH:\n this.current.setMonth(this.current.getMonth() + this.step);\n break;\n case links.Timeline.StepDate.SCALE.YEAR:\n this.current.setFullYear(this.current.getFullYear() + this.step);\n break;\n default:\n break;\n }\n }\n if (this.step != 1) {\n // round down to the correct major value\n switch (this.scale) {\n case links.Timeline.StepDate.SCALE.MILLISECOND:\n if (this.current.getMilliseconds() < this.step) this.current.setMilliseconds(0);\n break;\n case links.Timeline.StepDate.SCALE.SECOND:\n if (this.current.getSeconds() < this.step) this.current.setSeconds(0);\n break;\n case links.Timeline.StepDate.SCALE.MINUTE:\n if (this.current.getMinutes() < this.step) this.current.setMinutes(0);\n break;\n case links.Timeline.StepDate.SCALE.HOUR:\n if (this.current.getHours() < this.step) this.current.setHours(0);\n break;\n case links.Timeline.StepDate.SCALE.WEEKDAY: // intentional fall through\n case links.Timeline.StepDate.SCALE.DAY:\n if (this.current.getDate() < this.step + 1) this.current.setDate(1);\n break;\n case links.Timeline.StepDate.SCALE.MONTH:\n if (this.current.getMonth() < this.step) this.current.setMonth(0);\n break;\n case links.Timeline.StepDate.SCALE.YEAR:\n break;\n // nothing to do for year\n default:\n break;\n }\n }\n\n // safety mechanism: if current time is still unchanged, move to the end\n if (this.current.valueOf() == prev) {\n this.current = new Date(this._end.valueOf());\n }\n};\n\n/**\n * Get the current datetime\n * @return {Date} current The current date\n */\nlinks.Timeline.StepDate.prototype.getCurrent = function () {\n return this.current;\n};\n\n/**\n * Set a custom scale. Autoscaling will be disabled.\n * For example setScale(SCALE.MINUTES, 5) will result\n * in minor steps of 5 minutes, and major steps of an hour.\n *\n * @param {links.Timeline.StepDate.SCALE} newScale\n * A scale. Choose from SCALE.MILLISECOND,\n * SCALE.SECOND, SCALE.MINUTE, SCALE.HOUR,\n * SCALE.WEEKDAY, SCALE.DAY, SCALE.MONTH,\n * SCALE.YEAR.\n * @param {Number} newStep A step size, by default 1. Choose for\n * example 1, 2, 5, or 10.\n */\nlinks.Timeline.StepDate.prototype.setScale = function (newScale, newStep) {\n this.scale = newScale;\n if (newStep > 0) {\n this.step = newStep;\n }\n this.autoScale = false;\n};\n\n/**\n * Enable or disable autoscaling\n * @param {boolean} enable If true, autoascaling is set true\n */\nlinks.Timeline.StepDate.prototype.setAutoScale = function (enable) {\n this.autoScale = enable;\n};\n\n/**\n * Automatically determine the scale that bests fits the provided minimum step\n * @param {Number} minimumStep The minimum step size in milliseconds\n */\nlinks.Timeline.StepDate.prototype.setMinimumStep = function (minimumStep) {\n if (minimumStep == undefined) {\n return;\n }\n var stepYear = 1000 * 60 * 60 * 24 * 30 * 12;\n var stepMonth = 1000 * 60 * 60 * 24 * 30;\n var stepDay = 1000 * 60 * 60 * 24;\n var stepHour = 1000 * 60 * 60;\n var stepMinute = 1000 * 60;\n var stepSecond = 1000;\n var stepMillisecond = 1;\n\n // find the smallest step that is larger than the provided minimumStep\n if (stepYear * 1000 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.YEAR;\n this.step = 1000;\n }\n if (stepYear * 500 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.YEAR;\n this.step = 500;\n }\n if (stepYear * 100 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.YEAR;\n this.step = 100;\n }\n if (stepYear * 50 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.YEAR;\n this.step = 50;\n }\n if (stepYear * 10 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.YEAR;\n this.step = 10;\n }\n if (stepYear * 5 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.YEAR;\n this.step = 5;\n }\n if (stepYear > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.YEAR;\n this.step = 1;\n }\n if (stepMonth * 3 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MONTH;\n this.step = 3;\n }\n if (stepMonth > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MONTH;\n this.step = 1;\n }\n if (stepDay * 5 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.DAY;\n this.step = 5;\n }\n if (stepDay * 2 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.DAY;\n this.step = 2;\n }\n if (stepDay > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.DAY;\n this.step = 1;\n }\n if (stepDay / 2 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.WEEKDAY;\n this.step = 1;\n }\n if (stepHour * 4 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.HOUR;\n this.step = 4;\n }\n if (stepHour > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.HOUR;\n this.step = 1;\n }\n if (stepMinute * 15 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MINUTE;\n this.step = 15;\n }\n if (stepMinute * 10 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MINUTE;\n this.step = 10;\n }\n if (stepMinute * 5 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MINUTE;\n this.step = 5;\n }\n if (stepMinute > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MINUTE;\n this.step = 1;\n }\n if (stepSecond * 15 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.SECOND;\n this.step = 15;\n }\n if (stepSecond * 10 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.SECOND;\n this.step = 10;\n }\n if (stepSecond * 5 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.SECOND;\n this.step = 5;\n }\n if (stepSecond > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.SECOND;\n this.step = 1;\n }\n if (stepMillisecond * 200 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MILLISECOND;\n this.step = 200;\n }\n if (stepMillisecond * 100 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MILLISECOND;\n this.step = 100;\n }\n if (stepMillisecond * 50 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MILLISECOND;\n this.step = 50;\n }\n if (stepMillisecond * 10 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MILLISECOND;\n this.step = 10;\n }\n if (stepMillisecond * 5 > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MILLISECOND;\n this.step = 5;\n }\n if (stepMillisecond > minimumStep) {\n this.scale = links.Timeline.StepDate.SCALE.MILLISECOND;\n this.step = 1;\n }\n};\n\n/**\n * Snap a date to a rounded value. The snap intervals are dependent on the\n * current scale and step.\n * @param {Date} date the date to be snapped\n */\nlinks.Timeline.StepDate.prototype.snap = function (date) {\n if (this.scale == links.Timeline.StepDate.SCALE.YEAR) {\n var year = date.getFullYear() + Math.round(date.getMonth() / 12);\n date.setFullYear(Math.round(year / this.step) * this.step);\n date.setMonth(0);\n date.setDate(0);\n date.setHours(0);\n date.setMinutes(0);\n date.setSeconds(0);\n date.setMilliseconds(0);\n } else if (this.scale == links.Timeline.StepDate.SCALE.MONTH) {\n if (date.getDate() > 15) {\n date.setDate(1);\n date.setMonth(date.getMonth() + 1);\n // important: first set Date to 1, after that change the month.\n } else {\n date.setDate(1);\n }\n date.setHours(0);\n date.setMinutes(0);\n date.setSeconds(0);\n date.setMilliseconds(0);\n } else if (this.scale == links.Timeline.StepDate.SCALE.DAY || this.scale == links.Timeline.StepDate.SCALE.WEEKDAY) {\n switch (this.step) {\n case 5:\n case 2:\n date.setHours(Math.round(date.getHours() / 24) * 24);\n break;\n default:\n date.setHours(Math.round(date.getHours() / 12) * 12);\n break;\n }\n date.setMinutes(0);\n date.setSeconds(0);\n date.setMilliseconds(0);\n } else if (this.scale == links.Timeline.StepDate.SCALE.HOUR) {\n switch (this.step) {\n case 4:\n date.setMinutes(Math.round(date.getMinutes() / 60) * 60);\n break;\n default:\n date.setMinutes(Math.round(date.getMinutes() / 30) * 30);\n break;\n }\n date.setSeconds(0);\n date.setMilliseconds(0);\n } else if (this.scale == links.Timeline.StepDate.SCALE.MINUTE) {\n switch (this.step) {\n case 15:\n case 10:\n date.setMinutes(Math.round(date.getMinutes() / 5) * 5);\n date.setSeconds(0);\n break;\n case 5:\n date.setSeconds(Math.round(date.getSeconds() / 60) * 60);\n break;\n default:\n date.setSeconds(Math.round(date.getSeconds() / 30) * 30);\n break;\n }\n date.setMilliseconds(0);\n } else if (this.scale == links.Timeline.StepDate.SCALE.SECOND) {\n switch (this.step) {\n case 15:\n case 10:\n date.setSeconds(Math.round(date.getSeconds() / 5) * 5);\n date.setMilliseconds(0);\n break;\n case 5:\n date.setMilliseconds(Math.round(date.getMilliseconds() / 1000) * 1000);\n break;\n default:\n date.setMilliseconds(Math.round(date.getMilliseconds() / 500) * 500);\n break;\n }\n } else if (this.scale == links.Timeline.StepDate.SCALE.MILLISECOND) {\n var step = this.step > 5 ? this.step / 2 : 1;\n date.setMilliseconds(Math.round(date.getMilliseconds() / step) * step);\n }\n};\n\n/**\n * Check if the current step is a major step (for example when the step\n * is DAY, a major step is each first day of the MONTH)\n * @return {boolean} true if current date is major, else false.\n */\nlinks.Timeline.StepDate.prototype.isMajor = function () {\n switch (this.scale) {\n case links.Timeline.StepDate.SCALE.MILLISECOND:\n return this.current.getMilliseconds() == 0;\n case links.Timeline.StepDate.SCALE.SECOND:\n return this.current.getSeconds() == 0;\n case links.Timeline.StepDate.SCALE.MINUTE:\n return this.current.getHours() == 0 && this.current.getMinutes() == 0;\n // Note: this is no bug. Major label is equal for both minute and hour scale\n case links.Timeline.StepDate.SCALE.HOUR:\n return this.current.getHours() == 0;\n case links.Timeline.StepDate.SCALE.WEEKDAY: // intentional fall through\n case links.Timeline.StepDate.SCALE.DAY:\n return this.current.getDate() == 1;\n case links.Timeline.StepDate.SCALE.MONTH:\n return this.current.getMonth() == 0;\n case links.Timeline.StepDate.SCALE.YEAR:\n return false;\n default:\n return false;\n }\n};\n\n/**\n * Returns formatted text for the minor axislabel, depending on the current\n * date and the scale. For example when scale is MINUTE, the current time is\n * formatted as \"hh:mm\".\n * @param {Object} options\n * @param {Date} [date] custom date. if not provided, current date is taken\n */\nlinks.Timeline.StepDate.prototype.getLabelMinor = function (options, date) {\n if (date == undefined) {\n date = this.current;\n }\n switch (this.scale) {\n case links.Timeline.StepDate.SCALE.MILLISECOND:\n return String(date.getMilliseconds());\n case links.Timeline.StepDate.SCALE.SECOND:\n return String(date.getSeconds());\n case links.Timeline.StepDate.SCALE.MINUTE:\n return this.addZeros(date.getHours(), 2) + \":\" + this.addZeros(date.getMinutes(), 2);\n case links.Timeline.StepDate.SCALE.HOUR:\n return this.addZeros(date.getHours(), 2) + \":\" + this.addZeros(date.getMinutes(), 2);\n case links.Timeline.StepDate.SCALE.WEEKDAY:\n return options.DAYS_SHORT[date.getDay()] + ' ' + date.getDate();\n case links.Timeline.StepDate.SCALE.DAY:\n return String(date.getDate());\n case links.Timeline.StepDate.SCALE.MONTH:\n return options.MONTHS_SHORT[date.getMonth()];\n // month is zero based\n case links.Timeline.StepDate.SCALE.YEAR:\n return String(date.getFullYear());\n default:\n return \"\";\n }\n};\n\n/**\n * Returns formatted text for the major axislabel, depending on the current\n * date and the scale. For example when scale is MINUTE, the major scale is\n * hours, and the hour will be formatted as \"hh\".\n * @param {Object} options\n * @param {Date} [date] custom date. if not provided, current date is taken\n */\nlinks.Timeline.StepDate.prototype.getLabelMajor = function (options, date) {\n if (date == undefined) {\n date = this.current;\n }\n switch (this.scale) {\n case links.Timeline.StepDate.SCALE.MILLISECOND:\n return this.addZeros(date.getHours(), 2) + \":\" + this.addZeros(date.getMinutes(), 2) + \":\" + this.addZeros(date.getSeconds(), 2);\n case links.Timeline.StepDate.SCALE.SECOND:\n return date.getDate() + \" \" + options.MONTHS[date.getMonth()] + \" \" + this.addZeros(date.getHours(), 2) + \":\" + this.addZeros(date.getMinutes(), 2);\n case links.Timeline.StepDate.SCALE.MINUTE:\n return options.DAYS[date.getDay()] + \" \" + date.getDate() + \" \" + options.MONTHS[date.getMonth()] + \" \" + date.getFullYear();\n case links.Timeline.StepDate.SCALE.HOUR:\n return options.DAYS[date.getDay()] + \" \" + date.getDate() + \" \" + options.MONTHS[date.getMonth()] + \" \" + date.getFullYear();\n case links.Timeline.StepDate.SCALE.WEEKDAY:\n case links.Timeline.StepDate.SCALE.DAY:\n return options.MONTHS[date.getMonth()] + \" \" + date.getFullYear();\n case links.Timeline.StepDate.SCALE.MONTH:\n return String(date.getFullYear());\n default:\n return \"\";\n }\n};\n\n/**\n * Add leading zeros to the given value to match the desired length.\n * For example addZeros(123, 5) returns \"00123\"\n * @param {int} value A value\n * @param {int} len Desired final length\n * @return {string} value with leading zeros\n */\nlinks.Timeline.StepDate.prototype.addZeros = function (value, len) {\n var str = \"\" + value;\n while (str.length < len) {\n str = \"0\" + str;\n }\n return str;\n};\n\n/** ------------------------------------------------------------------------ **/\n\n/**\n * Image Loader service.\n * can be used to get a callback when a certain image is loaded\n *\n */\nlinks.imageloader = function () {\n var urls = {}; // the loaded urls\n var callbacks = {}; // the urls currently being loaded. Each key contains\n // an array with callbacks\n\n /**\n * Check if an image url is loaded\n * @param {String} url\n * @return {boolean} loaded True when loaded, false when not loaded\n * or when being loaded\n */\n function isLoaded(url) {\n if (urls[url] == true) {\n return true;\n }\n var image = new Image();\n image.src = url;\n if (image.complete) {\n return true;\n }\n return false;\n }\n\n /**\n * Check if an image url is being loaded\n * @param {String} url\n * @return {boolean} loading True when being loaded, false when not loading\n * or when already loaded\n */\n function isLoading(url) {\n return callbacks[url] != undefined;\n }\n\n /**\n * Load given image url\n * @param {String} url\n * @param {function} callback\n * @param {boolean} sendCallbackWhenAlreadyLoaded optional\n */\n function load(url, callback, sendCallbackWhenAlreadyLoaded) {\n if (sendCallbackWhenAlreadyLoaded == undefined) {\n sendCallbackWhenAlreadyLoaded = true;\n }\n if (isLoaded(url)) {\n if (sendCallbackWhenAlreadyLoaded) {\n callback(url);\n }\n return;\n }\n if (isLoading(url) && !sendCallbackWhenAlreadyLoaded) {\n return;\n }\n var c = callbacks[url];\n if (!c) {\n var image = new Image();\n image.src = url;\n c = [];\n callbacks[url] = c;\n image.onload = function (event) {\n urls[url] = true;\n delete callbacks[url];\n for (var i = 0; i < c.length; i++) {\n c[i](url);\n }\n };\n }\n if (c.indexOf(callback) == -1) {\n c.push(callback);\n }\n }\n\n /**\n * Load a set of images, and send a callback as soon as all images are\n * loaded\n * @param {String[]} urls\n * @param {function } callback\n * @param {boolean} sendCallbackWhenAlreadyLoaded\n */\n function loadAll(urls, callback, sendCallbackWhenAlreadyLoaded) {\n // list all urls which are not yet loaded\n var urlsLeft = [];\n urls.forEach(function (url) {\n if (!isLoaded(url)) {\n urlsLeft.push(url);\n }\n });\n if (urlsLeft.length) {\n // there are unloaded images\n var countLeft = urlsLeft.length;\n urlsLeft.forEach(function (url) {\n load(url, function () {\n countLeft--;\n if (countLeft == 0) {\n // done!\n callback();\n }\n }, sendCallbackWhenAlreadyLoaded);\n });\n } else {\n // we are already done!\n if (sendCallbackWhenAlreadyLoaded) {\n callback();\n }\n }\n }\n\n /**\n * Recursively retrieve all image urls from the images located inside a given\n * HTML element\n * @param {Node} elem\n * @param {String[]} urls Urls will be added here (no duplicates)\n */\n function filterImageUrls(elem, urls) {\n var child = elem.firstChild;\n while (child) {\n if (child.tagName == 'IMG') {\n var url = child.src;\n if (urls.indexOf(url) == -1) {\n urls.push(url);\n }\n }\n filterImageUrls(child, urls);\n child = child.nextSibling;\n }\n }\n return {\n 'isLoaded': isLoaded,\n 'isLoading': isLoading,\n 'load': load,\n 'loadAll': loadAll,\n 'filterImageUrls': filterImageUrls\n };\n}();\n\n/** ------------------------------------------------------------------------ **/\n\n/**\n * Add and event listener. Works for all browsers\n * @param {Element} element An html element\n * @param {string} action The action, for example \"click\",\n * without the prefix \"on\"\n * @param {function} listener The callback function to be executed\n * @param {boolean} useCapture\n */\nlinks.Timeline.addEventListener = function (element, action, listener, useCapture) {\n if (element.addEventListener) {\n if (useCapture === undefined) useCapture = false;\n if (action === \"mousewheel\" && navigator.userAgent.indexOf(\"Firefox\") >= 0) {\n action = \"DOMMouseScroll\"; // For Firefox\n }\n\n element.addEventListener(action, listener, useCapture);\n } else {\n element.attachEvent(\"on\" + action, listener); // IE browsers\n }\n};\n\n/**\n * Remove an event listener from an element\n * @param {Element} element An html dom element\n * @param {string} action The name of the event, for example \"mousedown\"\n * @param {function} listener The listener function\n * @param {boolean} useCapture\n */\nlinks.Timeline.removeEventListener = function (element, action, listener, useCapture) {\n if (element.removeEventListener) {\n // non-IE browsers\n if (useCapture === undefined) useCapture = false;\n if (action === \"mousewheel\" && navigator.userAgent.indexOf(\"Firefox\") >= 0) {\n action = \"DOMMouseScroll\"; // For Firefox\n }\n\n element.removeEventListener(action, listener, useCapture);\n } else {\n // IE browsers\n element.detachEvent(\"on\" + action, listener);\n }\n};\n\n/**\n * Get HTML element which is the target of the event\n * @param {Event} event\n * @return {Element} target element\n */\nlinks.Timeline.getTarget = function (event) {\n // code from http://www.quirksmode.org/js/events_properties.html\n if (!event) {\n event = window.event;\n }\n var target;\n if (event.target) {\n target = event.target;\n } else if (event.srcElement) {\n target = event.srcElement;\n }\n if (target.nodeType != undefined && target.nodeType == 3) {\n // defeat Safari bug\n target = target.parentNode;\n }\n return target;\n};\n\n/**\n * Stop event propagation\n */\nlinks.Timeline.stopPropagation = function (event) {\n if (!event) event = window.event;\n if (event.stopPropagation) {\n event.stopPropagation(); // non-IE browsers\n } else {\n event.cancelBubble = true; // IE browsers\n }\n};\n\n/**\n * Cancels the event if it is cancelable, without stopping further propagation of the event.\n */\nlinks.Timeline.preventDefault = function (event) {\n if (!event) event = window.event;\n if (event.preventDefault) {\n event.preventDefault(); // non-IE browsers\n } else {\n event.returnValue = false; // IE browsers\n }\n};\n\n/**\n * Retrieve the absolute left value of a DOM element\n * @param {Element} elem A dom element, for example a div\n * @return {number} left The absolute left position of this element\n * in the browser page.\n */\nlinks.Timeline.getAbsoluteLeft = function (elem) {\n var doc = document.documentElement;\n var body = document.body;\n var left = elem.offsetLeft;\n var e = elem.offsetParent;\n while (e != null && e != body && e != doc) {\n left += e.offsetLeft;\n left -= e.scrollLeft;\n e = e.offsetParent;\n }\n return left;\n};\n\n/**\n * Retrieve the absolute top value of a DOM element\n * @param {Element} elem A dom element, for example a div\n * @return {number} top The absolute top position of this element\n * in the browser page.\n */\nlinks.Timeline.getAbsoluteTop = function (elem) {\n var doc = document.documentElement;\n var body = document.body;\n var top = elem.offsetTop;\n var e = elem.offsetParent;\n while (e != null && e != body && e != doc) {\n top += e.offsetTop;\n top -= e.scrollTop;\n e = e.offsetParent;\n }\n return top;\n};\n\n/**\n * Get the absolute, vertical mouse position from an event.\n * @param {Event} event\n * @return {Number} pageY\n */\nlinks.Timeline.getPageY = function (event) {\n if ('targetTouches' in event && event.targetTouches.length) {\n event = event.targetTouches[0];\n }\n if ('pageY' in event) {\n return event.pageY;\n }\n\n // calculate pageY from clientY\n var clientY = event.clientY;\n var doc = document.documentElement;\n var body = document.body;\n return clientY + (doc && doc.scrollTop || body && body.scrollTop || 0) - (doc && doc.clientTop || body && body.clientTop || 0);\n};\n\n/**\n * Get the absolute, horizontal mouse position from an event.\n * @param {Event} event\n * @return {Number} pageX\n */\nlinks.Timeline.getPageX = function (event) {\n if ('targetTouches' in event && event.targetTouches.length) {\n event = event.targetTouches[0];\n }\n if ('pageX' in event) {\n return event.pageX;\n }\n\n // calculate pageX from clientX\n var clientX = event.clientX;\n var doc = document.documentElement;\n var body = document.body;\n return clientX + (doc && doc.scrollLeft || body && body.scrollLeft || 0) - (doc && doc.clientLeft || body && body.clientLeft || 0);\n};\n\n/**\n * Adds one or more className's to the given elements style\n * @param {Element} elem\n * @param {String} className\n */\nlinks.Timeline.addClassName = function (elem, className) {\n var classes = elem.className.split(' ');\n var classesToAdd = className.split(' ');\n var added = false;\n for (var i = 0; i < classesToAdd.length; i++) {\n if (classes.indexOf(classesToAdd[i]) == -1) {\n classes.push(classesToAdd[i]); // add the class to the array\n added = true;\n }\n }\n if (added) {\n elem.className = classes.join(' ');\n }\n};\n\n/**\n * Removes one or more className's from the given elements style\n * @param {Element} elem\n * @param {String} className\n */\nlinks.Timeline.removeClassName = function (elem, className) {\n var classes = elem.className.split(' ');\n var classesToRemove = className.split(' ');\n var removed = false;\n for (var i = 0; i < classesToRemove.length; i++) {\n var index = classes.indexOf(classesToRemove[i]);\n if (index != -1) {\n classes.splice(index, 1); // remove the class from the array\n removed = true;\n }\n }\n if (removed) {\n elem.className = classes.join(' ');\n }\n};\n\n/**\n * Check if given object is a Javascript Array\n * @param {*} obj\n * @return {Boolean} isArray true if the given object is an array\n */\n// See http://stackoverflow.com/questions/2943805/javascript-instanceof-typeof-in-gwt-jsni\nlinks.Timeline.isArray = function (obj) {\n if (obj instanceof Array) {\n return true;\n }\n return Object.prototype.toString.call(obj) === '[object Array]';\n};\n\n/**\n * Shallow clone an object\n * @param {Object} object\n * @return {Object} clone\n */\nlinks.Timeline.clone = function (object) {\n var clone = {};\n for (var prop in object) {\n if (object.hasOwnProperty(prop)) {\n clone[prop] = object[prop];\n }\n }\n return clone;\n};\n\n/**\n * parse a JSON date\n * @param {Date | String | Number} date Date object to be parsed. Can be:\n * - a Date object like new Date(),\n * - a long like 1356970529389,\n * an ISO String like \"2012-12-31T16:16:07.213Z\",\n * or a .Net Date string like\n * \"\\/Date(1356970529389)\\/\"\n * @return {Date} parsedDate\n */\nlinks.Timeline.parseJSONDate = function (date) {\n if (date == undefined) {\n return undefined;\n }\n\n //test for date\n if (date instanceof Date) {\n return date;\n }\n\n // test for MS format.\n // FIXME: will fail on a Number\n var m = date.match(/\\/Date\\((-?\\d+)([-\\+]?\\d{2})?(\\d{2})?\\)\\//i);\n if (m) {\n var offset = m[2] ? 3600000 * m[2] // hrs offset\n + 60000 * m[3] * (m[2] / Math.abs(m[2])) // mins offset\n : 0;\n return new Date(1 * m[1] // ticks\n + offset);\n }\n\n // failing that, try to parse whatever we've got.\n return Date.parse(date);\n};\n\n},{}],9:[function(require,module,exports){\n\"use strict\";\n\n/* eslint-disable no-undef */\n/**\r\n * Webpack will replace the __[]__ variables inline\r\n * So, for gulp we need to check if the variable is defined.\r\n * We need to check the type as the variable was never defined as undefined either.\r\n */\nvar ENV = {\n DEBUG: typeof __DEBUG__ !== \"undefined\" ? __DEBUG__ : false,\n DEBUG__DISABLE_MEMOIZE: typeof __DEBUG__DISABLE_MEMOIZE__ !== \"undefined\" ? __DEBUG__DISABLE_MEMOIZE__ : false\n};\nwindow.Environment = ENV;\nmodule.exports = ENV;\n\n},{}],10:[function(require,module,exports){\n\"use strict\";\n\n//require(\"./../css/nixps-container-Collapse.css\");\n\nvar Component = require(\"../../../General/Component/js/nixps-Component.js\");\nrequire(\"./nixps-container-Panel.js\");\nrequire(\"../../Form/js/nixps-form-Button.js\");\nrequire(\"../../Widget/js/nixps-widget-NiXPSWidget.js\");\nrequire(\"../../../General/Component/js/nixps-Component.js\");\n$.widget(\"nixps-container.Collapse\", $[\"nixps-widget\"][\"NiXPSWidget\"], {\n options: {\n /**\r\n * @description The title of the collapse\r\n * @name nixps-form.Collapse#title\r\n * @type string\r\n * @default \"\"\r\n */\n title: \"\",\n /**\r\n * @description Whether to add bootstrap classes or not\r\n * @name nixps-cloudflow.Collapse#useBootstrap\r\n * @type {boolean}\r\n * @default \"true\"\r\n */\n useBootstrap: true,\n /**\r\n * @description Whether the initial state of the panel is collapsed or not\r\n * @name nixps-cloudflow.Collapse#collapsed\r\n * @type {boolean}\r\n * @default \"false\"\r\n */\n collapsed: false,\n /**\r\n * @description An object with the icons to be displayed. Contains 2 members: \"open\" and \"closed\" which are iconClasses for buttons.\r\n * @name nixps-cloudflow.Collapse#iconClasses\r\n * @type {object}\r\n * @default \"{ open: \"fa fa-chevron-down\", closed: \"fa fa-chevron-right\" }\"\r\n */\n iconClasses: {\n open: \"fa fa-chevron-down\",\n closed: \"fa fa-chevron-right\"\n }\n },\n _create: function _create() {\n this._super();\n this._validateOptions();\n this.skipCollapse = [];\n this.element.addClass(this.widgetFullName);\n this._draw();\n this._on(this.element, {\n \"click .collapse-header\": this._clickCollapseHandler\n });\n },\n /**\r\n * @brief redraws list\r\n */\n redraw: function redraw() {\n this._draw();\n },\n /**\r\n * @brief draws the list according to the current state\r\n */\n _draw: function _draw() {\n this.element.empty();\n var header = $(\"
\").addClass(\"collapse-header\").appendTo(this.element);\n var downButton = $(\"