/*jslint browser:true */ Element.prototype.remove = function() { this.parentElement.removeChild(this); }; NodeList.prototype.remove = HTMLCollection.prototype.remove = function() { for(var i = this.length - 1; i >= 0; i--) { if(this[i] && this[i].parentElement) { this[i].parentElement.removeChild(this[i]); } } }; var RealexHpp = (function () { 'use strict'; var hppUrl = "https://pay.realexpayments.com/pay"; var allowedHppUrls = [ 'https://pay.realexpayments.com/pay', 'https://pay.sandbox.realexpayments.com/pay' ]; var randomId = randomId || Math.random().toString(16).substr(2,8); var setHppUrl = function(url) { hppUrl = url; }; var mobileXSLowerBound = 360; var setMobileXSLowerBound = function (lowerBound) { mobileXSLowerBound = lowerBound; }; var isWindowsMobileOs = /Windows Phone|IEMobile/.test(navigator.userAgent); var isAndroidOrIOs = /Android|iPad|iPhone|iPod/.test(navigator.userAgent); var isMobileXS = function () { return (((window.innerWidth > 0) ? window.innerWidth : screen.width) <= mobileXSLowerBound ? true : false) || (((window.innerHeight > 0) ? window.innerHeight : screen.Height) <= mobileXSLowerBound ? true : false); }; // Display IFrame on WIndows Phone OS mobile devices var isMobileIFrame = isWindowsMobileOs; // For IOs/Android and small screen devices always open in new tab/window var isMobileNewTab = function () { return !isWindowsMobileOs && (isAndroidOrIOs || isMobileXS()); }; var tabWindow; var redirectUrl; /** * Shared functionality across lightbox, embedded, and redirect display modes. */ var internal = { evtMsg: [], /** * Adds a new window message event listener and tracks it for later removal * * @param {Function} evtMsgFct */ addEvtMsgListener: function(evtMsgFct) { this.evtMsg.push({ fct: evtMsgFct, opt: false }); if (window.addEventListener) { window.addEventListener("message", evtMsgFct, false); } else { window.attachEvent('message', evtMsgFct); } }, /** * Removes a previously set window message event listener */ removeOldEvtMsgListener: function () { if (this.evtMsg.length > 0) { var evt = this.evtMsg.pop(); if (window.addEventListener) { window.removeEventListener("message", evt.fct, evt.opt); } else { window.detachEvent('message', evt.fct); } } }, /** * Shimmed base64 encode/decode support */ base64:{ encode:function(input) { var keyStr = "ABCDEFGHIJKLMNOP" + "QRSTUVWXYZabcdef" + "ghijklmnopqrstuv" + "wxyz0123456789+/" + "="; input = escape(input); var output = ""; var chr1, chr2, chr3 = ""; var enc1, enc2, enc3, enc4 = ""; var i = 0; do { chr1 = input.charCodeAt(i++); chr2 = input.charCodeAt(i++); chr3 = input.charCodeAt(i++); enc1 = chr1 >> 2; enc2 = ((chr1 & 3) << 4) | (chr2 >> 4); enc3 = ((chr2 & 15) << 2) | (chr3 >> 6); enc4 = chr3 & 63; if (isNaN(chr2)) { enc3 = enc4 = 64; } else if (isNaN(chr3)) { enc4 = 64; } output = output + keyStr.charAt(enc1) + keyStr.charAt(enc2) + keyStr.charAt(enc3) + keyStr.charAt(enc4); chr1 = chr2 = chr3 = ""; enc1 = enc2 = enc3 = enc4 = ""; } while (i < input.length); return output; }, decode:function(input) { if(typeof input === 'undefined') { return input; } var keyStr = "ABCDEFGHIJKLMNOP" + "QRSTUVWXYZabcdef" + "ghijklmnopqrstuv" + "wxyz0123456789+/" + "="; var output = ""; var chr1, chr2, chr3 = ""; var enc1, enc2, enc3, enc4 = ""; var i = 0; // remove all characters that are not A-Z, a-z, 0-9, +, /, or = var base64test = /[^A-Za-z0-9\+\/\=]/g; if (base64test.exec(input)) { throw new Error("There were invalid base64 characters in the input text.\n" + "Valid base64 characters are A-Z, a-z, 0-9, '+', '/',and '='\n" + "Expect errors in decoding."); } input = input.replace(/[^A-Za-z0-9\+\/\=]/g, ""); do { enc1 = keyStr.indexOf(input.charAt(i++)); enc2 = keyStr.indexOf(input.charAt(i++)); enc3 = keyStr.indexOf(input.charAt(i++)); enc4 = keyStr.indexOf(input.charAt(i++)); chr1 = (enc1 << 2) | (enc2 >> 4); chr2 = ((enc2 & 15) << 4) | (enc3 >> 2); chr3 = ((enc3 & 3) << 6) | enc4; output = output + String.fromCharCode(chr1); if (enc3 !== 64) { output = output + String.fromCharCode(chr2); } if (enc4 !== 64) { output = output + String.fromCharCode(chr3); } chr1 = chr2 = chr3 = ""; enc1 = enc2 = enc3 = enc4 = ""; } while (i < input.length); return unescape(output); } }, /** * Converts an HPP message to a developer-friendly version. * * The decode process has two steps: * * 1. Attempt to parse the string as JSON. If this fails, an error response * is provided as we expect that the HPP has errored out to the cardholder * 2. Attempt to base64 decode the data to cover both HPP versions 1 and 2. * * @param {any} answer * @returns null if answer is not a string, otherwise the data from the HPP */ decodeAnswer:function(answer){ //internal.decodeAnswer var _r; if (typeof answer !== "string") { return null; } try { _r=JSON.parse(answer); } catch (e) { _r = { error: true, message: answer }; } try { for(var r in _r){ if(_r[r]) { _r[r]=internal.base64.decode(_r[r]); } } } catch (e) { /** */ } return _r; }, /** * Creates a new input of type `hidden`. Does not append to DOM. * * @param {string} name Name for the new input * @param {string} value Value for the new input * @returns the created input */ createFormHiddenInput: function (name, value) { var el = document.createElement("input"); el.setAttribute("type", "hidden"); el.setAttribute("name", name); el.setAttribute("value", value); return el; }, /** * Determines a mobile device's orientation for width calculation * * @returns true if in landscape */ checkDevicesOrientation: function () { if (window.orientation === 90 || window.orientation === -90) { return true; } else { return false; } }, /** * Creates a semi-transparent overlay with full width/height to serve as * a background for the lightbox modal * * @returns the created overlay */ createOverlay: function () { var overlay = document.createElement("div"); overlay.setAttribute("id", "rxp-overlay-" + randomId); overlay.style.position = "fixed"; overlay.style.width = "100%"; overlay.style.height = "100%"; overlay.style.top = "0"; overlay.style.left = "0"; overlay.style.transition = "all 0.3s ease-in-out"; overlay.style.zIndex = "100"; if (isMobileIFrame) { overlay.style.position = "absolute !important"; overlay.style.WebkitOverflowScrolling = "touch"; overlay.style.overflowX = "hidden"; overlay.style.overflowY = "scroll"; } document.body.appendChild(overlay); setTimeout(function () { overlay.style.background = "rgba(0, 0, 0, 0.7)"; }, 1); return overlay; }, /** * Closes a lightbox modal and all associated elements * * @param {HTMLImageElement} closeButton * @param {HTMLIFrameElement} iFrame * @param {HTMLImageElement} spinner * @param {HTMLDivElement} overlayElement */ closeModal: function (closeButton, iFrame, spinner, overlayElement) { if (closeButton && closeButton.parentNode) { closeButton.parentNode.removeChild(closeButton); } if (iFrame && iFrame.parentNode) { iFrame.parentNode.removeChild(iFrame); } if (spinner && spinner.parentNode) { spinner.parentNode.removeChild(spinner); } if (!overlayElement) { return; } overlayElement.className = ""; setTimeout(function () { if (overlayElement.parentNode) { overlayElement.parentNode.removeChild(overlayElement); } }, 300); }, /** * Creates a close button for the lightbox modal * * @returns the created element */ createCloseButton: function (overlayElement) { if (document.getElementById("rxp-frame-close-" + randomId) !== null) { return; } var closeButton = document.createElement("img"); closeButton.setAttribute("id","rxp-frame-close-" + randomId); closeButton.setAttribute("src", ""); closeButton.setAttribute("style","transition: all 0.5s ease-in-out; opacity: 0; float: left; position: absolute; left: 50%; margin-left: 173px; z-index: 99999999; top: 30px;"); setTimeout(function () { closeButton.style.opacity = "1"; },500); if (isMobileIFrame) { closeButton.style.position = "absolute"; closeButton.style.float = "right"; closeButton.style.top = "20px"; closeButton.style.left = "initial"; closeButton.style.marginLeft = "0px"; closeButton.style.right = "20px"; } return closeButton; }, /** * Creates a form and appends the HPP request data as hidden input elements to * POST to the defined HPP URL. * * The created form is not appended to the DOM and is not submitted at this time. * * @param {Document} doc * @param {object} token HPP request data * @param {bool} ignorePostMessage If true, the HPP will redirect to the defined * defined redirect URL. If false, the HPP will send a postMessage * to the parent window to be handled by this library. * @returns the created form */ createForm: function (doc, token, ignorePostMessage) { var form = document.createElement("form"); form.setAttribute("method", "POST"); form.setAttribute("action", hppUrl); var versionSet = false; for (var key in token) { if (key === "HPP_VERSION"){ versionSet = true; } form.appendChild(internal.createFormHiddenInput(key, token[key])); } if (versionSet === false){ form.appendChild(internal.createFormHiddenInput("HPP_VERSION", "2")); } if (ignorePostMessage) { form.appendChild(internal.createFormHiddenInput("MERCHANT_RESPONSE_URL", redirectUrl)); } else { var parser = internal.getUrlParser(window.location.href); var hppOriginParam = parser.protocol + '//' + parser.host; form.appendChild(internal.createFormHiddenInput("HPP_POST_RESPONSE", hppOriginParam)); form.appendChild(internal.createFormHiddenInput("HPP_POST_DIMENSIONS", hppOriginParam)); } return form; }, /** * Creates a visual spinner element to be shown with the lightbox overlay while the * HPP's iframe loads * * @returns the created spinner element */ createSpinner: function () { var spinner = document.createElement("img"); spinner.setAttribute("src", ""); spinner.setAttribute("id", "rxp-loader-" + randomId); spinner.style.left = "50%"; spinner.style.position = "fixed"; spinner.style.background = "#FFFFFF"; spinner.style.borderRadius = "50%"; spinner.style.width = "30px"; spinner.style.zIndex = "200"; spinner.style.marginLeft = "-15px"; spinner.style.top = "120px"; return spinner; }, /** * Creates the HPP's form, spinner, iframe, and close button, appends them * to the DOM, and submits the form to load the HPP * * @param {HTMLDivElement} overlayElement * @param {object} token The HPP request data * @returns an object with the created spinner, iframe, and close button */ createIFrame: function (overlayElement, token) { //Create the spinner var spinner = internal.createSpinner(); document.body.appendChild(spinner); //Create the iframe var iFrame = document.createElement("iframe"); iFrame.setAttribute("name", "rxp-frame-" + randomId); iFrame.setAttribute("id", "rxp-frame-" + randomId); iFrame.setAttribute("height", "562px"); iFrame.setAttribute("frameBorder", "0"); iFrame.setAttribute("width", "360px"); iFrame.setAttribute("seamless", "seamless"); iFrame.setAttribute("allow", "payment " + internal.getBaseUrl(hppUrl)); iFrame.style.zIndex = "10001"; iFrame.style.position = "absolute"; iFrame.style.transition = "transform 0.5s ease-in-out"; iFrame.style.transform = "scale(0.7)"; iFrame.style.opacity = "0"; overlayElement.appendChild(iFrame); if (isMobileIFrame) { iFrame.style.top = "0px"; iFrame.style.bottom = "0px"; iFrame.style.left = "0px"; iFrame.style.marginLeft = "0px;"; iFrame.style.width = "100%"; iFrame.style.height = "100%"; iFrame.style.minHeight = "100%"; iFrame.style.WebkitTransform = "translate3d(0,0,0)"; iFrame.style.transform = "translate3d(0, 0, 0)"; var metaTag = document.createElement('meta'); metaTag.name = "viewport"; metaTag.content = "width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0"; document.getElementsByTagName('head')[0].appendChild(metaTag); } else { iFrame.style.top = "40px"; iFrame.style.left = "50%"; iFrame.style.marginLeft = "-180px"; } var closeButton; iFrame.onload = function () { iFrame.style.opacity = "1"; iFrame.style.transform = "scale(1)"; iFrame.style.backgroundColor = "#ffffff"; if (spinner && spinner.parentNode) { spinner.parentNode.removeChild(spinner); } closeButton = internal.createCloseButton(); if (overlayElement && closeButton) { overlayElement.appendChild(closeButton); closeButton.addEventListener("click", function () { internal.closeModal(closeButton, iFrame, spinner, overlayElement); }, true); } }; var form = internal.createForm(document, token); if (iFrame.contentWindow.document.body) { iFrame.contentWindow.document.body.appendChild(form); } else { iFrame.contentWindow.document.appendChild(form); } form.submit(); return { spinner: spinner, iFrame: iFrame, closeButton: closeButton }; }, /** * Opens the HPP in a new window * * Used in some mobile scenarios or when the browser viewport is * smaller than the HPP's inner width. * * Will automatically post the request data to the defined HPP * URL to load the HPP. * * @param {object} token The HPP request data * @returns the created window */ openWindow: function (token) { //open new window var tabWindow = window.open(); // browsers can prevent a new window from being created // e.g. mobile Safari if (!tabWindow) { return null; } var doc = tabWindow.document; //add meta tag to new window (needed for iOS 8 bug) var meta = doc.createElement("meta"); var name = doc.createAttribute("name"); name.value = "viewport"; meta.setAttributeNode(name); var content = doc.createAttribute("content"); content.value = "width=device-width"; meta.setAttributeNode(content); doc.head.appendChild(meta); //create form, append to new window and submit var form = internal.createForm(doc, token); doc.body.appendChild(form); form.submit(); return tabWindow; }, /** * Creates a rudimentary URL parser using an anchor element * * @param {string} url * @returns the created anchor element */ getUrlParser: function (url) { var parser = document.createElement('a'); parser.href = url; return parser; }, /** * Gets the hostname/origin from a URL. Used for origin checks * * @param {string} url * @returns the hostname/origin of the URL */ getHostnameFromUrl: function (url) { return internal.getUrlParser(url).hostname; }, /** * Gets the base URL from a URL. Used to set the 'allow payment' attribute. * * @param {string} url * @returns the base URL of the provided URL */ getBaseUrl: function (url) { var urlParser = internal.getUrlParser(url); return urlParser.protocol + '//' + urlParser.hostname; }, /** * Checks if the origin is HPP. * * @param {string} origin * @returns {boolean} */ isHppOrigin: function(origin) { var result = false; allowedHppUrls.forEach(function (url) { if (internal.getHostnameFromUrl(url) === origin) { result = true; } }); return result; }, /** * Compares the origins from both arguments to validate we have received a postMessage * from the expected source * * @param {string} origin The origin attached to the recieved message * @param {string} hppUrl Our expected source origin * @returns true if the origins match */ isMessageFromHpp: function (origin, hppUrl) { var originHostName = internal.getHostnameFromUrl(origin); return originHostName === internal.getHostnameFromUrl(hppUrl) || internal.isHppOrigin(originHostName); }, /** * Handles messages from the HPP * * Messages from the HPP are one of: * * - iframe resize event * - transaction response * - error information * * @param {MessageEvent} e */ receiveMessage: function (e) { //Check the origin of the response comes from HPP if (!internal.isMessageFromHpp(e.event.origin, hppUrl)) { return; } if (!e.event.data) { return; } var evtdata = internal.decodeAnswer(e.event.data); // we received an invalid message from the HPP iframe (e.g. from a browser plugin) // return early to prevent invalid processing if (evtdata === null) { return; } // check for iframe resize values if (evtdata.iframe) { if (!isMobileNewTab()) { var iframeWidth = evtdata.iframe.width; var iframeHeight = evtdata.iframe.height; var iFrame; var resized = false; if (e.embedded) { iFrame = e.instance.getIframe(); } else { iFrame = document.getElementById("rxp-frame-" + randomId); } if (e.instance.events && e.instance.events.onResize) { e.instance.events.onResize(evtdata.iframe); } if (iframeWidth === "390px" && iframeHeight === "440px") { iFrame.setAttribute("width", iframeWidth); iFrame.setAttribute("height", iframeHeight); resized = true; } iFrame.style.backgroundColor="#ffffff"; if (isMobileIFrame) { iFrame.style.marginLeft = "0px"; iFrame.style.WebkitOverflowScrolling = "touch"; iFrame.style.overflowX = "scroll"; iFrame.style.overflowY = "scroll"; if (!e.embedded) { var overlay = document.getElementById("rxp-overlay-" + randomId); overlay.style.overflowX = "scroll"; overlay.style.overflowY = "scroll"; } } else if (!e.embedded && resized) { iFrame.style.marginLeft = (parseInt(iframeWidth.replace("px", ""), 10) / 2 * -1) + "px"; } if (!e.embedded && resized) { // wrap the below in a setTimeout to prevent a timing issue on a // cache-miss load setTimeout(function () { var closeButton = document.getElementById("rxp-frame-close-" + randomId); closeButton.style.marginLeft = ((parseInt(iframeWidth.replace("px", ""), 10) / 2) -7) + "px"; }, 200); } } } else { var _close=function(){ if (isMobileNewTab() && tabWindow) { //Close the new window tabWindow.close(); } else { //Close the lightbox e.instance.close(); } var overlay=document.getElementById("rxp-overlay-" + randomId); if(overlay) { overlay.remove(); } }; var response = e.event.data; //allow the script to intercept the answer, instead of redirecting to another page. (which is really a 90s thing) if (typeof e.url === 'function'){ e.url(evtdata, _close); return; } _close(); //Create a form and submit the hpp response to the merchant's response url var form = document.createElement("form"); form.setAttribute("method", "POST"); form.setAttribute("action", e.url); form.appendChild(internal.createFormHiddenInput("hppResponse", response)); document.body.appendChild(form); form.submit(); } } }; /** * Public interface for the lightbox display mode */ var RxpLightbox = (function () { var instance; function init() { var overlayElement; var spinner; var iFrame; var closeButton; var token; var isLandscape = internal.checkDevicesOrientation(); if (isMobileIFrame) { if (window.addEventListener) { window.addEventListener("orientationchange", function () { isLandscape = internal.checkDevicesOrientation(); }, false); } } return { lightbox: function () { if (isMobileNewTab()) { tabWindow = internal.openWindow(token); } else { overlayElement = internal.createOverlay(); var temp = internal.createIFrame(overlayElement, token); spinner = temp.spinner; iFrame = temp.iFrame; closeButton = temp.closeButton; } }, close: function () { internal.closeModal(); }, setToken: function (hppToken) { token = hppToken; } }; } return { // Get the Singleton instance if one exists // or create one if it doesn't getInstance: function (hppToken) { if (!instance) { instance = init(); } //Set the hpp token instance.setToken(hppToken); return instance; }, init: function (idOfLightboxButton, merchantUrl, serverSdkJson) { //Get the lightbox instance (it's a singleton) and set the sdk json var lightboxInstance = RxpLightbox.getInstance(serverSdkJson); //if you want the form to load on function call, set to autoload if(idOfLightboxButton==='autoload'){ lightboxInstance.lightbox(); } // Sets the event listener on the PAY button. The click will invoke the lightbox method else if (document.getElementById(idOfLightboxButton).addEventListener) { document.getElementById(idOfLightboxButton).addEventListener("click", lightboxInstance.lightbox, true); } else { document.getElementById(idOfLightboxButton).attachEvent('onclick', lightboxInstance.lightbox); } //avoid multiple message event listener binded to the window object. internal.removeOldEvtMsgListener(); var evtMsgFct = function (event) { return internal.receiveMessage({ event: event, instance: lightboxInstance, url: merchantUrl, embedded: false }); }; internal.evtMsg.push({ fct: evtMsgFct, opt: false }); internal.addEvtMsgListener(evtMsgFct); } }; })(); /** * Public interface for the embedded display mode */ var RxpEmbedded = (function () { var instance; function init() { var overlayElement; var spinner; var iFrame; var closeButton; var token; return { embedded: function () { var form = internal.createForm(document, token); if (iFrame) { if (iFrame.contentWindow.document.body) { iFrame.contentWindow.document.body.appendChild(form); } else { iFrame.contentWindow.document.appendChild(form); } form.submit(); iFrame.style.display = "inherit"; } }, close: function () { iFrame.style.display = "none"; }, setToken: function (hppToken) { token = hppToken; }, setIframe: function (iframeId) { iFrame = document.getElementById(iframeId); if (iFrame) { iFrame.setAttribute("allow", "payment " + internal.getBaseUrl(hppUrl)); } }, getIframe: function () { return iFrame; } }; } return { // Get the Singleton instance if one exists // or create one if it doesn't getInstance: function (hppToken) { if (!instance) { instance = init(); } //Set the hpp token instance.setToken(hppToken); return instance; }, init: function (idOfEmbeddedButton, idOfTargetIframe, merchantUrl, serverSdkJson,events) { //Get the embedded instance (it's a singleton) and set the sdk json var embeddedInstance = RxpEmbedded.getInstance(serverSdkJson); embeddedInstance.events=events; embeddedInstance.setIframe(idOfTargetIframe); //if you want the form to load on function call, set to autoload if(idOfEmbeddedButton==='autoload'){ embeddedInstance.embedded(); } // Sets the event listener on the PAY button. The click will invoke the embedded method else if (document.getElementById(idOfEmbeddedButton).addEventListener) { document.getElementById(idOfEmbeddedButton).addEventListener("click", embeddedInstance.embedded, true); } else { document.getElementById(idOfEmbeddedButton).attachEvent('onclick', embeddedInstance.embedded); } //avoid multiple message event listener binded to the window object. internal.removeOldEvtMsgListener(); var evtMsgFct = function (event) { return internal.receiveMessage({ event: event, instance: embeddedInstance, url: merchantUrl, embedded: true }); }; internal.evtMsg.push({ fct: evtMsgFct, opt: false }); internal.addEvtMsgListener(evtMsgFct); } }; })(); /** * Public interface for the redirect display mode */ var RxpRedirect = (function () { var instance; function init() { var overlayElement; var spinner; var iFrame; var closeButton; var token; var isLandscape = internal.checkDevicesOrientation(); if (isMobileIFrame) { if (window.addEventListener) { window.addEventListener("orientationchange", function () { isLandscape = internal.checkDevicesOrientation(); }, false); } } return { redirect: function () { var form = internal.createForm(document, token, true); document.body.append(form); form.submit(); }, setToken: function (hppToken) { token = hppToken; } }; } return { // Get the singleton instance if one exists // or create one if it doesn't getInstance: function (hppToken) { if (!instance) { instance = init(); } // Set the hpp token instance.setToken(hppToken); return instance; }, init: function (idOfButton, merchantUrl, serverSdkJson) { // Get the redirect instance (it's a singleton) and set the sdk json var redirectInstance = RxpRedirect.getInstance(serverSdkJson); redirectUrl = merchantUrl; // Sets the event listener on the PAY button. The click will invoke the redirect method if (document.getElementById(idOfButton).addEventListener) { document.getElementById(idOfButton).addEventListener("click", redirectInstance.redirect, true); } else { document.getElementById(idOfButton).attachEvent('onclick', redirectInstance.redirect); } //avoid multiple message event listener binded to the window object. internal.removeOldEvtMsgListener(); var evtMsgFct = function (event) { return internal.receiveMessage({ event: event, instance: redirectInstance, url: merchantUrl, embedded: false }); }; internal.evtMsg.push({ fct: evtMsgFct, opt: false }); internal.addEvtMsgListener(evtMsgFct); } }; }()); /** * Public interface for the Realex HPP library */ return { init: RxpLightbox.init, lightbox: { init: RxpLightbox.init }, embedded: { init: RxpEmbedded.init }, redirect: { init: RxpRedirect.init }, setHppUrl: setHppUrl, setMobileXSLowerBound: setMobileXSLowerBound, _internal: internal }; }());