/*
 * File: iframeResizer.js
 * Desc: Force iframes to size to content.
 * Requires: iframeResizer.contentWindow.js to be loaded into the target frame.
 * Author: David J. Bradshaw - dave@bradshaw.net
 * Contributor: Jure Mav - jure.mav@gmail.com
 * Contributor: Reed Dadoune - reed@dadoune.com
 */
;(function() {
  'use strict';

  var
    count                 = 0,
    firstRun              = true,
    logEnabled            = false,
    msgHeader             = 'message',
    msgHeaderLen          = msgHeader.length,
    msgId                 = '[iFrameSizer]', //Must match iframe msg ID
    msgIdLen              = msgId.length,
    pagePosition          = null,
    requestAnimationFrame = window.requestAnimationFrame,
    resetRequiredMethods  = {max:1,scroll:1,bodyScroll:1,documentElementScroll:1},
    settings              = {},
    timer                 = null,

    defaults              = {
      autoResize                : true,
      bodyBackground            : null,
      bodyMargin                : null,
      bodyMarginV1              : 8,
      bodyPadding               : null,
      checkOrigin               : true,
      enableInPageLinks         : false,
      enablePublicMethods       : false,
      heightCalculationMethod   : 'offset',
      interval                  : 32,
      log                       : false,
      maxHeight                 : Infinity,
      maxWidth                  : Infinity,
      minHeight                 : 0,
      minWidth                  : 0,
      resizeFrom                : 'parent',
      scrolling                 : false,
      sizeHeight                : true,
      sizeWidth                 : false,
      tolerance                 : 0,
      closedCallback            : function(){},
      initCallback              : function(){},
      messageCallback           : function(){},
      resizedCallback           : function(){},
      scrollCallback            : function(){return true;}
    };

  function addEventListener(obj,evt,func){
    if ('addEventListener' in window){
      obj.addEventListener(evt,func, false);
    } else if ('attachEvent' in window){//IE
      obj.attachEvent('on'+evt,func);
    }
  }

  function setupRequestAnimationFrame(){
    var
      vendors = ['moz', 'webkit', 'o', 'ms'],
      x;

    // Remove vendor prefixing if prefixed and break early if not
    for (x = 0; x < vendors.length && !requestAnimationFrame; x += 1) {
      requestAnimationFrame = window[vendors[x] + 'RequestAnimationFrame'];
    }

    if (!(requestAnimationFrame)){
      log(' RequestAnimationFrame not supported');
    }
  }

  function getMyID(){
    var retStr = 'Host page';

    if (window.top!==window.self){
      if (window.parentIFrame){
        retStr = window.parentIFrame.getId();
      } else {
        retStr = 'Nested host page';
      }
    }

    return retStr;
  }

  function formatLogMsg(msg){
    return msgId + '[' + getMyID() + ']' + msg;
  }

  function log(msg){
    if (logEnabled && ('object' === typeof window.console)){
      console.log(formatLogMsg(msg));
    }
  }

  function warn(msg){
    if ('object' === typeof window.console){
      console.warn(formatLogMsg(msg));
    }
  }

  function iFrameListener(event){
    function resizeIFrame(){
      function resize(){
        setSize(messageData);
        setPagePosition();
        settings[iframeID].resizedCallback(messageData);
      }

      ensureInRange('Height');
      ensureInRange('Width');

      syncResize(resize,messageData,'resetPage');
    }

    function closeIFrame(iframe){
      var iframeID = iframe.id;

      log(' Removing iFrame: '+iframeID);
      iframe.parentNode.removeChild(iframe);
      settings[iframeID].closedCallback(iframeID);
      delete settings[iframeID];
      log(' --');
    }

    function processMsg(){
      var data = msg.substr(msgIdLen).split(':');

      return {
        iframe: document.getElementById(data[0]),
        id:     data[0],
        height: data[1],
        width:  data[2],
        type:   data[3]
      };
    }

    function ensureInRange(Dimension){
      var
        max  = Number(settings[iframeID]['max'+Dimension]),
        min  = Number(settings[iframeID]['min'+Dimension]),
        dimension = Dimension.toLowerCase(),
        size = Number(messageData[dimension]);

      if (min>max){
        throw new Error('Value for min'+Dimension+' can not be greater than max'+Dimension);
      }

      log(' Checking '+dimension+' is in range '+min+'-'+max);

      if (size<min) {
        size=min;
        log(' Set '+dimension+' to min value');
      }

      if (size>max) {
        size=max;
        log(' Set '+dimension+' to max value');
      }

      messageData[dimension]=''+size;
    }


    function isMessageFromIFrame(){
      function checkAllowedOrigin(){
        function checkList(){
          log(' Checking connection is from allowed list of origins: ' + checkOrigin);
          var i;
          for (i = 0; i < checkOrigin.length; i++) {
            if (checkOrigin[i] === origin) {
              return true;
            }
          }
          return false;
        }

        function checkSingle(){
          log(' Checking connection is from: '+remoteHost);
          return origin == remoteHost;
        }

        return checkOrigin.constructor === Array ? checkList() : checkSingle();
      }

      var
        origin     = event.origin,
        checkOrigin = settings[iframeID].checkOrigin,
        remoteHost = messageData.iframe.src.split('/').slice(0,3).join('/');

      if (checkOrigin) {
        if ((''+origin !== 'null') && !checkAllowedOrigin()) {
          throw new Error(
            'Unexpected message received from: ' + origin +
            ' for ' + messageData.iframe.id +
            '. Message was: ' + event.data +
            '. This error can be disabled by setting the checkOrigin: false option or by providing of array of trusted domains.'
          );
        }
      }

      return true;
    }

    function isMessageForUs(){
      return msgId === ('' + msg).substr(0,msgIdLen); //''+Protects against non-string msg
    }

    function isMessageFromMetaParent(){
      //test if this message is from a parent above us. This is an ugly test, however, updating
      //the message format would break backwards compatibity.
      var retCode = messageData.type in {'true':1,'false':1,'undefined':1};

      if (retCode){
        log(' Ignoring init message from meta parent page');
      }

      return retCode;
    }

    function getMsgBody(offset){
      return msg.substr(msg.indexOf(':')+msgHeaderLen+offset);
    }

    function forwardMsgFromIFrame(msgBody){
      log(' MessageCallback passed: {iframe: '+ messageData.iframe.id + ', message: ' + msgBody + '}');
      settings[iframeID].messageCallback({
        iframe: messageData.iframe,
        message: JSON.parse(msgBody)
      });
      log(' --');
    }

    function checkIFrameExists(){
      if (null === messageData.iframe) {
        warn(' IFrame ('+messageData.id+') not found');
        return false;
      }
      return true;
    }

    function getElementPosition(target){
      var
        iFramePosition = target.getBoundingClientRect();

      getPagePosition();

      return {
        x: parseInt(iFramePosition.left, 10) + parseInt(pagePosition.x, 10),
        y: parseInt(iFramePosition.top, 10)  + parseInt(pagePosition.y, 10)
      };
    }

    function scrollRequestFromChild(addOffset){
      function reposition(){
        pagePosition = newPosition;

        scrollTo();

        log(' --');
      }

      function calcOffset(){
        return {
          x: Number(messageData.width) + offset.x,
          y: Number(messageData.height) + offset.y
        };
      }

      var
        offset = addOffset ? getElementPosition(messageData.iframe) : {x:0,y:0},
        newPosition = calcOffset();

      log(' Reposition requested from iFrame (offset x:'+offset.x+' y:'+offset.y+')');

      if(window.top!==window.self){
        if (window.parentIFrame){
          if (addOffset){
            parentIFrame.scrollToOffset(newPosition.x,newPosition.y);
          } else {
            parentIFrame.scrollTo(messageData.width,messageData.height);
          }
        } else {
          warn(' Unable to scroll to requested position, window.parentIFrame not found');
        }
      } else {
        reposition();
      }

    }

    function scrollTo(){
      if (false !== settings[iframeID].scrollCallback(pagePosition)){
        setPagePosition();
      }
    }

    function findTarget(location){
      var hash = location.split("#")[1] || "";
      var hashData = decodeURIComponent(hash);

      function jumpToTarget(target){
        var jumpPosition = getElementPosition(target);

        log(' Moving to in page link (#'+hash+') at x: '+jumpPosition.x+' y: '+jumpPosition.y);
        pagePosition = {
          x: jumpPosition.x,
          y: jumpPosition.y
        };

        scrollTo();
        log(' --');
      }

      var target = document.getElementById(hashData) || document.getElementsByName(hashData)[0];

      if(window.top!==window.self){
        if (window.parentIFrame){
          parentIFrame.moveToAnchor(hash);
        } else {
          log(' In page link #'+hash+' not found and window.parentIFrame not found');
        }
      } else if (target){
        jumpToTarget(target);
      } else {
        log(' In page link #'+hash+' not found');
      }
    }

    function actionMsg(){
      switch(messageData.type){
        case 'close':
          closeIFrame(messageData.iframe);
          break;
        case 'message':
          forwardMsgFromIFrame(getMsgBody(6));
          break;
        case 'scrollTo':
          scrollRequestFromChild(false);
          break;
        case 'scrollToOffset':
          scrollRequestFromChild(true);
          break;
        case 'inPageLink':
          findTarget(getMsgBody(9));
          break;
        case 'reset':
          resetIFrame(messageData);
          break;
        case 'init':
          resizeIFrame();
          settings[iframeID].initCallback(messageData.iframe);
          break;
        default:
          resizeIFrame();
      }
    }

    function hasSettings(iframeID){
      var retBool = true;

      if (!settings[iframeID]){
        retBool = false;
        warn(messageData.type + ' No settings for ' + iframeID + '. Message was: ' + msg);
      }

      return retBool;
    }

    var
      msg = event.data,
      messageData = {},
      iframeID = null;

    if (isMessageForUs()){
      messageData = processMsg();
      iframeID    = messageData.id;

      if (!isMessageFromMetaParent() && hasSettings(iframeID)){
        logEnabled  = settings[iframeID].log;
        log(' Received: '+msg);

        if ( checkIFrameExists() && isMessageFromIFrame() ){
          actionMsg();
          firstRun = false;
        }
      }
    }
  }


  function getPagePosition (){
    if(null === pagePosition){
      pagePosition = {
        x: (window.pageXOffset !== undefined) ? window.pageXOffset : document.documentElement.scrollLeft,
        y: (window.pageYOffset !== undefined) ? window.pageYOffset : document.documentElement.scrollTop
      };
      log(' Get page position: '+pagePosition.x+','+pagePosition.y);
    }
  }

  function setPagePosition(){
    if(null !== pagePosition){
      window.scrollTo(pagePosition.x,pagePosition.y);
      log(' Set page position: '+pagePosition.x+','+pagePosition.y);
      pagePosition = null;
    }
  }

  function resetIFrame(messageData){
    function reset(){
      setSize(messageData);
      trigger('reset','reset',messageData.iframe,messageData.id);
    }

    log(' Size reset requested by '+('init'===messageData.type?'host page':'iFrame'));
    getPagePosition();
    syncResize(reset,messageData,'init');
  }

  function setSize(messageData){
    function setDimension(dimension){
      messageData.iframe.style[dimension] = messageData[dimension] + 'px';
      log(
        ' IFrame (' + iframeID +
        ') ' + dimension +
        ' set to ' + messageData[dimension] + 'px'
      );
    }
    var iframeID = messageData.iframe.id;
    if( settings[iframeID].sizeHeight) { setDimension('height'); }
    if( settings[iframeID].sizeWidth ) { setDimension('width'); }
  }

  function syncResize(func,messageData,doNotSync){
    if(doNotSync!==messageData.type && requestAnimationFrame){
      log(' Requesting animation frame');
      requestAnimationFrame(func);
    } else {
      func();
    }
  }

  function trigger(calleeMsg,msg,iframe,id){
    if(iframe && iframe.contentWindow){
      log('[' + calleeMsg + '] Sending msg to iframe ('+msg+')');
      iframe.contentWindow.postMessage( msgId + msg, '*' );
    } else {
      warn('[' + calleeMsg + '] IFrame not found');
      if(settings[id]) delete settings[id];
    }
  }


  function setupIFrame(options){
    function setLimits(){
      function addStyle(style){
        if ((Infinity !== settings[iframeID][style]) && (0 !== settings[iframeID][style])){
          iframe.style[style] = settings[iframeID][style] + 'px';
          log(' Set '+style+' = '+settings[iframeID][style]+'px');
        }
      }

      addStyle('maxHeight');
      addStyle('minHeight');
      addStyle('maxWidth');
      addStyle('minWidth');
    }

    function ensureHasId(iframeID){
      if (''===iframeID){
        iframe.id = iframeID = 'iFrameResizer' + count++;
        logEnabled = (options || {}).log;
        log(' Added missing iframe ID: '+ iframeID +' (' + iframe.src + ')');
      }

      return iframeID;
    }

    function setScrolling(){
      log(' IFrame scrolling ' + (settings[iframeID].scrolling ? 'enabled' : 'disabled') + ' for ' + iframeID);
      iframe.style.overflow = false === settings[iframeID].scrolling ? 'hidden' : 'auto';
      iframe.scrolling      = false === settings[iframeID].scrolling ? 'no' : 'yes';
    }

    //The V1 iFrame script expects an int, where as in V2 expects a CSS
    //string value such as '1px 3em', so if we have an int for V2, set V1=V2
    //and then convert V2 to a string PX value.
    function setupBodyMarginValues(){
      if (('number'===typeof(settings[iframeID].bodyMargin)) || ('0'===settings[iframeID].bodyMargin)){
        settings[iframeID].bodyMarginV1 = settings[iframeID].bodyMargin;
        settings[iframeID].bodyMargin   = '' + settings[iframeID].bodyMargin + 'px';
      }
    }

    function createOutgoingMsg(){
      return iframeID +
        ':' + settings[iframeID].bodyMarginV1 +
        ':' + settings[iframeID].sizeWidth +
        ':' + settings[iframeID].log +
        ':' + settings[iframeID].interval +
        ':' + settings[iframeID].enablePublicMethods +
        ':' + settings[iframeID].autoResize +
        ':' + settings[iframeID].bodyMargin +
        ':' + settings[iframeID].heightCalculationMethod +
        ':' + settings[iframeID].bodyBackground +
        ':' + settings[iframeID].bodyPadding +
        ':' + settings[iframeID].tolerance +
        ':' + settings[iframeID].enableInPageLinks +
        ':' + settings[iframeID].resizeFrom;
    }

    function init(msg){
      //We have to call trigger twice, as we can not be sure if all
      //iframes have completed loading when this code runs. The
      //event listener also catches the page changing in the iFrame.
      addEventListener(iframe,'load',function(){
        var fr = firstRun;   // Reduce scope of var to function, because IE8's JS execution
                                     // context stack is borked and this value gets externally
                                     // changed midway through running this function.
        trigger('iFrame.onload',msg,iframe);
        if (!fr && settings[iframeID].heightCalculationMethod in resetRequiredMethods){
          resetIFrame({
            iframe:iframe,
            height:0,
            width:0,
            type:'init'
          });
        }
      });
      trigger('init',msg,iframe);
    }

    function checkOptions(options){
      if ('object' !== typeof options){
        throw new TypeError('Options is not an object.');
      }
    }

    function processOptions(options){
      options = options || {};
      settings[iframeID] = {};

      checkOptions(options);

      for (var option in defaults) {
        if (defaults.hasOwnProperty(option)){
          settings[iframeID][option] = options.hasOwnProperty(option) ? options[option] : defaults[option];
        }
      }

      logEnabled = settings[iframeID].log;
    }

    var
      /*jshint validthis:true */
      iframe   = this,
      iframeID = ensureHasId(iframe.id);

    processOptions(options);
    setScrolling();
    setLimits();
    setupBodyMarginValues();
    init(createOutgoingMsg());
  }

  function throttle(fn,time){
    if (null === timer){
      timer = setTimeout(function(){
        timer = null;
        fn();
      }, time);
    }
  }

  function winResize(){
    throttle(function(){
      for (var iframeId in settings){
        if('parent' === settings[iframeId].resizeFrom){
          trigger('Window resize','resize',document.getElementById(iframeId),iframeId);
        }
      }
    },66);
  }

  function factory(){

    setupRequestAnimationFrame();
    addEventListener(window,'message',iFrameListener);
    addEventListener(window,'resize', winResize);

    function init(element, options){
      if(!element.tagName) {
        throw new TypeError('Object is not a valid DOM element');
      } else if ('IFRAME' !== element.tagName.toUpperCase()) {
        throw new TypeError('Expected <IFRAME> tag, found <'+element.tagName+'>.');
      } else {
        setupIFrame.call(element, options);
      }
    }

    return function iFrameResizeF(options,target){
      switch (typeof(target)){
        case 'undefined':
        case 'string':
          Array.prototype.forEach.call( document.querySelectorAll( target || 'iframe' ), function (element) {
            init(element, options);
          });
          break;
        case 'object':
          init(target, options);
          break;
        default:
          throw new TypeError('Unexpected data type ('+typeof(target)+').');
      }
    };
  }

  function createJQueryPublicMethod($){
    $.fn.iFrameResize = function $iFrameResizeF(options) {
      return this.filter('iframe').each(function (index, element) {
        setupIFrame.call(element, options);
      }).end();
    };
  }

  if (window.jQuery) { createJQueryPublicMethod(jQuery); }

  if (typeof define === 'function' && define.amd) {
    define([],factory);
  } else if (typeof module === 'object' && typeof module.exports === 'object') { //Node for browserfy
    module.exports = factory();
  } else {
    window.iFrameResize = window.iFrameResize || factory();
  }

})();