/* Minification failed. Returning unminified contents.
(53783,74-75): run-time error JS1195: Expected expression: .
(53783,81-82): run-time error JS1003: Expected ':': )
(53783,83-84): run-time error JS1195: Expected expression: >
(53783,88-89): run-time error JS1006: Expected ')': {
(59971,13-16): run-time error JS1009: Expected '}': ...
(59984,13-16): run-time error JS1009: Expected '}': ...
(59991,13-16): run-time error JS1009: Expected '}': ...
(59992,9-10): run-time error JS1002: Syntax error: }
(59998,13-16): run-time error JS1009: Expected '}': ...
(59999,9-10): run-time error JS1002: Syntax error: }
(60004,13-16): run-time error JS1010: Expected identifier: ...
(60004,24-25): run-time error JS1193: Expected ',' or ')': .
(60004,24-25): run-time error JS1006: Expected ')': .
(60004,24-25): run-time error JS1008: Expected '{': .
(60004,69): run-time error JS1004: Expected ';'
(60004,69-70): run-time error JS1195: Expected expression: )
(60005,10-11): run-time error JS1009: Expected '}': ;
(60008,13-16): run-time error JS1010: Expected identifier: ...
(60008,24-25): run-time error JS1193: Expected ',' or ')': .
(60008,24-25): run-time error JS1006: Expected ')': .
(60008,24-25): run-time error JS1008: Expected '{': .
(60008,69): run-time error JS1004: Expected ';'
(60008,69-70): run-time error JS1195: Expected expression: )
(60009,10-11): run-time error JS1009: Expected '}': ;
(60012,13-16): run-time error JS1010: Expected identifier: ...
(60012,24-25): run-time error JS1193: Expected ',' or ')': .
(60012,24-25): run-time error JS1006: Expected ')': .
(60012,24-25): run-time error JS1008: Expected '{': .
(60012,70): run-time error JS1004: Expected ';'
(60012,70-71): run-time error JS1195: Expected expression: )
(60013,10-11): run-time error JS1009: Expected '}': ;
(60022,13-16): run-time error JS1009: Expected '}': ...
(60023,9-10): run-time error JS1002: Syntax error: }
(60028,13-16): run-time error JS1010: Expected identifier: ...
(60028,24-25): run-time error JS1193: Expected ',' or ')': .
(60028,24-25): run-time error JS1006: Expected ')': .
(60028,24-25): run-time error JS1008: Expected '{': .
(60028,59): run-time error JS1004: Expected ';'
(60028,59-60): run-time error JS1195: Expected expression: )
(60029,10-11): run-time error JS1009: Expected '}': ;
(60034,13-16): run-time error JS1009: Expected '}': ...
(60035,9-10): run-time error JS1002: Syntax error: }
(60047,13-16): run-time error JS1009: Expected '}': ...
(60048,9-10): run-time error JS1002: Syntax error: }
(60072,5-6): run-time error JS1002: Syntax error: }
(61191,1-2): run-time error JS1002: Syntax error: }
(61194,48): run-time error JS1004: Expected ';'
(61194,58-59): run-time error JS1010: Expected identifier: (
(60004,13,60005,10): run-time error JS1314: Implicit property name must be identifier: ...(options.seatStyles ? options.seatStyles.default : {})
        }
(60008,13,60009,10): run-time error JS1314: Implicit property name must be identifier: ...(options.seatStyles ? options.seatStyles.marquee : {})
        }
(60012,13,60013,10): run-time error JS1314: Implicit property name must be identifier: ...(options.seatStyles ? options.seatStyles.selected : {})
        }
(60028,13,60029,10): run-time error JS1314: Implicit property name must be identifier: ...(options.mouse ? options.mouse.buttons : {})
        }
(61147,5,61190,6): run-time error JS1018: 'return' statement outside of function: return {
        "addUpdateCategory": addUpdateCategory,
        "canvas": canvas,
        "canvasControls": canvasControls,
        "canvasProperties": canvasProperties,
        "categories": categories,
        "clearSelectedSeats": clearSelectedSeats,
        "colourPalette": colourPalette,
        "enums": enums,
        "displayTooltip": displayTooltip,
        "events": events,
        "getAllBlocks": getAllBlocks,
        "getAllTexts": getAllTexts,
        "getBlockOfSeat": getBlockOfSeat,
        "getBlocks": getBlocks,
        "getHasRenderedFirstFrame": getHasRenderedFirstFrame,
        "getRenderInterval": getRenderInterval,
        "getRowOfSeat": getRowOfSeat,
        "getSeat": getSeat,
        "getSeatIndexInRow": getSeatIndexInRow,
        "getSeats": getSeats,
        "getMarqueeSeats": getMarqueeSeats,
        "getSelectedSeats": getSelectedSeats,
        "getTexts": getTexts,
        "getZoom": getZoom,
        "helpers": helpers,
        "hideTooltip": hideTooltip,
        "isSeatSelected": isSeatSelected,
        "labelProperties": labelProperties,
        "layout": layout,
        "mouse": mouse,
        "removeCategory": removeCategory,
        "seatStyles": seatStyles,
        "selectSeat": selectSeat,
        "setDefaultTooltipPosition": setDefaultTooltipPosition,
        "setZoom": setZoom,
        "tooltip": tooltip,
        "touch": touch,
        "translateToCanvasCoordinates": translateToCanvasCoordinates,
        "translateToPlanCoordinates": translateToPlanCoordinates,
        "unselectSeat": unselectSeat,
        "zoomToBlock": zoomToBlock,
        "zoomToSeat": zoomToSeat
    }
 */
/**
 * @license AngularJS v1.4.8
 * (c) 2010-2015 Google, Inc. http://angularjs.org
 * License: MIT
 */
(function(window, document, undefined) {'use strict';

/**
 * @description
 *
 * This object provides a utility for producing rich Error messages within
 * Angular. It can be called as follows:
 *
 * var exampleMinErr = minErr('example');
 * throw exampleMinErr('one', 'This {0} is {1}', foo, bar);
 *
 * The above creates an instance of minErr in the example namespace. The
 * resulting error will have a namespaced error code of example.one.  The
 * resulting error will replace {0} with the value of foo, and {1} with the
 * value of bar. The object is not restricted in the number of arguments it can
 * take.
 *
 * If fewer arguments are specified than necessary for interpolation, the extra
 * interpolation markers will be preserved in the final string.
 *
 * Since data will be parsed statically during a build step, some restrictions
 * are applied with respect to how minErr instances are created and called.
 * Instances should have names of the form namespaceMinErr for a minErr created
 * using minErr('namespace') . Error codes, namespaces and template strings
 * should all be static strings, not variables or general expressions.
 *
 * @param {string} module The namespace to use for the new minErr instance.
 * @param {function} ErrorConstructor Custom error constructor to be instantiated when returning
 *   error from returned function, for cases when a particular type of error is useful.
 * @returns {function(code:string, template:string, ...templateArgs): Error} minErr instance
 */

function minErr(module, ErrorConstructor) {
  ErrorConstructor = ErrorConstructor || Error;
  return function() {
    var SKIP_INDEXES = 2;

    var templateArgs = arguments,
      code = templateArgs[0],
      message = '[' + (module ? module + ':' : '') + code + '] ',
      template = templateArgs[1],
      paramPrefix, i;

    message += template.replace(/\{\d+\}/g, function(match) {
      var index = +match.slice(1, -1),
        shiftedIndex = index + SKIP_INDEXES;

      if (shiftedIndex < templateArgs.length) {
        return toDebugString(templateArgs[shiftedIndex]);
      }

      return match;
    });

    message += '\nhttp://errors.angularjs.org/1.4.8/' +
      (module ? module + '/' : '') + code;

    for (i = SKIP_INDEXES, paramPrefix = '?'; i < templateArgs.length; i++, paramPrefix = '&') {
      message += paramPrefix + 'p' + (i - SKIP_INDEXES) + '=' +
        encodeURIComponent(toDebugString(templateArgs[i]));
    }

    return new ErrorConstructor(message);
  };
}

/* We need to tell jshint what variables are being exported */
/* global angular: true,
  msie: true,
  jqLite: true,
  jQuery: true,
  slice: true,
  splice: true,
  push: true,
  toString: true,
  ngMinErr: true,
  angularModule: true,
  uid: true,
  REGEX_STRING_REGEXP: true,
  VALIDITY_STATE_PROPERTY: true,

  lowercase: true,
  uppercase: true,
  manualLowercase: true,
  manualUppercase: true,
  nodeName_: true,
  isArrayLike: true,
  forEach: true,
  forEachSorted: true,
  reverseParams: true,
  nextUid: true,
  setHashKey: true,
  extend: true,
  toInt: true,
  inherit: true,
  merge: true,
  noop: true,
  identity: true,
  valueFn: true,
  isUndefined: true,
  isDefined: true,
  isObject: true,
  isBlankObject: true,
  isString: true,
  isNumber: true,
  isDate: true,
  isArray: true,
  isFunction: true,
  isRegExp: true,
  isWindow: true,
  isScope: true,
  isFile: true,
  isFormData: true,
  isBlob: true,
  isBoolean: true,
  isPromiseLike: true,
  trim: true,
  escapeForRegexp: true,
  isElement: true,
  makeMap: true,
  includes: true,
  arrayRemove: true,
  copy: true,
  shallowCopy: true,
  equals: true,
  csp: true,
  jq: true,
  concat: true,
  sliceArgs: true,
  bind: true,
  toJsonReplacer: true,
  toJson: true,
  fromJson: true,
  convertTimezoneToLocal: true,
  timezoneToOffset: true,
  startingTag: true,
  tryDecodeURIComponent: true,
  parseKeyValue: true,
  toKeyValue: true,
  encodeUriSegment: true,
  encodeUriQuery: true,
  angularInit: true,
  bootstrap: true,
  getTestability: true,
  snake_case: true,
  bindJQuery: true,
  assertArg: true,
  assertArgFn: true,
  assertNotHasOwnProperty: true,
  getter: true,
  getBlockNodes: true,
  hasOwnProperty: true,
  createMap: true,

  NODE_TYPE_ELEMENT: true,
  NODE_TYPE_ATTRIBUTE: true,
  NODE_TYPE_TEXT: true,
  NODE_TYPE_COMMENT: true,
  NODE_TYPE_DOCUMENT: true,
  NODE_TYPE_DOCUMENT_FRAGMENT: true,
*/

////////////////////////////////////

/**
 * @ngdoc module
 * @name ng
 * @module ng
 * @description
 *
 * # ng (core module)
 * The ng module is loaded by default when an AngularJS application is started. The module itself
 * contains the essential components for an AngularJS application to function. The table below
 * lists a high level breakdown of each of the services/factories, filters, directives and testing
 * components available within this core module.
 *
 * <div doc-module-components="ng"></div>
 */

var REGEX_STRING_REGEXP = /^\/(.+)\/([a-z]*)$/;

// The name of a form control's ValidityState property.
// This is used so that it's possible for internal tests to create mock ValidityStates.
var VALIDITY_STATE_PROPERTY = 'validity';

/**
 * @ngdoc function
 * @name angular.lowercase
 * @module ng
 * @kind function
 *
 * @description Converts the specified string to lowercase.
 * @param {string} string String to be converted to lowercase.
 * @returns {string} Lowercased string.
 */
var lowercase = function(string) {return isString(string) ? string.toLowerCase() : string;};
var hasOwnProperty = Object.prototype.hasOwnProperty;

/**
 * @ngdoc function
 * @name angular.uppercase
 * @module ng
 * @kind function
 *
 * @description Converts the specified string to uppercase.
 * @param {string} string String to be converted to uppercase.
 * @returns {string} Uppercased string.
 */
var uppercase = function(string) {return isString(string) ? string.toUpperCase() : string;};


var manualLowercase = function(s) {
  /* jshint bitwise: false */
  return isString(s)
      ? s.replace(/[A-Z]/g, function(ch) {return String.fromCharCode(ch.charCodeAt(0) | 32);})
      : s;
};
var manualUppercase = function(s) {
  /* jshint bitwise: false */
  return isString(s)
      ? s.replace(/[a-z]/g, function(ch) {return String.fromCharCode(ch.charCodeAt(0) & ~32);})
      : s;
};


// String#toLowerCase and String#toUpperCase don't produce correct results in browsers with Turkish
// locale, for this reason we need to detect this case and redefine lowercase/uppercase methods
// with correct but slower alternatives.
if ('i' !== 'I'.toLowerCase()) {
  lowercase = manualLowercase;
  uppercase = manualUppercase;
}


var
    msie,             // holds major version number for IE, or NaN if UA is not IE.
    jqLite,           // delay binding since jQuery could be loaded after us.
    jQuery,           // delay binding
    slice             = [].slice,
    splice            = [].splice,
    push              = [].push,
    toString          = Object.prototype.toString,
    getPrototypeOf    = Object.getPrototypeOf,
    ngMinErr          = minErr('ng'),

    /** @name angular */
    angular           = window.angular || (window.angular = {}),
    angularModule,
    uid               = 0;

/**
 * documentMode is an IE-only property
 * http://msdn.microsoft.com/en-us/library/ie/cc196988(v=vs.85).aspx
 */
msie = document.documentMode;


/**
 * @private
 * @param {*} obj
 * @return {boolean} Returns true if `obj` is an array or array-like object (NodeList, Arguments,
 *                   String ...)
 */
function isArrayLike(obj) {

  // `null`, `undefined` and `window` are not array-like
  if (obj == null || isWindow(obj)) return false;

  // arrays, strings and jQuery/jqLite objects are array like
  // * jqLite is either the jQuery or jqLite constructor function
  // * we have to check the existance of jqLite first as this method is called
  //   via the forEach method when constructing the jqLite object in the first place
  if (isArray(obj) || isString(obj) || (jqLite && obj instanceof jqLite)) return true;

  // Support: iOS 8.2 (not reproducible in simulator)
  // "length" in obj used to prevent JIT error (gh-11508)
  var length = "length" in Object(obj) && obj.length;

  // NodeList objects (with `item` method) and
  // other objects with suitable length characteristics are array-like
  return isNumber(length) &&
    (length >= 0 && (length - 1) in obj || typeof obj.item == 'function');
}

/**
 * @ngdoc function
 * @name angular.forEach
 * @module ng
 * @kind function
 *
 * @description
 * Invokes the `iterator` function once for each item in `obj` collection, which can be either an
 * object or an array. The `iterator` function is invoked with `iterator(value, key, obj)`, where `value`
 * is the value of an object property or an array element, `key` is the object property key or
 * array element index and obj is the `obj` itself. Specifying a `context` for the function is optional.
 *
 * It is worth noting that `.forEach` does not iterate over inherited properties because it filters
 * using the `hasOwnProperty` method.
 *
 * Unlike ES262's
 * [Array.prototype.forEach](http://www.ecma-international.org/ecma-262/5.1/#sec-15.4.4.18),
 * Providing 'undefined' or 'null' values for `obj` will not throw a TypeError, but rather just
 * return the value provided.
 *
   ```js
     var values = {name: 'misko', gender: 'male'};
     var log = [];
     angular.forEach(values, function(value, key) {
       this.push(key + ': ' + value);
     }, log);
     expect(log).toEqual(['name: misko', 'gender: male']);
   ```
 *
 * @param {Object|Array} obj Object to iterate over.
 * @param {Function} iterator Iterator function.
 * @param {Object=} context Object to become context (`this`) for the iterator function.
 * @returns {Object|Array} Reference to `obj`.
 */

function forEach(obj, iterator, context) {
  var key, length;
  if (obj) {
    if (isFunction(obj)) {
      for (key in obj) {
        // Need to check if hasOwnProperty exists,
        // as on IE8 the result of querySelectorAll is an object without a hasOwnProperty function
        if (key != 'prototype' && key != 'length' && key != 'name' && (!obj.hasOwnProperty || obj.hasOwnProperty(key))) {
          iterator.call(context, obj[key], key, obj);
        }
      }
    } else if (isArray(obj) || isArrayLike(obj)) {
      var isPrimitive = typeof obj !== 'object';
      for (key = 0, length = obj.length; key < length; key++) {
        if (isPrimitive || key in obj) {
          iterator.call(context, obj[key], key, obj);
        }
      }
    } else if (obj.forEach && obj.forEach !== forEach) {
        obj.forEach(iterator, context, obj);
    } else if (isBlankObject(obj)) {
      // createMap() fast path --- Safe to avoid hasOwnProperty check because prototype chain is empty
      for (key in obj) {
        iterator.call(context, obj[key], key, obj);
      }
    } else if (typeof obj.hasOwnProperty === 'function') {
      // Slow path for objects inheriting Object.prototype, hasOwnProperty check needed
      for (key in obj) {
        if (obj.hasOwnProperty(key)) {
          iterator.call(context, obj[key], key, obj);
        }
      }
    } else {
      // Slow path for objects which do not have a method `hasOwnProperty`
      for (key in obj) {
        if (hasOwnProperty.call(obj, key)) {
          iterator.call(context, obj[key], key, obj);
        }
      }
    }
  }
  return obj;
}

function forEachSorted(obj, iterator, context) {
  var keys = Object.keys(obj).sort();
  for (var i = 0; i < keys.length; i++) {
    iterator.call(context, obj[keys[i]], keys[i]);
  }
  return keys;
}


/**
 * when using forEach the params are value, key, but it is often useful to have key, value.
 * @param {function(string, *)} iteratorFn
 * @returns {function(*, string)}
 */
function reverseParams(iteratorFn) {
  return function(value, key) { iteratorFn(key, value); };
}

/**
 * A consistent way of creating unique IDs in angular.
 *
 * Using simple numbers allows us to generate 28.6 million unique ids per second for 10 years before
 * we hit number precision issues in JavaScript.
 *
 * Math.pow(2,53) / 60 / 60 / 24 / 365 / 10 = 28.6M
 *
 * @returns {number} an unique alpha-numeric string
 */
function nextUid() {
  return ++uid;
}


/**
 * Set or clear the hashkey for an object.
 * @param obj object
 * @param h the hashkey (!truthy to delete the hashkey)
 */
function setHashKey(obj, h) {
  if (h) {
    obj.$$hashKey = h;
  } else {
    delete obj.$$hashKey;
  }
}


function baseExtend(dst, objs, deep) {
  var h = dst.$$hashKey;

  for (var i = 0, ii = objs.length; i < ii; ++i) {
    var obj = objs[i];
    if (!isObject(obj) && !isFunction(obj)) continue;
    var keys = Object.keys(obj);
    for (var j = 0, jj = keys.length; j < jj; j++) {
      var key = keys[j];
      var src = obj[key];

      if (deep && isObject(src)) {
        if (isDate(src)) {
          dst[key] = new Date(src.valueOf());
        } else if (isRegExp(src)) {
          dst[key] = new RegExp(src);
        } else if (src.nodeName) {
          dst[key] = src.cloneNode(true);
        } else if (isElement(src)) {
          dst[key] = src.clone();
        } else {
          if (!isObject(dst[key])) dst[key] = isArray(src) ? [] : {};
          baseExtend(dst[key], [src], true);
        }
      } else {
        dst[key] = src;
      }
    }
  }

  setHashKey(dst, h);
  return dst;
}

/**
 * @ngdoc function
 * @name angular.extend
 * @module ng
 * @kind function
 *
 * @description
 * Extends the destination object `dst` by copying own enumerable properties from the `src` object(s)
 * to `dst`. You can specify multiple `src` objects. If you want to preserve original objects, you can do so
 * by passing an empty object as the target: `var object = angular.extend({}, object1, object2)`.
 *
 * **Note:** Keep in mind that `angular.extend` does not support recursive merge (deep copy). Use
 * {@link angular.merge} for this.
 *
 * @param {Object} dst Destination object.
 * @param {...Object} src Source object(s).
 * @returns {Object} Reference to `dst`.
 */
function extend(dst) {
  return baseExtend(dst, slice.call(arguments, 1), false);
}


/**
* @ngdoc function
* @name angular.merge
* @module ng
* @kind function
*
* @description
* Deeply extends the destination object `dst` by copying own enumerable properties from the `src` object(s)
* to `dst`. You can specify multiple `src` objects. If you want to preserve original objects, you can do so
* by passing an empty object as the target: `var object = angular.merge({}, object1, object2)`.
*
* Unlike {@link angular.extend extend()}, `merge()` recursively descends into object properties of source
* objects, performing a deep copy.
*
* @param {Object} dst Destination object.
* @param {...Object} src Source object(s).
* @returns {Object} Reference to `dst`.
*/
function merge(dst) {
  return baseExtend(dst, slice.call(arguments, 1), true);
}



function toInt(str) {
  return parseInt(str, 10);
}


function inherit(parent, extra) {
  return extend(Object.create(parent), extra);
}

/**
 * @ngdoc function
 * @name angular.noop
 * @module ng
 * @kind function
 *
 * @description
 * A function that performs no operations. This function can be useful when writing code in the
 * functional style.
   ```js
     function foo(callback) {
       var result = calculateResult();
       (callback || angular.noop)(result);
     }
   ```
 */
function noop() {}
noop.$inject = [];


/**
 * @ngdoc function
 * @name angular.identity
 * @module ng
 * @kind function
 *
 * @description
 * A function that returns its first argument. This function is useful when writing code in the
 * functional style.
 *
   ```js
     function transformer(transformationFn, value) {
       return (transformationFn || angular.identity)(value);
     };
   ```
  * @param {*} value to be returned.
  * @returns {*} the value passed in.
 */
function identity($) {return $;}
identity.$inject = [];


function valueFn(value) {return function() {return value;};}

function hasCustomToString(obj) {
  return isFunction(obj.toString) && obj.toString !== toString;
}


/**
 * @ngdoc function
 * @name angular.isUndefined
 * @module ng
 * @kind function
 *
 * @description
 * Determines if a reference is undefined.
 *
 * @param {*} value Reference to check.
 * @returns {boolean} True if `value` is undefined.
 */
function isUndefined(value) {return typeof value === 'undefined';}


/**
 * @ngdoc function
 * @name angular.isDefined
 * @module ng
 * @kind function
 *
 * @description
 * Determines if a reference is defined.
 *
 * @param {*} value Reference to check.
 * @returns {boolean} True if `value` is defined.
 */
function isDefined(value) {return typeof value !== 'undefined';}


/**
 * @ngdoc function
 * @name angular.isObject
 * @module ng
 * @kind function
 *
 * @description
 * Determines if a reference is an `Object`. Unlike `typeof` in JavaScript, `null`s are not
 * considered to be objects. Note that JavaScript arrays are objects.
 *
 * @param {*} value Reference to check.
 * @returns {boolean} True if `value` is an `Object` but not `null`.
 */
function isObject(value) {
  // http://jsperf.com/isobject4
  return value !== null && typeof value === 'object';
}


/**
 * Determine if a value is an object with a null prototype
 *
 * @returns {boolean} True if `value` is an `Object` with a null prototype
 */
function isBlankObject(value) {
  return value !== null && typeof value === 'object' && !getPrototypeOf(value);
}


/**
 * @ngdoc function
 * @name angular.isString
 * @module ng
 * @kind function
 *
 * @description
 * Determines if a reference is a `String`.
 *
 * @param {*} value Reference to check.
 * @returns {boolean} True if `value` is a `String`.
 */
function isString(value) {return typeof value === 'string';}


/**
 * @ngdoc function
 * @name angular.isNumber
 * @module ng
 * @kind function
 *
 * @description
 * Determines if a reference is a `Number`.
 *
 * This includes the "special" numbers `NaN`, `+Infinity` and `-Infinity`.
 *
 * If you wish to exclude these then you can use the native
 * [`isFinite'](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/isFinite)
 * method.
 *
 * @param {*} value Reference to check.
 * @returns {boolean} True if `value` is a `Number`.
 */
function isNumber(value) {return typeof value === 'number';}


/**
 * @ngdoc function
 * @name angular.isDate
 * @module ng
 * @kind function
 *
 * @description
 * Determines if a value is a date.
 *
 * @param {*} value Reference to check.
 * @returns {boolean} True if `value` is a `Date`.
 */
function isDate(value) {
  return toString.call(value) === '[object Date]';
}


/**
 * @ngdoc function
 * @name angular.isArray
 * @module ng
 * @kind function
 *
 * @description
 * Determines if a reference is an `Array`.
 *
 * @param {*} value Reference to check.
 * @returns {boolean} True if `value` is an `Array`.
 */
var isArray = Array.isArray;

/**
 * @ngdoc function
 * @name angular.isFunction
 * @module ng
 * @kind function
 *
 * @description
 * Determines if a reference is a `Function`.
 *
 * @param {*} value Reference to check.
 * @returns {boolean} True if `value` is a `Function`.
 */
function isFunction(value) {return typeof value === 'function';}


/**
 * Determines if a value is a regular expression object.
 *
 * @private
 * @param {*} value Reference to check.
 * @returns {boolean} True if `value` is a `RegExp`.
 */
function isRegExp(value) {
  return toString.call(value) === '[object RegExp]';
}


/**
 * Checks if `obj` is a window object.
 *
 * @private
 * @param {*} obj Object to check
 * @returns {boolean} True if `obj` is a window obj.
 */
function isWindow(obj) {
  return obj && obj.window === obj;
}


function isScope(obj) {
  return obj && obj.$evalAsync && obj.$watch;
}


function isFile(obj) {
  return toString.call(obj) === '[object File]';
}


function isFormData(obj) {
  return toString.call(obj) === '[object FormData]';
}


function isBlob(obj) {
  return toString.call(obj) === '[object Blob]';
}


function isBoolean(value) {
  return typeof value === 'boolean';
}


function isPromiseLike(obj) {
  return obj && isFunction(obj.then);
}


var TYPED_ARRAY_REGEXP = /^\[object (?:Uint8|Uint8Clamped|Uint16|Uint32|Int8|Int16|Int32|Float32|Float64)Array\]$/;
function isTypedArray(value) {
  return value && isNumber(value.length) && TYPED_ARRAY_REGEXP.test(toString.call(value));
}


var trim = function(value) {
  return isString(value) ? value.trim() : value;
};

// Copied from:
// http://docs.closure-library.googlecode.com/git/local_closure_goog_string_string.js.source.html#line1021
// Prereq: s is a string.
var escapeForRegexp = function(s) {
  return s.replace(/([-()\[\]{}+?*.$\^|,:#<!\\])/g, '\\$1').
           replace(/\x08/g, '\\x08');
};


/**
 * @ngdoc function
 * @name angular.isElement
 * @module ng
 * @kind function
 *
 * @description
 * Determines if a reference is a DOM element (or wrapped jQuery element).
 *
 * @param {*} value Reference to check.
 * @returns {boolean} True if `value` is a DOM element (or wrapped jQuery element).
 */
function isElement(node) {
  return !!(node &&
    (node.nodeName  // we are a direct element
    || (node.prop && node.attr && node.find)));  // we have an on and find method part of jQuery API
}

/**
 * @param str 'key1,key2,...'
 * @returns {object} in the form of {key1:true, key2:true, ...}
 */
function makeMap(str) {
  var obj = {}, items = str.split(","), i;
  for (i = 0; i < items.length; i++) {
    obj[items[i]] = true;
  }
  return obj;
}


function nodeName_(element) {
  return lowercase(element.nodeName || (element[0] && element[0].nodeName));
}

function includes(array, obj) {
  return Array.prototype.indexOf.call(array, obj) != -1;
}

function arrayRemove(array, value) {
  var index = array.indexOf(value);
  if (index >= 0) {
    array.splice(index, 1);
  }
  return index;
}

/**
 * @ngdoc function
 * @name angular.copy
 * @module ng
 * @kind function
 *
 * @description
 * Creates a deep copy of `source`, which should be an object or an array.
 *
 * * If no destination is supplied, a copy of the object or array is created.
 * * If a destination is provided, all of its elements (for arrays) or properties (for objects)
 *   are deleted and then all elements/properties from the source are copied to it.
 * * If `source` is not an object or array (inc. `null` and `undefined`), `source` is returned.
 * * If `source` is identical to 'destination' an exception will be thrown.
 *
 * @param {*} source The source that will be used to make a copy.
 *                   Can be any type, including primitives, `null`, and `undefined`.
 * @param {(Object|Array)=} destination Destination into which the source is copied. If
 *     provided, must be of the same type as `source`.
 * @returns {*} The copy or updated `destination`, if `destination` was specified.
 *
 * @example
 <example module="copyExample">
 <file name="index.html">
 <div ng-controller="ExampleController">
 <form novalidate class="simple-form">
 Name: <input type="text" ng-model="user.name" /><br />
 E-mail: <input type="email" ng-model="user.email" /><br />
 Gender: <input type="radio" ng-model="user.gender" value="male" />male
 <input type="radio" ng-model="user.gender" value="female" />female<br />
 <button ng-click="reset()">RESET</button>
 <button ng-click="update(user)">SAVE</button>
 </form>
 <pre>form = {{user | json}}</pre>
 <pre>master = {{master | json}}</pre>
 </div>

 <script>
  angular.module('copyExample', [])
    .controller('ExampleController', ['$scope', function($scope) {
      $scope.master= {};

      $scope.update = function(user) {
        // Example with 1 argument
        $scope.master= angular.copy(user);
      };

      $scope.reset = function() {
        // Example with 2 arguments
        angular.copy($scope.master, $scope.user);
      };

      $scope.reset();
    }]);
 </script>
 </file>
 </example>
 */
function copy(source, destination) {
  var stackSource = [];
  var stackDest = [];

  if (destination) {
    if (isTypedArray(destination)) {
      throw ngMinErr('cpta', "Can't copy! TypedArray destination cannot be mutated.");
    }
    if (source === destination) {
      throw ngMinErr('cpi', "Can't copy! Source and destination are identical.");
    }

    // Empty the destination object
    if (isArray(destination)) {
      destination.length = 0;
    } else {
      forEach(destination, function(value, key) {
        if (key !== '$$hashKey') {
          delete destination[key];
        }
      });
    }

    stackSource.push(source);
    stackDest.push(destination);
    return copyRecurse(source, destination);
  }

  return copyElement(source);

  function copyRecurse(source, destination) {
    var h = destination.$$hashKey;
    var result, key;
    if (isArray(source)) {
      for (var i = 0, ii = source.length; i < ii; i++) {
        destination.push(copyElement(source[i]));
      }
    } else if (isBlankObject(source)) {
      // createMap() fast path --- Safe to avoid hasOwnProperty check because prototype chain is empty
      for (key in source) {
        destination[key] = copyElement(source[key]);
      }
    } else if (source && typeof source.hasOwnProperty === 'function') {
      // Slow path, which must rely on hasOwnProperty
      for (key in source) {
        if (source.hasOwnProperty(key)) {
          destination[key] = copyElement(source[key]);
        }
      }
    } else {
      // Slowest path --- hasOwnProperty can't be called as a method
      for (key in source) {
        if (hasOwnProperty.call(source, key)) {
          destination[key] = copyElement(source[key]);
        }
      }
    }
    setHashKey(destination, h);
    return destination;
  }

  function copyElement(source) {
    // Simple values
    if (!isObject(source)) {
      return source;
    }

    // Already copied values
    var index = stackSource.indexOf(source);
    if (index !== -1) {
      return stackDest[index];
    }

    if (isWindow(source) || isScope(source)) {
      throw ngMinErr('cpws',
        "Can't copy! Making copies of Window or Scope instances is not supported.");
    }

    var needsRecurse = false;
    var destination;

    if (isArray(source)) {
      destination = [];
      needsRecurse = true;
    } else if (isTypedArray(source)) {
      destination = new source.constructor(source);
    } else if (isDate(source)) {
      destination = new Date(source.getTime());
    } else if (isRegExp(source)) {
      destination = new RegExp(source.source, source.toString().match(/[^\/]*$/)[0]);
      destination.lastIndex = source.lastIndex;
    } else if (isFunction(source.cloneNode)) {
        destination = source.cloneNode(true);
    } else {
      destination = Object.create(getPrototypeOf(source));
      needsRecurse = true;
    }

    stackSource.push(source);
    stackDest.push(destination);

    return needsRecurse
      ? copyRecurse(source, destination)
      : destination;
  }
}

/**
 * Creates a shallow copy of an object, an array or a primitive.
 *
 * Assumes that there are no proto properties for objects.
 */
function shallowCopy(src, dst) {
  if (isArray(src)) {
    dst = dst || [];

    for (var i = 0, ii = src.length; i < ii; i++) {
      dst[i] = src[i];
    }
  } else if (isObject(src)) {
    dst = dst || {};

    for (var key in src) {
      if (!(key.charAt(0) === '$' && key.charAt(1) === '$')) {
        dst[key] = src[key];
      }
    }
  }

  return dst || src;
}


/**
 * @ngdoc function
 * @name angular.equals
 * @module ng
 * @kind function
 *
 * @description
 * Determines if two objects or two values are equivalent. Supports value types, regular
 * expressions, arrays and objects.
 *
 * Two objects or values are considered equivalent if at least one of the following is true:
 *
 * * Both objects or values pass `===` comparison.
 * * Both objects or values are of the same type and all of their properties are equal by
 *   comparing them with `angular.equals`.
 * * Both values are NaN. (In JavaScript, NaN == NaN => false. But we consider two NaN as equal)
 * * Both values represent the same regular expression (In JavaScript,
 *   /abc/ == /abc/ => false. But we consider two regular expressions as equal when their textual
 *   representation matches).
 *
 * During a property comparison, properties of `function` type and properties with names
 * that begin with `$` are ignored.
 *
 * Scope and DOMWindow objects are being compared only by identify (`===`).
 *
 * @param {*} o1 Object or value to compare.
 * @param {*} o2 Object or value to compare.
 * @returns {boolean} True if arguments are equal.
 */
function equals(o1, o2) {
  if (o1 === o2) return true;
  if (o1 === null || o2 === null) return false;
  if (o1 !== o1 && o2 !== o2) return true; // NaN === NaN
  var t1 = typeof o1, t2 = typeof o2, length, key, keySet;
  if (t1 == t2) {
    if (t1 == 'object') {
      if (isArray(o1)) {
        if (!isArray(o2)) return false;
        if ((length = o1.length) == o2.length) {
          for (key = 0; key < length; key++) {
            if (!equals(o1[key], o2[key])) return false;
          }
          return true;
        }
      } else if (isDate(o1)) {
        if (!isDate(o2)) return false;
        return equals(o1.getTime(), o2.getTime());
      } else if (isRegExp(o1)) {
        return isRegExp(o2) ? o1.toString() == o2.toString() : false;
      } else {
        if (isScope(o1) || isScope(o2) || isWindow(o1) || isWindow(o2) ||
          isArray(o2) || isDate(o2) || isRegExp(o2)) return false;
        keySet = createMap();
        for (key in o1) {
          if (key.charAt(0) === '$' || isFunction(o1[key])) continue;
          if (!equals(o1[key], o2[key])) return false;
          keySet[key] = true;
        }
        for (key in o2) {
          if (!(key in keySet) &&
              key.charAt(0) !== '$' &&
              isDefined(o2[key]) &&
              !isFunction(o2[key])) return false;
        }
        return true;
      }
    }
  }
  return false;
}

var csp = function() {
  if (!isDefined(csp.rules)) {


    var ngCspElement = (document.querySelector('[ng-csp]') ||
                    document.querySelector('[data-ng-csp]'));

    if (ngCspElement) {
      var ngCspAttribute = ngCspElement.getAttribute('ng-csp') ||
                    ngCspElement.getAttribute('data-ng-csp');
      csp.rules = {
        noUnsafeEval: !ngCspAttribute || (ngCspAttribute.indexOf('no-unsafe-eval') !== -1),
        noInlineStyle: !ngCspAttribute || (ngCspAttribute.indexOf('no-inline-style') !== -1)
      };
    } else {
      csp.rules = {
        noUnsafeEval: noUnsafeEval(),
        noInlineStyle: false
      };
    }
  }

  return csp.rules;

  function noUnsafeEval() {
    try {
      /* jshint -W031, -W054 */
      new Function('');
      /* jshint +W031, +W054 */
      return false;
    } catch (e) {
      return true;
    }
  }
};

/**
 * @ngdoc directive
 * @module ng
 * @name ngJq
 *
 * @element ANY
 * @param {string=} ngJq the name of the library available under `window`
 * to be used for angular.element
 * @description
 * Use this directive to force the angular.element library.  This should be
 * used to force either jqLite by leaving ng-jq blank or setting the name of
 * the jquery variable under window (eg. jQuery).
 *
 * Since angular looks for this directive when it is loaded (doesn't wait for the
 * DOMContentLoaded event), it must be placed on an element that comes before the script
 * which loads angular. Also, only the first instance of `ng-jq` will be used and all
 * others ignored.
 *
 * @example
 * This example shows how to force jqLite using the `ngJq` directive to the `html` tag.
 ```html
 <!doctype html>
 <html ng-app ng-jq>
 ...
 ...
 </html>
 ```
 * @example
 * This example shows how to use a jQuery based library of a different name.
 * The library name must be available at the top most 'window'.
 ```html
 <!doctype html>
 <html ng-app ng-jq="jQueryLib">
 ...
 ...
 </html>
 ```
 */
var jq = function() {
  if (isDefined(jq.name_)) return jq.name_;
  var el;
  var i, ii = ngAttrPrefixes.length, prefix, name;
  for (i = 0; i < ii; ++i) {
    prefix = ngAttrPrefixes[i];
    if (el = document.querySelector('[' + prefix.replace(':', '\\:') + 'jq]')) {
      name = el.getAttribute(prefix + 'jq');
      break;
    }
  }

  return (jq.name_ = name);
};

function concat(array1, array2, index) {
  return array1.concat(slice.call(array2, index));
}

function sliceArgs(args, startIndex) {
  return slice.call(args, startIndex || 0);
}


/* jshint -W101 */
/**
 * @ngdoc function
 * @name angular.bind
 * @module ng
 * @kind function
 *
 * @description
 * Returns a function which calls function `fn` bound to `self` (`self` becomes the `this` for
 * `fn`). You can supply optional `args` that are prebound to the function. This feature is also
 * known as [partial application](http://en.wikipedia.org/wiki/Partial_application), as
 * distinguished from [function currying](http://en.wikipedia.org/wiki/Currying#Contrast_with_partial_function_application).
 *
 * @param {Object} self Context which `fn` should be evaluated in.
 * @param {function()} fn Function to be bound.
 * @param {...*} args Optional arguments to be prebound to the `fn` function call.
 * @returns {function()} Function that wraps the `fn` with all the specified bindings.
 */
/* jshint +W101 */
function bind(self, fn) {
  var curryArgs = arguments.length > 2 ? sliceArgs(arguments, 2) : [];
  if (isFunction(fn) && !(fn instanceof RegExp)) {
    return curryArgs.length
      ? function() {
          return arguments.length
            ? fn.apply(self, concat(curryArgs, arguments, 0))
            : fn.apply(self, curryArgs);
        }
      : function() {
          return arguments.length
            ? fn.apply(self, arguments)
            : fn.call(self);
        };
  } else {
    // in IE, native methods are not functions so they cannot be bound (note: they don't need to be)
    return fn;
  }
}


function toJsonReplacer(key, value) {
  var val = value;

  if (typeof key === 'string' && key.charAt(0) === '$' && key.charAt(1) === '$') {
    val = undefined;
  } else if (isWindow(value)) {
    val = '$WINDOW';
  } else if (value &&  document === value) {
    val = '$DOCUMENT';
  } else if (isScope(value)) {
    val = '$SCOPE';
  }

  return val;
}


/**
 * @ngdoc function
 * @name angular.toJson
 * @module ng
 * @kind function
 *
 * @description
 * Serializes input into a JSON-formatted string. Properties with leading $$ characters will be
 * stripped since angular uses this notation internally.
 *
 * @param {Object|Array|Date|string|number} obj Input to be serialized into JSON.
 * @param {boolean|number} [pretty=2] If set to true, the JSON output will contain newlines and whitespace.
 *    If set to an integer, the JSON output will contain that many spaces per indentation.
 * @returns {string|undefined} JSON-ified string representing `obj`.
 */
function toJson(obj, pretty) {
  if (typeof obj === 'undefined') return undefined;
  if (!isNumber(pretty)) {
    pretty = pretty ? 2 : null;
  }
  return JSON.stringify(obj, toJsonReplacer, pretty);
}


/**
 * @ngdoc function
 * @name angular.fromJson
 * @module ng
 * @kind function
 *
 * @description
 * Deserializes a JSON string.
 *
 * @param {string} json JSON string to deserialize.
 * @returns {Object|Array|string|number} Deserialized JSON string.
 */
function fromJson(json) {
  return isString(json)
      ? JSON.parse(json)
      : json;
}


function timezoneToOffset(timezone, fallback) {
  var requestedTimezoneOffset = Date.parse('Jan 01, 1970 00:00:00 ' + timezone) / 60000;
  return isNaN(requestedTimezoneOffset) ? fallback : requestedTimezoneOffset;
}


function addDateMinutes(date, minutes) {
  date = new Date(date.getTime());
  date.setMinutes(date.getMinutes() + minutes);
  return date;
}


function convertTimezoneToLocal(date, timezone, reverse) {
  reverse = reverse ? -1 : 1;
  var timezoneOffset = timezoneToOffset(timezone, date.getTimezoneOffset());
  return addDateMinutes(date, reverse * (timezoneOffset - date.getTimezoneOffset()));
}


/**
 * @returns {string} Returns the string representation of the element.
 */
function startingTag(element) {
  element = jqLite(element).clone();
  try {
    // turns out IE does not let you set .html() on elements which
    // are not allowed to have children. So we just ignore it.
    element.empty();
  } catch (e) {}
  var elemHtml = jqLite('<div>').append(element).html();
  try {
    return element[0].nodeType === NODE_TYPE_TEXT ? lowercase(elemHtml) :
        elemHtml.
          match(/^(<[^>]+>)/)[1].
          replace(/^<([\w\-]+)/, function(match, nodeName) { return '<' + lowercase(nodeName); });
  } catch (e) {
    return lowercase(elemHtml);
  }

}


/////////////////////////////////////////////////

/**
 * Tries to decode the URI component without throwing an exception.
 *
 * @private
 * @param str value potential URI component to check.
 * @returns {boolean} True if `value` can be decoded
 * with the decodeURIComponent function.
 */
function tryDecodeURIComponent(value) {
  try {
    return decodeURIComponent(value);
  } catch (e) {
    // Ignore any invalid uri component
  }
}


/**
 * Parses an escaped url query string into key-value pairs.
 * @returns {Object.<string,boolean|Array>}
 */
function parseKeyValue(/**string*/keyValue) {
  var obj = {};
  forEach((keyValue || "").split('&'), function(keyValue) {
    var splitPoint, key, val;
    if (keyValue) {
      key = keyValue = keyValue.replace(/\+/g,'%20');
      splitPoint = keyValue.indexOf('=');
      if (splitPoint !== -1) {
        key = keyValue.substring(0, splitPoint);
        val = keyValue.substring(splitPoint + 1);
      }
      key = tryDecodeURIComponent(key);
      if (isDefined(key)) {
        val = isDefined(val) ? tryDecodeURIComponent(val) : true;
        if (!hasOwnProperty.call(obj, key)) {
          obj[key] = val;
        } else if (isArray(obj[key])) {
          obj[key].push(val);
        } else {
          obj[key] = [obj[key],val];
        }
      }
    }
  });
  return obj;
}

function toKeyValue(obj) {
  var parts = [];
  forEach(obj, function(value, key) {
    if (isArray(value)) {
      forEach(value, function(arrayValue) {
        parts.push(encodeUriQuery(key, true) +
                   (arrayValue === true ? '' : '=' + encodeUriQuery(arrayValue, true)));
      });
    } else {
    parts.push(encodeUriQuery(key, true) +
               (value === true ? '' : '=' + encodeUriQuery(value, true)));
    }
  });
  return parts.length ? parts.join('&') : '';
}


/**
 * We need our custom method because encodeURIComponent is too aggressive and doesn't follow
 * http://www.ietf.org/rfc/rfc3986.txt with regards to the character set (pchar) allowed in path
 * segments:
 *    segment       = *pchar
 *    pchar         = unreserved / pct-encoded / sub-delims / ":" / "@"
 *    pct-encoded   = "%" HEXDIG HEXDIG
 *    unreserved    = ALPHA / DIGIT / "-" / "." / "_" / "~"
 *    sub-delims    = "!" / "$" / "&" / "'" / "(" / ")"
 *                     / "*" / "+" / "," / ";" / "="
 */
function encodeUriSegment(val) {
  return encodeUriQuery(val, true).
             replace(/%26/gi, '&').
             replace(/%3D/gi, '=').
             replace(/%2B/gi, '+');
}


/**
 * This method is intended for encoding *key* or *value* parts of query component. We need a custom
 * method because encodeURIComponent is too aggressive and encodes stuff that doesn't have to be
 * encoded per http://tools.ietf.org/html/rfc3986:
 *    query       = *( pchar / "/" / "?" )
 *    pchar         = unreserved / pct-encoded / sub-delims / ":" / "@"
 *    unreserved    = ALPHA / DIGIT / "-" / "." / "_" / "~"
 *    pct-encoded   = "%" HEXDIG HEXDIG
 *    sub-delims    = "!" / "$" / "&" / "'" / "(" / ")"
 *                     / "*" / "+" / "," / ";" / "="
 */
function encodeUriQuery(val, pctEncodeSpaces) {
  return encodeURIComponent(val).
             replace(/%40/gi, '@').
             replace(/%3A/gi, ':').
             replace(/%24/g, '$').
             replace(/%2C/gi, ',').
             replace(/%3B/gi, ';').
             replace(/%20/g, (pctEncodeSpaces ? '%20' : '+'));
}

var ngAttrPrefixes = ['ng-', 'data-ng-', 'ng:', 'x-ng-'];

function getNgAttribute(element, ngAttr) {
  var attr, i, ii = ngAttrPrefixes.length;
  for (i = 0; i < ii; ++i) {
    attr = ngAttrPrefixes[i] + ngAttr;
    if (isString(attr = element.getAttribute(attr))) {
      return attr;
    }
  }
  return null;
}

/**
 * @ngdoc directive
 * @name ngApp
 * @module ng
 *
 * @element ANY
 * @param {angular.Module} ngApp an optional application
 *   {@link angular.module module} name to load.
 * @param {boolean=} ngStrictDi if this attribute is present on the app element, the injector will be
 *   created in "strict-di" mode. This means that the application will fail to invoke functions which
 *   do not use explicit function annotation (and are thus unsuitable for minification), as described
 *   in {@link guide/di the Dependency Injection guide}, and useful debugging info will assist in
 *   tracking down the root of these bugs.
 *
 * @description
 *
 * Use this directive to **auto-bootstrap** an AngularJS application. The `ngApp` directive
 * designates the **root element** of the application and is typically placed near the root element
 * of the page - e.g. on the `<body>` or `<html>` tags.
 *
 * Only one AngularJS application can be auto-bootstrapped per HTML document. The first `ngApp`
 * found in the document will be used to define the root element to auto-bootstrap as an
 * application. To run multiple applications in an HTML document you must manually bootstrap them using
 * {@link angular.bootstrap} instead. AngularJS applications cannot be nested within each other.
 *
 * You can specify an **AngularJS module** to be used as the root module for the application.  This
 * module will be loaded into the {@link auto.$injector} when the application is bootstrapped. It
 * should contain the application code needed or have dependencies on other modules that will
 * contain the code. See {@link angular.module} for more information.
 *
 * In the example below if the `ngApp` directive were not placed on the `html` element then the
 * document would not be compiled, the `AppController` would not be instantiated and the `{{ a+b }}`
 * would not be resolved to `3`.
 *
 * `ngApp` is the easiest, and most common way to bootstrap an application.
 *
 <example module="ngAppDemo">
   <file name="index.html">
   <div ng-controller="ngAppDemoController">
     I can add: {{a}} + {{b}} =  {{ a+b }}
   </div>
   </file>
   <file name="script.js">
   angular.module('ngAppDemo', []).controller('ngAppDemoController', function($scope) {
     $scope.a = 1;
     $scope.b = 2;
   });
   </file>
 </example>
 *
 * Using `ngStrictDi`, you would see something like this:
 *
 <example ng-app-included="true">
   <file name="index.html">
   <div ng-app="ngAppStrictDemo" ng-strict-di>
       <div ng-controller="GoodController1">
           I can add: {{a}} + {{b}} =  {{ a+b }}

           <p>This renders because the controller does not fail to
              instantiate, by using explicit annotation style (see
              script.js for details)
           </p>
       </div>

       <div ng-controller="GoodController2">
           Name: <input ng-model="name"><br />
           Hello, {{name}}!

           <p>This renders because the controller does not fail to
              instantiate, by using explicit annotation style
              (see script.js for details)
           </p>
       </div>

       <div ng-controller="BadController">
           I can add: {{a}} + {{b}} =  {{ a+b }}

           <p>The controller could not be instantiated, due to relying
              on automatic function annotations (which are disabled in
              strict mode). As such, the content of this section is not
              interpolated, and there should be an error in your web console.
           </p>
       </div>
   </div>
   </file>
   <file name="script.js">
   angular.module('ngAppStrictDemo', [])
     // BadController will fail to instantiate, due to relying on automatic function annotation,
     // rather than an explicit annotation
     .controller('BadController', function($scope) {
       $scope.a = 1;
       $scope.b = 2;
     })
     // Unlike BadController, GoodController1 and GoodController2 will not fail to be instantiated,
     // due to using explicit annotations using the array style and $inject property, respectively.
     .controller('GoodController1', ['$scope', function($scope) {
       $scope.a = 1;
       $scope.b = 2;
     }])
     .controller('GoodController2', GoodController2);
     function GoodController2($scope) {
       $scope.name = "World";
     }
     GoodController2.$inject = ['$scope'];
   </file>
   <file name="style.css">
   div[ng-controller] {
       margin-bottom: 1em;
       -webkit-border-radius: 4px;
       border-radius: 4px;
       border: 1px solid;
       padding: .5em;
   }
   div[ng-controller^=Good] {
       border-color: #d6e9c6;
       background-color: #dff0d8;
       color: #3c763d;
   }
   div[ng-controller^=Bad] {
       border-color: #ebccd1;
       background-color: #f2dede;
       color: #a94442;
       margin-bottom: 0;
   }
   </file>
 </example>
 */
function angularInit(element, bootstrap) {
  var appElement,
      module,
      config = {};

  // The element `element` has priority over any other element
  forEach(ngAttrPrefixes, function(prefix) {
    var name = prefix + 'app';

    if (!appElement && element.hasAttribute && element.hasAttribute(name)) {
      appElement = element;
      module = element.getAttribute(name);
    }
  });
  forEach(ngAttrPrefixes, function(prefix) {
    var name = prefix + 'app';
    var candidate;

    if (!appElement && (candidate = element.querySelector('[' + name.replace(':', '\\:') + ']'))) {
      appElement = candidate;
      module = candidate.getAttribute(name);
    }
  });
  if (appElement) {
    config.strictDi = getNgAttribute(appElement, "strict-di") !== null;
    bootstrap(appElement, module ? [module] : [], config);
  }
}

/**
 * @ngdoc function
 * @name angular.bootstrap
 * @module ng
 * @description
 * Use this function to manually start up angular application.
 *
 * See: {@link guide/bootstrap Bootstrap}
 *
 * Note that Protractor based end-to-end tests cannot use this function to bootstrap manually.
 * They must use {@link ng.directive:ngApp ngApp}.
 *
 * Angular will detect if it has been loaded into the browser more than once and only allow the
 * first loaded script to be bootstrapped and will report a warning to the browser console for
 * each of the subsequent scripts. This prevents strange results in applications, where otherwise
 * multiple instances of Angular try to work on the DOM.
 *
 * ```html
 * <!doctype html>
 * <html>
 * <body>
 * <div ng-controller="WelcomeController">
 *   {{greeting}}
 * </div>
 *
 * <script src="angular.js"></script>
 * <script>
 *   var app = angular.module('demo', [])
 *   .controller('WelcomeController', function($scope) {
 *       $scope.greeting = 'Welcome!';
 *   });
 *   angular.bootstrap(document, ['demo']);
 * </script>
 * </body>
 * </html>
 * ```
 *
 * @param {DOMElement} element DOM element which is the root of angular application.
 * @param {Array<String|Function|Array>=} modules an array of modules to load into the application.
 *     Each item in the array should be the name of a predefined module or a (DI annotated)
 *     function that will be invoked by the injector as a `config` block.
 *     See: {@link angular.module modules}
 * @param {Object=} config an object for defining configuration options for the application. The
 *     following keys are supported:
 *
 * * `strictDi` - disable automatic function annotation for the application. This is meant to
 *   assist in finding bugs which break minified code. Defaults to `false`.
 *
 * @returns {auto.$injector} Returns the newly created injector for this app.
 */
function bootstrap(element, modules, config) {
  if (!isObject(config)) config = {};
  var defaultConfig = {
    strictDi: false
  };
  config = extend(defaultConfig, config);
  var doBootstrap = function() {
    element = jqLite(element);

    if (element.injector()) {
      var tag = (element[0] === document) ? 'document' : startingTag(element);
      //Encode angle brackets to prevent input from being sanitized to empty string #8683
      throw ngMinErr(
          'btstrpd',
          "App Already Bootstrapped with this Element '{0}'",
          tag.replace(/</,'&lt;').replace(/>/,'&gt;'));
    }

    modules = modules || [];
    modules.unshift(['$provide', function($provide) {
      $provide.value('$rootElement', element);
    }]);

    if (config.debugInfoEnabled) {
      // Pushing so that this overrides `debugInfoEnabled` setting defined in user's `modules`.
      modules.push(['$compileProvider', function($compileProvider) {
        $compileProvider.debugInfoEnabled(true);
      }]);
    }

    modules.unshift('ng');
    var injector = createInjector(modules, config.strictDi);
    injector.invoke(['$rootScope', '$rootElement', '$compile', '$injector',
       function bootstrapApply(scope, element, compile, injector) {
        scope.$apply(function() {
          element.data('$injector', injector);
          compile(element)(scope);
        });
      }]
    );
    return injector;
  };

  var NG_ENABLE_DEBUG_INFO = /^NG_ENABLE_DEBUG_INFO!/;
  var NG_DEFER_BOOTSTRAP = /^NG_DEFER_BOOTSTRAP!/;

  if (window && NG_ENABLE_DEBUG_INFO.test(window.name)) {
    config.debugInfoEnabled = true;
    window.name = window.name.replace(NG_ENABLE_DEBUG_INFO, '');
  }

  if (window && !NG_DEFER_BOOTSTRAP.test(window.name)) {
    return doBootstrap();
  }

  window.name = window.name.replace(NG_DEFER_BOOTSTRAP, '');
  angular.resumeBootstrap = function(extraModules) {
    forEach(extraModules, function(module) {
      modules.push(module);
    });
    return doBootstrap();
  };

  if (isFunction(angular.resumeDeferredBootstrap)) {
    angular.resumeDeferredBootstrap();
  }
}

/**
 * @ngdoc function
 * @name angular.reloadWithDebugInfo
 * @module ng
 * @description
 * Use this function to reload the current application with debug information turned on.
 * This takes precedence over a call to `$compileProvider.debugInfoEnabled(false)`.
 *
 * See {@link ng.$compileProvider#debugInfoEnabled} for more.
 */
function reloadWithDebugInfo() {
  window.name = 'NG_ENABLE_DEBUG_INFO!' + window.name;
  window.location.reload();
}

/**
 * @name angular.getTestability
 * @module ng
 * @description
 * Get the testability service for the instance of Angular on the given
 * element.
 * @param {DOMElement} element DOM element which is the root of angular application.
 */
function getTestability(rootElement) {
  var injector = angular.element(rootElement).injector();
  if (!injector) {
    throw ngMinErr('test',
      'no injector found for element argument to getTestability');
  }
  return injector.get('$$testability');
}

var SNAKE_CASE_REGEXP = /[A-Z]/g;
function snake_case(name, separator) {
  separator = separator || '_';
  return name.replace(SNAKE_CASE_REGEXP, function(letter, pos) {
    return (pos ? separator : '') + letter.toLowerCase();
  });
}

var bindJQueryFired = false;
var skipDestroyOnNextJQueryCleanData;
function bindJQuery() {
  var originalCleanData;

  if (bindJQueryFired) {
    return;
  }

  // bind to jQuery if present;
  var jqName = jq();
  jQuery = isUndefined(jqName) ? window.jQuery :   // use jQuery (if present)
           !jqName             ? undefined     :   // use jqLite
                                 window[jqName];   // use jQuery specified by `ngJq`

  // Use jQuery if it exists with proper functionality, otherwise default to us.
  // Angular 1.2+ requires jQuery 1.7+ for on()/off() support.
  // Angular 1.3+ technically requires at least jQuery 2.1+ but it may work with older
  // versions. It will not work for sure with jQuery <1.7, though.
  if (jQuery && jQuery.fn.on) {
    jqLite = jQuery;
    extend(jQuery.fn, {
      scope: JQLitePrototype.scope,
      isolateScope: JQLitePrototype.isolateScope,
      controller: JQLitePrototype.controller,
      injector: JQLitePrototype.injector,
      inheritedData: JQLitePrototype.inheritedData
    });

    // All nodes removed from the DOM via various jQuery APIs like .remove()
    // are passed through jQuery.cleanData. Monkey-patch this method to fire
    // the $destroy event on all removed nodes.
    originalCleanData = jQuery.cleanData;
    jQuery.cleanData = function(elems) {
      var events;
      if (!skipDestroyOnNextJQueryCleanData) {
        for (var i = 0, elem; (elem = elems[i]) != null; i++) {
          events = jQuery._data(elem, "events");
          if (events && events.$destroy) {
            jQuery(elem).triggerHandler('$destroy');
          }
        }
      } else {
        skipDestroyOnNextJQueryCleanData = false;
      }
      originalCleanData(elems);
    };
  } else {
    jqLite = JQLite;
  }

  angular.element = jqLite;

  // Prevent double-proxying.
  bindJQueryFired = true;
}

/**
 * throw error if the argument is falsy.
 */
function assertArg(arg, name, reason) {
  if (!arg) {
    throw ngMinErr('areq', "Argument '{0}' is {1}", (name || '?'), (reason || "required"));
  }
  return arg;
}

function assertArgFn(arg, name, acceptArrayAnnotation) {
  if (acceptArrayAnnotation && isArray(arg)) {
      arg = arg[arg.length - 1];
  }

  assertArg(isFunction(arg), name, 'not a function, got ' +
      (arg && typeof arg === 'object' ? arg.constructor.name || 'Object' : typeof arg));
  return arg;
}

/**
 * throw error if the name given is hasOwnProperty
 * @param  {String} name    the name to test
 * @param  {String} context the context in which the name is used, such as module or directive
 */
function assertNotHasOwnProperty(name, context) {
  if (name === 'hasOwnProperty') {
    throw ngMinErr('badname', "hasOwnProperty is not a valid {0} name", context);
  }
}

/**
 * Return the value accessible from the object by path. Any undefined traversals are ignored
 * @param {Object} obj starting object
 * @param {String} path path to traverse
 * @param {boolean} [bindFnToScope=true]
 * @returns {Object} value as accessible by path
 */
//TODO(misko): this function needs to be removed
function getter(obj, path, bindFnToScope) {
  if (!path) return obj;
  var keys = path.split('.');
  var key;
  var lastInstance = obj;
  var len = keys.length;

  for (var i = 0; i < len; i++) {
    key = keys[i];
    if (obj) {
      obj = (lastInstance = obj)[key];
    }
  }
  if (!bindFnToScope && isFunction(obj)) {
    return bind(lastInstance, obj);
  }
  return obj;
}

/**
 * Return the DOM siblings between the first and last node in the given array.
 * @param {Array} array like object
 * @returns {Array} the inputted object or a jqLite collection containing the nodes
 */
function getBlockNodes(nodes) {
  // TODO(perf): update `nodes` instead of creating a new object?
  var node = nodes[0];
  var endNode = nodes[nodes.length - 1];
  var blockNodes;

  for (var i = 1; node !== endNode && (node = node.nextSibling); i++) {
    if (blockNodes || nodes[i] !== node) {
      if (!blockNodes) {
        blockNodes = jqLite(slice.call(nodes, 0, i));
      }
      blockNodes.push(node);
    }
  }

  return blockNodes || nodes;
}


/**
 * Creates a new object without a prototype. This object is useful for lookup without having to
 * guard against prototypically inherited properties via hasOwnProperty.
 *
 * Related micro-benchmarks:
 * - http://jsperf.com/object-create2
 * - http://jsperf.com/proto-map-lookup/2
 * - http://jsperf.com/for-in-vs-object-keys2
 *
 * @returns {Object}
 */
function createMap() {
  return Object.create(null);
}

var NODE_TYPE_ELEMENT = 1;
var NODE_TYPE_ATTRIBUTE = 2;
var NODE_TYPE_TEXT = 3;
var NODE_TYPE_COMMENT = 8;
var NODE_TYPE_DOCUMENT = 9;
var NODE_TYPE_DOCUMENT_FRAGMENT = 11;

/**
 * @ngdoc type
 * @name angular.Module
 * @module ng
 * @description
 *
 * Interface for configuring angular {@link angular.module modules}.
 */

function setupModuleLoader(window) {

  var $injectorMinErr = minErr('$injector');
  var ngMinErr = minErr('ng');

  function ensure(obj, name, factory) {
    return obj[name] || (obj[name] = factory());
  }

  var angular = ensure(window, 'angular', Object);

  // We need to expose `angular.$$minErr` to modules such as `ngResource` that reference it during bootstrap
  angular.$$minErr = angular.$$minErr || minErr;

  return ensure(angular, 'module', function() {
    /** @type {Object.<string, angular.Module>} */
    var modules = {};

    /**
     * @ngdoc function
     * @name angular.module
     * @module ng
     * @description
     *
     * The `angular.module` is a global place for creating, registering and retrieving Angular
     * modules.
     * All modules (angular core or 3rd party) that should be available to an application must be
     * registered using this mechanism.
     *
     * Passing one argument retrieves an existing {@link angular.Module},
     * whereas passing more than one argument creates a new {@link angular.Module}
     *
     *
     * # Module
     *
     * A module is a collection of services, directives, controllers, filters, and configuration information.
     * `angular.module` is used to configure the {@link auto.$injector $injector}.
     *
     * ```js
     * // Create a new module
     * var myModule = angular.module('myModule', []);
     *
     * // register a new service
     * myModule.value('appName', 'MyCoolApp');
     *
     * // configure existing services inside initialization blocks.
     * myModule.config(['$locationProvider', function($locationProvider) {
     *   // Configure existing providers
     *   $locationProvider.hashPrefix('!');
     * }]);
     * ```
     *
     * Then you can create an injector and load your modules like this:
     *
     * ```js
     * var injector = angular.injector(['ng', 'myModule'])
     * ```
     *
     * However it's more likely that you'll just use
     * {@link ng.directive:ngApp ngApp} or
     * {@link angular.bootstrap} to simplify this process for you.
     *
     * @param {!string} name The name of the module to create or retrieve.
     * @param {!Array.<string>=} requires If specified then new module is being created. If
     *        unspecified then the module is being retrieved for further configuration.
     * @param {Function=} configFn Optional configuration function for the module. Same as
     *        {@link angular.Module#config Module#config()}.
     * @returns {module} new module with the {@link angular.Module} api.
     */
    return function module(name, requires, configFn) {
      var assertNotHasOwnProperty = function(name, context) {
        if (name === 'hasOwnProperty') {
          throw ngMinErr('badname', 'hasOwnProperty is not a valid {0} name', context);
        }
      };

      assertNotHasOwnProperty(name, 'module');
      if (requires && modules.hasOwnProperty(name)) {
        modules[name] = null;
      }
      return ensure(modules, name, function() {
        if (!requires) {
          throw $injectorMinErr('nomod', "Module '{0}' is not available! You either misspelled " +
             "the module name or forgot to load it. If registering a module ensure that you " +
             "specify the dependencies as the second argument.", name);
        }

        /** @type {!Array.<Array.<*>>} */
        var invokeQueue = [];

        /** @type {!Array.<Function>} */
        var configBlocks = [];

        /** @type {!Array.<Function>} */
        var runBlocks = [];

        var config = invokeLater('$injector', 'invoke', 'push', configBlocks);

        /** @type {angular.Module} */
        var moduleInstance = {
          // Private state
          _invokeQueue: invokeQueue,
          _configBlocks: configBlocks,
          _runBlocks: runBlocks,

          /**
           * @ngdoc property
           * @name angular.Module#requires
           * @module ng
           *
           * @description
           * Holds the list of modules which the injector will load before the current module is
           * loaded.
           */
          requires: requires,

          /**
           * @ngdoc property
           * @name angular.Module#name
           * @module ng
           *
           * @description
           * Name of the module.
           */
          name: name,


          /**
           * @ngdoc method
           * @name angular.Module#provider
           * @module ng
           * @param {string} name service name
           * @param {Function} providerType Construction function for creating new instance of the
           *                                service.
           * @description
           * See {@link auto.$provide#provider $provide.provider()}.
           */
          provider: invokeLaterAndSetModuleName('$provide', 'provider'),

          /**
           * @ngdoc method
           * @name angular.Module#factory
           * @module ng
           * @param {string} name service name
           * @param {Function} providerFunction Function for creating new instance of the service.
           * @description
           * See {@link auto.$provide#factory $provide.factory()}.
           */
          factory: invokeLaterAndSetModuleName('$provide', 'factory'),

          /**
           * @ngdoc method
           * @name angular.Module#service
           * @module ng
           * @param {string} name service name
           * @param {Function} constructor A constructor function that will be instantiated.
           * @description
           * See {@link auto.$provide#service $provide.service()}.
           */
          service: invokeLaterAndSetModuleName('$provide', 'service'),

          /**
           * @ngdoc method
           * @name angular.Module#value
           * @module ng
           * @param {string} name service name
           * @param {*} object Service instance object.
           * @description
           * See {@link auto.$provide#value $provide.value()}.
           */
          value: invokeLater('$provide', 'value'),

          /**
           * @ngdoc method
           * @name angular.Module#constant
           * @module ng
           * @param {string} name constant name
           * @param {*} object Constant value.
           * @description
           * Because the constants are fixed, they get applied before other provide methods.
           * See {@link auto.$provide#constant $provide.constant()}.
           */
          constant: invokeLater('$provide', 'constant', 'unshift'),

           /**
           * @ngdoc method
           * @name angular.Module#decorator
           * @module ng
           * @param {string} The name of the service to decorate.
           * @param {Function} This function will be invoked when the service needs to be
           *                                    instantiated and should return the decorated service instance.
           * @description
           * See {@link auto.$provide#decorator $provide.decorator()}.
           */
          decorator: invokeLaterAndSetModuleName('$provide', 'decorator'),

          /**
           * @ngdoc method
           * @name angular.Module#animation
           * @module ng
           * @param {string} name animation name
           * @param {Function} animationFactory Factory function for creating new instance of an
           *                                    animation.
           * @description
           *
           * **NOTE**: animations take effect only if the **ngAnimate** module is loaded.
           *
           *
           * Defines an animation hook that can be later used with
           * {@link $animate $animate} service and directives that use this service.
           *
           * ```js
           * module.animation('.animation-name', function($inject1, $inject2) {
           *   return {
           *     eventName : function(element, done) {
           *       //code to run the animation
           *       //once complete, then run done()
           *       return function cancellationFunction(element) {
           *         //code to cancel the animation
           *       }
           *     }
           *   }
           * })
           * ```
           *
           * See {@link ng.$animateProvider#register $animateProvider.register()} and
           * {@link ngAnimate ngAnimate module} for more information.
           */
          animation: invokeLaterAndSetModuleName('$animateProvider', 'register'),

          /**
           * @ngdoc method
           * @name angular.Module#filter
           * @module ng
           * @param {string} name Filter name - this must be a valid angular expression identifier
           * @param {Function} filterFactory Factory function for creating new instance of filter.
           * @description
           * See {@link ng.$filterProvider#register $filterProvider.register()}.
           *
           * <div class="alert alert-warning">
           * **Note:** Filter names must be valid angular {@link expression} identifiers, such as `uppercase` or `orderBy`.
           * Names with special characters, such as hyphens and dots, are not allowed. If you wish to namespace
           * your filters, then you can use capitalization (`myappSubsectionFilterx`) or underscores
           * (`myapp_subsection_filterx`).
           * </div>
           */
          filter: invokeLaterAndSetModuleName('$filterProvider', 'register'),

          /**
           * @ngdoc method
           * @name angular.Module#controller
           * @module ng
           * @param {string|Object} name Controller name, or an object map of controllers where the
           *    keys are the names and the values are the constructors.
           * @param {Function} constructor Controller constructor function.
           * @description
           * See {@link ng.$controllerProvider#register $controllerProvider.register()}.
           */
          controller: invokeLaterAndSetModuleName('$controllerProvider', 'register'),

          /**
           * @ngdoc method
           * @name angular.Module#directive
           * @module ng
           * @param {string|Object} name Directive name, or an object map of directives where the
           *    keys are the names and the values are the factories.
           * @param {Function} directiveFactory Factory function for creating new instance of
           * directives.
           * @description
           * See {@link ng.$compileProvider#directive $compileProvider.directive()}.
           */
          directive: invokeLaterAndSetModuleName('$compileProvider', 'directive'),

          /**
           * @ngdoc method
           * @name angular.Module#config
           * @module ng
           * @param {Function} configFn Execute this function on module load. Useful for service
           *    configuration.
           * @description
           * Use this method to register work which needs to be performed on module loading.
           * For more about how to configure services, see
           * {@link providers#provider-recipe Provider Recipe}.
           */
          config: config,

          /**
           * @ngdoc method
           * @name angular.Module#run
           * @module ng
           * @param {Function} initializationFn Execute this function after injector creation.
           *    Useful for application initialization.
           * @description
           * Use this method to register work which should be performed when the injector is done
           * loading all modules.
           */
          run: function(block) {
            runBlocks.push(block);
            return this;
          }
        };

        if (configFn) {
          config(configFn);
        }

        return moduleInstance;

        /**
         * @param {string} provider
         * @param {string} method
         * @param {String=} insertMethod
         * @returns {angular.Module}
         */
        function invokeLater(provider, method, insertMethod, queue) {
          if (!queue) queue = invokeQueue;
          return function() {
            queue[insertMethod || 'push']([provider, method, arguments]);
            return moduleInstance;
          };
        }

        /**
         * @param {string} provider
         * @param {string} method
         * @returns {angular.Module}
         */
        function invokeLaterAndSetModuleName(provider, method) {
          return function(recipeName, factoryFunction) {
            if (factoryFunction && isFunction(factoryFunction)) factoryFunction.$$moduleName = name;
            invokeQueue.push([provider, method, arguments]);
            return moduleInstance;
          };
        }
      });
    };
  });

}

/* global: toDebugString: true */

function serializeObject(obj) {
  var seen = [];

  return JSON.stringify(obj, function(key, val) {
    val = toJsonReplacer(key, val);
    if (isObject(val)) {

      if (seen.indexOf(val) >= 0) return '...';

      seen.push(val);
    }
    return val;
  });
}

function toDebugString(obj) {
  if (typeof obj === 'function') {
    return obj.toString().replace(/ \{[\s\S]*$/, '');
  } else if (isUndefined(obj)) {
    return 'undefined';
  } else if (typeof obj !== 'string') {
    return serializeObject(obj);
  }
  return obj;
}

/* global angularModule: true,
  version: true,

  $CompileProvider,

  htmlAnchorDirective,
  inputDirective,
  inputDirective,
  formDirective,
  scriptDirective,
  selectDirective,
  styleDirective,
  optionDirective,
  ngBindDirective,
  ngBindHtmlDirective,
  ngBindTemplateDirective,
  ngClassDirective,
  ngClassEvenDirective,
  ngClassOddDirective,
  ngCloakDirective,
  ngControllerDirective,
  ngFormDirective,
  ngHideDirective,
  ngIfDirective,
  ngIncludeDirective,
  ngIncludeFillContentDirective,
  ngInitDirective,
  ngNonBindableDirective,
  ngPluralizeDirective,
  ngRepeatDirective,
  ngShowDirective,
  ngStyleDirective,
  ngSwitchDirective,
  ngSwitchWhenDirective,
  ngSwitchDefaultDirective,
  ngOptionsDirective,
  ngTranscludeDirective,
  ngModelDirective,
  ngListDirective,
  ngChangeDirective,
  patternDirective,
  patternDirective,
  requiredDirective,
  requiredDirective,
  minlengthDirective,
  minlengthDirective,
  maxlengthDirective,
  maxlengthDirective,
  ngValueDirective,
  ngModelOptionsDirective,
  ngAttributeAliasDirectives,
  ngEventDirectives,

  $AnchorScrollProvider,
  $AnimateProvider,
  $CoreAnimateCssProvider,
  $$CoreAnimateQueueProvider,
  $$CoreAnimateRunnerProvider,
  $BrowserProvider,
  $CacheFactoryProvider,
  $ControllerProvider,
  $DocumentProvider,
  $ExceptionHandlerProvider,
  $FilterProvider,
  $$ForceReflowProvider,
  $InterpolateProvider,
  $IntervalProvider,
  $$HashMapProvider,
  $HttpProvider,
  $HttpParamSerializerProvider,
  $HttpParamSerializerJQLikeProvider,
  $HttpBackendProvider,
  $xhrFactoryProvider,
  $LocationProvider,
  $LogProvider,
  $ParseProvider,
  $RootScopeProvider,
  $QProvider,
  $$QProvider,
  $$SanitizeUriProvider,
  $SceProvider,
  $SceDelegateProvider,
  $SnifferProvider,
  $TemplateCacheProvider,
  $TemplateRequestProvider,
  $$TestabilityProvider,
  $TimeoutProvider,
  $$RAFProvider,
  $WindowProvider,
  $$jqLiteProvider,
  $$CookieReaderProvider
*/


/**
 * @ngdoc object
 * @name angular.version
 * @module ng
 * @description
 * An object that contains information about the current AngularJS version.
 *
 * This object has the following properties:
 *
 * - `full` â€“ `{string}` â€“ Full version string, such as "0.9.18".
 * - `major` â€“ `{number}` â€“ Major version number, such as "0".
 * - `minor` â€“ `{number}` â€“ Minor version number, such as "9".
 * - `dot` â€“ `{number}` â€“ Dot version number, such as "18".
 * - `codeName` â€“ `{string}` â€“ Code name of the release, such as "jiggling-armfat".
 */
var version = {
  full: '1.4.8',    // all of these placeholder strings will be replaced by grunt's
  major: 1,    // package task
  minor: 4,
  dot: 8,
  codeName: 'ice-manipulation'
};


function publishExternalAPI(angular) {
  extend(angular, {
    'bootstrap': bootstrap,
    'copy': copy,
    'extend': extend,
    'merge': merge,
    'equals': equals,
    'element': jqLite,
    'forEach': forEach,
    'injector': createInjector,
    'noop': noop,
    'bind': bind,
    'toJson': toJson,
    'fromJson': fromJson,
    'identity': identity,
    'isUndefined': isUndefined,
    'isDefined': isDefined,
    'isString': isString,
    'isFunction': isFunction,
    'isObject': isObject,
    'isNumber': isNumber,
    'isElement': isElement,
    'isArray': isArray,
    'version': version,
    'isDate': isDate,
    'lowercase': lowercase,
    'uppercase': uppercase,
    'callbacks': {counter: 0},
    'getTestability': getTestability,
    '$$minErr': minErr,
    '$$csp': csp,
    'reloadWithDebugInfo': reloadWithDebugInfo
  });

  angularModule = setupModuleLoader(window);

  angularModule('ng', ['ngLocale'], ['$provide',
    function ngModule($provide) {
      // $$sanitizeUriProvider needs to be before $compileProvider as it is used by it.
      $provide.provider({
        $$sanitizeUri: $$SanitizeUriProvider
      });
      $provide.provider('$compile', $CompileProvider).
        directive({
            a: htmlAnchorDirective,
            input: inputDirective,
            textarea: inputDirective,
            form: formDirective,
            script: scriptDirective,
            select: selectDirective,
            style: styleDirective,
            option: optionDirective,
            ngBind: ngBindDirective,
            ngBindHtml: ngBindHtmlDirective,
            ngBindTemplate: ngBindTemplateDirective,
            ngClass: ngClassDirective,
            ngClassEven: ngClassEvenDirective,
            ngClassOdd: ngClassOddDirective,
            ngCloak: ngCloakDirective,
            ngController: ngControllerDirective,
            ngForm: ngFormDirective,
            ngHide: ngHideDirective,
            ngIf: ngIfDirective,
            ngInclude: ngIncludeDirective,
            ngInit: ngInitDirective,
            ngNonBindable: ngNonBindableDirective,
            ngPluralize: ngPluralizeDirective,
            ngRepeat: ngRepeatDirective,
            ngShow: ngShowDirective,
            ngStyle: ngStyleDirective,
            ngSwitch: ngSwitchDirective,
            ngSwitchWhen: ngSwitchWhenDirective,
            ngSwitchDefault: ngSwitchDefaultDirective,
            ngOptions: ngOptionsDirective,
            ngTransclude: ngTranscludeDirective,
            ngModel: ngModelDirective,
            ngList: ngListDirective,
            ngChange: ngChangeDirective,
            pattern: patternDirective,
            ngPattern: patternDirective,
            required: requiredDirective,
            ngRequired: requiredDirective,
            minlength: minlengthDirective,
            ngMinlength: minlengthDirective,
            maxlength: maxlengthDirective,
            ngMaxlength: maxlengthDirective,
            ngValue: ngValueDirective,
            ngModelOptions: ngModelOptionsDirective
        }).
        directive({
          ngInclude: ngIncludeFillContentDirective
        }).
        directive(ngAttributeAliasDirectives).
        directive(ngEventDirectives);
      $provide.provider({
        $anchorScroll: $AnchorScrollProvider,
        $animate: $AnimateProvider,
        $animateCss: $CoreAnimateCssProvider,
        $$animateQueue: $$CoreAnimateQueueProvider,
        $$AnimateRunner: $$CoreAnimateRunnerProvider,
        $browser: $BrowserProvider,
        $cacheFactory: $CacheFactoryProvider,
        $controller: $ControllerProvider,
        $document: $DocumentProvider,
        $exceptionHandler: $ExceptionHandlerProvider,
        $filter: $FilterProvider,
        $$forceReflow: $$ForceReflowProvider,
        $interpolate: $InterpolateProvider,
        $interval: $IntervalProvider,
        $http: $HttpProvider,
        $httpParamSerializer: $HttpParamSerializerProvider,
        $httpParamSerializerJQLike: $HttpParamSerializerJQLikeProvider,
        $httpBackend: $HttpBackendProvider,
        $xhrFactory: $xhrFactoryProvider,
        $location: $LocationProvider,
        $log: $LogProvider,
        $parse: $ParseProvider,
        $rootScope: $RootScopeProvider,
        $q: $QProvider,
        $$q: $$QProvider,
        $sce: $SceProvider,
        $sceDelegate: $SceDelegateProvider,
        $sniffer: $SnifferProvider,
        $templateCache: $TemplateCacheProvider,
        $templateRequest: $TemplateRequestProvider,
        $$testability: $$TestabilityProvider,
        $timeout: $TimeoutProvider,
        $window: $WindowProvider,
        $$rAF: $$RAFProvider,
        $$jqLite: $$jqLiteProvider,
        $$HashMap: $$HashMapProvider,
        $$cookieReader: $$CookieReaderProvider
      });
    }
  ]);
}

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
 *     Any commits to this file should be reviewed with security in mind.  *
 *   Changes to this file can potentially create security vulnerabilities. *
 *          An approval from 2 Core members with history of modifying      *
 *                         this file is required.                          *
 *                                                                         *
 *  Does the change somehow allow for arbitrary javascript to be executed? *
 *    Or allows for someone to change the prototype of built-in objects?   *
 *     Or gives undesired access to variables likes document or window?    *
 * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */

/* global JQLitePrototype: true,
  addEventListenerFn: true,
  removeEventListenerFn: true,
  BOOLEAN_ATTR: true,
  ALIASED_ATTR: true,
*/

//////////////////////////////////
//JQLite
//////////////////////////////////

/**
 * @ngdoc function
 * @name angular.element
 * @module ng
 * @kind function
 *
 * @description
 * Wraps a raw DOM element or HTML string as a [jQuery](http://jquery.com) element.
 *
 * If jQuery is available, `angular.element` is an alias for the
 * [jQuery](http://api.jquery.com/jQuery/) function. If jQuery is not available, `angular.element`
 * delegates to Angular's built-in subset of jQuery, called "jQuery lite" or "jqLite."
 *
 * <div class="alert alert-success">jqLite is a tiny, API-compatible subset of jQuery that allows
 * Angular to manipulate the DOM in a cross-browser compatible way. **jqLite** implements only the most
 * commonly needed functionality with the goal of having a very small footprint.</div>
 *
 * To use `jQuery`, simply ensure it is loaded before the `angular.js` file.
 *
 * <div class="alert">**Note:** all element references in Angular are always wrapped with jQuery or
 * jqLite; they are never raw DOM references.</div>
 *
 * ## Angular's jqLite
 * jqLite provides only the following jQuery methods:
 *
 * - [`addClass()`](http://api.jquery.com/addClass/)
 * - [`after()`](http://api.jquery.com/after/)
 * - [`append()`](http://api.jquery.com/append/)
 * - [`attr()`](http://api.jquery.com/attr/) - Does not support functions as parameters
 * - [`bind()`](http://api.jquery.com/bind/) - Does not support namespaces, selectors or eventData
 * - [`children()`](http://api.jquery.com/children/) - Does not support selectors
 * - [`clone()`](http://api.jquery.com/clone/)
 * - [`contents()`](http://api.jquery.com/contents/)
 * - [`css()`](http://api.jquery.com/css/) - Only retrieves inline-styles, does not call `getComputedStyle()`. As a setter, does not convert numbers to strings or append 'px'.
 * - [`data()`](http://api.jquery.com/data/)
 * - [`detach()`](http://api.jquery.com/detach/)
 * - [`empty()`](http://api.jquery.com/empty/)
 * - [`eq()`](http://api.jquery.com/eq/)
 * - [`find()`](http://api.jquery.com/find/) - Limited to lookups by tag name
 * - [`hasClass()`](http://api.jquery.com/hasClass/)
 * - [`html()`](http://api.jquery.com/html/)
 * - [`next()`](http://api.jquery.com/next/) - Does not support selectors
 * - [`on()`](http://api.jquery.com/on/) - Does not support namespaces, selectors or eventData
 * - [`off()`](http://api.jquery.com/off/) - Does not support namespaces, selectors or event object as parameter
 * - [`one()`](http://api.jquery.com/one/) - Does not support namespaces or selectors
 * - [`parent()`](http://api.jquery.com/parent/) - Does not support selectors
 * - [`prepend()`](http://api.jquery.com/prepend/)
 * - [`prop()`](http://api.jquery.com/prop/)
 * - [`ready()`](http://api.jquery.com/ready/)
 * - [`remove()`](http://api.jquery.com/remove/)
 * - [`removeAttr()`](http://api.jquery.com/removeAttr/)
 * - [`removeClass()`](http://api.jquery.com/removeClass/)
 * - [`removeData()`](http://api.jquery.com/removeData/)
 * - [`replaceWith()`](http://api.jquery.com/replaceWith/)
 * - [`text()`](http://api.jquery.com/text/)
 * - [`toggleClass()`](http://api.jquery.com/toggleClass/)
 * - [`triggerHandler()`](http://api.jquery.com/triggerHandler/) - Passes a dummy event object to handlers.
 * - [`unbind()`](http://api.jquery.com/unbind/) - Does not support namespaces or event object as parameter
 * - [`val()`](http://api.jquery.com/val/)
 * - [`wrap()`](http://api.jquery.com/wrap/)
 *
 * ## jQuery/jqLite Extras
 * Angular also provides the following additional methods and events to both jQuery and jqLite:
 *
 * ### Events
 * - `$destroy` - AngularJS intercepts all jqLite/jQuery's DOM destruction apis and fires this event
 *    on all DOM nodes being removed.  This can be used to clean up any 3rd party bindings to the DOM
 *    element before it is removed.
 *
 * ### Methods
 * - `controller(name)` - retrieves the controller of the current element or its parent. By default
 *   retrieves controller associated with the `ngController` directive. If `name` is provided as
 *   camelCase directive name, then the controller for this directive will be retrieved (e.g.
 *   `'ngModel'`).
 * - `injector()` - retrieves the injector of the current element or its parent.
 * - `scope()` - retrieves the {@link ng.$rootScope.Scope scope} of the current
 *   element or its parent. Requires {@link guide/production#disabling-debug-data Debug Data} to
 *   be enabled.
 * - `isolateScope()` - retrieves an isolate {@link ng.$rootScope.Scope scope} if one is attached directly to the
 *   current element. This getter should be used only on elements that contain a directive which starts a new isolate
 *   scope. Calling `scope()` on this element always returns the original non-isolate scope.
 *   Requires {@link guide/production#disabling-debug-data Debug Data} to be enabled.
 * - `inheritedData()` - same as `data()`, but walks up the DOM until a value is found or the top
 *   parent element is reached.
 *
 * @param {string|DOMElement} element HTML string or DOMElement to be wrapped into jQuery.
 * @returns {Object} jQuery object.
 */

JQLite.expando = 'ng339';

var jqCache = JQLite.cache = {},
    jqId = 1,
    addEventListenerFn = function(element, type, fn) {
      element.addEventListener(type, fn, false);
    },
    removeEventListenerFn = function(element, type, fn) {
      element.removeEventListener(type, fn, false);
    };

/*
 * !!! This is an undocumented "private" function !!!
 */
JQLite._data = function(node) {
  //jQuery always returns an object on cache miss
  return this.cache[node[this.expando]] || {};
};

function jqNextId() { return ++jqId; }


var SPECIAL_CHARS_REGEXP = /([\:\-\_]+(.))/g;
var MOZ_HACK_REGEXP = /^moz([A-Z])/;
var MOUSE_EVENT_MAP= { mouseleave: "mouseout", mouseenter: "mouseover"};
var jqLiteMinErr = minErr('jqLite');

/**
 * Converts snake_case to camelCase.
 * Also there is special case for Moz prefix starting with upper case letter.
 * @param name Name to normalize
 */
function camelCase(name) {
  return name.
    replace(SPECIAL_CHARS_REGEXP, function(_, separator, letter, offset) {
      return offset ? letter.toUpperCase() : letter;
    }).
    replace(MOZ_HACK_REGEXP, 'Moz$1');
}

var SINGLE_TAG_REGEXP = /^<([\w-]+)\s*\/?>(?:<\/\1>|)$/;
var HTML_REGEXP = /<|&#?\w+;/;
var TAG_NAME_REGEXP = /<([\w:-]+)/;
var XHTML_TAG_REGEXP = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:-]+)[^>]*)\/>/gi;

var wrapMap = {
  'option': [1, '<select multiple="multiple">', '</select>'],

  'thead': [1, '<table>', '</table>'],
  'col': [2, '<table><colgroup>', '</colgroup></table>'],
  'tr': [2, '<table><tbody>', '</tbody></table>'],
  'td': [3, '<table><tbody><tr>', '</tr></tbody></table>'],
  '_default': [0, "", ""]
};

wrapMap.optgroup = wrapMap.option;
wrapMap.tbody = wrapMap.tfoot = wrapMap.colgroup = wrapMap.caption = wrapMap.thead;
wrapMap.th = wrapMap.td;


function jqLiteIsTextNode(html) {
  return !HTML_REGEXP.test(html);
}

function jqLiteAcceptsData(node) {
  // The window object can accept data but has no nodeType
  // Otherwise we are only interested in elements (1) and documents (9)
  var nodeType = node.nodeType;
  return nodeType === NODE_TYPE_ELEMENT || !nodeType || nodeType === NODE_TYPE_DOCUMENT;
}

function jqLiteHasData(node) {
  for (var key in jqCache[node.ng339]) {
    return true;
  }
  return false;
}

function jqLiteBuildFragment(html, context) {
  var tmp, tag, wrap,
      fragment = context.createDocumentFragment(),
      nodes = [], i;

  if (jqLiteIsTextNode(html)) {
    // Convert non-html into a text node
    nodes.push(context.createTextNode(html));
  } else {
    // Convert html into DOM nodes
    tmp = tmp || fragment.appendChild(context.createElement("div"));
    tag = (TAG_NAME_REGEXP.exec(html) || ["", ""])[1].toLowerCase();
    wrap = wrapMap[tag] || wrapMap._default;
    tmp.innerHTML = wrap[1] + html.replace(XHTML_TAG_REGEXP, "<$1></$2>") + wrap[2];

    // Descend through wrappers to the right content
    i = wrap[0];
    while (i--) {
      tmp = tmp.lastChild;
    }

    nodes = concat(nodes, tmp.childNodes);

    tmp = fragment.firstChild;
    tmp.textContent = "";
  }

  // Remove wrapper from fragment
  fragment.textContent = "";
  fragment.innerHTML = ""; // Clear inner HTML
  forEach(nodes, function(node) {
    fragment.appendChild(node);
  });

  return fragment;
}

function jqLiteParseHTML(html, context) {
  context = context || document;
  var parsed;

  if ((parsed = SINGLE_TAG_REGEXP.exec(html))) {
    return [context.createElement(parsed[1])];
  }

  if ((parsed = jqLiteBuildFragment(html, context))) {
    return parsed.childNodes;
  }

  return [];
}


// IE9-11 has no method "contains" in SVG element and in Node.prototype. Bug #10259.
var jqLiteContains = Node.prototype.contains || function(arg) {
  // jshint bitwise: false
  return !!(this.compareDocumentPosition(arg) & 16);
  // jshint bitwise: true
};

/////////////////////////////////////////////
function JQLite(element) {
  if (element instanceof JQLite) {
    return element;
  }

  var argIsString;

  if (isString(element)) {
    element = trim(element);
    argIsString = true;
  }
  if (!(this instanceof JQLite)) {
    if (argIsString && element.charAt(0) != '<') {
      throw jqLiteMinErr('nosel', 'Looking up elements via selectors is not supported by jqLite! See: http://docs.angularjs.org/api/angular.element');
    }
    return new JQLite(element);
  }

  if (argIsString) {
    jqLiteAddNodes(this, jqLiteParseHTML(element));
  } else {
    jqLiteAddNodes(this, element);
  }
}

function jqLiteClone(element) {
  return element.cloneNode(true);
}

function jqLiteDealoc(element, onlyDescendants) {
  if (!onlyDescendants) jqLiteRemoveData(element);

  if (element.querySelectorAll) {
    var descendants = element.querySelectorAll('*');
    for (var i = 0, l = descendants.length; i < l; i++) {
      jqLiteRemoveData(descendants[i]);
    }
  }
}

function jqLiteOff(element, type, fn, unsupported) {
  if (isDefined(unsupported)) throw jqLiteMinErr('offargs', 'jqLite#off() does not support the `selector` argument');

  var expandoStore = jqLiteExpandoStore(element);
  var events = expandoStore && expandoStore.events;
  var handle = expandoStore && expandoStore.handle;

  if (!handle) return; //no listeners registered

  if (!type) {
    for (type in events) {
      if (type !== '$destroy') {
        removeEventListenerFn(element, type, handle);
      }
      delete events[type];
    }
  } else {

    var removeHandler = function(type) {
      var listenerFns = events[type];
      if (isDefined(fn)) {
        arrayRemove(listenerFns || [], fn);
      }
      if (!(isDefined(fn) && listenerFns && listenerFns.length > 0)) {
        removeEventListenerFn(element, type, handle);
        delete events[type];
      }
    };

    forEach(type.split(' '), function(type) {
      removeHandler(type);
      if (MOUSE_EVENT_MAP[type]) {
        removeHandler(MOUSE_EVENT_MAP[type]);
      }
    });
  }
}

function jqLiteRemoveData(element, name) {
  var expandoId = element.ng339;
  var expandoStore = expandoId && jqCache[expandoId];

  if (expandoStore) {
    if (name) {
      delete expandoStore.data[name];
      return;
    }

    if (expandoStore.handle) {
      if (expandoStore.events.$destroy) {
        expandoStore.handle({}, '$destroy');
      }
      jqLiteOff(element);
    }
    delete jqCache[expandoId];
    element.ng339 = undefined; // don't delete DOM expandos. IE and Chrome don't like it
  }
}


function jqLiteExpandoStore(element, createIfNecessary) {
  var expandoId = element.ng339,
      expandoStore = expandoId && jqCache[expandoId];

  if (createIfNecessary && !expandoStore) {
    element.ng339 = expandoId = jqNextId();
    expandoStore = jqCache[expandoId] = {events: {}, data: {}, handle: undefined};
  }

  return expandoStore;
}


function jqLiteData(element, key, value) {
  if (jqLiteAcceptsData(element)) {

    var isSimpleSetter = isDefined(value);
    var isSimpleGetter = !isSimpleSetter && key && !isObject(key);
    var massGetter = !key;
    var expandoStore = jqLiteExpandoStore(element, !isSimpleGetter);
    var data = expandoStore && expandoStore.data;

    if (isSimpleSetter) { // data('key', value)
      data[key] = value;
    } else {
      if (massGetter) {  // data()
        return data;
      } else {
        if (isSimpleGetter) { // data('key')
          // don't force creation of expandoStore if it doesn't exist yet
          return data && data[key];
        } else { // mass-setter: data({key1: val1, key2: val2})
          extend(data, key);
        }
      }
    }
  }
}

function jqLiteHasClass(element, selector) {
  if (!element.getAttribute) return false;
  return ((" " + (element.getAttribute('class') || '') + " ").replace(/[\n\t]/g, " ").
      indexOf(" " + selector + " ") > -1);
}

function jqLiteRemoveClass(element, cssClasses) {
  if (cssClasses && element.setAttribute) {
    forEach(cssClasses.split(' '), function(cssClass) {
      element.setAttribute('class', trim(
          (" " + (element.getAttribute('class') || '') + " ")
          .replace(/[\n\t]/g, " ")
          .replace(" " + trim(cssClass) + " ", " "))
      );
    });
  }
}

function jqLiteAddClass(element, cssClasses) {
  if (cssClasses && element.setAttribute) {
    var existingClasses = (' ' + (element.getAttribute('class') || '') + ' ')
                            .replace(/[\n\t]/g, " ");

    forEach(cssClasses.split(' '), function(cssClass) {
      cssClass = trim(cssClass);
      if (existingClasses.indexOf(' ' + cssClass + ' ') === -1) {
        existingClasses += cssClass + ' ';
      }
    });

    element.setAttribute('class', trim(existingClasses));
  }
}


function jqLiteAddNodes(root, elements) {
  // THIS CODE IS VERY HOT. Don't make changes without benchmarking.

  if (elements) {

    // if a Node (the most common case)
    if (elements.nodeType) {
      root[root.length++] = elements;
    } else {
      var length = elements.length;

      // if an Array or NodeList and not a Window
      if (typeof length === 'number' && elements.window !== elements) {
        if (length) {
          for (var i = 0; i < length; i++) {
            root[root.length++] = elements[i];
          }
        }
      } else {
        root[root.length++] = elements;
      }
    }
  }
}


function jqLiteController(element, name) {
  return jqLiteInheritedData(element, '$' + (name || 'ngController') + 'Controller');
}

function jqLiteInheritedData(element, name, value) {
  // if element is the document object work with the html element instead
  // this makes $(document).scope() possible
  if (element.nodeType == NODE_TYPE_DOCUMENT) {
    element = element.documentElement;
  }
  var names = isArray(name) ? name : [name];

  while (element) {
    for (var i = 0, ii = names.length; i < ii; i++) {
      if (isDefined(value = jqLite.data(element, names[i]))) return value;
    }

    // If dealing with a document fragment node with a host element, and no parent, use the host
    // element as the parent. This enables directives within a Shadow DOM or polyfilled Shadow DOM
    // to lookup parent controllers.
    element = element.parentNode || (element.nodeType === NODE_TYPE_DOCUMENT_FRAGMENT && element.host);
  }
}

function jqLiteEmpty(element) {
  jqLiteDealoc(element, true);
  while (element.firstChild) {
    element.removeChild(element.firstChild);
  }
}

function jqLiteRemove(element, keepData) {
  if (!keepData) jqLiteDealoc(element);
  var parent = element.parentNode;
  if (parent) parent.removeChild(element);
}


function jqLiteDocumentLoaded(action, win) {
  win = win || window;
  if (win.document.readyState === 'complete') {
    // Force the action to be run async for consistent behaviour
    // from the action's point of view
    // i.e. it will definitely not be in a $apply
    win.setTimeout(action);
  } else {
    // No need to unbind this handler as load is only ever called once
    jqLite(win).on('load', action);
  }
}

//////////////////////////////////////////
// Functions which are declared directly.
//////////////////////////////////////////
var JQLitePrototype = JQLite.prototype = {
  ready: function(fn) {
    var fired = false;

    function trigger() {
      if (fired) return;
      fired = true;
      fn();
    }

    // check if document is already loaded
    if (document.readyState === 'complete') {
      setTimeout(trigger);
    } else {
      this.on('DOMContentLoaded', trigger); // works for modern browsers and IE9
      // we can not use jqLite since we are not done loading and jQuery could be loaded later.
      // jshint -W064
      JQLite(window).on('load', trigger); // fallback to window.onload for others
      // jshint +W064
    }
  },
  toString: function() {
    var value = [];
    forEach(this, function(e) { value.push('' + e);});
    return '[' + value.join(', ') + ']';
  },

  eq: function(index) {
      return (index >= 0) ? jqLite(this[index]) : jqLite(this[this.length + index]);
  },

  length: 0,
  push: push,
  sort: [].sort,
  splice: [].splice
};

//////////////////////////////////////////
// Functions iterating getter/setters.
// these functions return self on setter and
// value on get.
//////////////////////////////////////////
var BOOLEAN_ATTR = {};
forEach('multiple,selected,checked,disabled,readOnly,required,open'.split(','), function(value) {
  BOOLEAN_ATTR[lowercase(value)] = value;
});
var BOOLEAN_ELEMENTS = {};
forEach('input,select,option,textarea,button,form,details'.split(','), function(value) {
  BOOLEAN_ELEMENTS[value] = true;
});
var ALIASED_ATTR = {
  'ngMinlength': 'minlength',
  'ngMaxlength': 'maxlength',
  'ngMin': 'min',
  'ngMax': 'max',
  'ngPattern': 'pattern'
};

function getBooleanAttrName(element, name) {
  // check dom last since we will most likely fail on name
  var booleanAttr = BOOLEAN_ATTR[name.toLowerCase()];

  // booleanAttr is here twice to minimize DOM access
  return booleanAttr && BOOLEAN_ELEMENTS[nodeName_(element)] && booleanAttr;
}

function getAliasedAttrName(name) {
  return ALIASED_ATTR[name];
}

forEach({
  data: jqLiteData,
  removeData: jqLiteRemoveData,
  hasData: jqLiteHasData
}, function(fn, name) {
  JQLite[name] = fn;
});

forEach({
  data: jqLiteData,
  inheritedData: jqLiteInheritedData,

  scope: function(element) {
    // Can't use jqLiteData here directly so we stay compatible with jQuery!
    return jqLite.data(element, '$scope') || jqLiteInheritedData(element.parentNode || element, ['$isolateScope', '$scope']);
  },

  isolateScope: function(element) {
    // Can't use jqLiteData here directly so we stay compatible with jQuery!
    return jqLite.data(element, '$isolateScope') || jqLite.data(element, '$isolateScopeNoTemplate');
  },

  controller: jqLiteController,

  injector: function(element) {
    return jqLiteInheritedData(element, '$injector');
  },

  removeAttr: function(element, name) {
    element.removeAttribute(name);
  },

  hasClass: jqLiteHasClass,

  css: function(element, name, value) {
    name = camelCase(name);

    if (isDefined(value)) {
      element.style[name] = value;
    } else {
      return element.style[name];
    }
  },

  attr: function(element, name, value) {
    var nodeType = element.nodeType;
    if (nodeType === NODE_TYPE_TEXT || nodeType === NODE_TYPE_ATTRIBUTE || nodeType === NODE_TYPE_COMMENT) {
      return;
    }
    var lowercasedName = lowercase(name);
    if (BOOLEAN_ATTR[lowercasedName]) {
      if (isDefined(value)) {
        if (!!value) {
          element[name] = true;
          element.setAttribute(name, lowercasedName);
        } else {
          element[name] = false;
          element.removeAttribute(lowercasedName);
        }
      } else {
        return (element[name] ||
                 (element.attributes.getNamedItem(name) || noop).specified)
               ? lowercasedName
               : undefined;
      }
    } else if (isDefined(value)) {
      element.setAttribute(name, value);
    } else if (element.getAttribute) {
      // the extra argument "2" is to get the right thing for a.href in IE, see jQuery code
      // some elements (e.g. Document) don't have get attribute, so return undefined
      var ret = element.getAttribute(name, 2);
      // normalize non-existing attributes to undefined (as jQuery)
      return ret === null ? undefined : ret;
    }
  },

  prop: function(element, name, value) {
    if (isDefined(value)) {
      element[name] = value;
    } else {
      return element[name];
    }
  },

  text: (function() {
    getText.$dv = '';
    return getText;

    function getText(element, value) {
      if (isUndefined(value)) {
        var nodeType = element.nodeType;
        return (nodeType === NODE_TYPE_ELEMENT || nodeType === NODE_TYPE_TEXT) ? element.textContent : '';
      }
      element.textContent = value;
    }
  })(),

  val: function(element, value) {
    if (isUndefined(value)) {
      if (element.multiple && nodeName_(element) === 'select') {
        var result = [];
        forEach(element.options, function(option) {
          if (option.selected) {
            result.push(option.value || option.text);
          }
        });
        return result.length === 0 ? null : result;
      }
      return element.value;
    }
    element.value = value;
  },

  html: function(element, value) {
    if (isUndefined(value)) {
      return element.innerHTML;
    }
    jqLiteDealoc(element, true);
    element.innerHTML = value;
  },

  empty: jqLiteEmpty
}, function(fn, name) {
  /**
   * Properties: writes return selection, reads return first value
   */
  JQLite.prototype[name] = function(arg1, arg2) {
    var i, key;
    var nodeCount = this.length;

    // jqLiteHasClass has only two arguments, but is a getter-only fn, so we need to special-case it
    // in a way that survives minification.
    // jqLiteEmpty takes no arguments but is a setter.
    if (fn !== jqLiteEmpty &&
        (isUndefined((fn.length == 2 && (fn !== jqLiteHasClass && fn !== jqLiteController)) ? arg1 : arg2))) {
      if (isObject(arg1)) {

        // we are a write, but the object properties are the key/values
        for (i = 0; i < nodeCount; i++) {
          if (fn === jqLiteData) {
            // data() takes the whole object in jQuery
            fn(this[i], arg1);
          } else {
            for (key in arg1) {
              fn(this[i], key, arg1[key]);
            }
          }
        }
        // return self for chaining
        return this;
      } else {
        // we are a read, so read the first child.
        // TODO: do we still need this?
        var value = fn.$dv;
        // Only if we have $dv do we iterate over all, otherwise it is just the first element.
        var jj = (isUndefined(value)) ? Math.min(nodeCount, 1) : nodeCount;
        for (var j = 0; j < jj; j++) {
          var nodeValue = fn(this[j], arg1, arg2);
          value = value ? value + nodeValue : nodeValue;
        }
        return value;
      }
    } else {
      // we are a write, so apply to all children
      for (i = 0; i < nodeCount; i++) {
        fn(this[i], arg1, arg2);
      }
      // return self for chaining
      return this;
    }
  };
});

function createEventHandler(element, events) {
  var eventHandler = function(event, type) {
    // jQuery specific api
    event.isDefaultPrevented = function() {
      return event.defaultPrevented;
    };

    var eventFns = events[type || event.type];
    var eventFnsLength = eventFns ? eventFns.length : 0;

    if (!eventFnsLength) return;

    if (isUndefined(event.immediatePropagationStopped)) {
      var originalStopImmediatePropagation = event.stopImmediatePropagation;
      event.stopImmediatePropagation = function() {
        event.immediatePropagationStopped = true;

        if (event.stopPropagation) {
          event.stopPropagation();
        }

        if (originalStopImmediatePropagation) {
          originalStopImmediatePropagation.call(event);
        }
      };
    }

    event.isImmediatePropagationStopped = function() {
      return event.immediatePropagationStopped === true;
    };

    // Some events have special handlers that wrap the real handler
    var handlerWrapper = eventFns.specialHandlerWrapper || defaultHandlerWrapper;

    // Copy event handlers in case event handlers array is modified during execution.
    if ((eventFnsLength > 1)) {
      eventFns = shallowCopy(eventFns);
    }

    for (var i = 0; i < eventFnsLength; i++) {
      if (!event.isImmediatePropagationStopped()) {
        handlerWrapper(element, event, eventFns[i]);
      }
    }
  };

  // TODO: this is a hack for angularMocks/clearDataCache that makes it possible to deregister all
  //       events on `element`
  eventHandler.elem = element;
  return eventHandler;
}

function defaultHandlerWrapper(element, event, handler) {
  handler.call(element, event);
}

function specialMouseHandlerWrapper(target, event, handler) {
  // Refer to jQuery's implementation of mouseenter & mouseleave
  // Read about mouseenter and mouseleave:
  // http://www.quirksmode.org/js/events_mouse.html#link8
  var related = event.relatedTarget;
  // For mousenter/leave call the handler if related is outside the target.
  // NB: No relatedTarget if the mouse left/entered the browser window
  if (!related || (related !== target && !jqLiteContains.call(target, related))) {
    handler.call(target, event);
  }
}

//////////////////////////////////////////
// Functions iterating traversal.
// These functions chain results into a single
// selector.
//////////////////////////////////////////
forEach({
  removeData: jqLiteRemoveData,

  on: function jqLiteOn(element, type, fn, unsupported) {
    if (isDefined(unsupported)) throw jqLiteMinErr('onargs', 'jqLite#on() does not support the `selector` or `eventData` parameters');

    // Do not add event handlers to non-elements because they will not be cleaned up.
    if (!jqLiteAcceptsData(element)) {
      return;
    }

    var expandoStore = jqLiteExpandoStore(element, true);
    var events = expandoStore.events;
    var handle = expandoStore.handle;

    if (!handle) {
      handle = expandoStore.handle = createEventHandler(element, events);
    }

    // http://jsperf.com/string-indexof-vs-split
    var types = type.indexOf(' ') >= 0 ? type.split(' ') : [type];
    var i = types.length;

    var addHandler = function(type, specialHandlerWrapper, noEventListener) {
      var eventFns = events[type];

      if (!eventFns) {
        eventFns = events[type] = [];
        eventFns.specialHandlerWrapper = specialHandlerWrapper;
        if (type !== '$destroy' && !noEventListener) {
          addEventListenerFn(element, type, handle);
        }
      }

      eventFns.push(fn);
    };

    while (i--) {
      type = types[i];
      if (MOUSE_EVENT_MAP[type]) {
        addHandler(MOUSE_EVENT_MAP[type], specialMouseHandlerWrapper);
        addHandler(type, undefined, true);
      } else {
        addHandler(type);
      }
    }
  },

  off: jqLiteOff,

  one: function(element, type, fn) {
    element = jqLite(element);

    //add the listener twice so that when it is called
    //you can remove the original function and still be
    //able to call element.off(ev, fn) normally
    element.on(type, function onFn() {
      element.off(type, fn);
      element.off(type, onFn);
    });
    element.on(type, fn);
  },

  replaceWith: function(element, replaceNode) {
    var index, parent = element.parentNode;
    jqLiteDealoc(element);
    forEach(new JQLite(replaceNode), function(node) {
      if (index) {
        parent.insertBefore(node, index.nextSibling);
      } else {
        parent.replaceChild(node, element);
      }
      index = node;
    });
  },

  children: function(element) {
    var children = [];
    forEach(element.childNodes, function(element) {
      if (element.nodeType === NODE_TYPE_ELEMENT) {
        children.push(element);
      }
    });
    return children;
  },

  contents: function(element) {
    return element.contentDocument || element.childNodes || [];
  },

  append: function(element, node) {
    var nodeType = element.nodeType;
    if (nodeType !== NODE_TYPE_ELEMENT && nodeType !== NODE_TYPE_DOCUMENT_FRAGMENT) return;

    node = new JQLite(node);

    for (var i = 0, ii = node.length; i < ii; i++) {
      var child = node[i];
      element.appendChild(child);
    }
  },

  prepend: function(element, node) {
    if (element.nodeType === NODE_TYPE_ELEMENT) {
      var index = element.firstChild;
      forEach(new JQLite(node), function(child) {
        element.insertBefore(child, index);
      });
    }
  },

  wrap: function(element, wrapNode) {
    wrapNode = jqLite(wrapNode).eq(0).clone()[0];
    var parent = element.parentNode;
    if (parent) {
      parent.replaceChild(wrapNode, element);
    }
    wrapNode.appendChild(element);
  },

  remove: jqLiteRemove,

  detach: function(element) {
    jqLiteRemove(element, true);
  },

  after: function(element, newElement) {
    var index = element, parent = element.parentNode;
    newElement = new JQLite(newElement);

    for (var i = 0, ii = newElement.length; i < ii; i++) {
      var node = newElement[i];
      parent.insertBefore(node, index.nextSibling);
      index = node;
    }
  },

  addClass: jqLiteAddClass,
  removeClass: jqLiteRemoveClass,

  toggleClass: function(element, selector, condition) {
    if (selector) {
      forEach(selector.split(' '), function(className) {
        var classCondition = condition;
        if (isUndefined(classCondition)) {
          classCondition = !jqLiteHasClass(element, className);
        }
        (classCondition ? jqLiteAddClass : jqLiteRemoveClass)(element, className);
      });
    }
  },

  parent: function(element) {
    var parent = element.parentNode;
    return parent && parent.nodeType !== NODE_TYPE_DOCUMENT_FRAGMENT ? parent : null;
  },

  next: function(element) {
    return element.nextElementSibling;
  },

  find: function(element, selector) {
    if (element.getElementsByTagName) {
      return element.getElementsByTagName(selector);
    } else {
      return [];
    }
  },

  clone: jqLiteClone,

  triggerHandler: function(element, event, extraParameters) {

    var dummyEvent, eventFnsCopy, handlerArgs;
    var eventName = event.type || event;
    var expandoStore = jqLiteExpandoStore(element);
    var events = expandoStore && expandoStore.events;
    var eventFns = events && events[eventName];

    if (eventFns) {
      // Create a dummy event to pass to the handlers
      dummyEvent = {
        preventDefault: function() { this.defaultPrevented = true; },
        isDefaultPrevented: function() { return this.defaultPrevented === true; },
        stopImmediatePropagation: function() { this.immediatePropagationStopped = true; },
        isImmediatePropagationStopped: function() { return this.immediatePropagationStopped === true; },
        stopPropagation: noop,
        type: eventName,
        target: element
      };

      // If a custom event was provided then extend our dummy event with it
      if (event.type) {
        dummyEvent = extend(dummyEvent, event);
      }

      // Copy event handlers in case event handlers array is modified during execution.
      eventFnsCopy = shallowCopy(eventFns);
      handlerArgs = extraParameters ? [dummyEvent].concat(extraParameters) : [dummyEvent];

      forEach(eventFnsCopy, function(fn) {
        if (!dummyEvent.isImmediatePropagationStopped()) {
          fn.apply(element, handlerArgs);
        }
      });
    }
  }
}, function(fn, name) {
  /**
   * chaining functions
   */
  JQLite.prototype[name] = function(arg1, arg2, arg3) {
    var value;

    for (var i = 0, ii = this.length; i < ii; i++) {
      if (isUndefined(value)) {
        value = fn(this[i], arg1, arg2, arg3);
        if (isDefined(value)) {
          // any function which returns a value needs to be wrapped
          value = jqLite(value);
        }
      } else {
        jqLiteAddNodes(value, fn(this[i], arg1, arg2, arg3));
      }
    }
    return isDefined(value) ? value : this;
  };

  // bind legacy bind/unbind to on/off
  JQLite.prototype.bind = JQLite.prototype.on;
  JQLite.prototype.unbind = JQLite.prototype.off;
});


// Provider for private $$jqLite service
function $$jqLiteProvider() {
  this.$get = function $$jqLite() {
    return extend(JQLite, {
      hasClass: function(node, classes) {
        if (node.attr) node = node[0];
        return jqLiteHasClass(node, classes);
      },
      addClass: function(node, classes) {
        if (node.attr) node = node[0];
        return jqLiteAddClass(node, classes);
      },
      removeClass: function(node, classes) {
        if (node.attr) node = node[0];
        return jqLiteRemoveClass(node, classes);
      }
    });
  };
}

/**
 * Computes a hash of an 'obj'.
 * Hash of a:
 *  string is string
 *  number is number as string
 *  object is either result of calling $$hashKey function on the object or uniquely generated id,
 *         that is also assigned to the $$hashKey property of the object.
 *
 * @param obj
 * @returns {string} hash string such that the same input will have the same hash string.
 *         The resulting string key is in 'type:hashKey' format.
 */
function hashKey(obj, nextUidFn) {
  var key = obj && obj.$$hashKey;

  if (key) {
    if (typeof key === 'function') {
      key = obj.$$hashKey();
    }
    return key;
  }

  var objType = typeof obj;
  if (objType == 'function' || (objType == 'object' && obj !== null)) {
    key = obj.$$hashKey = objType + ':' + (nextUidFn || nextUid)();
  } else {
    key = objType + ':' + obj;
  }

  return key;
}

/**
 * HashMap which can use objects as keys
 */
function HashMap(array, isolatedUid) {
  if (isolatedUid) {
    var uid = 0;
    this.nextUid = function() {
      return ++uid;
    };
  }
  forEach(array, this.put, this);
}
HashMap.prototype = {
  /**
   * Store key value pair
   * @param key key to store can be any type
   * @param value value to store can be any type
   */
  put: function(key, value) {
    this[hashKey(key, this.nextUid)] = value;
  },

  /**
   * @param key
   * @returns {Object} the value for the key
   */
  get: function(key) {
    return this[hashKey(key, this.nextUid)];
  },

  /**
   * Remove the key/value pair
   * @param key
   */
  remove: function(key) {
    var value = this[key = hashKey(key, this.nextUid)];
    delete this[key];
    return value;
  }
};

var $$HashMapProvider = [function() {
  this.$get = [function() {
    return HashMap;
  }];
}];

/**
 * @ngdoc function
 * @module ng
 * @name angular.injector
 * @kind function
 *
 * @description
 * Creates an injector object that can be used for retrieving services as well as for
 * dependency injection (see {@link guide/di dependency injection}).
 *
 * @param {Array.<string|Function>} modules A list of module functions or their aliases. See
 *     {@link angular.module}. The `ng` module must be explicitly added.
 * @param {boolean=} [strictDi=false] Whether the injector should be in strict mode, which
 *     disallows argument name annotation inference.
 * @returns {injector} Injector object. See {@link auto.$injector $injector}.
 *
 * @example
 * Typical usage
 * ```js
 *   // create an injector
 *   var $injector = angular.injector(['ng']);
 *
 *   // use the injector to kick off your application
 *   // use the type inference to auto inject arguments, or use implicit injection
 *   $injector.invoke(function($rootScope, $compile, $document) {
 *     $compile($document)($rootScope);
 *     $rootScope.$digest();
 *   });
 * ```
 *
 * Sometimes you want to get access to the injector of a currently running Angular app
 * from outside Angular. Perhaps, you want to inject and compile some markup after the
 * application has been bootstrapped. You can do this using the extra `injector()` added
 * to JQuery/jqLite elements. See {@link angular.element}.
 *
 * *This is fairly rare but could be the case if a third party library is injecting the
 * markup.*
 *
 * In the following example a new block of HTML containing a `ng-controller`
 * directive is added to the end of the document body by JQuery. We then compile and link
 * it into the current AngularJS scope.
 *
 * ```js
 * var $div = $('<div ng-controller="MyCtrl">{{content.label}}</div>');
 * $(document.body).append($div);
 *
 * angular.element(document).injector().invoke(function($compile) {
 *   var scope = angular.element($div).scope();
 *   $compile($div)(scope);
 * });
 * ```
 */


/**
 * @ngdoc module
 * @name auto
 * @description
 *
 * Implicit module which gets automatically added to each {@link auto.$injector $injector}.
 */

var FN_ARGS = /^[^\(]*\(\s*([^\)]*)\)/m;
var FN_ARG_SPLIT = /,/;
var FN_ARG = /^\s*(_?)(\S+?)\1\s*$/;
var STRIP_COMMENTS = /((\/\/.*$)|(\/\*[\s\S]*?\*\/))/mg;
var $injectorMinErr = minErr('$injector');

function anonFn(fn) {
  // For anonymous functions, showing at the very least the function signature can help in
  // debugging.
  var fnText = fn.toString().replace(STRIP_COMMENTS, ''),
      args = fnText.match(FN_ARGS);
  if (args) {
    return 'function(' + (args[1] || '').replace(/[\s\r\n]+/, ' ') + ')';
  }
  return 'fn';
}

function annotate(fn, strictDi, name) {
  var $inject,
      fnText,
      argDecl,
      last;

  if (typeof fn === 'function') {
    if (!($inject = fn.$inject)) {
      $inject = [];
      if (fn.length) {
        if (strictDi) {
          if (!isString(name) || !name) {
            name = fn.name || anonFn(fn);
          }
          throw $injectorMinErr('strictdi',
            '{0} is not using explicit annotation and cannot be invoked in strict mode', name);
        }
        fnText = fn.toString().replace(STRIP_COMMENTS, '');
        argDecl = fnText.match(FN_ARGS);
        forEach(argDecl[1].split(FN_ARG_SPLIT), function(arg) {
          arg.replace(FN_ARG, function(all, underscore, name) {
            $inject.push(name);
          });
        });
      }
      fn.$inject = $inject;
    }
  } else if (isArray(fn)) {
    last = fn.length - 1;
    assertArgFn(fn[last], 'fn');
    $inject = fn.slice(0, last);
  } else {
    assertArgFn(fn, 'fn', true);
  }
  return $inject;
}

///////////////////////////////////////

/**
 * @ngdoc service
 * @name $injector
 *
 * @description
 *
 * `$injector` is used to retrieve object instances as defined by
 * {@link auto.$provide provider}, instantiate types, invoke methods,
 * and load modules.
 *
 * The following always holds true:
 *
 * ```js
 *   var $injector = angular.injector();
 *   expect($injector.get('$injector')).toBe($injector);
 *   expect($injector.invoke(function($injector) {
 *     return $injector;
 *   })).toBe($injector);
 * ```
 *
 * # Injection Function Annotation
 *
 * JavaScript does not have annotations, and annotations are needed for dependency injection. The
 * following are all valid ways of annotating function with injection arguments and are equivalent.
 *
 * ```js
 *   // inferred (only works if code not minified/obfuscated)
 *   $injector.invoke(function(serviceA){});
 *
 *   // annotated
 *   function explicit(serviceA) {};
 *   explicit.$inject = ['serviceA'];
 *   $injector.invoke(explicit);
 *
 *   // inline
 *   $injector.invoke(['serviceA', function(serviceA){}]);
 * ```
 *
 * ## Inference
 *
 * In JavaScript calling `toString()` on a function returns the function definition. The definition
 * can then be parsed and the function arguments can be extracted. This method of discovering
 * annotations is disallowed when the injector is in strict mode.
 * *NOTE:* This does not work with minification, and obfuscation tools since these tools change the
 * argument names.
 *
 * ## `$inject` Annotation
 * By adding an `$inject` property onto a function the injection parameters can be specified.
 *
 * ## Inline
 * As an array of injection names, where the last item in the array is the function to call.
 */

/**
 * @ngdoc method
 * @name $injector#get
 *
 * @description
 * Return an instance of the service.
 *
 * @param {string} name The name of the instance to retrieve.
 * @param {string=} caller An optional string to provide the origin of the function call for error messages.
 * @return {*} The instance.
 */

/**
 * @ngdoc method
 * @name $injector#invoke
 *
 * @description
 * Invoke the method and supply the method arguments from the `$injector`.
 *
 * @param {Function|Array.<string|Function>} fn The injectable function to invoke. Function parameters are
 *   injected according to the {@link guide/di $inject Annotation} rules.
 * @param {Object=} self The `this` for the invoked method.
 * @param {Object=} locals Optional object. If preset then any argument names are read from this
 *                         object first, before the `$injector` is consulted.
 * @returns {*} the value returned by the invoked `fn` function.
 */

/**
 * @ngdoc method
 * @name $injector#has
 *
 * @description
 * Allows the user to query if the particular service exists.
 *
 * @param {string} name Name of the service to query.
 * @returns {boolean} `true` if injector has given service.
 */

/**
 * @ngdoc method
 * @name $injector#instantiate
 * @description
 * Create a new instance of JS type. The method takes a constructor function, invokes the new
 * operator, and supplies all of the arguments to the constructor function as specified by the
 * constructor annotation.
 *
 * @param {Function} Type Annotated constructor function.
 * @param {Object=} locals Optional object. If preset then any argument names are read from this
 * object first, before the `$injector` is consulted.
 * @returns {Object} new instance of `Type`.
 */

/**
 * @ngdoc method
 * @name $injector#annotate
 *
 * @description
 * Returns an array of service names which the function is requesting for injection. This API is
 * used by the injector to determine which services need to be injected into the function when the
 * function is invoked. There are three ways in which the function can be annotated with the needed
 * dependencies.
 *
 * # Argument names
 *
 * The simplest form is to extract the dependencies from the arguments of the function. This is done
 * by converting the function into a string using `toString()` method and extracting the argument
 * names.
 * ```js
 *   // Given
 *   function MyController($scope, $route) {
 *     // ...
 *   }
 *
 *   // Then
 *   expect(injector.annotate(MyController)).toEqual(['$scope', '$route']);
 * ```
 *
 * You can disallow this method by using strict injection mode.
 *
 * This method does not work with code minification / obfuscation. For this reason the following
 * annotation strategies are supported.
 *
 * # The `$inject` property
 *
 * If a function has an `$inject` property and its value is an array of strings, then the strings
 * represent names of services to be injected into the function.
 * ```js
 *   // Given
 *   var MyController = function(obfuscatedScope, obfuscatedRoute) {
 *     // ...
 *   }
 *   // Define function dependencies
 *   MyController['$inject'] = ['$scope', '$route'];
 *
 *   // Then
 *   expect(injector.annotate(MyController)).toEqual(['$scope', '$route']);
 * ```
 *
 * # The array notation
 *
 * It is often desirable to inline Injected functions and that's when setting the `$inject` property
 * is very inconvenient. In these situations using the array notation to specify the dependencies in
 * a way that survives minification is a better choice:
 *
 * ```js
 *   // We wish to write this (not minification / obfuscation safe)
 *   injector.invoke(function($compile, $rootScope) {
 *     // ...
 *   });
 *
 *   // We are forced to write break inlining
 *   var tmpFn = function(obfuscatedCompile, obfuscatedRootScope) {
 *     // ...
 *   };
 *   tmpFn.$inject = ['$compile', '$rootScope'];
 *   injector.invoke(tmpFn);
 *
 *   // To better support inline function the inline annotation is supported
 *   injector.invoke(['$compile', '$rootScope', function(obfCompile, obfRootScope) {
 *     // ...
 *   }]);
 *
 *   // Therefore
 *   expect(injector.annotate(
 *      ['$compile', '$rootScope', function(obfus_$compile, obfus_$rootScope) {}])
 *    ).toEqual(['$compile', '$rootScope']);
 * ```
 *
 * @param {Function|Array.<string|Function>} fn Function for which dependent service names need to
 * be retrieved as described above.
 *
 * @param {boolean=} [strictDi=false] Disallow argument name annotation inference.
 *
 * @returns {Array.<string>} The names of the services which the function requires.
 */




/**
 * @ngdoc service
 * @name $provide
 *
 * @description
 *
 * The {@link auto.$provide $provide} service has a number of methods for registering components
 * with the {@link auto.$injector $injector}. Many of these functions are also exposed on
 * {@link angular.Module}.
 *
 * An Angular **service** is a singleton object created by a **service factory**.  These **service
 * factories** are functions which, in turn, are created by a **service provider**.
 * The **service providers** are constructor functions. When instantiated they must contain a
 * property called `$get`, which holds the **service factory** function.
 *
 * When you request a service, the {@link auto.$injector $injector} is responsible for finding the
 * correct **service provider**, instantiating it and then calling its `$get` **service factory**
 * function to get the instance of the **service**.
 *
 * Often services have no configuration options and there is no need to add methods to the service
 * provider.  The provider will be no more than a constructor function with a `$get` property. For
 * these cases the {@link auto.$provide $provide} service has additional helper methods to register
 * services without specifying a provider.
 *
 * * {@link auto.$provide#provider provider(provider)} - registers a **service provider** with the
 *     {@link auto.$injector $injector}
 * * {@link auto.$provide#constant constant(obj)} - registers a value/object that can be accessed by
 *     providers and services.
 * * {@link auto.$provide#value value(obj)} - registers a value/object that can only be accessed by
 *     services, not providers.
 * * {@link auto.$provide#factory factory(fn)} - registers a service **factory function**, `fn`,
 *     that will be wrapped in a **service provider** object, whose `$get` property will contain the
 *     given factory function.
 * * {@link auto.$provide#service service(class)} - registers a **constructor function**, `class`
 *     that will be wrapped in a **service provider** object, whose `$get` property will instantiate
 *      a new object using the given constructor function.
 *
 * See the individual methods for more information and examples.
 */

/**
 * @ngdoc method
 * @name $provide#provider
 * @description
 *
 * Register a **provider function** with the {@link auto.$injector $injector}. Provider functions
 * are constructor functions, whose instances are responsible for "providing" a factory for a
 * service.
 *
 * Service provider names start with the name of the service they provide followed by `Provider`.
 * For example, the {@link ng.$log $log} service has a provider called
 * {@link ng.$logProvider $logProvider}.
 *
 * Service provider objects can have additional methods which allow configuration of the provider
 * and its service. Importantly, you can configure what kind of service is created by the `$get`
 * method, or how that service will act. For example, the {@link ng.$logProvider $logProvider} has a
 * method {@link ng.$logProvider#debugEnabled debugEnabled}
 * which lets you specify whether the {@link ng.$log $log} service will log debug messages to the
 * console or not.
 *
 * @param {string} name The name of the instance. NOTE: the provider will be available under `name +
                        'Provider'` key.
 * @param {(Object|function())} provider If the provider is:
 *
 *   - `Object`: then it should have a `$get` method. The `$get` method will be invoked using
 *     {@link auto.$injector#invoke $injector.invoke()} when an instance needs to be created.
 *   - `Constructor`: a new instance of the provider will be created using
 *     {@link auto.$injector#instantiate $injector.instantiate()}, then treated as `object`.
 *
 * @returns {Object} registered provider instance

 * @example
 *
 * The following example shows how to create a simple event tracking service and register it using
 * {@link auto.$provide#provider $provide.provider()}.
 *
 * ```js
 *  // Define the eventTracker provider
 *  function EventTrackerProvider() {
 *    var trackingUrl = '/track';
 *
 *    // A provider method for configuring where the tracked events should been saved
 *    this.setTrackingUrl = function(url) {
 *      trackingUrl = url;
 *    };
 *
 *    // The service factory function
 *    this.$get = ['$http', function($http) {
 *      var trackedEvents = {};
 *      return {
 *        // Call this to track an event
 *        event: function(event) {
 *          var count = trackedEvents[event] || 0;
 *          count += 1;
 *          trackedEvents[event] = count;
 *          return count;
 *        },
 *        // Call this to save the tracked events to the trackingUrl
 *        save: function() {
 *          $http.post(trackingUrl, trackedEvents);
 *        }
 *      };
 *    }];
 *  }
 *
 *  describe('eventTracker', function() {
 *    var postSpy;
 *
 *    beforeEach(module(function($provide) {
 *      // Register the eventTracker provider
 *      $provide.provider('eventTracker', EventTrackerProvider);
 *    }));
 *
 *    beforeEach(module(function(eventTrackerProvider) {
 *      // Configure eventTracker provider
 *      eventTrackerProvider.setTrackingUrl('/custom-track');
 *    }));
 *
 *    it('tracks events', inject(function(eventTracker) {
 *      expect(eventTracker.event('login')).toEqual(1);
 *      expect(eventTracker.event('login')).toEqual(2);
 *    }));
 *
 *    it('saves to the tracking url', inject(function(eventTracker, $http) {
 *      postSpy = spyOn($http, 'post');
 *      eventTracker.event('login');
 *      eventTracker.save();
 *      expect(postSpy).toHaveBeenCalled();
 *      expect(postSpy.mostRecentCall.args[0]).not.toEqual('/track');
 *      expect(postSpy.mostRecentCall.args[0]).toEqual('/custom-track');
 *      expect(postSpy.mostRecentCall.args[1]).toEqual({ 'login': 1 });
 *    }));
 *  });
 * ```
 */

/**
 * @ngdoc method
 * @name $provide#factory
 * @description
 *
 * Register a **service factory**, which will be called to return the service instance.
 * This is short for registering a service where its provider consists of only a `$get` property,
 * which is the given service factory function.
 * You should use {@link auto.$provide#factory $provide.factory(getFn)} if you do not need to
 * configure your service in a provider.
 *
 * @param {string} name The name of the instance.
 * @param {Function|Array.<string|Function>} $getFn The injectable $getFn for the instance creation.
 *                      Internally this is a short hand for `$provide.provider(name, {$get: $getFn})`.
 * @returns {Object} registered provider instance
 *
 * @example
 * Here is an example of registering a service
 * ```js
 *   $provide.factory('ping', ['$http', function($http) {
 *     return function ping() {
 *       return $http.send('/ping');
 *     };
 *   }]);
 * ```
 * You would then inject and use this service like this:
 * ```js
 *   someModule.controller('Ctrl', ['ping', function(ping) {
 *     ping();
 *   }]);
 * ```
 */


/**
 * @ngdoc method
 * @name $provide#service
 * @description
 *
 * Register a **service constructor**, which will be invoked with `new` to create the service
 * instance.
 * This is short for registering a service where its provider's `$get` property is the service
 * constructor function that will be used to instantiate the service instance.
 *
 * You should use {@link auto.$provide#service $provide.service(class)} if you define your service
 * as a type/class.
 *
 * @param {string} name The name of the instance.
 * @param {Function|Array.<string|Function>} constructor An injectable class (constructor function)
 *     that will be instantiated.
 * @returns {Object} registered provider instance
 *
 * @example
 * Here is an example of registering a service using
 * {@link auto.$provide#service $provide.service(class)}.
 * ```js
 *   var Ping = function($http) {
 *     this.$http = $http;
 *   };
 *
 *   Ping.$inject = ['$http'];
 *
 *   Ping.prototype.send = function() {
 *     return this.$http.get('/ping');
 *   };
 *   $provide.service('ping', Ping);
 * ```
 * You would then inject and use this service like this:
 * ```js
 *   someModule.controller('Ctrl', ['ping', function(ping) {
 *     ping.send();
 *   }]);
 * ```
 */


/**
 * @ngdoc method
 * @name $provide#value
 * @description
 *
 * Register a **value service** with the {@link auto.$injector $injector}, such as a string, a
 * number, an array, an object or a function.  This is short for registering a service where its
 * provider's `$get` property is a factory function that takes no arguments and returns the **value
 * service**.
 *
 * Value services are similar to constant services, except that they cannot be injected into a
 * module configuration function (see {@link angular.Module#config}) but they can be overridden by
 * an Angular
 * {@link auto.$provide#decorator decorator}.
 *
 * @param {string} name The name of the instance.
 * @param {*} value The value.
 * @returns {Object} registered provider instance
 *
 * @example
 * Here are some examples of creating value services.
 * ```js
 *   $provide.value('ADMIN_USER', 'admin');
 *
 *   $provide.value('RoleLookup', { admin: 0, writer: 1, reader: 2 });
 *
 *   $provide.value('halfOf', function(value) {
 *     return value / 2;
 *   });
 * ```
 */


/**
 * @ngdoc method
 * @name $provide#constant
 * @description
 *
 * Register a **constant service**, such as a string, a number, an array, an object or a function,
 * with the {@link auto.$injector $injector}. Unlike {@link auto.$provide#value value} it can be
 * injected into a module configuration function (see {@link angular.Module#config}) and it cannot
 * be overridden by an Angular {@link auto.$provide#decorator decorator}.
 *
 * @param {string} name The name of the constant.
 * @param {*} value The constant value.
 * @returns {Object} registered instance
 *
 * @example
 * Here a some examples of creating constants:
 * ```js
 *   $provide.constant('SHARD_HEIGHT', 306);
 *
 *   $provide.constant('MY_COLOURS', ['red', 'blue', 'grey']);
 *
 *   $provide.constant('double', function(value) {
 *     return value * 2;
 *   });
 * ```
 */


/**
 * @ngdoc method
 * @name $provide#decorator
 * @description
 *
 * Register a **service decorator** with the {@link auto.$injector $injector}. A service decorator
 * intercepts the creation of a service, allowing it to override or modify the behaviour of the
 * service. The object returned by the decorator may be the original service, or a new service
 * object which replaces or wraps and delegates to the original service.
 *
 * @param {string} name The name of the service to decorate.
 * @param {Function|Array.<string|Function>} decorator This function will be invoked when the service needs to be
 *    instantiated and should return the decorated service instance. The function is called using
 *    the {@link auto.$injector#invoke injector.invoke} method and is therefore fully injectable.
 *    Local injection arguments:
 *
 *    * `$delegate` - The original service instance, which can be monkey patched, configured,
 *      decorated or delegated to.
 *
 * @example
 * Here we decorate the {@link ng.$log $log} service to convert warnings to errors by intercepting
 * calls to {@link ng.$log#error $log.warn()}.
 * ```js
 *   $provide.decorator('$log', ['$delegate', function($delegate) {
 *     $delegate.warn = $delegate.error;
 *     return $delegate;
 *   }]);
 * ```
 */


function createInjector(modulesToLoad, strictDi) {
  strictDi = (strictDi === true);
  var INSTANTIATING = {},
      providerSuffix = 'Provider',
      path = [],
      loadedModules = new HashMap([], true),
      providerCache = {
        $provide: {
            provider: supportObject(provider),
            factory: supportObject(factory),
            service: supportObject(service),
            value: supportObject(value),
            constant: supportObject(constant),
            decorator: decorator
          }
      },
      providerInjector = (providerCache.$injector =
          createInternalInjector(providerCache, function(serviceName, caller) {
            if (angular.isString(caller)) {
              path.push(caller);
            }
            throw $injectorMinErr('unpr', "Unknown provider: {0}", path.join(' <- '));
          })),
      instanceCache = {},
      instanceInjector = (instanceCache.$injector =
          createInternalInjector(instanceCache, function(serviceName, caller) {
            var provider = providerInjector.get(serviceName + providerSuffix, caller);
            return instanceInjector.invoke(provider.$get, provider, undefined, serviceName);
          }));


  forEach(loadModules(modulesToLoad), function(fn) { if (fn) instanceInjector.invoke(fn); });

  return instanceInjector;

  ////////////////////////////////////
  // $provider
  ////////////////////////////////////

  function supportObject(delegate) {
    return function(key, value) {
      if (isObject(key)) {
        forEach(key, reverseParams(delegate));
      } else {
        return delegate(key, value);
      }
    };
  }

  function provider(name, provider_) {
    assertNotHasOwnProperty(name, 'service');
    if (isFunction(provider_) || isArray(provider_)) {
      provider_ = providerInjector.instantiate(provider_);
    }
    if (!provider_.$get) {
      throw $injectorMinErr('pget', "Provider '{0}' must define $get factory method.", name);
    }
    return providerCache[name + providerSuffix] = provider_;
  }

  function enforceReturnValue(name, factory) {
    return function enforcedReturnValue() {
      var result = instanceInjector.invoke(factory, this);
      if (isUndefined(result)) {
        throw $injectorMinErr('undef', "Provider '{0}' must return a value from $get factory method.", name);
      }
      return result;
    };
  }

  function factory(name, factoryFn, enforce) {
    return provider(name, {
      $get: enforce !== false ? enforceReturnValue(name, factoryFn) : factoryFn
    });
  }

  function service(name, constructor) {
    return factory(name, ['$injector', function($injector) {
      return $injector.instantiate(constructor);
    }]);
  }

  function value(name, val) { return factory(name, valueFn(val), false); }

  function constant(name, value) {
    assertNotHasOwnProperty(name, 'constant');
    providerCache[name] = value;
    instanceCache[name] = value;
  }

  function decorator(serviceName, decorFn) {
    var origProvider = providerInjector.get(serviceName + providerSuffix),
        orig$get = origProvider.$get;

    origProvider.$get = function() {
      var origInstance = instanceInjector.invoke(orig$get, origProvider);
      return instanceInjector.invoke(decorFn, null, {$delegate: origInstance});
    };
  }

  ////////////////////////////////////
  // Module Loading
  ////////////////////////////////////
  function loadModules(modulesToLoad) {
    assertArg(isUndefined(modulesToLoad) || isArray(modulesToLoad), 'modulesToLoad', 'not an array');
    var runBlocks = [], moduleFn;
    forEach(modulesToLoad, function(module) {
      if (loadedModules.get(module)) return;
      loadedModules.put(module, true);

      function runInvokeQueue(queue) {
        var i, ii;
        for (i = 0, ii = queue.length; i < ii; i++) {
          var invokeArgs = queue[i],
              provider = providerInjector.get(invokeArgs[0]);

          provider[invokeArgs[1]].apply(provider, invokeArgs[2]);
        }
      }

      try {
        if (isString(module)) {
          moduleFn = angularModule(module);
          runBlocks = runBlocks.concat(loadModules(moduleFn.requires)).concat(moduleFn._runBlocks);
          runInvokeQueue(moduleFn._invokeQueue);
          runInvokeQueue(moduleFn._configBlocks);
        } else if (isFunction(module)) {
            runBlocks.push(providerInjector.invoke(module));
        } else if (isArray(module)) {
            runBlocks.push(providerInjector.invoke(module));
        } else {
          assertArgFn(module, 'module');
        }
      } catch (e) {
        if (isArray(module)) {
          module = module[module.length - 1];
        }
        if (e.message && e.stack && e.stack.indexOf(e.message) == -1) {
          // Safari & FF's stack traces don't contain error.message content
          // unlike those of Chrome and IE
          // So if stack doesn't contain message, we create a new string that contains both.
          // Since error.stack is read-only in Safari, I'm overriding e and not e.stack here.
          /* jshint -W022 */
          e = e.message + '\n' + e.stack;
        }
        throw $injectorMinErr('modulerr', "Failed to instantiate module {0} due to:\n{1}",
                  module, e.stack || e.message || e);
      }
    });
    return runBlocks;
  }

  ////////////////////////////////////
  // internal Injector
  ////////////////////////////////////

  function createInternalInjector(cache, factory) {

    function getService(serviceName, caller) {
      if (cache.hasOwnProperty(serviceName)) {
        if (cache[serviceName] === INSTANTIATING) {
          throw $injectorMinErr('cdep', 'Circular dependency found: {0}',
                    serviceName + ' <- ' + path.join(' <- '));
        }
        return cache[serviceName];
      } else {
        try {
          path.unshift(serviceName);
          cache[serviceName] = INSTANTIATING;
          return cache[serviceName] = factory(serviceName, caller);
        } catch (err) {
          if (cache[serviceName] === INSTANTIATING) {
            delete cache[serviceName];
          }
          throw err;
        } finally {
          path.shift();
        }
      }
    }

    function invoke(fn, self, locals, serviceName) {
      if (typeof locals === 'string') {
        serviceName = locals;
        locals = null;
      }

      var args = [],
          $inject = createInjector.$$annotate(fn, strictDi, serviceName),
          length, i,
          key;

      for (i = 0, length = $inject.length; i < length; i++) {
        key = $inject[i];
        if (typeof key !== 'string') {
          throw $injectorMinErr('itkn',
                  'Incorrect injection token! Expected service name as string, got {0}', key);
        }
        args.push(
          locals && locals.hasOwnProperty(key)
          ? locals[key]
          : getService(key, serviceName)
        );
      }
      if (isArray(fn)) {
        fn = fn[length];
      }

      // http://jsperf.com/angularjs-invoke-apply-vs-switch
      // #5388
      return fn.apply(self, args);
    }

    function instantiate(Type, locals, serviceName) {
      // Check if Type is annotated and use just the given function at n-1 as parameter
      // e.g. someModule.factory('greeter', ['$window', function(renamed$window) {}]);
      // Object creation: http://jsperf.com/create-constructor/2
      var instance = Object.create((isArray(Type) ? Type[Type.length - 1] : Type).prototype || null);
      var returnedValue = invoke(Type, instance, locals, serviceName);

      return isObject(returnedValue) || isFunction(returnedValue) ? returnedValue : instance;
    }

    return {
      invoke: invoke,
      instantiate: instantiate,
      get: getService,
      annotate: createInjector.$$annotate,
      has: function(name) {
        return providerCache.hasOwnProperty(name + providerSuffix) || cache.hasOwnProperty(name);
      }
    };
  }
}

createInjector.$$annotate = annotate;

/**
 * @ngdoc provider
 * @name $anchorScrollProvider
 *
 * @description
 * Use `$anchorScrollProvider` to disable automatic scrolling whenever
 * {@link ng.$location#hash $location.hash()} changes.
 */
function $AnchorScrollProvider() {

  var autoScrollingEnabled = true;

  /**
   * @ngdoc method
   * @name $anchorScrollProvider#disableAutoScrolling
   *
   * @description
   * By default, {@link ng.$anchorScroll $anchorScroll()} will automatically detect changes to
   * {@link ng.$location#hash $location.hash()} and scroll to the element matching the new hash.<br />
   * Use this method to disable automatic scrolling.
   *
   * If automatic scrolling is disabled, one must explicitly call
   * {@link ng.$anchorScroll $anchorScroll()} in order to scroll to the element related to the
   * current hash.
   */
  this.disableAutoScrolling = function() {
    autoScrollingEnabled = false;
  };

  /**
   * @ngdoc service
   * @name $anchorScroll
   * @kind function
   * @requires $window
   * @requires $location
   * @requires $rootScope
   *
   * @description
   * When called, it scrolls to the element related to the specified `hash` or (if omitted) to the
   * current value of {@link ng.$location#hash $location.hash()}, according to the rules specified
   * in the
   * [HTML5 spec](http://www.w3.org/html/wg/drafts/html/master/browsers.html#the-indicated-part-of-the-document).
   *
   * It also watches the {@link ng.$location#hash $location.hash()} and automatically scrolls to
   * match any anchor whenever it changes. This can be disabled by calling
   * {@link ng.$anchorScrollProvider#disableAutoScrolling $anchorScrollProvider.disableAutoScrolling()}.
   *
   * Additionally, you can use its {@link ng.$anchorScroll#yOffset yOffset} property to specify a
   * vertical scroll-offset (either fixed or dynamic).
   *
   * @param {string=} hash The hash specifying the element to scroll to. If omitted, the value of
   *                       {@link ng.$location#hash $location.hash()} will be used.
   *
   * @property {(number|function|jqLite)} yOffset
   * If set, specifies a vertical scroll-offset. This is often useful when there are fixed
   * positioned elements at the top of the page, such as navbars, headers etc.
   *
   * `yOffset` can be specified in various ways:
   * - **number**: A fixed number of pixels to be used as offset.<br /><br />
   * - **function**: A getter function called everytime `$anchorScroll()` is executed. Must return
   *   a number representing the offset (in pixels).<br /><br />
   * - **jqLite**: A jqLite/jQuery element to be used for specifying the offset. The distance from
   *   the top of the page to the element's bottom will be used as offset.<br />
   *   **Note**: The element will be taken into account only as long as its `position` is set to
   *   `fixed`. This option is useful, when dealing with responsive navbars/headers that adjust
   *   their height and/or positioning according to the viewport's size.
   *
   * <br />
   * <div class="alert alert-warning">
   * In order for `yOffset` to work properly, scrolling should take place on the document's root and
   * not some child element.
   * </div>
   *
   * @example
     <example module="anchorScrollExample">
       <file name="index.html">
         <div id="scrollArea" ng-controller="ScrollController">
           <a ng-click="gotoBottom()">Go to bottom</a>
           <a id="bottom"></a> You're at the bottom!
         </div>
       </file>
       <file name="script.js">
         angular.module('anchorScrollExample', [])
           .controller('ScrollController', ['$scope', '$location', '$anchorScroll',
             function ($scope, $location, $anchorScroll) {
               $scope.gotoBottom = function() {
                 // set the location.hash to the id of
                 // the element you wish to scroll to.
                 $location.hash('bottom');

                 // call $anchorScroll()
                 $anchorScroll();
               };
             }]);
       </file>
       <file name="style.css">
         #scrollArea {
           height: 280px;
           overflow: auto;
         }

         #bottom {
           display: block;
           margin-top: 2000px;
         }
       </file>
     </example>
   *
   * <hr />
   * The example below illustrates the use of a vertical scroll-offset (specified as a fixed value).
   * See {@link ng.$anchorScroll#yOffset $anchorScroll.yOffset} for more details.
   *
   * @example
     <example module="anchorScrollOffsetExample">
       <file name="index.html">
         <div class="fixed-header" ng-controller="headerCtrl">
           <a href="" ng-click="gotoAnchor(x)" ng-repeat="x in [1,2,3,4,5]">
             Go to anchor {{x}}
           </a>
         </div>
         <div id="anchor{{x}}" class="anchor" ng-repeat="x in [1,2,3,4,5]">
           Anchor {{x}} of 5
         </div>
       </file>
       <file name="script.js">
         angular.module('anchorScrollOffsetExample', [])
           .run(['$anchorScroll', function($anchorScroll) {
             $anchorScroll.yOffset = 50;   // always scroll by 50 extra pixels
           }])
           .controller('headerCtrl', ['$anchorScroll', '$location', '$scope',
             function ($anchorScroll, $location, $scope) {
               $scope.gotoAnchor = function(x) {
                 var newHash = 'anchor' + x;
                 if ($location.hash() !== newHash) {
                   // set the $location.hash to `newHash` and
                   // $anchorScroll will automatically scroll to it
                   $location.hash('anchor' + x);
                 } else {
                   // call $anchorScroll() explicitly,
                   // since $location.hash hasn't changed
                   $anchorScroll();
                 }
               };
             }
           ]);
       </file>
       <file name="style.css">
         body {
           padding-top: 50px;
         }

         .anchor {
           border: 2px dashed DarkOrchid;
           padding: 10px 10px 200px 10px;
         }

         .fixed-header {
           background-color: rgba(0, 0, 0, 0.2);
           height: 50px;
           position: fixed;
           top: 0; left: 0; right: 0;
         }

         .fixed-header > a {
           display: inline-block;
           margin: 5px 15px;
         }
       </file>
     </example>
   */
  this.$get = ['$window', '$location', '$rootScope', function($window, $location, $rootScope) {
    var document = $window.document;

    // Helper function to get first anchor from a NodeList
    // (using `Array#some()` instead of `angular#forEach()` since it's more performant
    //  and working in all supported browsers.)
    function getFirstAnchor(list) {
      var result = null;
      Array.prototype.some.call(list, function(element) {
        if (nodeName_(element) === 'a') {
          result = element;
          return true;
        }
      });
      return result;
    }

    function getYOffset() {

      var offset = scroll.yOffset;

      if (isFunction(offset)) {
        offset = offset();
      } else if (isElement(offset)) {
        var elem = offset[0];
        var style = $window.getComputedStyle(elem);
        if (style.position !== 'fixed') {
          offset = 0;
        } else {
          offset = elem.getBoundingClientRect().bottom;
        }
      } else if (!isNumber(offset)) {
        offset = 0;
      }

      return offset;
    }

    function scrollTo(elem) {
      if (elem) {
        elem.scrollIntoView();

        var offset = getYOffset();

        if (offset) {
          // `offset` is the number of pixels we should scroll UP in order to align `elem` properly.
          // This is true ONLY if the call to `elem.scrollIntoView()` initially aligns `elem` at the
          // top of the viewport.
          //
          // IF the number of pixels from the top of `elem` to the end of the page's content is less
          // than the height of the viewport, then `elem.scrollIntoView()` will align the `elem` some
          // way down the page.
          //
          // This is often the case for elements near the bottom of the page.
          //
          // In such cases we do not need to scroll the whole `offset` up, just the difference between
          // the top of the element and the offset, which is enough to align the top of `elem` at the
          // desired position.
          var elemTop = elem.getBoundingClientRect().top;
          $window.scrollBy(0, elemTop - offset);
        }
      } else {
        $window.scrollTo(0, 0);
      }
    }

    function scroll(hash) {
      hash = isString(hash) ? hash : $location.hash();
      var elm;

      // empty hash, scroll to the top of the page
      if (!hash) scrollTo(null);

      // element with given id
      else if ((elm = document.getElementById(hash))) scrollTo(elm);

      // first anchor with given name :-D
      else if ((elm = getFirstAnchor(document.getElementsByName(hash)))) scrollTo(elm);

      // no element and hash == 'top', scroll to the top of the page
      else if (hash === 'top') scrollTo(null);
    }

    // does not scroll when user clicks on anchor link that is currently on
    // (no url change, no $location.hash() change), browser native does scroll
    if (autoScrollingEnabled) {
      $rootScope.$watch(function autoScrollWatch() {return $location.hash();},
        function autoScrollWatchAction(newVal, oldVal) {
          // skip the initial scroll if $location.hash is empty
          if (newVal === oldVal && newVal === '') return;

          jqLiteDocumentLoaded(function() {
            $rootScope.$evalAsync(scroll);
          });
        });
    }

    return scroll;
  }];
}

var $animateMinErr = minErr('$animate');
var ELEMENT_NODE = 1;
var NG_ANIMATE_CLASSNAME = 'ng-animate';

function mergeClasses(a,b) {
  if (!a && !b) return '';
  if (!a) return b;
  if (!b) return a;
  if (isArray(a)) a = a.join(' ');
  if (isArray(b)) b = b.join(' ');
  return a + ' ' + b;
}

function extractElementNode(element) {
  for (var i = 0; i < element.length; i++) {
    var elm = element[i];
    if (elm.nodeType === ELEMENT_NODE) {
      return elm;
    }
  }
}

function splitClasses(classes) {
  if (isString(classes)) {
    classes = classes.split(' ');
  }

  // Use createMap() to prevent class assumptions involving property names in
  // Object.prototype
  var obj = createMap();
  forEach(classes, function(klass) {
    // sometimes the split leaves empty string values
    // incase extra spaces were applied to the options
    if (klass.length) {
      obj[klass] = true;
    }
  });
  return obj;
}

// if any other type of options value besides an Object value is
// passed into the $animate.method() animation then this helper code
// will be run which will ignore it. While this patch is not the
// greatest solution to this, a lot of existing plugins depend on
// $animate to either call the callback (< 1.2) or return a promise
// that can be changed. This helper function ensures that the options
// are wiped clean incase a callback function is provided.
function prepareAnimateOptions(options) {
  return isObject(options)
      ? options
      : {};
}

var $$CoreAnimateRunnerProvider = function() {
  this.$get = ['$q', '$$rAF', function($q, $$rAF) {
    function AnimateRunner() {}
    AnimateRunner.all = noop;
    AnimateRunner.chain = noop;
    AnimateRunner.prototype = {
      end: noop,
      cancel: noop,
      resume: noop,
      pause: noop,
      complete: noop,
      then: function(pass, fail) {
        return $q(function(resolve) {
          $$rAF(function() {
            resolve();
          });
        }).then(pass, fail);
      }
    };
    return AnimateRunner;
  }];
};

// this is prefixed with Core since it conflicts with
// the animateQueueProvider defined in ngAnimate/animateQueue.js
var $$CoreAnimateQueueProvider = function() {
  var postDigestQueue = new HashMap();
  var postDigestElements = [];

  this.$get = ['$$AnimateRunner', '$rootScope',
       function($$AnimateRunner,   $rootScope) {
    return {
      enabled: noop,
      on: noop,
      off: noop,
      pin: noop,

      push: function(element, event, options, domOperation) {
        domOperation        && domOperation();

        options = options || {};
        options.from        && element.css(options.from);
        options.to          && element.css(options.to);

        if (options.addClass || options.removeClass) {
          addRemoveClassesPostDigest(element, options.addClass, options.removeClass);
        }

        return new $$AnimateRunner(); // jshint ignore:line
      }
    };


    function updateData(data, classes, value) {
      var changed = false;
      if (classes) {
        classes = isString(classes) ? classes.split(' ') :
                  isArray(classes) ? classes : [];
        forEach(classes, function(className) {
          if (className) {
            changed = true;
            data[className] = value;
          }
        });
      }
      return changed;
    }

    function handleCSSClassChanges() {
      forEach(postDigestElements, function(element) {
        var data = postDigestQueue.get(element);
        if (data) {
          var existing = splitClasses(element.attr('class'));
          var toAdd = '';
          var toRemove = '';
          forEach(data, function(status, className) {
            var hasClass = !!existing[className];
            if (status !== hasClass) {
              if (status) {
                toAdd += (toAdd.length ? ' ' : '') + className;
              } else {
                toRemove += (toRemove.length ? ' ' : '') + className;
              }
            }
          });

          forEach(element, function(elm) {
            toAdd    && jqLiteAddClass(elm, toAdd);
            toRemove && jqLiteRemoveClass(elm, toRemove);
          });
          postDigestQueue.remove(element);
        }
      });
      postDigestElements.length = 0;
    }


    function addRemoveClassesPostDigest(element, add, remove) {
      var data = postDigestQueue.get(element) || {};

      var classesAdded = updateData(data, add, true);
      var classesRemoved = updateData(data, remove, false);

      if (classesAdded || classesRemoved) {

        postDigestQueue.put(element, data);
        postDigestElements.push(element);

        if (postDigestElements.length === 1) {
          $rootScope.$$postDigest(handleCSSClassChanges);
        }
      }
    }
  }];
};

/**
 * @ngdoc provider
 * @name $animateProvider
 *
 * @description
 * Default implementation of $animate that doesn't perform any animations, instead just
 * synchronously performs DOM updates and resolves the returned runner promise.
 *
 * In order to enable animations the `ngAnimate` module has to be loaded.
 *
 * To see the functional implementation check out `src/ngAnimate/animate.js`.
 */
var $AnimateProvider = ['$provide', function($provide) {
  var provider = this;

  this.$$registeredAnimations = Object.create(null);

   /**
   * @ngdoc method
   * @name $animateProvider#register
   *
   * @description
   * Registers a new injectable animation factory function. The factory function produces the
   * animation object which contains callback functions for each event that is expected to be
   * animated.
   *
   *   * `eventFn`: `function(element, ... , doneFunction, options)`
   *   The element to animate, the `doneFunction` and the options fed into the animation. Depending
   *   on the type of animation additional arguments will be injected into the animation function. The
   *   list below explains the function signatures for the different animation methods:
   *
   *   - setClass: function(element, addedClasses, removedClasses, doneFunction, options)
   *   - addClass: function(element, addedClasses, doneFunction, options)
   *   - removeClass: function(element, removedClasses, doneFunction, options)
   *   - enter, leave, move: function(element, doneFunction, options)
   *   - animate: function(element, fromStyles, toStyles, doneFunction, options)
   *
   *   Make sure to trigger the `doneFunction` once the animation is fully complete.
   *
   * ```js
   *   return {
   *     //enter, leave, move signature
   *     eventFn : function(element, done, options) {
   *       //code to run the animation
   *       //once complete, then run done()
   *       return function endFunction(wasCancelled) {
   *         //code to cancel the animation
   *       }
   *     }
   *   }
   * ```
   *
   * @param {string} name The name of the animation (this is what the class-based CSS value will be compared to).
   * @param {Function} factory The factory function that will be executed to return the animation
   *                           object.
   */
  this.register = function(name, factory) {
    if (name && name.charAt(0) !== '.') {
      throw $animateMinErr('notcsel', "Expecting class selector starting with '.' got '{0}'.", name);
    }

    var key = name + '-animation';
    provider.$$registeredAnimations[name.substr(1)] = key;
    $provide.factory(key, factory);
  };

  /**
   * @ngdoc method
   * @name $animateProvider#classNameFilter
   *
   * @description
   * Sets and/or returns the CSS class regular expression that is checked when performing
   * an animation. Upon bootstrap the classNameFilter value is not set at all and will
   * therefore enable $animate to attempt to perform an animation on any element that is triggered.
   * When setting the `classNameFilter` value, animations will only be performed on elements
   * that successfully match the filter expression. This in turn can boost performance
   * for low-powered devices as well as applications containing a lot of structural operations.
   * @param {RegExp=} expression The className expression which will be checked against all animations
   * @return {RegExp} The current CSS className expression value. If null then there is no expression value
   */
  this.classNameFilter = function(expression) {
    if (arguments.length === 1) {
      this.$$classNameFilter = (expression instanceof RegExp) ? expression : null;
      if (this.$$classNameFilter) {
        var reservedRegex = new RegExp("(\\s+|\\/)" + NG_ANIMATE_CLASSNAME + "(\\s+|\\/)");
        if (reservedRegex.test(this.$$classNameFilter.toString())) {
          throw $animateMinErr('nongcls','$animateProvider.classNameFilter(regex) prohibits accepting a regex value which matches/contains the "{0}" CSS class.', NG_ANIMATE_CLASSNAME);

        }
      }
    }
    return this.$$classNameFilter;
  };

  this.$get = ['$$animateQueue', function($$animateQueue) {
    function domInsert(element, parentElement, afterElement) {
      // if for some reason the previous element was removed
      // from the dom sometime before this code runs then let's
      // just stick to using the parent element as the anchor
      if (afterElement) {
        var afterNode = extractElementNode(afterElement);
        if (afterNode && !afterNode.parentNode && !afterNode.previousElementSibling) {
          afterElement = null;
        }
      }
      afterElement ? afterElement.after(element) : parentElement.prepend(element);
    }

    /**
     * @ngdoc service
     * @name $animate
     * @description The $animate service exposes a series of DOM utility methods that provide support
     * for animation hooks. The default behavior is the application of DOM operations, however,
     * when an animation is detected (and animations are enabled), $animate will do the heavy lifting
     * to ensure that animation runs with the triggered DOM operation.
     *
     * By default $animate doesn't trigger any animations. This is because the `ngAnimate` module isn't
     * included and only when it is active then the animation hooks that `$animate` triggers will be
     * functional. Once active then all structural `ng-` directives will trigger animations as they perform
     * their DOM-related operations (enter, leave and move). Other directives such as `ngClass`,
     * `ngShow`, `ngHide` and `ngMessages` also provide support for animations.
     *
     * It is recommended that the`$animate` service is always used when executing DOM-related procedures within directives.
     *
     * To learn more about enabling animation support, click here to visit the
     * {@link ngAnimate ngAnimate module page}.
     */
    return {
      // we don't call it directly since non-existant arguments may
      // be interpreted as null within the sub enabled function

      /**
       *
       * @ngdoc method
       * @name $animate#on
       * @kind function
       * @description Sets up an event listener to fire whenever the animation event (enter, leave, move, etc...)
       *    has fired on the given element or among any of its children. Once the listener is fired, the provided callback
       *    is fired with the following params:
       *
       * ```js
       * $animate.on('enter', container,
       *    function callback(element, phase) {
       *      // cool we detected an enter animation within the container
       *    }
       * );
       * ```
       *
       * @param {string} event the animation event that will be captured (e.g. enter, leave, move, addClass, removeClass, etc...)
       * @param {DOMElement} container the container element that will capture each of the animation events that are fired on itself
       *     as well as among its children
       * @param {Function} callback the callback function that will be fired when the listener is triggered
       *
       * The arguments present in the callback function are:
       * * `element` - The captured DOM element that the animation was fired on.
       * * `phase` - The phase of the animation. The two possible phases are **start** (when the animation starts) and **close** (when it ends).
       */
      on: $$animateQueue.on,

      /**
       *
       * @ngdoc method
       * @name $animate#off
       * @kind function
       * @description Deregisters an event listener based on the event which has been associated with the provided element. This method
       * can be used in three different ways depending on the arguments:
       *
       * ```js
       * // remove all the animation event listeners listening for `enter`
       * $animate.off('enter');
       *
       * // remove all the animation event listeners listening for `enter` on the given element and its children
       * $animate.off('enter', container);
       *
       * // remove the event listener function provided by `listenerFn` that is set
       * // to listen for `enter` on the given `element` as well as its children
       * $animate.off('enter', container, callback);
       * ```
       *
       * @param {string} event the animation event (e.g. enter, leave, move, addClass, removeClass, etc...)
       * @param {DOMElement=} container the container element the event listener was placed on
       * @param {Function=} callback the callback function that was registered as the listener
       */
      off: $$animateQueue.off,

      /**
       * @ngdoc method
       * @name $animate#pin
       * @kind function
       * @description Associates the provided element with a host parent element to allow the element to be animated even if it exists
       *    outside of the DOM structure of the Angular application. By doing so, any animation triggered via `$animate` can be issued on the
       *    element despite being outside the realm of the application or within another application. Say for example if the application
       *    was bootstrapped on an element that is somewhere inside of the `<body>` tag, but we wanted to allow for an element to be situated
       *    as a direct child of `document.body`, then this can be achieved by pinning the element via `$animate.pin(element)`. Keep in mind
       *    that calling `$animate.pin(element, parentElement)` will not actually insert into the DOM anywhere; it will just create the association.
       *
       *    Note that this feature is only active when the `ngAnimate` module is used.
       *
       * @param {DOMElement} element the external element that will be pinned
       * @param {DOMElement} parentElement the host parent element that will be associated with the external element
       */
      pin: $$animateQueue.pin,

      /**
       *
       * @ngdoc method
       * @name $animate#enabled
       * @kind function
       * @description Used to get and set whether animations are enabled or not on the entire application or on an element and its children. This
       * function can be called in four ways:
       *
       * ```js
       * // returns true or false
       * $animate.enabled();
       *
       * // changes the enabled state for all animations
       * $animate.enabled(false);
       * $animate.enabled(true);
       *
       * // returns true or false if animations are enabled for an element
       * $animate.enabled(element);
       *
       * // changes the enabled state for an element and its children
       * $animate.enabled(element, true);
       * $animate.enabled(element, false);
       * ```
       *
       * @param {DOMElement=} element the element that will be considered for checking/setting the enabled state
       * @param {boolean=} enabled whether or not the animations will be enabled for the element
       *
       * @return {boolean} whether or not animations are enabled
       */
      enabled: $$animateQueue.enabled,

      /**
       * @ngdoc method
       * @name $animate#cancel
       * @kind function
       * @description Cancels the provided animation.
       *
       * @param {Promise} animationPromise The animation promise that is returned when an animation is started.
       */
      cancel: function(runner) {
        runner.end && runner.end();
      },

      /**
       *
       * @ngdoc method
       * @name $animate#enter
       * @kind function
       * @description Inserts the element into the DOM either after the `after` element (if provided) or
       *   as the first child within the `parent` element and then triggers an animation.
       *   A promise is returned that will be resolved during the next digest once the animation
       *   has completed.
       *
       * @param {DOMElement} element the element which will be inserted into the DOM
       * @param {DOMElement} parent the parent element which will append the element as
       *   a child (so long as the after element is not present)
       * @param {DOMElement=} after the sibling element after which the element will be appended
       * @param {object=} options an optional collection of options/styles that will be applied to the element
       *
       * @return {Promise} the animation callback promise
       */
      enter: function(element, parent, after, options) {
        parent = parent && jqLite(parent);
        after = after && jqLite(after);
        parent = parent || after.parent();
        domInsert(element, parent, after);
        return $$animateQueue.push(element, 'enter', prepareAnimateOptions(options));
      },

      /**
       *
       * @ngdoc method
       * @name $animate#move
       * @kind function
       * @description Inserts (moves) the element into its new position in the DOM either after
       *   the `after` element (if provided) or as the first child within the `parent` element
       *   and then triggers an animation. A promise is returned that will be resolved
       *   during the next digest once the animation has completed.
       *
       * @param {DOMElement} element the element which will be moved into the new DOM position
       * @param {DOMElement} parent the parent element which will append the element as
       *   a child (so long as the after element is not present)
       * @param {DOMElement=} after the sibling element after which the element will be appended
       * @param {object=} options an optional collection of options/styles that will be applied to the element
       *
       * @return {Promise} the animation callback promise
       */
      move: function(element, parent, after, options) {
        parent = parent && jqLite(parent);
        after = after && jqLite(after);
        parent = parent || after.parent();
        domInsert(element, parent, after);
        return $$animateQueue.push(element, 'move', prepareAnimateOptions(options));
      },

      /**
       * @ngdoc method
       * @name $animate#leave
       * @kind function
       * @description Triggers an animation and then removes the element from the DOM.
       * When the function is called a promise is returned that will be resolved during the next
       * digest once the animation has completed.
       *
       * @param {DOMElement} element the element which will be removed from the DOM
       * @param {object=} options an optional collection of options/styles that will be applied to the element
       *
       * @return {Promise} the animation callback promise
       */
      leave: function(element, options) {
        return $$animateQueue.push(element, 'leave', prepareAnimateOptions(options), function() {
          element.remove();
        });
      },

      /**
       * @ngdoc method
       * @name $animate#addClass
       * @kind function
       *
       * @description Triggers an addClass animation surrounding the addition of the provided CSS class(es). Upon
       *   execution, the addClass operation will only be handled after the next digest and it will not trigger an
       *   animation if element already contains the CSS class or if the class is removed at a later step.
       *   Note that class-based animations are treated differently compared to structural animations
       *   (like enter, move and leave) since the CSS classes may be added/removed at different points
       *   depending if CSS or JavaScript animations are used.
       *
       * @param {DOMElement} element the element which the CSS classes will be applied to
       * @param {string} className the CSS class(es) that will be added (multiple classes are separated via spaces)
       * @param {object=} options an optional collection of options/styles that will be applied to the element
       *
       * @return {Promise} the animation callback promise
       */
      addClass: function(element, className, options) {
        options = prepareAnimateOptions(options);
        options.addClass = mergeClasses(options.addclass, className);
        return $$animateQueue.push(element, 'addClass', options);
      },

      /**
       * @ngdoc method
       * @name $animate#removeClass
       * @kind function
       *
       * @description Triggers a removeClass animation surrounding the removal of the provided CSS class(es). Upon
       *   execution, the removeClass operation will only be handled after the next digest and it will not trigger an
       *   animation if element does not contain the CSS class or if the class is added at a later step.
       *   Note that class-based animations are treated differently compared to structural animations
       *   (like enter, move and leave) since the CSS classes may be added/removed at different points
       *   depending if CSS or JavaScript animations are used.
       *
       * @param {DOMElement} element the element which the CSS classes will be applied to
       * @param {string} className the CSS class(es) that will be removed (multiple classes are separated via spaces)
       * @param {object=} options an optional collection of options/styles that will be applied to the element
       *
       * @return {Promise} the animation callback promise
       */
      removeClass: function(element, className, options) {
        options = prepareAnimateOptions(options);
        options.removeClass = mergeClasses(options.removeClass, className);
        return $$animateQueue.push(element, 'removeClass', options);
      },

      /**
       * @ngdoc method
       * @name $animate#setClass
       * @kind function
       *
       * @description Performs both the addition and removal of a CSS classes on an element and (during the process)
       *    triggers an animation surrounding the class addition/removal. Much like `$animate.addClass` and
       *    `$animate.removeClass`, `setClass` will only evaluate the classes being added/removed once a digest has
       *    passed. Note that class-based animations are treated differently compared to structural animations
       *    (like enter, move and leave) since the CSS classes may be added/removed at different points
       *    depending if CSS or JavaScript animations are used.
       *
       * @param {DOMElement} element the element which the CSS classes will be applied to
       * @param {string} add the CSS class(es) that will be added (multiple classes are separated via spaces)
       * @param {string} remove the CSS class(es) that will be removed (multiple classes are separated via spaces)
       * @param {object=} options an optional collection of options/styles that will be applied to the element
       *
       * @return {Promise} the animation callback promise
       */
      setClass: function(element, add, remove, options) {
        options = prepareAnimateOptions(options);
        options.addClass = mergeClasses(options.addClass, add);
        options.removeClass = mergeClasses(options.removeClass, remove);
        return $$animateQueue.push(element, 'setClass', options);
      },

      /**
       * @ngdoc method
       * @name $animate#animate
       * @kind function
       *
       * @description Performs an inline animation on the element which applies the provided to and from CSS styles to the element.
       * If any detected CSS transition, keyframe or JavaScript matches the provided className value then the animation will take
       * on the provided styles. For example, if a transition animation is set for the given className then the provided from and
       * to styles will be applied alongside the given transition. If a JavaScript animation is detected then the provided styles
       * will be given in as function paramters into the `animate` method (or as apart of the `options` parameter).
       *
       * @param {DOMElement} element the element which the CSS styles will be applied to
       * @param {object} from the from (starting) CSS styles that will be applied to the element and across the animation.
       * @param {object} to the to (destination) CSS styles that will be applied to the element and across the animation.
       * @param {string=} className an optional CSS class that will be applied to the element for the duration of the animation. If
       *    this value is left as empty then a CSS class of `ng-inline-animate` will be applied to the element.
       *    (Note that if no animation is detected then this value will not be appplied to the element.)
       * @param {object=} options an optional collection of options/styles that will be applied to the element
       *
       * @return {Promise} the animation callback promise
       */
      animate: function(element, from, to, className, options) {
        options = prepareAnimateOptions(options);
        options.from = options.from ? extend(options.from, from) : from;
        options.to   = options.to   ? extend(options.to, to)     : to;

        className = className || 'ng-inline-animate';
        options.tempClasses = mergeClasses(options.tempClasses, className);
        return $$animateQueue.push(element, 'animate', options);
      }
    };
  }];
}];

/**
 * @ngdoc service
 * @name $animateCss
 * @kind object
 *
 * @description
 * This is the core version of `$animateCss`. By default, only when the `ngAnimate` is included,
 * then the `$animateCss` service will actually perform animations.
 *
 * Click here {@link ngAnimate.$animateCss to read the documentation for $animateCss}.
 */
var $CoreAnimateCssProvider = function() {
  this.$get = ['$$rAF', '$q', function($$rAF, $q) {

    var RAFPromise = function() {};
    RAFPromise.prototype = {
      done: function(cancel) {
        this.defer && this.defer[cancel === true ? 'reject' : 'resolve']();
      },
      end: function() {
        this.done();
      },
      cancel: function() {
        this.done(true);
      },
      getPromise: function() {
        if (!this.defer) {
          this.defer = $q.defer();
        }
        return this.defer.promise;
      },
      then: function(f1,f2) {
        return this.getPromise().then(f1,f2);
      },
      'catch': function(f1) {
        return this.getPromise()['catch'](f1);
      },
      'finally': function(f1) {
        return this.getPromise()['finally'](f1);
      }
    };

    return function(element, options) {
      // there is no point in applying the styles since
      // there is no animation that goes on at all in
      // this version of $animateCss.
      if (options.cleanupStyles) {
        options.from = options.to = null;
      }

      if (options.from) {
        element.css(options.from);
        options.from = null;
      }

      var closed, runner = new RAFPromise();
      return {
        start: run,
        end: run
      };

      function run() {
        $$rAF(function() {
          close();
          if (!closed) {
            runner.done();
          }
          closed = true;
        });
        return runner;
      }

      function close() {
        if (options.addClass) {
          element.addClass(options.addClass);
          options.addClass = null;
        }
        if (options.removeClass) {
          element.removeClass(options.removeClass);
          options.removeClass = null;
        }
        if (options.to) {
          element.css(options.to);
          options.to = null;
        }
      }
    };
  }];
};

/* global stripHash: true */

/**
 * ! This is a private undocumented service !
 *
 * @name $browser
 * @requires $log
 * @description
 * This object has two goals:
 *
 * - hide all the global state in the browser caused by the window object
 * - abstract away all the browser specific features and inconsistencies
 *
 * For tests we provide {@link ngMock.$browser mock implementation} of the `$browser`
 * service, which can be used for convenient testing of the application without the interaction with
 * the real browser apis.
 */
/**
 * @param {object} window The global window object.
 * @param {object} document jQuery wrapped document.
 * @param {object} $log window.console or an object with the same interface.
 * @param {object} $sniffer $sniffer service
 */
function Browser(window, document, $log, $sniffer) {
  var self = this,
      rawDocument = document[0],
      location = window.location,
      history = window.history,
      setTimeout = window.setTimeout,
      clearTimeout = window.clearTimeout,
      pendingDeferIds = {};

  self.isMock = false;

  var outstandingRequestCount = 0;
  var outstandingRequestCallbacks = [];

  // TODO(vojta): remove this temporary api
  self.$$completeOutstandingRequest = completeOutstandingRequest;
  self.$$incOutstandingRequestCount = function() { outstandingRequestCount++; };

  /**
   * Executes the `fn` function(supports currying) and decrements the `outstandingRequestCallbacks`
   * counter. If the counter reaches 0, all the `outstandingRequestCallbacks` are executed.
   */
  function completeOutstandingRequest(fn) {
    try {
      fn.apply(null, sliceArgs(arguments, 1));
    } finally {
      outstandingRequestCount--;
      if (outstandingRequestCount === 0) {
        while (outstandingRequestCallbacks.length) {
          try {
            outstandingRequestCallbacks.pop()();
          } catch (e) {
            $log.error(e);
          }
        }
      }
    }
  }

  function getHash(url) {
    var index = url.indexOf('#');
    return index === -1 ? '' : url.substr(index);
  }

  /**
   * @private
   * Note: this method is used only by scenario runner
   * TODO(vojta): prefix this method with $$ ?
   * @param {function()} callback Function that will be called when no outstanding request
   */
  self.notifyWhenNoOutstandingRequests = function(callback) {
    if (outstandingRequestCount === 0) {
      callback();
    } else {
      outstandingRequestCallbacks.push(callback);
    }
  };

  //////////////////////////////////////////////////////////////
  // URL API
  //////////////////////////////////////////////////////////////

  var cachedState, lastHistoryState,
      lastBrowserUrl = location.href,
      baseElement = document.find('base'),
      pendingLocation = null;

  cacheState();
  lastHistoryState = cachedState;

  /**
   * @name $browser#url
   *
   * @description
   * GETTER:
   * Without any argument, this method just returns current value of location.href.
   *
   * SETTER:
   * With at least one argument, this method sets url to new value.
   * If html5 history api supported, pushState/replaceState is used, otherwise
   * location.href/location.replace is used.
   * Returns its own instance to allow chaining
   *
   * NOTE: this api is intended for use only by the $location service. Please use the
   * {@link ng.$location $location service} to change url.
   *
   * @param {string} url New url (when used as setter)
   * @param {boolean=} replace Should new url replace current history record?
   * @param {object=} state object to use with pushState/replaceState
   */
  self.url = function(url, replace, state) {
    // In modern browsers `history.state` is `null` by default; treating it separately
    // from `undefined` would cause `$browser.url('/foo')` to change `history.state`
    // to undefined via `pushState`. Instead, let's change `undefined` to `null` here.
    if (isUndefined(state)) {
      state = null;
    }

    // Android Browser BFCache causes location, history reference to become stale.
    if (location !== window.location) location = window.location;
    if (history !== window.history) history = window.history;

    // setter
    if (url) {
      var sameState = lastHistoryState === state;

      // Don't change anything if previous and current URLs and states match. This also prevents
      // IE<10 from getting into redirect loop when in LocationHashbangInHtml5Url mode.
      // See https://github.com/angular/angular.js/commit/ffb2701
      if (lastBrowserUrl === url && (!$sniffer.history || sameState)) {
        return self;
      }
      var sameBase = lastBrowserUrl && stripHash(lastBrowserUrl) === stripHash(url);
      lastBrowserUrl = url;
      lastHistoryState = state;
      // Don't use history API if only the hash changed
      // due to a bug in IE10/IE11 which leads
      // to not firing a `hashchange` nor `popstate` event
      // in some cases (see #9143).
      if ($sniffer.history && (!sameBase || !sameState)) {
        history[replace ? 'replaceState' : 'pushState'](state, '', url);
        cacheState();
        // Do the assignment again so that those two variables are referentially identical.
        lastHistoryState = cachedState;
      } else {
        if (!sameBase || pendingLocation) {
          pendingLocation = url;
        }
        if (replace) {
          location.replace(url);
        } else if (!sameBase) {
          location.href = url;
        } else {
          location.hash = getHash(url);
        }
        if (location.href !== url) {
          pendingLocation = url;
        }
      }
      return self;
    // getter
    } else {
      // - pendingLocation is needed as browsers don't allow to read out
      //   the new location.href if a reload happened or if there is a bug like in iOS 9 (see
      //   https://openradar.appspot.com/22186109).
      // - the replacement is a workaround for https://bugzilla.mozilla.org/show_bug.cgi?id=407172
      return pendingLocation || location.href.replace(/%27/g,"'");
    }
  };

  /**
   * @name $browser#state
   *
   * @description
   * This method is a getter.
   *
   * Return history.state or null if history.state is undefined.
   *
   * @returns {object} state
   */
  self.state = function() {
    return cachedState;
  };

  var urlChangeListeners = [],
      urlChangeInit = false;

  function cacheStateAndFireUrlChange() {
    pendingLocation = null;
    cacheState();
    fireUrlChange();
  }

  function getCurrentState() {
    try {
      return history.state;
    } catch (e) {
      // MSIE can reportedly throw when there is no state (UNCONFIRMED).
    }
  }

  // This variable should be used *only* inside the cacheState function.
  var lastCachedState = null;
  function cacheState() {
    // This should be the only place in $browser where `history.state` is read.
    cachedState = getCurrentState();
    cachedState = isUndefined(cachedState) ? null : cachedState;

    // Prevent callbacks fo fire twice if both hashchange & popstate were fired.
    if (equals(cachedState, lastCachedState)) {
      cachedState = lastCachedState;
    }
    lastCachedState = cachedState;
  }

  function fireUrlChange() {
    if (lastBrowserUrl === self.url() && lastHistoryState === cachedState) {
      return;
    }

    lastBrowserUrl = self.url();
    lastHistoryState = cachedState;
    forEach(urlChangeListeners, function(listener) {
      listener(self.url(), cachedState);
    });
  }

  /**
   * @name $browser#onUrlChange
   *
   * @description
   * Register callback function that will be called, when url changes.
   *
   * It's only called when the url is changed from outside of angular:
   * - user types different url into address bar
   * - user clicks on history (forward/back) button
   * - user clicks on a link
   *
   * It's not called when url is changed by $browser.url() method
   *
   * The listener gets called with new url as parameter.
   *
   * NOTE: this api is intended for use only by the $location service. Please use the
   * {@link ng.$location $location service} to monitor url changes in angular apps.
   *
   * @param {function(string)} listener Listener function to be called when url changes.
   * @return {function(string)} Returns the registered listener fn - handy if the fn is anonymous.
   */
  self.onUrlChange = function(callback) {
    // TODO(vojta): refactor to use node's syntax for events
    if (!urlChangeInit) {
      // We listen on both (hashchange/popstate) when available, as some browsers (e.g. Opera)
      // don't fire popstate when user change the address bar and don't fire hashchange when url
      // changed by push/replaceState

      // html5 history api - popstate event
      if ($sniffer.history) jqLite(window).on('popstate', cacheStateAndFireUrlChange);
      // hashchange event
      jqLite(window).on('hashchange', cacheStateAndFireUrlChange);

      urlChangeInit = true;
    }

    urlChangeListeners.push(callback);
    return callback;
  };

  /**
   * @private
   * Remove popstate and hashchange handler from window.
   *
   * NOTE: this api is intended for use only by $rootScope.
   */
  self.$$applicationDestroyed = function() {
    jqLite(window).off('hashchange popstate', cacheStateAndFireUrlChange);
  };

  /**
   * Checks whether the url has changed outside of Angular.
   * Needs to be exported to be able to check for changes that have been done in sync,
   * as hashchange/popstate events fire in async.
   */
  self.$$checkUrlChange = fireUrlChange;

  //////////////////////////////////////////////////////////////
  // Misc API
  //////////////////////////////////////////////////////////////

  /**
   * @name $browser#baseHref
   *
   * @description
   * Returns current <base href>
   * (always relative - without domain)
   *
   * @returns {string} The current base href
   */
  self.baseHref = function() {
    var href = baseElement.attr('href');
    return href ? href.replace(/^(https?\:)?\/\/[^\/]*/, '') : '';
  };

  /**
   * @name $browser#defer
   * @param {function()} fn A function, who's execution should be deferred.
   * @param {number=} [delay=0] of milliseconds to defer the function execution.
   * @returns {*} DeferId that can be used to cancel the task via `$browser.defer.cancel()`.
   *
   * @description
   * Executes a fn asynchronously via `setTimeout(fn, delay)`.
   *
   * Unlike when calling `setTimeout` directly, in test this function is mocked and instead of using
   * `setTimeout` in tests, the fns are queued in an array, which can be programmatically flushed
   * via `$browser.defer.flush()`.
   *
   */
  self.defer = function(fn, delay) {
    var timeoutId;
    outstandingRequestCount++;
    timeoutId = setTimeout(function() {
      delete pendingDeferIds[timeoutId];
      completeOutstandingRequest(fn);
    }, delay || 0);
    pendingDeferIds[timeoutId] = true;
    return timeoutId;
  };


  /**
   * @name $browser#defer.cancel
   *
   * @description
   * Cancels a deferred task identified with `deferId`.
   *
   * @param {*} deferId Token returned by the `$browser.defer` function.
   * @returns {boolean} Returns `true` if the task hasn't executed yet and was successfully
   *                    canceled.
   */
  self.defer.cancel = function(deferId) {
    if (pendingDeferIds[deferId]) {
      delete pendingDeferIds[deferId];
      clearTimeout(deferId);
      completeOutstandingRequest(noop);
      return true;
    }
    return false;
  };

}

function $BrowserProvider() {
  this.$get = ['$window', '$log', '$sniffer', '$document',
      function($window, $log, $sniffer, $document) {
        return new Browser($window, $document, $log, $sniffer);
      }];
}

/**
 * @ngdoc service
 * @name $cacheFactory
 *
 * @description
 * Factory that constructs {@link $cacheFactory.Cache Cache} objects and gives access to
 * them.
 *
 * ```js
 *
 *  var cache = $cacheFactory('cacheId');
 *  expect($cacheFactory.get('cacheId')).toBe(cache);
 *  expect($cacheFactory.get('noSuchCacheId')).not.toBeDefined();
 *
 *  cache.put("key", "value");
 *  cache.put("another key", "another value");
 *
 *  // We've specified no options on creation
 *  expect(cache.info()).toEqual({id: 'cacheId', size: 2});
 *
 * ```
 *
 *
 * @param {string} cacheId Name or id of the newly created cache.
 * @param {object=} options Options object that specifies the cache behavior. Properties:
 *
 *   - `{number=}` `capacity` â€” turns the cache into LRU cache.
 *
 * @returns {object} Newly created cache object with the following set of methods:
 *
 * - `{object}` `info()` â€” Returns id, size, and options of cache.
 * - `{{*}}` `put({string} key, {*} value)` â€” Puts a new key-value pair into the cache and returns
 *   it.
 * - `{{*}}` `get({string} key)` â€” Returns cached value for `key` or undefined for cache miss.
 * - `{void}` `remove({string} key)` â€” Removes a key-value pair from the cache.
 * - `{void}` `removeAll()` â€” Removes all cached values.
 * - `{void}` `destroy()` â€” Removes references to this cache from $cacheFactory.
 *
 * @example
   <example module="cacheExampleApp">
     <file name="index.html">
       <div ng-controller="CacheController">
         <input ng-model="newCacheKey" placeholder="Key">
         <input ng-model="newCacheValue" placeholder="Value">
         <button ng-click="put(newCacheKey, newCacheValue)">Cache</button>

         <p ng-if="keys.length">Cached Values</p>
         <div ng-repeat="key in keys">
           <span ng-bind="key"></span>
           <span>: </span>
           <b ng-bind="cache.get(key)"></b>
         </div>

         <p>Cache Info</p>
         <div ng-repeat="(key, value) in cache.info()">
           <span ng-bind="key"></span>
           <span>: </span>
           <b ng-bind="value"></b>
         </div>
       </div>
     </file>
     <file name="script.js">
       angular.module('cacheExampleApp', []).
         controller('CacheController', ['$scope', '$cacheFactory', function($scope, $cacheFactory) {
           $scope.keys = [];
           $scope.cache = $cacheFactory('cacheId');
           $scope.put = function(key, value) {
             if (angular.isUndefined($scope.cache.get(key))) {
               $scope.keys.push(key);
             }
             $scope.cache.put(key, angular.isUndefined(value) ? null : value);
           };
         }]);
     </file>
     <file name="style.css">
       p {
         margin: 10px 0 3px;
       }
     </file>
   </example>
 */
function $CacheFactoryProvider() {

  this.$get = function() {
    var caches = {};

    function cacheFactory(cacheId, options) {
      if (cacheId in caches) {
        throw minErr('$cacheFactory')('iid', "CacheId '{0}' is already taken!", cacheId);
      }

      var size = 0,
          stats = extend({}, options, {id: cacheId}),
          data = createMap(),
          capacity = (options && options.capacity) || Number.MAX_VALUE,
          lruHash = createMap(),
          freshEnd = null,
          staleEnd = null;

      /**
       * @ngdoc type
       * @name $cacheFactory.Cache
       *
       * @description
       * A cache object used to store and retrieve data, primarily used by
       * {@link $http $http} and the {@link ng.directive:script script} directive to cache
       * templates and other data.
       *
       * ```js
       *  angular.module('superCache')
       *    .factory('superCache', ['$cacheFactory', function($cacheFactory) {
       *      return $cacheFactory('super-cache');
       *    }]);
       * ```
       *
       * Example test:
       *
       * ```js
       *  it('should behave like a cache', inject(function(superCache) {
       *    superCache.put('key', 'value');
       *    superCache.put('another key', 'another value');
       *
       *    expect(superCache.info()).toEqual({
       *      id: 'super-cache',
       *      size: 2
       *    });
       *
       *    superCache.remove('another key');
       *    expect(superCache.get('another key')).toBeUndefined();
       *
       *    superCache.removeAll();
       *    expect(superCache.info()).toEqual({
       *      id: 'super-cache',
       *      size: 0
       *    });
       *  }));
       * ```
       */
      return caches[cacheId] = {

        /**
         * @ngdoc method
         * @name $cacheFactory.Cache#put
         * @kind function
         *
         * @description
         * Inserts a named entry into the {@link $cacheFactory.Cache Cache} object to be
         * retrieved later, and incrementing the size of the cache if the key was not already
         * present in the cache. If behaving like an LRU cache, it will also remove stale
         * entries from the set.
         *
         * It will not insert undefined values into the cache.
         *
         * @param {string} key the key under which the cached data is stored.
         * @param {*} value the value to store alongside the key. If it is undefined, the key
         *    will not be stored.
         * @returns {*} the value stored.
         */
        put: function(key, value) {
          if (isUndefined(value)) return;
          if (capacity < Number.MAX_VALUE) {
            var lruEntry = lruHash[key] || (lruHash[key] = {key: key});

            refresh(lruEntry);
          }

          if (!(key in data)) size++;
          data[key] = value;

          if (size > capacity) {
            this.remove(staleEnd.key);
          }

          return value;
        },

        /**
         * @ngdoc method
         * @name $cacheFactory.Cache#get
         * @kind function
         *
         * @description
         * Retrieves named data stored in the {@link $cacheFactory.Cache Cache} object.
         *
         * @param {string} key the key of the data to be retrieved
         * @returns {*} the value stored.
         */
        get: function(key) {
          if (capacity < Number.MAX_VALUE) {
            var lruEntry = lruHash[key];

            if (!lruEntry) return;

            refresh(lruEntry);
          }

          return data[key];
        },


        /**
         * @ngdoc method
         * @name $cacheFactory.Cache#remove
         * @kind function
         *
         * @description
         * Removes an entry from the {@link $cacheFactory.Cache Cache} object.
         *
         * @param {string} key the key of the entry to be removed
         */
        remove: function(key) {
          if (capacity < Number.MAX_VALUE) {
            var lruEntry = lruHash[key];

            if (!lruEntry) return;

            if (lruEntry == freshEnd) freshEnd = lruEntry.p;
            if (lruEntry == staleEnd) staleEnd = lruEntry.n;
            link(lruEntry.n,lruEntry.p);

            delete lruHash[key];
          }

          if (!(key in data)) return;

          delete data[key];
          size--;
        },


        /**
         * @ngdoc method
         * @name $cacheFactory.Cache#removeAll
         * @kind function
         *
         * @description
         * Clears the cache object of any entries.
         */
        removeAll: function() {
          data = createMap();
          size = 0;
          lruHash = createMap();
          freshEnd = staleEnd = null;
        },


        /**
         * @ngdoc method
         * @name $cacheFactory.Cache#destroy
         * @kind function
         *
         * @description
         * Destroys the {@link $cacheFactory.Cache Cache} object entirely,
         * removing it from the {@link $cacheFactory $cacheFactory} set.
         */
        destroy: function() {
          data = null;
          stats = null;
          lruHash = null;
          delete caches[cacheId];
        },


        /**
         * @ngdoc method
         * @name $cacheFactory.Cache#info
         * @kind function
         *
         * @description
         * Retrieve information regarding a particular {@link $cacheFactory.Cache Cache}.
         *
         * @returns {object} an object with the following properties:
         *   <ul>
         *     <li>**id**: the id of the cache instance</li>
         *     <li>**size**: the number of entries kept in the cache instance</li>
         *     <li>**...**: any additional properties from the options object when creating the
         *       cache.</li>
         *   </ul>
         */
        info: function() {
          return extend({}, stats, {size: size});
        }
      };


      /**
       * makes the `entry` the freshEnd of the LRU linked list
       */
      function refresh(entry) {
        if (entry != freshEnd) {
          if (!staleEnd) {
            staleEnd = entry;
          } else if (staleEnd == entry) {
            staleEnd = entry.n;
          }

          link(entry.n, entry.p);
          link(entry, freshEnd);
          freshEnd = entry;
          freshEnd.n = null;
        }
      }


      /**
       * bidirectionally links two entries of the LRU linked list
       */
      function link(nextEntry, prevEntry) {
        if (nextEntry != prevEntry) {
          if (nextEntry) nextEntry.p = prevEntry; //p stands for previous, 'prev' didn't minify
          if (prevEntry) prevEntry.n = nextEntry; //n stands for next, 'next' didn't minify
        }
      }
    }


  /**
   * @ngdoc method
   * @name $cacheFactory#info
   *
   * @description
   * Get information about all the caches that have been created
   *
   * @returns {Object} - key-value map of `cacheId` to the result of calling `cache#info`
   */
    cacheFactory.info = function() {
      var info = {};
      forEach(caches, function(cache, cacheId) {
        info[cacheId] = cache.info();
      });
      return info;
    };


  /**
   * @ngdoc method
   * @name $cacheFactory#get
   *
   * @description
   * Get access to a cache object by the `cacheId` used when it was created.
   *
   * @param {string} cacheId Name or id of a cache to access.
   * @returns {object} Cache object identified by the cacheId or undefined if no such cache.
   */
    cacheFactory.get = function(cacheId) {
      return caches[cacheId];
    };


    return cacheFactory;
  };
}

/**
 * @ngdoc service
 * @name $templateCache
 *
 * @description
 * The first time a template is used, it is loaded in the template cache for quick retrieval. You
 * can load templates directly into the cache in a `script` tag, or by consuming the
 * `$templateCache` service directly.
 *
 * Adding via the `script` tag:
 *
 * ```html
 *   <script type="text/ng-template" id="templateId.html">
 *     <p>This is the content of the template</p>
 *   </script>
 * ```
 *
 * **Note:** the `script` tag containing the template does not need to be included in the `head` of
 * the document, but it must be a descendent of the {@link ng.$rootElement $rootElement} (IE,
 * element with ng-app attribute), otherwise the template will be ignored.
 *
 * Adding via the `$templateCache` service:
 *
 * ```js
 * var myApp = angular.module('myApp', []);
 * myApp.run(function($templateCache) {
 *   $templateCache.put('templateId.html', 'This is the content of the template');
 * });
 * ```
 *
 * To retrieve the template later, simply use it in your HTML:
 * ```html
 * <div ng-include=" 'templateId.html' "></div>
 * ```
 *
 * or get it via Javascript:
 * ```js
 * $templateCache.get('templateId.html')
 * ```
 *
 * See {@link ng.$cacheFactory $cacheFactory}.
 *
 */
function $TemplateCacheProvider() {
  this.$get = ['$cacheFactory', function($cacheFactory) {
    return $cacheFactory('templates');
  }];
}

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
 *     Any commits to this file should be reviewed with security in mind.  *
 *   Changes to this file can potentially create security vulnerabilities. *
 *          An approval from 2 Core members with history of modifying      *
 *                         this file is required.                          *
 *                                                                         *
 *  Does the change somehow allow for arbitrary javascript to be executed? *
 *    Or allows for someone to change the prototype of built-in objects?   *
 *     Or gives undesired access to variables likes document or window?    *
 * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */

/* ! VARIABLE/FUNCTION NAMING CONVENTIONS THAT APPLY TO THIS FILE!
 *
 * DOM-related variables:
 *
 * - "node" - DOM Node
 * - "element" - DOM Element or Node
 * - "$node" or "$element" - jqLite-wrapped node or element
 *
 *
 * Compiler related stuff:
 *
 * - "linkFn" - linking fn of a single directive
 * - "nodeLinkFn" - function that aggregates all linking fns for a particular node
 * - "childLinkFn" -  function that aggregates all linking fns for child nodes of a particular node
 * - "compositeLinkFn" - function that aggregates all linking fns for a compilation root (nodeList)
 */


/**
 * @ngdoc service
 * @name $compile
 * @kind function
 *
 * @description
 * Compiles an HTML string or DOM into a template and produces a template function, which
 * can then be used to link {@link ng.$rootScope.Scope `scope`} and the template together.
 *
 * The compilation is a process of walking the DOM tree and matching DOM elements to
 * {@link ng.$compileProvider#directive directives}.
 *
 * <div class="alert alert-warning">
 * **Note:** This document is an in-depth reference of all directive options.
 * For a gentle introduction to directives with examples of common use cases,
 * see the {@link guide/directive directive guide}.
 * </div>
 *
 * ## Comprehensive Directive API
 *
 * There are many different options for a directive.
 *
 * The difference resides in the return value of the factory function.
 * You can either return a "Directive Definition Object" (see below) that defines the directive properties,
 * or just the `postLink` function (all other properties will have the default values).
 *
 * <div class="alert alert-success">
 * **Best Practice:** It's recommended to use the "directive definition object" form.
 * </div>
 *
 * Here's an example directive declared with a Directive Definition Object:
 *
 * ```js
 *   var myModule = angular.module(...);
 *
 *   myModule.directive('directiveName', function factory(injectables) {
 *     var directiveDefinitionObject = {
 *       priority: 0,
 *       template: '<div></div>', // or // function(tElement, tAttrs) { ... },
 *       // or
 *       // templateUrl: 'directive.html', // or // function(tElement, tAttrs) { ... },
 *       transclude: false,
 *       restrict: 'A',
 *       templateNamespace: 'html',
 *       scope: false,
 *       controller: function($scope, $element, $attrs, $transclude, otherInjectables) { ... },
 *       controllerAs: 'stringIdentifier',
 *       bindToController: false,
 *       require: 'siblingDirectiveName', // or // ['^parentDirectiveName', '?optionalDirectiveName', '?^optionalParent'],
 *       compile: function compile(tElement, tAttrs, transclude) {
 *         return {
 *           pre: function preLink(scope, iElement, iAttrs, controller) { ... },
 *           post: function postLink(scope, iElement, iAttrs, controller) { ... }
 *         }
 *         // or
 *         // return function postLink( ... ) { ... }
 *       },
 *       // or
 *       // link: {
 *       //  pre: function preLink(scope, iElement, iAttrs, controller) { ... },
 *       //  post: function postLink(scope, iElement, iAttrs, controller) { ... }
 *       // }
 *       // or
 *       // link: function postLink( ... ) { ... }
 *     };
 *     return directiveDefinitionObject;
 *   });
 * ```
 *
 * <div class="alert alert-warning">
 * **Note:** Any unspecified options will use the default value. You can see the default values below.
 * </div>
 *
 * Therefore the above can be simplified as:
 *
 * ```js
 *   var myModule = angular.module(...);
 *
 *   myModule.directive('directiveName', function factory(injectables) {
 *     var directiveDefinitionObject = {
 *       link: function postLink(scope, iElement, iAttrs) { ... }
 *     };
 *     return directiveDefinitionObject;
 *     // or
 *     // return function postLink(scope, iElement, iAttrs) { ... }
 *   });
 * ```
 *
 *
 *
 * ### Directive Definition Object
 *
 * The directive definition object provides instructions to the {@link ng.$compile
 * compiler}. The attributes are:
 *
 * #### `multiElement`
 * When this property is set to true, the HTML compiler will collect DOM nodes between
 * nodes with the attributes `directive-name-start` and `directive-name-end`, and group them
 * together as the directive elements. It is recommended that this feature be used on directives
 * which are not strictly behavioural (such as {@link ngClick}), and which
 * do not manipulate or replace child nodes (such as {@link ngInclude}).
 *
 * #### `priority`
 * When there are multiple directives defined on a single DOM element, sometimes it
 * is necessary to specify the order in which the directives are applied. The `priority` is used
 * to sort the directives before their `compile` functions get called. Priority is defined as a
 * number. Directives with greater numerical `priority` are compiled first. Pre-link functions
 * are also run in priority order, but post-link functions are run in reverse order. The order
 * of directives with the same priority is undefined. The default priority is `0`.
 *
 * #### `terminal`
 * If set to true then the current `priority` will be the last set of directives
 * which will execute (any directives at the current priority will still execute
 * as the order of execution on same `priority` is undefined). Note that expressions
 * and other directives used in the directive's template will also be excluded from execution.
 *
 * #### `scope`
 * The scope property can be `true`, an object or a falsy value:
 *
 * * **falsy:** No scope will be created for the directive. The directive will use its parent's scope.
 *
 * * **`true`:** A new child scope that prototypically inherits from its parent will be created for
 * the directive's element. If multiple directives on the same element request a new scope,
 * only one new scope is created. The new scope rule does not apply for the root of the template
 * since the root of the template always gets a new scope.
 *
 * * **`{...}` (an object hash):** A new "isolate" scope is created for the directive's element. The
 * 'isolate' scope differs from normal scope in that it does not prototypically inherit from its parent
 * scope. This is useful when creating reusable components, which should not accidentally read or modify
 * data in the parent scope.
 *
 * The 'isolate' scope object hash defines a set of local scope properties derived from attributes on the
 * directive's element. These local properties are useful for aliasing values for templates. The keys in
 * the object hash map to the name of the property on the isolate scope; the values define how the property
 * is bound to the parent scope, via matching attributes on the directive's element:
 *
 * * `@` or `@attr` - bind a local scope property to the value of DOM attribute. The result is
 *   always a string since DOM attributes are strings. If no `attr` name is specified  then the
 *   attribute name is assumed to be the same as the local name.
 *   Given `<widget my-attr="hello {{name}}">` and widget definition
 *   of `scope: { localName:'@myAttr' }`, then widget scope property `localName` will reflect
 *   the interpolated value of `hello {{name}}`. As the `name` attribute changes so will the
 *   `localName` property on the widget scope. The `name` is read from the parent scope (not
 *   component scope).
 *
 * * `=` or `=attr` - set up bi-directional binding between a local scope property and the
 *   parent scope property of name defined via the value of the `attr` attribute. If no `attr`
 *   name is specified then the attribute name is assumed to be the same as the local name.
 *   Given `<widget my-attr="parentModel">` and widget definition of
 *   `scope: { localModel:'=myAttr' }`, then widget scope property `localModel` will reflect the
 *   value of `parentModel` on the parent scope. Any changes to `parentModel` will be reflected
 *   in `localModel` and any changes in `localModel` will reflect in `parentModel`. If the parent
 *   scope property doesn't exist, it will throw a NON_ASSIGNABLE_MODEL_EXPRESSION exception. You
 *   can avoid this behavior using `=?` or `=?attr` in order to flag the property as optional. If
 *   you want to shallow watch for changes (i.e. $watchCollection instead of $watch) you can use
 *   `=*` or `=*attr` (`=*?` or `=*?attr` if the property is optional).
 *
 * * `&` or `&attr` - provides a way to execute an expression in the context of the parent scope.
 *   If no `attr` name is specified then the attribute name is assumed to be the same as the
 *   local name. Given `<widget my-attr="count = count + value">` and widget definition of
 *   `scope: { localFn:'&myAttr' }`, then isolate scope property `localFn` will point to
 *   a function wrapper for the `count = count + value` expression. Often it's desirable to
 *   pass data from the isolated scope via an expression to the parent scope, this can be
 *   done by passing a map of local variable names and values into the expression wrapper fn.
 *   For example, if the expression is `increment(amount)` then we can specify the amount value
 *   by calling the `localFn` as `localFn({amount: 22})`.
 *
 * In general it's possible to apply more than one directive to one element, but there might be limitations
 * depending on the type of scope required by the directives. The following points will help explain these limitations.
 * For simplicity only two directives are taken into account, but it is also applicable for several directives:
 *
 * * **no scope** + **no scope** => Two directives which don't require their own scope will use their parent's scope
 * * **child scope** + **no scope** =>  Both directives will share one single child scope
 * * **child scope** + **child scope** =>  Both directives will share one single child scope
 * * **isolated scope** + **no scope** =>  The isolated directive will use it's own created isolated scope. The other directive will use
 * its parent's scope
 * * **isolated scope** + **child scope** =>  **Won't work!** Only one scope can be related to one element. Therefore these directives cannot
 * be applied to the same element.
 * * **isolated scope** + **isolated scope**  =>  **Won't work!** Only one scope can be related to one element. Therefore these directives
 * cannot be applied to the same element.
 *
 *
 * #### `bindToController`
 * When an isolate scope is used for a component (see above), and `controllerAs` is used, `bindToController: true` will
 * allow a component to have its properties bound to the controller, rather than to scope. When the controller
 * is instantiated, the initial values of the isolate scope bindings are already available.
 *
 * #### `controller`
 * Controller constructor function. The controller is instantiated before the
 * pre-linking phase and can be accessed by other directives (see
 * `require` attribute). This allows the directives to communicate with each other and augment
 * each other's behavior. The controller is injectable (and supports bracket notation) with the following locals:
 *
 * * `$scope` - Current scope associated with the element
 * * `$element` - Current element
 * * `$attrs` - Current attributes object for the element
 * * `$transclude` - A transclude linking function pre-bound to the correct transclusion scope:
 *   `function([scope], cloneLinkingFn, futureParentElement)`.
 *    * `scope`: optional argument to override the scope.
 *    * `cloneLinkingFn`: optional argument to create clones of the original transcluded content.
 *    * `futureParentElement`:
 *        * defines the parent to which the `cloneLinkingFn` will add the cloned elements.
 *        * default: `$element.parent()` resp. `$element` for `transclude:'element'` resp. `transclude:true`.
 *        * only needed for transcludes that are allowed to contain non html elements (e.g. SVG elements)
 *          and when the `cloneLinkinFn` is passed,
 *          as those elements need to created and cloned in a special way when they are defined outside their
 *          usual containers (e.g. like `<svg>`).
 *        * See also the `directive.templateNamespace` property.
 *
 *
 * #### `require`
 * Require another directive and inject its controller as the fourth argument to the linking function. The
 * `require` takes a string name (or array of strings) of the directive(s) to pass in. If an array is used, the
 * injected argument will be an array in corresponding order. If no such directive can be
 * found, or if the directive does not have a controller, then an error is raised (unless no link function
 * is specified, in which case error checking is skipped). The name can be prefixed with:
 *
 * * (no prefix) - Locate the required controller on the current element. Throw an error if not found.
 * * `?` - Attempt to locate the required controller or pass `null` to the `link` fn if not found.
 * * `^` - Locate the required controller by searching the element and its parents. Throw an error if not found.
 * * `^^` - Locate the required controller by searching the element's parents. Throw an error if not found.
 * * `?^` - Attempt to locate the required controller by searching the element and its parents or pass
 *   `null` to the `link` fn if not found.
 * * `?^^` - Attempt to locate the required controller by searching the element's parents, or pass
 *   `null` to the `link` fn if not found.
 *
 *
 * #### `controllerAs`
 * Identifier name for a reference to the controller in the directive's scope.
 * This allows the controller to be referenced from the directive template. This is especially
 * useful when a directive is used as component, i.e. with an `isolate` scope. It's also possible
 * to use it in a directive without an `isolate` / `new` scope, but you need to be aware that the
 * `controllerAs` reference might overwrite a property that already exists on the parent scope.
 *
 *
 * #### `restrict`
 * String of subset of `EACM` which restricts the directive to a specific directive
 * declaration style. If omitted, the defaults (elements and attributes) are used.
 *
 * * `E` - Element name (default): `<my-directive></my-directive>`
 * * `A` - Attribute (default): `<div my-directive="exp"></div>`
 * * `C` - Class: `<div class="my-directive: exp;"></div>`
 * * `M` - Comment: `<!-- directive: my-directive exp -->`
 *
 *
 * #### `templateNamespace`
 * String representing the document type used by the markup in the template.
 * AngularJS needs this information as those elements need to be created and cloned
 * in a special way when they are defined outside their usual containers like `<svg>` and `<math>`.
 *
 * * `html` - All root nodes in the template are HTML. Root nodes may also be
 *   top-level elements such as `<svg>` or `<math>`.
 * * `svg` - The root nodes in the template are SVG elements (excluding `<math>`).
 * * `math` - The root nodes in the template are MathML elements (excluding `<svg>`).
 *
 * If no `templateNamespace` is specified, then the namespace is considered to be `html`.
 *
 * #### `template`
 * HTML markup that may:
 * * Replace the contents of the directive's element (default).
 * * Replace the directive's element itself (if `replace` is true - DEPRECATED).
 * * Wrap the contents of the directive's element (if `transclude` is true).
 *
 * Value may be:
 *
 * * A string. For example `<div red-on-hover>{{delete_str}}</div>`.
 * * A function which takes two arguments `tElement` and `tAttrs` (described in the `compile`
 *   function api below) and returns a string value.
 *
 *
 * #### `templateUrl`
 * This is similar to `template` but the template is loaded from the specified URL, asynchronously.
 *
 * Because template loading is asynchronous the compiler will suspend compilation of directives on that element
 * for later when the template has been resolved.  In the meantime it will continue to compile and link
 * sibling and parent elements as though this element had not contained any directives.
 *
 * The compiler does not suspend the entire compilation to wait for templates to be loaded because this
 * would result in the whole app "stalling" until all templates are loaded asynchronously - even in the
 * case when only one deeply nested directive has `templateUrl`.
 *
 * Template loading is asynchronous even if the template has been preloaded into the {@link $templateCache}
 *
 * You can specify `templateUrl` as a string representing the URL or as a function which takes two
 * arguments `tElement` and `tAttrs` (described in the `compile` function api below) and returns
 * a string value representing the url.  In either case, the template URL is passed through {@link
 * $sce#getTrustedResourceUrl $sce.getTrustedResourceUrl}.
 *
 *
 * #### `replace` ([*DEPRECATED*!], will be removed in next major release - i.e. v2.0)
 * specify what the template should replace. Defaults to `false`.
 *
 * * `true` - the template will replace the directive's element.
 * * `false` - the template will replace the contents of the directive's element.
 *
 * The replacement process migrates all of the attributes / classes from the old element to the new
 * one. See the {@link guide/directive#template-expanding-directive
 * Directives Guide} for an example.
 *
 * There are very few scenarios where element replacement is required for the application function,
 * the main one being reusable custom components that are used within SVG contexts
 * (because SVG doesn't work with custom elements in the DOM tree).
 *
 * #### `transclude`
 * Extract the contents of the element where the directive appears and make it available to the directive.
 * The contents are compiled and provided to the directive as a **transclusion function**. See the
 * {@link $compile#transclusion Transclusion} section below.
 *
 * There are two kinds of transclusion depending upon whether you want to transclude just the contents of the
 * directive's element or the entire element:
 *
 * * `true` - transclude the content (i.e. the child nodes) of the directive's element.
 * * `'element'` - transclude the whole of the directive's element including any directives on this
 *   element that defined at a lower priority than this directive. When used, the `template`
 *   property is ignored.
 *
 *
 * #### `compile`
 *
 * ```js
 *   function compile(tElement, tAttrs, transclude) { ... }
 * ```
 *
 * The compile function deals with transforming the template DOM. Since most directives do not do
 * template transformation, it is not used often. The compile function takes the following arguments:
 *
 *   * `tElement` - template element - The element where the directive has been declared. It is
 *     safe to do template transformation on the element and child elements only.
 *
 *   * `tAttrs` - template attributes - Normalized list of attributes declared on this element shared
 *     between all directive compile functions.
 *
 *   * `transclude` -  [*DEPRECATED*!] A transclude linking function: `function(scope, cloneLinkingFn)`
 *
 * <div class="alert alert-warning">
 * **Note:** The template instance and the link instance may be different objects if the template has
 * been cloned. For this reason it is **not** safe to do anything other than DOM transformations that
 * apply to all cloned DOM nodes within the compile function. Specifically, DOM listener registration
 * should be done in a linking function rather than in a compile function.
 * </div>

 * <div class="alert alert-warning">
 * **Note:** The compile function cannot handle directives that recursively use themselves in their
 * own templates or compile functions. Compiling these directives results in an infinite loop and a
 * stack overflow errors.
 *
 * This can be avoided by manually using $compile in the postLink function to imperatively compile
 * a directive's template instead of relying on automatic template compilation via `template` or
 * `templateUrl` declaration or manual compilation inside the compile function.
 * </div>
 *
 * <div class="alert alert-danger">
 * **Note:** The `transclude` function that is passed to the compile function is deprecated, as it
 *   e.g. does not know about the right outer scope. Please use the transclude function that is passed
 *   to the link function instead.
 * </div>

 * A compile function can have a return value which can be either a function or an object.
 *
 * * returning a (post-link) function - is equivalent to registering the linking function via the
 *   `link` property of the config object when the compile function is empty.
 *
 * * returning an object with function(s) registered via `pre` and `post` properties - allows you to
 *   control when a linking function should be called during the linking phase. See info about
 *   pre-linking and post-linking functions below.
 *
 *
 * #### `link`
 * This property is used only if the `compile` property is not defined.
 *
 * ```js
 *   function link(scope, iElement, iAttrs, controller, transcludeFn) { ... }
 * ```
 *
 * The link function is responsible for registering DOM listeners as well as updating the DOM. It is
 * executed after the template has been cloned. This is where most of the directive logic will be
 * put.
 *
 *   * `scope` - {@link ng.$rootScope.Scope Scope} - The scope to be used by the
 *     directive for registering {@link ng.$rootScope.Scope#$watch watches}.
 *
 *   * `iElement` - instance element - The element where the directive is to be used. It is safe to
 *     manipulate the children of the element only in `postLink` function since the children have
 *     already been linked.
 *
 *   * `iAttrs` - instance attributes - Normalized list of attributes declared on this element shared
 *     between all directive linking functions.
 *
 *   * `controller` - the directive's required controller instance(s) - Instances are shared
 *     among all directives, which allows the directives to use the controllers as a communication
 *     channel. The exact value depends on the directive's `require` property:
 *       * no controller(s) required: the directive's own controller, or `undefined` if it doesn't have one
 *       * `string`: the controller instance
 *       * `array`: array of controller instances
 *
 *     If a required controller cannot be found, and it is optional, the instance is `null`,
 *     otherwise the {@link error:$compile:ctreq Missing Required Controller} error is thrown.
 *
 *     Note that you can also require the directive's own controller - it will be made available like
 *     any other controller.
 *
 *   * `transcludeFn` - A transclude linking function pre-bound to the correct transclusion scope.
 *     This is the same as the `$transclude`
 *     parameter of directive controllers, see there for details.
 *     `function([scope], cloneLinkingFn, futureParentElement)`.
 *
 * #### Pre-linking function
 *
 * Executed before the child elements are linked. Not safe to do DOM transformation since the
 * compiler linking function will fail to locate the correct elements for linking.
 *
 * #### Post-linking function
 *
 * Executed after the child elements are linked.
 *
 * Note that child elements that contain `templateUrl` directives will not have been compiled
 * and linked since they are waiting for their template to load asynchronously and their own
 * compilation and linking has been suspended until that occurs.
 *
 * It is safe to do DOM transformation in the post-linking function on elements that are not waiting
 * for their async templates to be resolved.
 *
 *
 * ### Transclusion
 *
 * Transclusion is the process of extracting a collection of DOM elements from one part of the DOM and
 * copying them to another part of the DOM, while maintaining their connection to the original AngularJS
 * scope from where they were taken.
 *
 * Transclusion is used (often with {@link ngTransclude}) to insert the
 * original contents of a directive's element into a specified place in the template of the directive.
 * The benefit of transclusion, over simply moving the DOM elements manually, is that the transcluded
 * content has access to the properties on the scope from which it was taken, even if the directive
 * has isolated scope.
 * See the {@link guide/directive#creating-a-directive-that-wraps-other-elements Directives Guide}.
 *
 * This makes it possible for the widget to have private state for its template, while the transcluded
 * content has access to its originating scope.
 *
 * <div class="alert alert-warning">
 * **Note:** When testing an element transclude directive you must not place the directive at the root of the
 * DOM fragment that is being compiled. See {@link guide/unit-testing#testing-transclusion-directives
 * Testing Transclusion Directives}.
 * </div>
 *
 * #### Transclusion Functions
 *
 * When a directive requests transclusion, the compiler extracts its contents and provides a **transclusion
 * function** to the directive's `link` function and `controller`. This transclusion function is a special
 * **linking function** that will return the compiled contents linked to a new transclusion scope.
 *
 * <div class="alert alert-info">
 * If you are just using {@link ngTransclude} then you don't need to worry about this function, since
 * ngTransclude will deal with it for us.
 * </div>
 *
 * If you want to manually control the insertion and removal of the transcluded content in your directive
 * then you must use this transclude function. When you call a transclude function it returns a a jqLite/JQuery
 * object that contains the compiled DOM, which is linked to the correct transclusion scope.
 *
 * When you call a transclusion function you can pass in a **clone attach function**. This function accepts
 * two parameters, `function(clone, scope) { ... }`, where the `clone` is a fresh compiled copy of your transcluded
 * content and the `scope` is the newly created transclusion scope, to which the clone is bound.
 *
 * <div class="alert alert-info">
 * **Best Practice**: Always provide a `cloneFn` (clone attach function) when you call a translude function
 * since you then get a fresh clone of the original DOM and also have access to the new transclusion scope.
 * </div>
 *
 * It is normal practice to attach your transcluded content (`clone`) to the DOM inside your **clone
 * attach function**:
 *
 * ```js
 * var transcludedContent, transclusionScope;
 *
 * $transclude(function(clone, scope) {
 *   element.append(clone);
 *   transcludedContent = clone;
 *   transclusionScope = scope;
 * });
 * ```
 *
 * Later, if you want to remove the transcluded content from your DOM then you should also destroy the
 * associated transclusion scope:
 *
 * ```js
 * transcludedContent.remove();
 * transclusionScope.$destroy();
 * ```
 *
 * <div class="alert alert-info">
 * **Best Practice**: if you intend to add and remove transcluded content manually in your directive
 * (by calling the transclude function to get the DOM and calling `element.remove()` to remove it),
 * then you are also responsible for calling `$destroy` on the transclusion scope.
 * </div>
 *
 * The built-in DOM manipulation directives, such as {@link ngIf}, {@link ngSwitch} and {@link ngRepeat}
 * automatically destroy their transluded clones as necessary so you do not need to worry about this if
 * you are simply using {@link ngTransclude} to inject the transclusion into your directive.
 *
 *
 * #### Transclusion Scopes
 *
 * When you call a transclude function it returns a DOM fragment that is pre-bound to a **transclusion
 * scope**. This scope is special, in that it is a child of the directive's scope (and so gets destroyed
 * when the directive's scope gets destroyed) but it inherits the properties of the scope from which it
 * was taken.
 *
 * For example consider a directive that uses transclusion and isolated scope. The DOM hierarchy might look
 * like this:
 *
 * ```html
 * <div ng-app>
 *   <div isolate>
 *     <div transclusion>
 *     </div>
 *   </div>
 * </div>
 * ```
 *
 * The `$parent` scope hierarchy will look like this:
 *
 * ```
 * - $rootScope
 *   - isolate
 *     - transclusion
 * ```
 *
 * but the scopes will inherit prototypically from different scopes to their `$parent`.
 *
 * ```
 * - $rootScope
 *   - transclusion
 * - isolate
 * ```
 *
 *
 * ### Attributes
 *
 * The {@link ng.$compile.directive.Attributes Attributes} object - passed as a parameter in the
 * `link()` or `compile()` functions. It has a variety of uses.
 *
 * accessing *Normalized attribute names:*
 * Directives like 'ngBind' can be expressed in many ways: 'ng:bind', `data-ng-bind`, or 'x-ng-bind'.
 * the attributes object allows for normalized access to
 *   the attributes.
 *
 * * *Directive inter-communication:* All directives share the same instance of the attributes
 *   object which allows the directives to use the attributes object as inter directive
 *   communication.
 *
 * * *Supports interpolation:* Interpolation attributes are assigned to the attribute object
 *   allowing other directives to read the interpolated value.
 *
 * * *Observing interpolated attributes:* Use `$observe` to observe the value changes of attributes
 *   that contain interpolation (e.g. `src="{{bar}}"`). Not only is this very efficient but it's also
 *   the only way to easily get the actual value because during the linking phase the interpolation
 *   hasn't been evaluated yet and so the value is at this time set to `undefined`.
 *
 * ```js
 * function linkingFn(scope, elm, attrs, ctrl) {
 *   // get the attribute value
 *   console.log(attrs.ngModel);
 *
 *   // change the attribute
 *   attrs.$set('ngModel', 'new value');
 *
 *   // observe changes to interpolated attribute
 *   attrs.$observe('ngModel', function(value) {
 *     console.log('ngModel has changed value to ' + value);
 *   });
 * }
 * ```
 *
 * ## Example
 *
 * <div class="alert alert-warning">
 * **Note**: Typically directives are registered with `module.directive`. The example below is
 * to illustrate how `$compile` works.
 * </div>
 *
 <example module="compileExample">
   <file name="index.html">
    <script>
      angular.module('compileExample', [], function($compileProvider) {
        // configure new 'compile' directive by passing a directive
        // factory function. The factory function injects the '$compile'
        $compileProvider.directive('compile', function($compile) {
          // directive factory creates a link function
          return function(scope, element, attrs) {
            scope.$watch(
              function(scope) {
                 // watch the 'compile' expression for changes
                return scope.$eval(attrs.compile);
              },
              function(value) {
                // when the 'compile' expression changes
                // assign it into the current DOM
                element.html(value);

                // compile the new DOM and link it to the current
                // scope.
                // NOTE: we only compile .childNodes so that
                // we don't get into infinite loop compiling ourselves
                $compile(element.contents())(scope);
              }
            );
          };
        });
      })
      .controller('GreeterController', ['$scope', function($scope) {
        $scope.name = 'Angular';
        $scope.html = 'Hello {{name}}';
      }]);
    </script>
    <div ng-controller="GreeterController">
      <input ng-model="name"> <br/>
      <textarea ng-model="html"></textarea> <br/>
      <div compile="html"></div>
    </div>
   </file>
   <file name="protractor.js" type="protractor">
     it('should auto compile', function() {
       var textarea = $('textarea');
       var output = $('div[compile]');
       // The initial state reads 'Hello Angular'.
       expect(output.getText()).toBe('Hello Angular');
       textarea.clear();
       textarea.sendKeys('{{name}}!');
       expect(output.getText()).toBe('Angular!');
     });
   </file>
 </example>

 *
 *
 * @param {string|DOMElement} element Element or HTML string to compile into a template function.
 * @param {function(angular.Scope, cloneAttachFn=)} transclude function available to directives - DEPRECATED.
 *
 * <div class="alert alert-danger">
 * **Note:** Passing a `transclude` function to the $compile function is deprecated, as it
 *   e.g. will not use the right outer scope. Please pass the transclude function as a
 *   `parentBoundTranscludeFn` to the link function instead.
 * </div>
 *
 * @param {number} maxPriority only apply directives lower than given priority (Only effects the
 *                 root element(s), not their children)
 * @returns {function(scope, cloneAttachFn=, options=)} a link function which is used to bind template
 * (a DOM element/tree) to a scope. Where:
 *
 *  * `scope` - A {@link ng.$rootScope.Scope Scope} to bind to.
 *  * `cloneAttachFn` - If `cloneAttachFn` is provided, then the link function will clone the
 *  `template` and call the `cloneAttachFn` function allowing the caller to attach the
 *  cloned elements to the DOM document at the appropriate place. The `cloneAttachFn` is
 *  called as: <br/> `cloneAttachFn(clonedElement, scope)` where:
 *
 *      * `clonedElement` - is a clone of the original `element` passed into the compiler.
 *      * `scope` - is the current scope with which the linking function is working with.
 *
 *  * `options` - An optional object hash with linking options. If `options` is provided, then the following
 *  keys may be used to control linking behavior:
 *
 *      * `parentBoundTranscludeFn` - the transclude function made available to
 *        directives; if given, it will be passed through to the link functions of
 *        directives found in `element` during compilation.
 *      * `transcludeControllers` - an object hash with keys that map controller names
 *        to controller instances; if given, it will make the controllers
 *        available to directives.
 *      * `futureParentElement` - defines the parent to which the `cloneAttachFn` will add
 *        the cloned elements; only needed for transcludes that are allowed to contain non html
 *        elements (e.g. SVG elements). See also the directive.controller property.
 *
 * Calling the linking function returns the element of the template. It is either the original
 * element passed in, or the clone of the element if the `cloneAttachFn` is provided.
 *
 * After linking the view is not updated until after a call to $digest which typically is done by
 * Angular automatically.
 *
 * If you need access to the bound view, there are two ways to do it:
 *
 * - If you are not asking the linking function to clone the template, create the DOM element(s)
 *   before you send them to the compiler and keep this reference around.
 *   ```js
 *     var element = $compile('<p>{{total}}</p>')(scope);
 *   ```
 *
 * - if on the other hand, you need the element to be cloned, the view reference from the original
 *   example would not point to the clone, but rather to the original template that was cloned. In
 *   this case, you can access the clone via the cloneAttachFn:
 *   ```js
 *     var templateElement = angular.element('<p>{{total}}</p>'),
 *         scope = ....;
 *
 *     var clonedElement = $compile(templateElement)(scope, function(clonedElement, scope) {
 *       //attach the clone to DOM document at the right place
 *     });
 *
 *     //now we have reference to the cloned DOM via `clonedElement`
 *   ```
 *
 *
 * For information on how the compiler works, see the
 * {@link guide/compiler Angular HTML Compiler} section of the Developer Guide.
 */

var $compileMinErr = minErr('$compile');

/**
 * @ngdoc provider
 * @name $compileProvider
 *
 * @description
 */
$CompileProvider.$inject = ['$provide', '$$sanitizeUriProvider'];
function $CompileProvider($provide, $$sanitizeUriProvider) {
  var hasDirectives = {},
      Suffix = 'Directive',
      COMMENT_DIRECTIVE_REGEXP = /^\s*directive\:\s*([\w\-]+)\s+(.*)$/,
      CLASS_DIRECTIVE_REGEXP = /(([\w\-]+)(?:\:([^;]+))?;?)/,
      ALL_OR_NOTHING_ATTRS = makeMap('ngSrc,ngSrcset,src,srcset'),
      REQUIRE_PREFIX_REGEXP = /^(?:(\^\^?)?(\?)?(\^\^?)?)?/;

  // Ref: http://developers.whatwg.org/webappapis.html#event-handler-idl-attributes
  // The assumption is that future DOM event attribute names will begin with
  // 'on' and be composed of only English letters.
  var EVENT_HANDLER_ATTR_REGEXP = /^(on[a-z]+|formaction)$/;

  function parseIsolateBindings(scope, directiveName, isController) {
    var LOCAL_REGEXP = /^\s*([@&]|=(\*?))(\??)\s*(\w*)\s*$/;

    var bindings = {};

    forEach(scope, function(definition, scopeName) {
      var match = definition.match(LOCAL_REGEXP);

      if (!match) {
        throw $compileMinErr('iscp',
            "Invalid {3} for directive '{0}'." +
            " Definition: {... {1}: '{2}' ...}",
            directiveName, scopeName, definition,
            (isController ? "controller bindings definition" :
            "isolate scope definition"));
      }

      bindings[scopeName] = {
        mode: match[1][0],
        collection: match[2] === '*',
        optional: match[3] === '?',
        attrName: match[4] || scopeName
      };
    });

    return bindings;
  }

  function parseDirectiveBindings(directive, directiveName) {
    var bindings = {
      isolateScope: null,
      bindToController: null
    };
    if (isObject(directive.scope)) {
      if (directive.bindToController === true) {
        bindings.bindToController = parseIsolateBindings(directive.scope,
                                                         directiveName, true);
        bindings.isolateScope = {};
      } else {
        bindings.isolateScope = parseIsolateBindings(directive.scope,
                                                     directiveName, false);
      }
    }
    if (isObject(directive.bindToController)) {
      bindings.bindToController =
          parseIsolateBindings(directive.bindToController, directiveName, true);
    }
    if (isObject(bindings.bindToController)) {
      var controller = directive.controller;
      var controllerAs = directive.controllerAs;
      if (!controller) {
        // There is no controller, there may or may not be a controllerAs property
        throw $compileMinErr('noctrl',
              "Cannot bind to controller without directive '{0}'s controller.",
              directiveName);
      } else if (!identifierForController(controller, controllerAs)) {
        // There is a controller, but no identifier or controllerAs property
        throw $compileMinErr('noident',
              "Cannot bind to controller without identifier for directive '{0}'.",
              directiveName);
      }
    }
    return bindings;
  }

  function assertValidDirectiveName(name) {
    var letter = name.charAt(0);
    if (!letter || letter !== lowercase(letter)) {
      throw $compileMinErr('baddir', "Directive name '{0}' is invalid. The first character must be a lowercase letter", name);
    }
    if (name !== name.trim()) {
      throw $compileMinErr('baddir',
            "Directive name '{0}' is invalid. The name should not contain leading or trailing whitespaces",
            name);
    }
  }

  /**
   * @ngdoc method
   * @name $compileProvider#directive
   * @kind function
   *
   * @description
   * Register a new directive with the compiler.
   *
   * @param {string|Object} name Name of the directive in camel-case (i.e. <code>ngBind</code> which
   *    will match as <code>ng-bind</code>), or an object map of directives where the keys are the
   *    names and the values are the factories.
   * @param {Function|Array} directiveFactory An injectable directive factory function. See
   *    {@link guide/directive} for more info.
   * @returns {ng.$compileProvider} Self for chaining.
   */
   this.directive = function registerDirective(name, directiveFactory) {
    assertNotHasOwnProperty(name, 'directive');
    if (isString(name)) {
      assertValidDirectiveName(name);
      assertArg(directiveFactory, 'directiveFactory');
      if (!hasDirectives.hasOwnProperty(name)) {
        hasDirectives[name] = [];
        $provide.factory(name + Suffix, ['$injector', '$exceptionHandler',
          function($injector, $exceptionHandler) {
            var directives = [];
            forEach(hasDirectives[name], function(directiveFactory, index) {
              try {
                var directive = $injector.invoke(directiveFactory);
                if (isFunction(directive)) {
                  directive = { compile: valueFn(directive) };
                } else if (!directive.compile && directive.link) {
                  directive.compile = valueFn(directive.link);
                }
                directive.priority = directive.priority || 0;
                directive.index = index;
                directive.name = directive.name || name;
                directive.require = directive.require || (directive.controller && directive.name);
                directive.restrict = directive.restrict || 'EA';
                var bindings = directive.$$bindings =
                    parseDirectiveBindings(directive, directive.name);
                if (isObject(bindings.isolateScope)) {
                  directive.$$isolateBindings = bindings.isolateScope;
                }
                directive.$$moduleName = directiveFactory.$$moduleName;
                directives.push(directive);
              } catch (e) {
                $exceptionHandler(e);
              }
            });
            return directives;
          }]);
      }
      hasDirectives[name].push(directiveFactory);
    } else {
      forEach(name, reverseParams(registerDirective));
    }
    return this;
  };


  /**
   * @ngdoc method
   * @name $compileProvider#aHrefSanitizationWhitelist
   * @kind function
   *
   * @description
   * Retrieves or overrides the default regular expression that is used for whitelisting of safe
   * urls during a[href] sanitization.
   *
   * The sanitization is a security measure aimed at preventing XSS attacks via html links.
   *
   * Any url about to be assigned to a[href] via data-binding is first normalized and turned into
   * an absolute url. Afterwards, the url is matched against the `aHrefSanitizationWhitelist`
   * regular expression. If a match is found, the original url is written into the dom. Otherwise,
   * the absolute url is prefixed with `'unsafe:'` string and only then is it written into the DOM.
   *
   * @param {RegExp=} regexp New regexp to whitelist urls with.
   * @returns {RegExp|ng.$compileProvider} Current RegExp if called without value or self for
   *    chaining otherwise.
   */
  this.aHrefSanitizationWhitelist = function(regexp) {
    if (isDefined(regexp)) {
      $$sanitizeUriProvider.aHrefSanitizationWhitelist(regexp);
      return this;
    } else {
      return $$sanitizeUriProvider.aHrefSanitizationWhitelist();
    }
  };


  /**
   * @ngdoc method
   * @name $compileProvider#imgSrcSanitizationWhitelist
   * @kind function
   *
   * @description
   * Retrieves or overrides the default regular expression that is used for whitelisting of safe
   * urls during img[src] sanitization.
   *
   * The sanitization is a security measure aimed at prevent XSS attacks via html links.
   *
   * Any url about to be assigned to img[src] via data-binding is first normalized and turned into
   * an absolute url. Afterwards, the url is matched against the `imgSrcSanitizationWhitelist`
   * regular expression. If a match is found, the original url is written into the dom. Otherwise,
   * the absolute url is prefixed with `'unsafe:'` string and only then is it written into the DOM.
   *
   * @param {RegExp=} regexp New regexp to whitelist urls with.
   * @returns {RegExp|ng.$compileProvider} Current RegExp if called without value or self for
   *    chaining otherwise.
   */
  this.imgSrcSanitizationWhitelist = function(regexp) {
    if (isDefined(regexp)) {
      $$sanitizeUriProvider.imgSrcSanitizationWhitelist(regexp);
      return this;
    } else {
      return $$sanitizeUriProvider.imgSrcSanitizationWhitelist();
    }
  };

  /**
   * @ngdoc method
   * @name  $compileProvider#debugInfoEnabled
   *
   * @param {boolean=} enabled update the debugInfoEnabled state if provided, otherwise just return the
   * current debugInfoEnabled state
   * @returns {*} current value if used as getter or itself (chaining) if used as setter
   *
   * @kind function
   *
   * @description
   * Call this method to enable/disable various debug runtime information in the compiler such as adding
   * binding information and a reference to the current scope on to DOM elements.
   * If enabled, the compiler will add the following to DOM elements that have been bound to the scope
   * * `ng-binding` CSS class
   * * `$binding` data property containing an array of the binding expressions
   *
   * You may want to disable this in production for a significant performance boost. See
   * {@link guide/production#disabling-debug-data Disabling Debug Data} for more.
   *
   * The default value is true.
   */
  var debugInfoEnabled = true;
  this.debugInfoEnabled = function(enabled) {
    if (isDefined(enabled)) {
      debugInfoEnabled = enabled;
      return this;
    }
    return debugInfoEnabled;
  };

  this.$get = [
            '$injector', '$interpolate', '$exceptionHandler', '$templateRequest', '$parse',
            '$controller', '$rootScope', '$document', '$sce', '$animate', '$$sanitizeUri',
    function($injector,   $interpolate,   $exceptionHandler,   $templateRequest,   $parse,
             $controller,   $rootScope,   $document,   $sce,   $animate,   $$sanitizeUri) {

    var Attributes = function(element, attributesToCopy) {
      if (attributesToCopy) {
        var keys = Object.keys(attributesToCopy);
        var i, l, key;

        for (i = 0, l = keys.length; i < l; i++) {
          key = keys[i];
          this[key] = attributesToCopy[key];
        }
      } else {
        this.$attr = {};
      }

      this.$$element = element;
    };

    Attributes.prototype = {
      /**
       * @ngdoc method
       * @name $compile.directive.Attributes#$normalize
       * @kind function
       *
       * @description
       * Converts an attribute name (e.g. dash/colon/underscore-delimited string, optionally prefixed with `x-` or
       * `data-`) to its normalized, camelCase form.
       *
       * Also there is special case for Moz prefix starting with upper case letter.
       *
       * For further information check out the guide on {@link guide/directive#matching-directives Matching Directives}
       *
       * @param {string} name Name to normalize
       */
      $normalize: directiveNormalize,


      /**
       * @ngdoc method
       * @name $compile.directive.Attributes#$addClass
       * @kind function
       *
       * @description
       * Adds the CSS class value specified by the classVal parameter to the element. If animations
       * are enabled then an animation will be triggered for the class addition.
       *
       * @param {string} classVal The className value that will be added to the element
       */
      $addClass: function(classVal) {
        if (classVal && classVal.length > 0) {
          $animate.addClass(this.$$element, classVal);
        }
      },

      /**
       * @ngdoc method
       * @name $compile.directive.Attributes#$removeClass
       * @kind function
       *
       * @description
       * Removes the CSS class value specified by the classVal parameter from the element. If
       * animations are enabled then an animation will be triggered for the class removal.
       *
       * @param {string} classVal The className value that will be removed from the element
       */
      $removeClass: function(classVal) {
        if (classVal && classVal.length > 0) {
          $animate.removeClass(this.$$element, classVal);
        }
      },

      /**
       * @ngdoc method
       * @name $compile.directive.Attributes#$updateClass
       * @kind function
       *
       * @description
       * Adds and removes the appropriate CSS class values to the element based on the difference
       * between the new and old CSS class values (specified as newClasses and oldClasses).
       *
       * @param {string} newClasses The current CSS className value
       * @param {string} oldClasses The former CSS className value
       */
      $updateClass: function(newClasses, oldClasses) {
        var toAdd = tokenDifference(newClasses, oldClasses);
        if (toAdd && toAdd.length) {
          $animate.addClass(this.$$element, toAdd);
        }

        var toRemove = tokenDifference(oldClasses, newClasses);
        if (toRemove && toRemove.length) {
          $animate.removeClass(this.$$element, toRemove);
        }
      },

      /**
       * Set a normalized attribute on the element in a way such that all directives
       * can share the attribute. This function properly handles boolean attributes.
       * @param {string} key Normalized key. (ie ngAttribute)
       * @param {string|boolean} value The value to set. If `null` attribute will be deleted.
       * @param {boolean=} writeAttr If false, does not write the value to DOM element attribute.
       *     Defaults to true.
       * @param {string=} attrName Optional none normalized name. Defaults to key.
       */
      $set: function(key, value, writeAttr, attrName) {
        // TODO: decide whether or not to throw an error if "class"
        //is set through this function since it may cause $updateClass to
        //become unstable.

        var node = this.$$element[0],
            booleanKey = getBooleanAttrName(node, key),
            aliasedKey = getAliasedAttrName(key),
            observer = key,
            nodeName;

        if (booleanKey) {
          this.$$element.prop(key, value);
          attrName = booleanKey;
        } else if (aliasedKey) {
          this[aliasedKey] = value;
          observer = aliasedKey;
        }

        this[key] = value;

        // translate normalized key to actual key
        if (attrName) {
          this.$attr[key] = attrName;
        } else {
          attrName = this.$attr[key];
          if (!attrName) {
            this.$attr[key] = attrName = snake_case(key, '-');
          }
        }

        nodeName = nodeName_(this.$$element);

        if ((nodeName === 'a' && key === 'href') ||
            (nodeName === 'img' && key === 'src')) {
          // sanitize a[href] and img[src] values
          this[key] = value = $$sanitizeUri(value, key === 'src');
        } else if (nodeName === 'img' && key === 'srcset') {
          // sanitize img[srcset] values
          var result = "";

          // first check if there are spaces because it's not the same pattern
          var trimmedSrcset = trim(value);
          //                (   999x   ,|   999w   ,|   ,|,   )
          var srcPattern = /(\s+\d+x\s*,|\s+\d+w\s*,|\s+,|,\s+)/;
          var pattern = /\s/.test(trimmedSrcset) ? srcPattern : /(,)/;

          // split srcset into tuple of uri and descriptor except for the last item
          var rawUris = trimmedSrcset.split(pattern);

          // for each tuples
          var nbrUrisWith2parts = Math.floor(rawUris.length / 2);
          for (var i = 0; i < nbrUrisWith2parts; i++) {
            var innerIdx = i * 2;
            // sanitize the uri
            result += $$sanitizeUri(trim(rawUris[innerIdx]), true);
            // add the descriptor
            result += (" " + trim(rawUris[innerIdx + 1]));
          }

          // split the last item into uri and descriptor
          var lastTuple = trim(rawUris[i * 2]).split(/\s/);

          // sanitize the last uri
          result += $$sanitizeUri(trim(lastTuple[0]), true);

          // and add the last descriptor if any
          if (lastTuple.length === 2) {
            result += (" " + trim(lastTuple[1]));
          }
          this[key] = value = result;
        }

        if (writeAttr !== false) {
          if (value === null || isUndefined(value)) {
            this.$$element.removeAttr(attrName);
          } else {
            this.$$element.attr(attrName, value);
          }
        }

        // fire observers
        var $$observers = this.$$observers;
        $$observers && forEach($$observers[observer], function(fn) {
          try {
            fn(value);
          } catch (e) {
            $exceptionHandler(e);
          }
        });
      },


      /**
       * @ngdoc method
       * @name $compile.directive.Attributes#$observe
       * @kind function
       *
       * @description
       * Observes an interpolated attribute.
       *
       * The observer function will be invoked once during the next `$digest` following
       * compilation. The observer is then invoked whenever the interpolated value
       * changes.
       *
       * @param {string} key Normalized key. (ie ngAttribute) .
       * @param {function(interpolatedValue)} fn Function that will be called whenever
                the interpolated value of the attribute changes.
       *        See the {@link guide/directive#text-and-attribute-bindings Directives} guide for more info.
       * @returns {function()} Returns a deregistration function for this observer.
       */
      $observe: function(key, fn) {
        var attrs = this,
            $$observers = (attrs.$$observers || (attrs.$$observers = createMap())),
            listeners = ($$observers[key] || ($$observers[key] = []));

        listeners.push(fn);
        $rootScope.$evalAsync(function() {
          if (!listeners.$$inter && attrs.hasOwnProperty(key) && !isUndefined(attrs[key])) {
            // no one registered attribute interpolation function, so lets call it manually
            fn(attrs[key]);
          }
        });

        return function() {
          arrayRemove(listeners, fn);
        };
      }
    };


    function safeAddClass($element, className) {
      try {
        $element.addClass(className);
      } catch (e) {
        // ignore, since it means that we are trying to set class on
        // SVG element, where class name is read-only.
      }
    }


    var startSymbol = $interpolate.startSymbol(),
        endSymbol = $interpolate.endSymbol(),
        denormalizeTemplate = (startSymbol == '{{' || endSymbol  == '}}')
            ? identity
            : function denormalizeTemplate(template) {
              return template.replace(/\{\{/g, startSymbol).replace(/}}/g, endSymbol);
        },
        NG_ATTR_BINDING = /^ngAttr[A-Z]/;
    var MULTI_ELEMENT_DIR_RE = /^(.+)Start$/;

    compile.$$addBindingInfo = debugInfoEnabled ? function $$addBindingInfo($element, binding) {
      var bindings = $element.data('$binding') || [];

      if (isArray(binding)) {
        bindings = bindings.concat(binding);
      } else {
        bindings.push(binding);
      }

      $element.data('$binding', bindings);
    } : noop;

    compile.$$addBindingClass = debugInfoEnabled ? function $$addBindingClass($element) {
      safeAddClass($element, 'ng-binding');
    } : noop;

    compile.$$addScopeInfo = debugInfoEnabled ? function $$addScopeInfo($element, scope, isolated, noTemplate) {
      var dataName = isolated ? (noTemplate ? '$isolateScopeNoTemplate' : '$isolateScope') : '$scope';
      $element.data(dataName, scope);
    } : noop;

    compile.$$addScopeClass = debugInfoEnabled ? function $$addScopeClass($element, isolated) {
      safeAddClass($element, isolated ? 'ng-isolate-scope' : 'ng-scope');
    } : noop;

    return compile;

    //================================

    function compile($compileNodes, transcludeFn, maxPriority, ignoreDirective,
                        previousCompileContext) {
      if (!($compileNodes instanceof jqLite)) {
        // jquery always rewraps, whereas we need to preserve the original selector so that we can
        // modify it.
        $compileNodes = jqLite($compileNodes);
      }
      // We can not compile top level text elements since text nodes can be merged and we will
      // not be able to attach scope data to them, so we will wrap them in <span>
      forEach($compileNodes, function(node, index) {
        if (node.nodeType == NODE_TYPE_TEXT && node.nodeValue.match(/\S+/) /* non-empty */ ) {
          $compileNodes[index] = jqLite(node).wrap('<span></span>').parent()[0];
        }
      });
      var compositeLinkFn =
              compileNodes($compileNodes, transcludeFn, $compileNodes,
                           maxPriority, ignoreDirective, previousCompileContext);
      compile.$$addScopeClass($compileNodes);
      var namespace = null;
      return function publicLinkFn(scope, cloneConnectFn, options) {
        assertArg(scope, 'scope');

        if (previousCompileContext && previousCompileContext.needsNewScope) {
          // A parent directive did a replace and a directive on this element asked
          // for transclusion, which caused us to lose a layer of element on which
          // we could hold the new transclusion scope, so we will create it manually
          // here.
          scope = scope.$parent.$new();
        }

        options = options || {};
        var parentBoundTranscludeFn = options.parentBoundTranscludeFn,
          transcludeControllers = options.transcludeControllers,
          futureParentElement = options.futureParentElement;

        // When `parentBoundTranscludeFn` is passed, it is a
        // `controllersBoundTransclude` function (it was previously passed
        // as `transclude` to directive.link) so we must unwrap it to get
        // its `boundTranscludeFn`
        if (parentBoundTranscludeFn && parentBoundTranscludeFn.$$boundTransclude) {
          parentBoundTranscludeFn = parentBoundTranscludeFn.$$boundTransclude;
        }

        if (!namespace) {
          namespace = detectNamespaceForChildElements(futureParentElement);
        }
        var $linkNode;
        if (namespace !== 'html') {
          // When using a directive with replace:true and templateUrl the $compileNodes
          // (or a child element inside of them)
          // might change, so we need to recreate the namespace adapted compileNodes
          // for call to the link function.
          // Note: This will already clone the nodes...
          $linkNode = jqLite(
            wrapTemplate(namespace, jqLite('<div>').append($compileNodes).html())
          );
        } else if (cloneConnectFn) {
          // important!!: we must call our jqLite.clone() since the jQuery one is trying to be smart
          // and sometimes changes the structure of the DOM.
          $linkNode = JQLitePrototype.clone.call($compileNodes);
        } else {
          $linkNode = $compileNodes;
        }

        if (transcludeControllers) {
          for (var controllerName in transcludeControllers) {
            $linkNode.data('$' + controllerName + 'Controller', transcludeControllers[controllerName].instance);
          }
        }

        compile.$$addScopeInfo($linkNode, scope);

        if (cloneConnectFn) cloneConnectFn($linkNode, scope);
        if (compositeLinkFn) compositeLinkFn(scope, $linkNode, $linkNode, parentBoundTranscludeFn);
        return $linkNode;
      };
    }

    function detectNamespaceForChildElements(parentElement) {
      // TODO: Make this detect MathML as well...
      var node = parentElement && parentElement[0];
      if (!node) {
        return 'html';
      } else {
        return nodeName_(node) !== 'foreignobject' && node.toString().match(/SVG/) ? 'svg' : 'html';
      }
    }

    /**
     * Compile function matches each node in nodeList against the directives. Once all directives
     * for a particular node are collected their compile functions are executed. The compile
     * functions return values - the linking functions - are combined into a composite linking
     * function, which is the a linking function for the node.
     *
     * @param {NodeList} nodeList an array of nodes or NodeList to compile
     * @param {function(angular.Scope, cloneAttachFn=)} transcludeFn A linking function, where the
     *        scope argument is auto-generated to the new child of the transcluded parent scope.
     * @param {DOMElement=} $rootElement If the nodeList is the root of the compilation tree then
     *        the rootElement must be set the jqLite collection of the compile root. This is
     *        needed so that the jqLite collection items can be replaced with widgets.
     * @param {number=} maxPriority Max directive priority.
     * @returns {Function} A composite linking function of all of the matched directives or null.
     */
    function compileNodes(nodeList, transcludeFn, $rootElement, maxPriority, ignoreDirective,
                            previousCompileContext) {
      var linkFns = [],
          attrs, directives, nodeLinkFn, childNodes, childLinkFn, linkFnFound, nodeLinkFnFound;

      for (var i = 0; i < nodeList.length; i++) {
        attrs = new Attributes();

        // we must always refer to nodeList[i] since the nodes can be replaced underneath us.
        directives = collectDirectives(nodeList[i], [], attrs, i === 0 ? maxPriority : undefined,
                                        ignoreDirective);

        nodeLinkFn = (directives.length)
            ? applyDirectivesToNode(directives, nodeList[i], attrs, transcludeFn, $rootElement,
                                      null, [], [], previousCompileContext)
            : null;

        if (nodeLinkFn && nodeLinkFn.scope) {
          compile.$$addScopeClass(attrs.$$element);
        }

        childLinkFn = (nodeLinkFn && nodeLinkFn.terminal ||
                      !(childNodes = nodeList[i].childNodes) ||
                      !childNodes.length)
            ? null
            : compileNodes(childNodes,
                 nodeLinkFn ? (
                  (nodeLinkFn.transcludeOnThisElement || !nodeLinkFn.templateOnThisElement)
                     && nodeLinkFn.transclude) : transcludeFn);

        if (nodeLinkFn || childLinkFn) {
          linkFns.push(i, nodeLinkFn, childLinkFn);
          linkFnFound = true;
          nodeLinkFnFound = nodeLinkFnFound || nodeLinkFn;
        }

        //use the previous context only for the first element in the virtual group
        previousCompileContext = null;
      }

      // return a linking function if we have found anything, null otherwise
      return linkFnFound ? compositeLinkFn : null;

      function compositeLinkFn(scope, nodeList, $rootElement, parentBoundTranscludeFn) {
        var nodeLinkFn, childLinkFn, node, childScope, i, ii, idx, childBoundTranscludeFn;
        var stableNodeList;


        if (nodeLinkFnFound) {
          // copy nodeList so that if a nodeLinkFn removes or adds an element at this DOM level our
          // offsets don't get screwed up
          var nodeListLength = nodeList.length;
          stableNodeList = new Array(nodeListLength);

          // create a sparse array by only copying the elements which have a linkFn
          for (i = 0; i < linkFns.length; i+=3) {
            idx = linkFns[i];
            stableNodeList[idx] = nodeList[idx];
          }
        } else {
          stableNodeList = nodeList;
        }

        for (i = 0, ii = linkFns.length; i < ii;) {
          node = stableNodeList[linkFns[i++]];
          nodeLinkFn = linkFns[i++];
          childLinkFn = linkFns[i++];

          if (nodeLinkFn) {
            if (nodeLinkFn.scope) {
              childScope = scope.$new();
              compile.$$addScopeInfo(jqLite(node), childScope);
            } else {
              childScope = scope;
            }

            if (nodeLinkFn.transcludeOnThisElement) {
              childBoundTranscludeFn = createBoundTranscludeFn(
                  scope, nodeLinkFn.transclude, parentBoundTranscludeFn);

            } else if (!nodeLinkFn.templateOnThisElement && parentBoundTranscludeFn) {
              childBoundTranscludeFn = parentBoundTranscludeFn;

            } else if (!parentBoundTranscludeFn && transcludeFn) {
              childBoundTranscludeFn = createBoundTranscludeFn(scope, transcludeFn);

            } else {
              childBoundTranscludeFn = null;
            }

            nodeLinkFn(childLinkFn, childScope, node, $rootElement, childBoundTranscludeFn);

          } else if (childLinkFn) {
            childLinkFn(scope, node.childNodes, undefined, parentBoundTranscludeFn);
          }
        }
      }
    }

    function createBoundTranscludeFn(scope, transcludeFn, previousBoundTranscludeFn) {

      var boundTranscludeFn = function(transcludedScope, cloneFn, controllers, futureParentElement, containingScope) {

        if (!transcludedScope) {
          transcludedScope = scope.$new(false, containingScope);
          transcludedScope.$$transcluded = true;
        }

        return transcludeFn(transcludedScope, cloneFn, {
          parentBoundTranscludeFn: previousBoundTranscludeFn,
          transcludeControllers: controllers,
          futureParentElement: futureParentElement
        });
      };

      return boundTranscludeFn;
    }

    /**
     * Looks for directives on the given node and adds them to the directive collection which is
     * sorted.
     *
     * @param node Node to search.
     * @param directives An array to which the directives are added to. This array is sorted before
     *        the function returns.
     * @param attrs The shared attrs object which is used to populate the normalized attributes.
     * @param {number=} maxPriority Max directive priority.
     */
    function collectDirectives(node, directives, attrs, maxPriority, ignoreDirective) {
      var nodeType = node.nodeType,
          attrsMap = attrs.$attr,
          match,
          className;

      switch (nodeType) {
        case NODE_TYPE_ELEMENT: /* Element */
          // use the node name: <directive>
          addDirective(directives,
              directiveNormalize(nodeName_(node)), 'E', maxPriority, ignoreDirective);

          // iterate over the attributes
          for (var attr, name, nName, ngAttrName, value, isNgAttr, nAttrs = node.attributes,
                   j = 0, jj = nAttrs && nAttrs.length; j < jj; j++) {
            var attrStartName = false;
            var attrEndName = false;

            attr = nAttrs[j];
            name = attr.name;
            value = trim(attr.value);

            // support ngAttr attribute binding
            ngAttrName = directiveNormalize(name);
            if (isNgAttr = NG_ATTR_BINDING.test(ngAttrName)) {
              name = name.replace(PREFIX_REGEXP, '')
                .substr(8).replace(/_(.)/g, function(match, letter) {
                  return letter.toUpperCase();
                });
            }

            var multiElementMatch = ngAttrName.match(MULTI_ELEMENT_DIR_RE);
            if (multiElementMatch && directiveIsMultiElement(multiElementMatch[1])) {
              attrStartName = name;
              attrEndName = name.substr(0, name.length - 5) + 'end';
              name = name.substr(0, name.length - 6);
            }

            nName = directiveNormalize(name.toLowerCase());
            attrsMap[nName] = name;
            if (isNgAttr || !attrs.hasOwnProperty(nName)) {
                attrs[nName] = value;
                if (getBooleanAttrName(node, nName)) {
                  attrs[nName] = true; // presence means true
                }
            }
            addAttrInterpolateDirective(node, directives, value, nName, isNgAttr);
            addDirective(directives, nName, 'A', maxPriority, ignoreDirective, attrStartName,
                          attrEndName);
          }

          // use class as directive
          className = node.className;
          if (isObject(className)) {
              // Maybe SVGAnimatedString
              className = className.animVal;
          }
          if (isString(className) && className !== '') {
            while (match = CLASS_DIRECTIVE_REGEXP.exec(className)) {
              nName = directiveNormalize(match[2]);
              if (addDirective(directives, nName, 'C', maxPriority, ignoreDirective)) {
                attrs[nName] = trim(match[3]);
              }
              className = className.substr(match.index + match[0].length);
            }
          }
          break;
        case NODE_TYPE_TEXT: /* Text Node */
          if (msie === 11) {
            // Workaround for #11781
            while (node.parentNode && node.nextSibling && node.nextSibling.nodeType === NODE_TYPE_TEXT) {
              node.nodeValue = node.nodeValue + node.nextSibling.nodeValue;
              node.parentNode.removeChild(node.nextSibling);
            }
          }
          addTextInterpolateDirective(directives, node.nodeValue);
          break;
        case NODE_TYPE_COMMENT: /* Comment */
          try {
            match = COMMENT_DIRECTIVE_REGEXP.exec(node.nodeValue);
            if (match) {
              nName = directiveNormalize(match[1]);
              if (addDirective(directives, nName, 'M', maxPriority, ignoreDirective)) {
                attrs[nName] = trim(match[2]);
              }
            }
          } catch (e) {
            // turns out that under some circumstances IE9 throws errors when one attempts to read
            // comment's node value.
            // Just ignore it and continue. (Can't seem to reproduce in test case.)
          }
          break;
      }

      directives.sort(byPriority);
      return directives;
    }

    /**
     * Given a node with an directive-start it collects all of the siblings until it finds
     * directive-end.
     * @param node
     * @param attrStart
     * @param attrEnd
     * @returns {*}
     */
    function groupScan(node, attrStart, attrEnd) {
      var nodes = [];
      var depth = 0;
      if (attrStart && node.hasAttribute && node.hasAttribute(attrStart)) {
        do {
          if (!node) {
            throw $compileMinErr('uterdir',
                      "Unterminated attribute, found '{0}' but no matching '{1}' found.",
                      attrStart, attrEnd);
          }
          if (node.nodeType == NODE_TYPE_ELEMENT) {
            if (node.hasAttribute(attrStart)) depth++;
            if (node.hasAttribute(attrEnd)) depth--;
          }
          nodes.push(node);
          node = node.nextSibling;
        } while (depth > 0);
      } else {
        nodes.push(node);
      }

      return jqLite(nodes);
    }

    /**
     * Wrapper for linking function which converts normal linking function into a grouped
     * linking function.
     * @param linkFn
     * @param attrStart
     * @param attrEnd
     * @returns {Function}
     */
    function groupElementsLinkFnWrapper(linkFn, attrStart, attrEnd) {
      return function(scope, element, attrs, controllers, transcludeFn) {
        element = groupScan(element[0], attrStart, attrEnd);
        return linkFn(scope, element, attrs, controllers, transcludeFn);
      };
    }

    /**
     * Once the directives have been collected, their compile functions are executed. This method
     * is responsible for inlining directive templates as well as terminating the application
     * of the directives if the terminal directive has been reached.
     *
     * @param {Array} directives Array of collected directives to execute their compile function.
     *        this needs to be pre-sorted by priority order.
     * @param {Node} compileNode The raw DOM node to apply the compile functions to
     * @param {Object} templateAttrs The shared attribute function
     * @param {function(angular.Scope, cloneAttachFn=)} transcludeFn A linking function, where the
     *                                                  scope argument is auto-generated to the new
     *                                                  child of the transcluded parent scope.
     * @param {JQLite} jqCollection If we are working on the root of the compile tree then this
     *                              argument has the root jqLite array so that we can replace nodes
     *                              on it.
     * @param {Object=} originalReplaceDirective An optional directive that will be ignored when
     *                                           compiling the transclusion.
     * @param {Array.<Function>} preLinkFns
     * @param {Array.<Function>} postLinkFns
     * @param {Object} previousCompileContext Context used for previous compilation of the current
     *                                        node
     * @returns {Function} linkFn
     */
    function applyDirectivesToNode(directives, compileNode, templateAttrs, transcludeFn,
                                   jqCollection, originalReplaceDirective, preLinkFns, postLinkFns,
                                   previousCompileContext) {
      previousCompileContext = previousCompileContext || {};

      var terminalPriority = -Number.MAX_VALUE,
          newScopeDirective = previousCompileContext.newScopeDirective,
          controllerDirectives = previousCompileContext.controllerDirectives,
          newIsolateScopeDirective = previousCompileContext.newIsolateScopeDirective,
          templateDirective = previousCompileContext.templateDirective,
          nonTlbTranscludeDirective = previousCompileContext.nonTlbTranscludeDirective,
          hasTranscludeDirective = false,
          hasTemplate = false,
          hasElementTranscludeDirective = previousCompileContext.hasElementTranscludeDirective,
          $compileNode = templateAttrs.$$element = jqLite(compileNode),
          directive,
          directiveName,
          $template,
          replaceDirective = originalReplaceDirective,
          childTranscludeFn = transcludeFn,
          linkFn,
          directiveValue;

      // executes all directives on the current element
      for (var i = 0, ii = directives.length; i < ii; i++) {
        directive = directives[i];
        var attrStart = directive.$$start;
        var attrEnd = directive.$$end;

        // collect multiblock sections
        if (attrStart) {
          $compileNode = groupScan(compileNode, attrStart, attrEnd);
        }
        $template = undefined;

        if (terminalPriority > directive.priority) {
          break; // prevent further processing of directives
        }

        if (directiveValue = directive.scope) {

          // skip the check for directives with async templates, we'll check the derived sync
          // directive when the template arrives
          if (!directive.templateUrl) {
            if (isObject(directiveValue)) {
              // This directive is trying to add an isolated scope.
              // Check that there is no scope of any kind already
              assertNoDuplicate('new/isolated scope', newIsolateScopeDirective || newScopeDirective,
                                directive, $compileNode);
              newIsolateScopeDirective = directive;
            } else {
              // This directive is trying to add a child scope.
              // Check that there is no isolated scope already
              assertNoDuplicate('new/isolated scope', newIsolateScopeDirective, directive,
                                $compileNode);
            }
          }

          newScopeDirective = newScopeDirective || directive;
        }

        directiveName = directive.name;

        if (!directive.templateUrl && directive.controller) {
          directiveValue = directive.controller;
          controllerDirectives = controllerDirectives || createMap();
          assertNoDuplicate("'" + directiveName + "' controller",
              controllerDirectives[directiveName], directive, $compileNode);
          controllerDirectives[directiveName] = directive;
        }

        if (directiveValue = directive.transclude) {
          hasTranscludeDirective = true;

          // Special case ngIf and ngRepeat so that we don't complain about duplicate transclusion.
          // This option should only be used by directives that know how to safely handle element transclusion,
          // where the transcluded nodes are added or replaced after linking.
          if (!directive.$$tlb) {
            assertNoDuplicate('transclusion', nonTlbTranscludeDirective, directive, $compileNode);
            nonTlbTranscludeDirective = directive;
          }

          if (directiveValue == 'element') {
            hasElementTranscludeDirective = true;
            terminalPriority = directive.priority;
            $template = $compileNode;
            $compileNode = templateAttrs.$$element =
                jqLite(document.createComment(' ' + directiveName + ': ' +
                                              templateAttrs[directiveName] + ' '));
            compileNode = $compileNode[0];
            replaceWith(jqCollection, sliceArgs($template), compileNode);

            childTranscludeFn = compile($template, transcludeFn, terminalPriority,
                                        replaceDirective && replaceDirective.name, {
                                          // Don't pass in:
                                          // - controllerDirectives - otherwise we'll create duplicates controllers
                                          // - newIsolateScopeDirective or templateDirective - combining templates with
                                          //   element transclusion doesn't make sense.
                                          //
                                          // We need only nonTlbTranscludeDirective so that we prevent putting transclusion
                                          // on the same element more than once.
                                          nonTlbTranscludeDirective: nonTlbTranscludeDirective
                                        });
          } else {
            $template = jqLite(jqLiteClone(compileNode)).contents();
            $compileNode.empty(); // clear contents
            childTranscludeFn = compile($template, transcludeFn, undefined,
                undefined, { needsNewScope: directive.$$isolateScope || directive.$$newScope});
          }
        }

        if (directive.template) {
          hasTemplate = true;
          assertNoDuplicate('template', templateDirective, directive, $compileNode);
          templateDirective = directive;

          directiveValue = (isFunction(directive.template))
              ? directive.template($compileNode, templateAttrs)
              : directive.template;

          directiveValue = denormalizeTemplate(directiveValue);

          if (directive.replace) {
            replaceDirective = directive;
            if (jqLiteIsTextNode(directiveValue)) {
              $template = [];
            } else {
              $template = removeComments(wrapTemplate(directive.templateNamespace, trim(directiveValue)));
            }
            compileNode = $template[0];

            if ($template.length != 1 || compileNode.nodeType !== NODE_TYPE_ELEMENT) {
              throw $compileMinErr('tplrt',
                  "Template for directive '{0}' must have exactly one root element. {1}",
                  directiveName, '');
            }

            replaceWith(jqCollection, $compileNode, compileNode);

            var newTemplateAttrs = {$attr: {}};

            // combine directives from the original node and from the template:
            // - take the array of directives for this element
            // - split it into two parts, those that already applied (processed) and those that weren't (unprocessed)
            // - collect directives from the template and sort them by priority
            // - combine directives as: processed + template + unprocessed
            var templateDirectives = collectDirectives(compileNode, [], newTemplateAttrs);
            var unprocessedDirectives = directives.splice(i + 1, directives.length - (i + 1));

            if (newIsolateScopeDirective || newScopeDirective) {
              // The original directive caused the current element to be replaced but this element
              // also needs to have a new scope, so we need to tell the template directives
              // that they would need to get their scope from further up, if they require transclusion
              markDirectiveScope(templateDirectives, newIsolateScopeDirective, newScopeDirective);
            }
            directives = directives.concat(templateDirectives).concat(unprocessedDirectives);
            mergeTemplateAttributes(templateAttrs, newTemplateAttrs);

            ii = directives.length;
          } else {
            $compileNode.html(directiveValue);
          }
        }

        if (directive.templateUrl) {
          hasTemplate = true;
          assertNoDuplicate('template', templateDirective, directive, $compileNode);
          templateDirective = directive;

          if (directive.replace) {
            replaceDirective = directive;
          }

          nodeLinkFn = compileTemplateUrl(directives.splice(i, directives.length - i), $compileNode,
              templateAttrs, jqCollection, hasTranscludeDirective && childTranscludeFn, preLinkFns, postLinkFns, {
                controllerDirectives: controllerDirectives,
                newScopeDirective: (newScopeDirective !== directive) && newScopeDirective,
                newIsolateScopeDirective: newIsolateScopeDirective,
                templateDirective: templateDirective,
                nonTlbTranscludeDirective: nonTlbTranscludeDirective
              });
          ii = directives.length;
        } else if (directive.compile) {
          try {
            linkFn = directive.compile($compileNode, templateAttrs, childTranscludeFn);
            if (isFunction(linkFn)) {
              addLinkFns(null, linkFn, attrStart, attrEnd);
            } else if (linkFn) {
              addLinkFns(linkFn.pre, linkFn.post, attrStart, attrEnd);
            }
          } catch (e) {
            $exceptionHandler(e, startingTag($compileNode));
          }
        }

        if (directive.terminal) {
          nodeLinkFn.terminal = true;
          terminalPriority = Math.max(terminalPriority, directive.priority);
        }

      }

      nodeLinkFn.scope = newScopeDirective && newScopeDirective.scope === true;
      nodeLinkFn.transcludeOnThisElement = hasTranscludeDirective;
      nodeLinkFn.templateOnThisElement = hasTemplate;
      nodeLinkFn.transclude = childTranscludeFn;

      previousCompileContext.hasElementTranscludeDirective = hasElementTranscludeDirective;

      // might be normal or delayed nodeLinkFn depending on if templateUrl is present
      return nodeLinkFn;

      ////////////////////

      function addLinkFns(pre, post, attrStart, attrEnd) {
        if (pre) {
          if (attrStart) pre = groupElementsLinkFnWrapper(pre, attrStart, attrEnd);
          pre.require = directive.require;
          pre.directiveName = directiveName;
          if (newIsolateScopeDirective === directive || directive.$$isolateScope) {
            pre = cloneAndAnnotateFn(pre, {isolateScope: true});
          }
          preLinkFns.push(pre);
        }
        if (post) {
          if (attrStart) post = groupElementsLinkFnWrapper(post, attrStart, attrEnd);
          post.require = directive.require;
          post.directiveName = directiveName;
          if (newIsolateScopeDirective === directive || directive.$$isolateScope) {
            post = cloneAndAnnotateFn(post, {isolateScope: true});
          }
          postLinkFns.push(post);
        }
      }


      function getControllers(directiveName, require, $element, elementControllers) {
        var value;

        if (isString(require)) {
          var match = require.match(REQUIRE_PREFIX_REGEXP);
          var name = require.substring(match[0].length);
          var inheritType = match[1] || match[3];
          var optional = match[2] === '?';

          //If only parents then start at the parent element
          if (inheritType === '^^') {
            $element = $element.parent();
          //Otherwise attempt getting the controller from elementControllers in case
          //the element is transcluded (and has no data) and to avoid .data if possible
          } else {
            value = elementControllers && elementControllers[name];
            value = value && value.instance;
          }

          if (!value) {
            var dataName = '$' + name + 'Controller';
            value = inheritType ? $element.inheritedData(dataName) : $element.data(dataName);
          }

          if (!value && !optional) {
            throw $compileMinErr('ctreq',
                "Controller '{0}', required by directive '{1}', can't be found!",
                name, directiveName);
          }
        } else if (isArray(require)) {
          value = [];
          for (var i = 0, ii = require.length; i < ii; i++) {
            value[i] = getControllers(directiveName, require[i], $element, elementControllers);
          }
        }

        return value || null;
      }

      function setupControllers($element, attrs, transcludeFn, controllerDirectives, isolateScope, scope) {
        var elementControllers = createMap();
        for (var controllerKey in controllerDirectives) {
          var directive = controllerDirectives[controllerKey];
          var locals = {
            $scope: directive === newIsolateScopeDirective || directive.$$isolateScope ? isolateScope : scope,
            $element: $element,
            $attrs: attrs,
            $transclude: transcludeFn
          };

          var controller = directive.controller;
          if (controller == '@') {
            controller = attrs[directive.name];
          }

          var controllerInstance = $controller(controller, locals, true, directive.controllerAs);

          // For directives with element transclusion the element is a comment,
          // but jQuery .data doesn't support attaching data to comment nodes as it's hard to
          // clean up (http://bugs.jquery.com/ticket/8335).
          // Instead, we save the controllers for the element in a local hash and attach to .data
          // later, once we have the actual element.
          elementControllers[directive.name] = controllerInstance;
          if (!hasElementTranscludeDirective) {
            $element.data('$' + directive.name + 'Controller', controllerInstance.instance);
          }
        }
        return elementControllers;
      }

      function nodeLinkFn(childLinkFn, scope, linkNode, $rootElement, boundTranscludeFn) {
        var linkFn, isolateScope, controllerScope, elementControllers, transcludeFn, $element,
            attrs, removeScopeBindingWatches, removeControllerBindingWatches;

        if (compileNode === linkNode) {
          attrs = templateAttrs;
          $element = templateAttrs.$$element;
        } else {
          $element = jqLite(linkNode);
          attrs = new Attributes($element, templateAttrs);
        }

        controllerScope = scope;
        if (newIsolateScopeDirective) {
          isolateScope = scope.$new(true);
        } else if (newScopeDirective) {
          controllerScope = scope.$parent;
        }

        if (boundTranscludeFn) {
          // track `boundTranscludeFn` so it can be unwrapped if `transcludeFn`
          // is later passed as `parentBoundTranscludeFn` to `publicLinkFn`
          transcludeFn = controllersBoundTransclude;
          transcludeFn.$$boundTransclude = boundTranscludeFn;
        }

        if (controllerDirectives) {
          elementControllers = setupControllers($element, attrs, transcludeFn, controllerDirectives, isolateScope, scope);
        }

        if (newIsolateScopeDirective) {
          // Initialize isolate scope bindings for new isolate scope directive.
          compile.$$addScopeInfo($element, isolateScope, true, !(templateDirective && (templateDirective === newIsolateScopeDirective ||
              templateDirective === newIsolateScopeDirective.$$originalDirective)));
          compile.$$addScopeClass($element, true);
          isolateScope.$$isolateBindings =
              newIsolateScopeDirective.$$isolateBindings;
          removeScopeBindingWatches = initializeDirectiveBindings(scope, attrs, isolateScope,
                                        isolateScope.$$isolateBindings,
                                        newIsolateScopeDirective);
          if (removeScopeBindingWatches) {
            isolateScope.$on('$destroy', removeScopeBindingWatches);
          }
        }

        // Initialize bindToController bindings
        for (var name in elementControllers) {
          var controllerDirective = controllerDirectives[name];
          var controller = elementControllers[name];
          var bindings = controllerDirective.$$bindings.bindToController;

          if (controller.identifier && bindings) {
            removeControllerBindingWatches =
              initializeDirectiveBindings(controllerScope, attrs, controller.instance, bindings, controllerDirective);
          }

          var controllerResult = controller();
          if (controllerResult !== controller.instance) {
            // If the controller constructor has a return value, overwrite the instance
            // from setupControllers
            controller.instance = controllerResult;
            $element.data('$' + controllerDirective.name + 'Controller', controllerResult);
            removeControllerBindingWatches && removeControllerBindingWatches();
            removeControllerBindingWatches =
              initializeDirectiveBindings(controllerScope, attrs, controller.instance, bindings, controllerDirective);
          }
        }

        // PRELINKING
        for (i = 0, ii = preLinkFns.length; i < ii; i++) {
          linkFn = preLinkFns[i];
          invokeLinkFn(linkFn,
              linkFn.isolateScope ? isolateScope : scope,
              $element,
              attrs,
              linkFn.require && getControllers(linkFn.directiveName, linkFn.require, $element, elementControllers),
              transcludeFn
          );
        }

        // RECURSION
        // We only pass the isolate scope, if the isolate directive has a template,
        // otherwise the child elements do not belong to the isolate directive.
        var scopeToChild = scope;
        if (newIsolateScopeDirective && (newIsolateScopeDirective.template || newIsolateScopeDirective.templateUrl === null)) {
          scopeToChild = isolateScope;
        }
        childLinkFn && childLinkFn(scopeToChild, linkNode.childNodes, undefined, boundTranscludeFn);

        // POSTLINKING
        for (i = postLinkFns.length - 1; i >= 0; i--) {
          linkFn = postLinkFns[i];
          invokeLinkFn(linkFn,
              linkFn.isolateScope ? isolateScope : scope,
              $element,
              attrs,
              linkFn.require && getControllers(linkFn.directiveName, linkFn.require, $element, elementControllers),
              transcludeFn
          );
        }

        // This is the function that is injected as `$transclude`.
        // Note: all arguments are optional!
        function controllersBoundTransclude(scope, cloneAttachFn, futureParentElement) {
          var transcludeControllers;

          // No scope passed in:
          if (!isScope(scope)) {
            futureParentElement = cloneAttachFn;
            cloneAttachFn = scope;
            scope = undefined;
          }

          if (hasElementTranscludeDirective) {
            transcludeControllers = elementControllers;
          }
          if (!futureParentElement) {
            futureParentElement = hasElementTranscludeDirective ? $element.parent() : $element;
          }
          return boundTranscludeFn(scope, cloneAttachFn, transcludeControllers, futureParentElement, scopeToChild);
        }
      }
    }

    // Depending upon the context in which a directive finds itself it might need to have a new isolated
    // or child scope created. For instance:
    // * if the directive has been pulled into a template because another directive with a higher priority
    // asked for element transclusion
    // * if the directive itself asks for transclusion but it is at the root of a template and the original
    // element was replaced. See https://github.com/angular/angular.js/issues/12936
    function markDirectiveScope(directives, isolateScope, newScope) {
      for (var j = 0, jj = directives.length; j < jj; j++) {
        directives[j] = inherit(directives[j], {$$isolateScope: isolateScope, $$newScope: newScope});
      }
    }

    /**
     * looks up the directive and decorates it with exception handling and proper parameters. We
     * call this the boundDirective.
     *
     * @param {string} name name of the directive to look up.
     * @param {string} location The directive must be found in specific format.
     *   String containing any of theses characters:
     *
     *   * `E`: element name
     *   * `A': attribute
     *   * `C`: class
     *   * `M`: comment
     * @returns {boolean} true if directive was added.
     */
    function addDirective(tDirectives, name, location, maxPriority, ignoreDirective, startAttrName,
                          endAttrName) {
      if (name === ignoreDirective) return null;
      var match = null;
      if (hasDirectives.hasOwnProperty(name)) {
        for (var directive, directives = $injector.get(name + Suffix),
            i = 0, ii = directives.length; i < ii; i++) {
          try {
            directive = directives[i];
            if ((isUndefined(maxPriority) || maxPriority > directive.priority) &&
                 directive.restrict.indexOf(location) != -1) {
              if (startAttrName) {
                directive = inherit(directive, {$$start: startAttrName, $$end: endAttrName});
              }
              tDirectives.push(directive);
              match = directive;
            }
          } catch (e) { $exceptionHandler(e); }
        }
      }
      return match;
    }


    /**
     * looks up the directive and returns true if it is a multi-element directive,
     * and therefore requires DOM nodes between -start and -end markers to be grouped
     * together.
     *
     * @param {string} name name of the directive to look up.
     * @returns true if directive was registered as multi-element.
     */
    function directiveIsMultiElement(name) {
      if (hasDirectives.hasOwnProperty(name)) {
        for (var directive, directives = $injector.get(name + Suffix),
            i = 0, ii = directives.length; i < ii; i++) {
          directive = directives[i];
          if (directive.multiElement) {
            return true;
          }
        }
      }
      return false;
    }

    /**
     * When the element is replaced with HTML template then the new attributes
     * on the template need to be merged with the existing attributes in the DOM.
     * The desired effect is to have both of the attributes present.
     *
     * @param {object} dst destination attributes (original DOM)
     * @param {object} src source attributes (from the directive template)
     */
    function mergeTemplateAttributes(dst, src) {
      var srcAttr = src.$attr,
          dstAttr = dst.$attr,
          $element = dst.$$element;

      // reapply the old attributes to the new element
      forEach(dst, function(value, key) {
        if (key.charAt(0) != '$') {
          if (src[key] && src[key] !== value) {
            value += (key === 'style' ? ';' : ' ') + src[key];
          }
          dst.$set(key, value, true, srcAttr[key]);
        }
      });

      // copy the new attributes on the old attrs object
      forEach(src, function(value, key) {
        if (key == 'class') {
          safeAddClass($element, value);
          dst['class'] = (dst['class'] ? dst['class'] + ' ' : '') + value;
        } else if (key == 'style') {
          $element.attr('style', $element.attr('style') + ';' + value);
          dst['style'] = (dst['style'] ? dst['style'] + ';' : '') + value;
          // `dst` will never contain hasOwnProperty as DOM parser won't let it.
          // You will get an "InvalidCharacterError: DOM Exception 5" error if you
          // have an attribute like "has-own-property" or "data-has-own-property", etc.
        } else if (key.charAt(0) != '$' && !dst.hasOwnProperty(key)) {
          dst[key] = value;
          dstAttr[key] = srcAttr[key];
        }
      });
    }


    function compileTemplateUrl(directives, $compileNode, tAttrs,
        $rootElement, childTranscludeFn, preLinkFns, postLinkFns, previousCompileContext) {
      var linkQueue = [],
          afterTemplateNodeLinkFn,
          afterTemplateChildLinkFn,
          beforeTemplateCompileNode = $compileNode[0],
          origAsyncDirective = directives.shift(),
          derivedSyncDirective = inherit(origAsyncDirective, {
            templateUrl: null, transclude: null, replace: null, $$originalDirective: origAsyncDirective
          }),
          templateUrl = (isFunction(origAsyncDirective.templateUrl))
              ? origAsyncDirective.templateUrl($compileNode, tAttrs)
              : origAsyncDirective.templateUrl,
          templateNamespace = origAsyncDirective.templateNamespace;

      $compileNode.empty();

      $templateRequest(templateUrl)
        .then(function(content) {
          var compileNode, tempTemplateAttrs, $template, childBoundTranscludeFn;

          content = denormalizeTemplate(content);

          if (origAsyncDirective.replace) {
            if (jqLiteIsTextNode(content)) {
              $template = [];
            } else {
              $template = removeComments(wrapTemplate(templateNamespace, trim(content)));
            }
            compileNode = $template[0];

            if ($template.length != 1 || compileNode.nodeType !== NODE_TYPE_ELEMENT) {
              throw $compileMinErr('tplrt',
                  "Template for directive '{0}' must have exactly one root element. {1}",
                  origAsyncDirective.name, templateUrl);
            }

            tempTemplateAttrs = {$attr: {}};
            replaceWith($rootElement, $compileNode, compileNode);
            var templateDirectives = collectDirectives(compileNode, [], tempTemplateAttrs);

            if (isObject(origAsyncDirective.scope)) {
              // the original directive that caused the template to be loaded async required
              // an isolate scope
              markDirectiveScope(templateDirectives, true);
            }
            directives = templateDirectives.concat(directives);
            mergeTemplateAttributes(tAttrs, tempTemplateAttrs);
          } else {
            compileNode = beforeTemplateCompileNode;
            $compileNode.html(content);
          }

          directives.unshift(derivedSyncDirective);

          afterTemplateNodeLinkFn = applyDirectivesToNode(directives, compileNode, tAttrs,
              childTranscludeFn, $compileNode, origAsyncDirective, preLinkFns, postLinkFns,
              previousCompileContext);
          forEach($rootElement, function(node, i) {
            if (node == compileNode) {
              $rootElement[i] = $compileNode[0];
            }
          });
          afterTemplateChildLinkFn = compileNodes($compileNode[0].childNodes, childTranscludeFn);

          while (linkQueue.length) {
            var scope = linkQueue.shift(),
                beforeTemplateLinkNode = linkQueue.shift(),
                linkRootElement = linkQueue.shift(),
                boundTranscludeFn = linkQueue.shift(),
                linkNode = $compileNode[0];

            if (scope.$$destroyed) continue;

            if (beforeTemplateLinkNode !== beforeTemplateCompileNode) {
              var oldClasses = beforeTemplateLinkNode.className;

              if (!(previousCompileContext.hasElementTranscludeDirective &&
                  origAsyncDirective.replace)) {
                // it was cloned therefore we have to clone as well.
                linkNode = jqLiteClone(compileNode);
              }
              replaceWith(linkRootElement, jqLite(beforeTemplateLinkNode), linkNode);

              // Copy in CSS classes from original node
              safeAddClass(jqLite(linkNode), oldClasses);
            }
            if (afterTemplateNodeLinkFn.transcludeOnThisElement) {
              childBoundTranscludeFn = createBoundTranscludeFn(scope, afterTemplateNodeLinkFn.transclude, boundTranscludeFn);
            } else {
              childBoundTranscludeFn = boundTranscludeFn;
            }
            afterTemplateNodeLinkFn(afterTemplateChildLinkFn, scope, linkNode, $rootElement,
              childBoundTranscludeFn);
          }
          linkQueue = null;
        });

      return function delayedNodeLinkFn(ignoreChildLinkFn, scope, node, rootElement, boundTranscludeFn) {
        var childBoundTranscludeFn = boundTranscludeFn;
        if (scope.$$destroyed) return;
        if (linkQueue) {
          linkQueue.push(scope,
                         node,
                         rootElement,
                         childBoundTranscludeFn);
        } else {
          if (afterTemplateNodeLinkFn.transcludeOnThisElement) {
            childBoundTranscludeFn = createBoundTranscludeFn(scope, afterTemplateNodeLinkFn.transclude, boundTranscludeFn);
          }
          afterTemplateNodeLinkFn(afterTemplateChildLinkFn, scope, node, rootElement, childBoundTranscludeFn);
        }
      };
    }


    /**
     * Sorting function for bound directives.
     */
    function byPriority(a, b) {
      var diff = b.priority - a.priority;
      if (diff !== 0) return diff;
      if (a.name !== b.name) return (a.name < b.name) ? -1 : 1;
      return a.index - b.index;
    }

    function assertNoDuplicate(what, previousDirective, directive, element) {

      function wrapModuleNameIfDefined(moduleName) {
        return moduleName ?
          (' (module: ' + moduleName + ')') :
          '';
      }

      if (previousDirective) {
        throw $compileMinErr('multidir', 'Multiple directives [{0}{1}, {2}{3}] asking for {4} on: {5}',
            previousDirective.name, wrapModuleNameIfDefined(previousDirective.$$moduleName),
            directive.name, wrapModuleNameIfDefined(directive.$$moduleName), what, startingTag(element));
      }
    }


    function addTextInterpolateDirective(directives, text) {
      var interpolateFn = $interpolate(text, true);
      if (interpolateFn) {
        directives.push({
          priority: 0,
          compile: function textInterpolateCompileFn(templateNode) {
            var templateNodeParent = templateNode.parent(),
                hasCompileParent = !!templateNodeParent.length;

            // When transcluding a template that has bindings in the root
            // we don't have a parent and thus need to add the class during linking fn.
            if (hasCompileParent) compile.$$addBindingClass(templateNodeParent);

            return function textInterpolateLinkFn(scope, node) {
              var parent = node.parent();
              if (!hasCompileParent) compile.$$addBindingClass(parent);
              compile.$$addBindingInfo(parent, interpolateFn.expressions);
              scope.$watch(interpolateFn, function interpolateFnWatchAction(value) {
                node[0].nodeValue = value;
              });
            };
          }
        });
      }
    }


    function wrapTemplate(type, template) {
      type = lowercase(type || 'html');
      switch (type) {
      case 'svg':
      case 'math':
        var wrapper = document.createElement('div');
        wrapper.innerHTML = '<' + type + '>' + template + '</' + type + '>';
        return wrapper.childNodes[0].childNodes;
      default:
        return template;
      }
    }


    function getTrustedContext(node, attrNormalizedName) {
      if (attrNormalizedName == "srcdoc") {
        return $sce.HTML;
      }
      var tag = nodeName_(node);
      // maction[xlink:href] can source SVG.  It's not limited to <maction>.
      if (attrNormalizedName == "xlinkHref" ||
          (tag == "form" && attrNormalizedName == "action") ||
          (tag != "img" && (attrNormalizedName == "src" ||
                            attrNormalizedName == "ngSrc"))) {
        return $sce.RESOURCE_URL;
      }
    }


    function addAttrInterpolateDirective(node, directives, value, name, allOrNothing) {
      var trustedContext = getTrustedContext(node, name);
      allOrNothing = ALL_OR_NOTHING_ATTRS[name] || allOrNothing;

      var interpolateFn = $interpolate(value, true, trustedContext, allOrNothing);

      // no interpolation found -> ignore
      if (!interpolateFn) return;


      if (name === "multiple" && nodeName_(node) === "select") {
        throw $compileMinErr("selmulti",
            "Binding to the 'multiple' attribute is not supported. Element: {0}",
            startingTag(node));
      }

      directives.push({
        priority: 100,
        compile: function() {
            return {
              pre: function attrInterpolatePreLinkFn(scope, element, attr) {
                var $$observers = (attr.$$observers || (attr.$$observers = createMap()));

                if (EVENT_HANDLER_ATTR_REGEXP.test(name)) {
                  throw $compileMinErr('nodomevents',
                      "Interpolations for HTML DOM event attributes are disallowed.  Please use the " +
                          "ng- versions (such as ng-click instead of onclick) instead.");
                }

                // If the attribute has changed since last $interpolate()ed
                var newValue = attr[name];
                if (newValue !== value) {
                  // we need to interpolate again since the attribute value has been updated
                  // (e.g. by another directive's compile function)
                  // ensure unset/empty values make interpolateFn falsy
                  interpolateFn = newValue && $interpolate(newValue, true, trustedContext, allOrNothing);
                  value = newValue;
                }

                // if attribute was updated so that there is no interpolation going on we don't want to
                // register any observers
                if (!interpolateFn) return;

                // initialize attr object so that it's ready in case we need the value for isolate
                // scope initialization, otherwise the value would not be available from isolate
                // directive's linking fn during linking phase
                attr[name] = interpolateFn(scope);

                ($$observers[name] || ($$observers[name] = [])).$$inter = true;
                (attr.$$observers && attr.$$observers[name].$$scope || scope).
                  $watch(interpolateFn, function interpolateFnWatchAction(newValue, oldValue) {
                    //special case for class attribute addition + removal
                    //so that class changes can tap into the animation
                    //hooks provided by the $animate service. Be sure to
                    //skip animations when the first digest occurs (when
                    //both the new and the old values are the same) since
                    //the CSS classes are the non-interpolated values
                    if (name === 'class' && newValue != oldValue) {
                      attr.$updateClass(newValue, oldValue);
                    } else {
                      attr.$set(name, newValue);
                    }
                  });
              }
            };
          }
      });
    }


    /**
     * This is a special jqLite.replaceWith, which can replace items which
     * have no parents, provided that the containing jqLite collection is provided.
     *
     * @param {JqLite=} $rootElement The root of the compile tree. Used so that we can replace nodes
     *                               in the root of the tree.
     * @param {JqLite} elementsToRemove The jqLite element which we are going to replace. We keep
     *                                  the shell, but replace its DOM node reference.
     * @param {Node} newNode The new DOM node.
     */
    function replaceWith($rootElement, elementsToRemove, newNode) {
      var firstElementToRemove = elementsToRemove[0],
          removeCount = elementsToRemove.length,
          parent = firstElementToRemove.parentNode,
          i, ii;

      if ($rootElement) {
        for (i = 0, ii = $rootElement.length; i < ii; i++) {
          if ($rootElement[i] == firstElementToRemove) {
            $rootElement[i++] = newNode;
            for (var j = i, j2 = j + removeCount - 1,
                     jj = $rootElement.length;
                 j < jj; j++, j2++) {
              if (j2 < jj) {
                $rootElement[j] = $rootElement[j2];
              } else {
                delete $rootElement[j];
              }
            }
            $rootElement.length -= removeCount - 1;

            // If the replaced element is also the jQuery .context then replace it
            // .context is a deprecated jQuery api, so we should set it only when jQuery set it
            // http://api.jquery.com/context/
            if ($rootElement.context === firstElementToRemove) {
              $rootElement.context = newNode;
            }
            break;
          }
        }
      }

      if (parent) {
        parent.replaceChild(newNode, firstElementToRemove);
      }

      // TODO(perf): what's this document fragment for? is it needed? can we at least reuse it?
      var fragment = document.createDocumentFragment();
      fragment.appendChild(firstElementToRemove);

      if (jqLite.hasData(firstElementToRemove)) {
        // Copy over user data (that includes Angular's $scope etc.). Don't copy private
        // data here because there's no public interface in jQuery to do that and copying over
        // event listeners (which is the main use of private data) wouldn't work anyway.
        jqLite.data(newNode, jqLite.data(firstElementToRemove));

        // Remove data of the replaced element. We cannot just call .remove()
        // on the element it since that would deallocate scope that is needed
        // for the new node. Instead, remove the data "manually".
        if (!jQuery) {
          delete jqLite.cache[firstElementToRemove[jqLite.expando]];
        } else {
          // jQuery 2.x doesn't expose the data storage. Use jQuery.cleanData to clean up after
          // the replaced element. The cleanData version monkey-patched by Angular would cause
          // the scope to be trashed and we do need the very same scope to work with the new
          // element. However, we cannot just cache the non-patched version and use it here as
          // that would break if another library patches the method after Angular does (one
          // example is jQuery UI). Instead, set a flag indicating scope destroying should be
          // skipped this one time.
          skipDestroyOnNextJQueryCleanData = true;
          jQuery.cleanData([firstElementToRemove]);
        }
      }

      for (var k = 1, kk = elementsToRemove.length; k < kk; k++) {
        var element = elementsToRemove[k];
        jqLite(element).remove(); // must do this way to clean up expando
        fragment.appendChild(element);
        delete elementsToRemove[k];
      }

      elementsToRemove[0] = newNode;
      elementsToRemove.length = 1;
    }


    function cloneAndAnnotateFn(fn, annotation) {
      return extend(function() { return fn.apply(null, arguments); }, fn, annotation);
    }


    function invokeLinkFn(linkFn, scope, $element, attrs, controllers, transcludeFn) {
      try {
        linkFn(scope, $element, attrs, controllers, transcludeFn);
      } catch (e) {
        $exceptionHandler(e, startingTag($element));
      }
    }


    // Set up $watches for isolate scope and controller bindings. This process
    // only occurs for isolate scopes and new scopes with controllerAs.
    function initializeDirectiveBindings(scope, attrs, destination, bindings, directive) {
      var removeWatchCollection = [];
      forEach(bindings, function(definition, scopeName) {
        var attrName = definition.attrName,
        optional = definition.optional,
        mode = definition.mode, // @, =, or &
        lastValue,
        parentGet, parentSet, compare;

        switch (mode) {

          case '@':
            if (!optional && !hasOwnProperty.call(attrs, attrName)) {
              destination[scopeName] = attrs[attrName] = void 0;
            }
            attrs.$observe(attrName, function(value) {
              if (isString(value)) {
                destination[scopeName] = value;
              }
            });
            attrs.$$observers[attrName].$$scope = scope;
            if (isString(attrs[attrName])) {
              // If the attribute has been provided then we trigger an interpolation to ensure
              // the value is there for use in the link fn
              destination[scopeName] = $interpolate(attrs[attrName])(scope);
            }
            break;

          case '=':
            if (!hasOwnProperty.call(attrs, attrName)) {
              if (optional) break;
              attrs[attrName] = void 0;
            }
            if (optional && !attrs[attrName]) break;

            parentGet = $parse(attrs[attrName]);
            if (parentGet.literal) {
              compare = equals;
            } else {
              compare = function(a, b) { return a === b || (a !== a && b !== b); };
            }
            parentSet = parentGet.assign || function() {
              // reset the change, or we will throw this exception on every $digest
              lastValue = destination[scopeName] = parentGet(scope);
              throw $compileMinErr('nonassign',
                  "Expression '{0}' used with directive '{1}' is non-assignable!",
                  attrs[attrName], directive.name);
            };
            lastValue = destination[scopeName] = parentGet(scope);
            var parentValueWatch = function parentValueWatch(parentValue) {
              if (!compare(parentValue, destination[scopeName])) {
                // we are out of sync and need to copy
                if (!compare(parentValue, lastValue)) {
                  // parent changed and it has precedence
                  destination[scopeName] = parentValue;
                } else {
                  // if the parent can be assigned then do so
                  parentSet(scope, parentValue = destination[scopeName]);
                }
              }
              return lastValue = parentValue;
            };
            parentValueWatch.$stateful = true;
            var removeWatch;
            if (definition.collection) {
              removeWatch = scope.$watchCollection(attrs[attrName], parentValueWatch);
            } else {
              removeWatch = scope.$watch($parse(attrs[attrName], parentValueWatch), null, parentGet.literal);
            }
            removeWatchCollection.push(removeWatch);
            break;

          case '&':
            // Don't assign Object.prototype method to scope
            parentGet = attrs.hasOwnProperty(attrName) ? $parse(attrs[attrName]) : noop;

            // Don't assign noop to destination if expression is not valid
            if (parentGet === noop && optional) break;

            destination[scopeName] = function(locals) {
              return parentGet(scope, locals);
            };
            break;
        }
      });

      return removeWatchCollection.length && function removeWatches() {
        for (var i = 0, ii = removeWatchCollection.length; i < ii; ++i) {
          removeWatchCollection[i]();
        }
      };
    }
  }];
}

var PREFIX_REGEXP = /^((?:x|data)[\:\-_])/i;
/**
 * Converts all accepted directives format into proper directive name.
 * @param name Name to normalize
 */
function directiveNormalize(name) {
  return camelCase(name.replace(PREFIX_REGEXP, ''));
}

/**
 * @ngdoc type
 * @name $compile.directive.Attributes
 *
 * @description
 * A shared object between directive compile / linking functions which contains normalized DOM
 * element attributes. The values reflect current binding state `{{ }}`. The normalization is
 * needed since all of these are treated as equivalent in Angular:
 *
 * ```
 *    <span ng:bind="a" ng-bind="a" data-ng-bind="a" x-ng-bind="a">
 * ```
 */

/**
 * @ngdoc property
 * @name $compile.directive.Attributes#$attr
 *
 * @description
 * A map of DOM element attribute names to the normalized name. This is
 * needed to do reverse lookup from normalized name back to actual name.
 */


/**
 * @ngdoc method
 * @name $compile.directive.Attributes#$set
 * @kind function
 *
 * @description
 * Set DOM element attribute value.
 *
 *
 * @param {string} name Normalized element attribute name of the property to modify. The name is
 *          reverse-translated using the {@link ng.$compile.directive.Attributes#$attr $attr}
 *          property to the original name.
 * @param {string} value Value to set the attribute to. The value can be an interpolated string.
 */



/**
 * Closure compiler type information
 */

function nodesetLinkingFn(
  /* angular.Scope */ scope,
  /* NodeList */ nodeList,
  /* Element */ rootElement,
  /* function(Function) */ boundTranscludeFn
) {}

function directiveLinkingFn(
  /* nodesetLinkingFn */ nodesetLinkingFn,
  /* angular.Scope */ scope,
  /* Node */ node,
  /* Element */ rootElement,
  /* function(Function) */ boundTranscludeFn
) {}

function tokenDifference(str1, str2) {
  var values = '',
      tokens1 = str1.split(/\s+/),
      tokens2 = str2.split(/\s+/);

  outer:
  for (var i = 0; i < tokens1.length; i++) {
    var token = tokens1[i];
    for (var j = 0; j < tokens2.length; j++) {
      if (token == tokens2[j]) continue outer;
    }
    values += (values.length > 0 ? ' ' : '') + token;
  }
  return values;
}

function removeComments(jqNodes) {
  jqNodes = jqLite(jqNodes);
  var i = jqNodes.length;

  if (i <= 1) {
    return jqNodes;
  }

  while (i--) {
    var node = jqNodes[i];
    if (node.nodeType === NODE_TYPE_COMMENT) {
      splice.call(jqNodes, i, 1);
    }
  }
  return jqNodes;
}

var $controllerMinErr = minErr('$controller');


var CNTRL_REG = /^(\S+)(\s+as\s+(\w+))?$/;
function identifierForController(controller, ident) {
  if (ident && isString(ident)) return ident;
  if (isString(controller)) {
    var match = CNTRL_REG.exec(controller);
    if (match) return match[3];
  }
}


/**
 * @ngdoc provider
 * @name $controllerProvider
 * @description
 * The {@link ng.$controller $controller service} is used by Angular to create new
 * controllers.
 *
 * This provider allows controller registration via the
 * {@link ng.$controllerProvider#register register} method.
 */
function $ControllerProvider() {
  var controllers = {},
      globals = false;

  /**
   * @ngdoc method
   * @name $controllerProvider#register
   * @param {string|Object} name Controller name, or an object map of controllers where the keys are
   *    the names and the values are the constructors.
   * @param {Function|Array} constructor Controller constructor fn (optionally decorated with DI
   *    annotations in the array notation).
   */
  this.register = function(name, constructor) {
    assertNotHasOwnProperty(name, 'controller');
    if (isObject(name)) {
      extend(controllers, name);
    } else {
      controllers[name] = constructor;
    }
  };

  /**
   * @ngdoc method
   * @name $controllerProvider#allowGlobals
   * @description If called, allows `$controller` to find controller constructors on `window`
   */
  this.allowGlobals = function() {
    globals = true;
  };


  this.$get = ['$injector', '$window', function($injector, $window) {

    /**
     * @ngdoc service
     * @name $controller
     * @requires $injector
     *
     * @param {Function|string} constructor If called with a function then it's considered to be the
     *    controller constructor function. Otherwise it's considered to be a string which is used
     *    to retrieve the controller constructor using the following steps:
     *
     *    * check if a controller with given name is registered via `$controllerProvider`
     *    * check if evaluating the string on the current scope returns a constructor
     *    * if $controllerProvider#allowGlobals, check `window[constructor]` on the global
     *      `window` object (not recommended)
     *
     *    The string can use the `controller as property` syntax, where the controller instance is published
     *    as the specified property on the `scope`; the `scope` must be injected into `locals` param for this
     *    to work correctly.
     *
     * @param {Object} locals Injection locals for Controller.
     * @return {Object} Instance of given controller.
     *
     * @description
     * `$controller` service is responsible for instantiating controllers.
     *
     * It's just a simple call to {@link auto.$injector $injector}, but extracted into
     * a service, so that one can override this service with [BC version](https://gist.github.com/1649788).
     */
    return function(expression, locals, later, ident) {
      // PRIVATE API:
      //   param `later` --- indicates that the controller's constructor is invoked at a later time.
      //                     If true, $controller will allocate the object with the correct
      //                     prototype chain, but will not invoke the controller until a returned
      //                     callback is invoked.
      //   param `ident` --- An optional label which overrides the label parsed from the controller
      //                     expression, if any.
      var instance, match, constructor, identifier;
      later = later === true;
      if (ident && isString(ident)) {
        identifier = ident;
      }

      if (isString(expression)) {
        match = expression.match(CNTRL_REG);
        if (!match) {
          throw $controllerMinErr('ctrlfmt',
            "Badly formed controller string '{0}'. " +
            "Must match `__name__ as __id__` or `__name__`.", expression);
        }
        constructor = match[1],
        identifier = identifier || match[3];
        expression = controllers.hasOwnProperty(constructor)
            ? controllers[constructor]
            : getter(locals.$scope, constructor, true) ||
                (globals ? getter($window, constructor, true) : undefined);

        assertArgFn(expression, constructor, true);
      }

      if (later) {
        // Instantiate controller later:
        // This machinery is used to create an instance of the object before calling the
        // controller's constructor itself.
        //
        // This allows properties to be added to the controller before the constructor is
        // invoked. Primarily, this is used for isolate scope bindings in $compile.
        //
        // This feature is not intended for use by applications, and is thus not documented
        // publicly.
        // Object creation: http://jsperf.com/create-constructor/2
        var controllerPrototype = (isArray(expression) ?
          expression[expression.length - 1] : expression).prototype;
        instance = Object.create(controllerPrototype || null);

        if (identifier) {
          addIdentifier(locals, identifier, instance, constructor || expression.name);
        }

        var instantiate;
        return instantiate = extend(function() {
          var result = $injector.invoke(expression, instance, locals, constructor);
          if (result !== instance && (isObject(result) || isFunction(result))) {
            instance = result;
            if (identifier) {
              // If result changed, re-assign controllerAs value to scope.
              addIdentifier(locals, identifier, instance, constructor || expression.name);
            }
          }
          return instance;
        }, {
          instance: instance,
          identifier: identifier
        });
      }

      instance = $injector.instantiate(expression, locals, constructor);

      if (identifier) {
        addIdentifier(locals, identifier, instance, constructor || expression.name);
      }

      return instance;
    };

    function addIdentifier(locals, identifier, instance, name) {
      if (!(locals && isObject(locals.$scope))) {
        throw minErr('$controller')('noscp',
          "Cannot export controller '{0}' as '{1}'! No $scope object provided via `locals`.",
          name, identifier);
      }

      locals.$scope[identifier] = instance;
    }
  }];
}

/**
 * @ngdoc service
 * @name $document
 * @requires $window
 *
 * @description
 * A {@link angular.element jQuery or jqLite} wrapper for the browser's `window.document` object.
 *
 * @example
   <example module="documentExample">
     <file name="index.html">
       <div ng-controller="ExampleController">
         <p>$document title: <b ng-bind="title"></b></p>
         <p>window.document title: <b ng-bind="windowTitle"></b></p>
       </div>
     </file>
     <file name="script.js">
       angular.module('documentExample', [])
         .controller('ExampleController', ['$scope', '$document', function($scope, $document) {
           $scope.title = $document[0].title;
           $scope.windowTitle = angular.element(window.document)[0].title;
         }]);
     </file>
   </example>
 */
function $DocumentProvider() {
  this.$get = ['$window', function(window) {
    return jqLite(window.document);
  }];
}

/**
 * @ngdoc service
 * @name $exceptionHandler
 * @requires ng.$log
 *
 * @description
 * Any uncaught exception in angular expressions is delegated to this service.
 * The default implementation simply delegates to `$log.error` which logs it into
 * the browser console.
 *
 * In unit tests, if `angular-mocks.js` is loaded, this service is overridden by
 * {@link ngMock.$exceptionHandler mock $exceptionHandler} which aids in testing.
 *
 * ## Example:
 *
 * ```js
 *   angular.module('exceptionOverride', []).factory('$exceptionHandler', function() {
 *     return function(exception, cause) {
 *       exception.message += ' (caused by "' + cause + '")';
 *       throw exception;
 *     };
 *   });
 * ```
 *
 * This example will override the normal action of `$exceptionHandler`, to make angular
 * exceptions fail hard when they happen, instead of just logging to the console.
 *
 * <hr />
 * Note, that code executed in event-listeners (even those registered using jqLite's `on`/`bind`
 * methods) does not delegate exceptions to the {@link ng.$exceptionHandler $exceptionHandler}
 * (unless executed during a digest).
 *
 * If you wish, you can manually delegate exceptions, e.g.
 * `try { ... } catch(e) { $exceptionHandler(e); }`
 *
 * @param {Error} exception Exception associated with the error.
 * @param {string=} cause optional information about the context in which
 *       the error was thrown.
 *
 */
function $ExceptionHandlerProvider() {
  this.$get = ['$log', function($log) {
    return function(exception, cause) {
      $log.error.apply($log, arguments);
    };
  }];
}

var $$ForceReflowProvider = function() {
  this.$get = ['$document', function($document) {
    return function(domNode) {
      //the line below will force the browser to perform a repaint so
      //that all the animated elements within the animation frame will
      //be properly updated and drawn on screen. This is required to
      //ensure that the preparation animation is properly flushed so that
      //the active state picks up from there. DO NOT REMOVE THIS LINE.
      //DO NOT OPTIMIZE THIS LINE. THE MINIFIER WILL REMOVE IT OTHERWISE WHICH
      //WILL RESULT IN AN UNPREDICTABLE BUG THAT IS VERY HARD TO TRACK DOWN AND
      //WILL TAKE YEARS AWAY FROM YOUR LIFE.
      if (domNode) {
        if (!domNode.nodeType && domNode instanceof jqLite) {
          domNode = domNode[0];
        }
      } else {
        domNode = $document[0].body;
      }
      return domNode.offsetWidth + 1;
    };
  }];
};

var APPLICATION_JSON = 'application/json';
var CONTENT_TYPE_APPLICATION_JSON = {'Content-Type': APPLICATION_JSON + ';charset=utf-8'};
var JSON_START = /^\[|^\{(?!\{)/;
var JSON_ENDS = {
  '[': /]$/,
  '{': /}$/
};
var JSON_PROTECTION_PREFIX = /^\)\]\}',?\n/;
var $httpMinErr = minErr('$http');
var $httpMinErrLegacyFn = function(method) {
  return function() {
    throw $httpMinErr('legacy', 'The method `{0}` on the promise returned from `$http` has been disabled.', method);
  };
};

function serializeValue(v) {
  if (isObject(v)) {
    return isDate(v) ? v.toISOString() : toJson(v);
  }
  return v;
}


function $HttpParamSerializerProvider() {
  /**
   * @ngdoc service
   * @name $httpParamSerializer
   * @description
   *
   * Default {@link $http `$http`} params serializer that converts objects to strings
   * according to the following rules:
   *
   * * `{'foo': 'bar'}` results in `foo=bar`
   * * `{'foo': Date.now()}` results in `foo=2015-04-01T09%3A50%3A49.262Z` (`toISOString()` and encoded representation of a Date object)
   * * `{'foo': ['bar', 'baz']}` results in `foo=bar&foo=baz` (repeated key for each array element)
   * * `{'foo': {'bar':'baz'}}` results in `foo=%7B%22bar%22%3A%22baz%22%7D"` (stringified and encoded representation of an object)
   *
   * Note that serializer will sort the request parameters alphabetically.
   * */

  this.$get = function() {
    return function ngParamSerializer(params) {
      if (!params) return '';
      var parts = [];
      forEachSorted(params, function(value, key) {
        if (value === null || isUndefined(value)) return;
        if (isArray(value)) {
          forEach(value, function(v, k) {
            parts.push(encodeUriQuery(key)  + '=' + encodeUriQuery(serializeValue(v)));
          });
        } else {
          parts.push(encodeUriQuery(key) + '=' + encodeUriQuery(serializeValue(value)));
        }
      });

      return parts.join('&');
    };
  };
}

function $HttpParamSerializerJQLikeProvider() {
  /**
   * @ngdoc service
   * @name $httpParamSerializerJQLike
   * @description
   *
   * Alternative {@link $http `$http`} params serializer that follows
   * jQuery's [`param()`](http://api.jquery.com/jquery.param/) method logic.
   * The serializer will also sort the params alphabetically.
   *
   * To use it for serializing `$http` request parameters, set it as the `paramSerializer` property:
   *
   * ```js
   * $http({
   *   url: myUrl,
   *   method: 'GET',
   *   params: myParams,
   *   paramSerializer: '$httpParamSerializerJQLike'
   * });
   * ```
   *
   * It is also possible to set it as the default `paramSerializer` in the
   * {@link $httpProvider#defaults `$httpProvider`}.
   *
   * Additionally, you can inject the serializer and use it explicitly, for example to serialize
   * form data for submission:
   *
   * ```js
   * .controller(function($http, $httpParamSerializerJQLike) {
   *   //...
   *
   *   $http({
   *     url: myUrl,
   *     method: 'POST',
   *     data: $httpParamSerializerJQLike(myData),
   *     headers: {
   *       'Content-Type': 'application/x-www-form-urlencoded'
   *     }
   *   });
   *
   * });
   * ```
   *
   * */
  this.$get = function() {
    return function jQueryLikeParamSerializer(params) {
      if (!params) return '';
      var parts = [];
      serialize(params, '', true);
      return parts.join('&');

      function serialize(toSerialize, prefix, topLevel) {
        if (toSerialize === null || isUndefined(toSerialize)) return;
        if (isArray(toSerialize)) {
          forEach(toSerialize, function(value, index) {
            serialize(value, prefix + '[' + (isObject(value) ? index : '') + ']');
          });
        } else if (isObject(toSerialize) && !isDate(toSerialize)) {
          forEachSorted(toSerialize, function(value, key) {
            serialize(value, prefix +
                (topLevel ? '' : '[') +
                key +
                (topLevel ? '' : ']'));
          });
        } else {
          parts.push(encodeUriQuery(prefix) + '=' + encodeUriQuery(serializeValue(toSerialize)));
        }
      }
    };
  };
}

function defaultHttpResponseTransform(data, headers) {
  if (isString(data)) {
    // Strip json vulnerability protection prefix and trim whitespace
    var tempData = data.replace(JSON_PROTECTION_PREFIX, '').trim();

    if (tempData) {
      var contentType = headers('Content-Type');
      if ((contentType && (contentType.indexOf(APPLICATION_JSON) === 0)) || isJsonLike(tempData)) {
        data = fromJson(tempData);
      }
    }
  }

  return data;
}

function isJsonLike(str) {
    var jsonStart = str.match(JSON_START);
    return jsonStart && JSON_ENDS[jsonStart[0]].test(str);
}

/**
 * Parse headers into key value object
 *
 * @param {string} headers Raw headers as a string
 * @returns {Object} Parsed headers as key value object
 */
function parseHeaders(headers) {
  var parsed = createMap(), i;

  function fillInParsed(key, val) {
    if (key) {
      parsed[key] = parsed[key] ? parsed[key] + ', ' + val : val;
    }
  }

  if (isString(headers)) {
    forEach(headers.split('\n'), function(line) {
      i = line.indexOf(':');
      fillInParsed(lowercase(trim(line.substr(0, i))), trim(line.substr(i + 1)));
    });
  } else if (isObject(headers)) {
    forEach(headers, function(headerVal, headerKey) {
      fillInParsed(lowercase(headerKey), trim(headerVal));
    });
  }

  return parsed;
}


/**
 * Returns a function that provides access to parsed headers.
 *
 * Headers are lazy parsed when first requested.
 * @see parseHeaders
 *
 * @param {(string|Object)} headers Headers to provide access to.
 * @returns {function(string=)} Returns a getter function which if called with:
 *
 *   - if called with single an argument returns a single header value or null
 *   - if called with no arguments returns an object containing all headers.
 */
function headersGetter(headers) {
  var headersObj;

  return function(name) {
    if (!headersObj) headersObj =  parseHeaders(headers);

    if (name) {
      var value = headersObj[lowercase(name)];
      if (value === void 0) {
        value = null;
      }
      return value;
    }

    return headersObj;
  };
}


/**
 * Chain all given functions
 *
 * This function is used for both request and response transforming
 *
 * @param {*} data Data to transform.
 * @param {function(string=)} headers HTTP headers getter fn.
 * @param {number} status HTTP status code of the response.
 * @param {(Function|Array.<Function>)} fns Function or an array of functions.
 * @returns {*} Transformed data.
 */
function transformData(data, headers, status, fns) {
  if (isFunction(fns)) {
    return fns(data, headers, status);
  }

  forEach(fns, function(fn) {
    data = fn(data, headers, status);
  });

  return data;
}


function isSuccess(status) {
  return 200 <= status && status < 300;
}


/**
 * @ngdoc provider
 * @name $httpProvider
 * @description
 * Use `$httpProvider` to change the default behavior of the {@link ng.$http $http} service.
 * */
function $HttpProvider() {
  /**
   * @ngdoc property
   * @name $httpProvider#defaults
   * @description
   *
   * Object containing default values for all {@link ng.$http $http} requests.
   *
   * - **`defaults.cache`** - {Object} - an object built with {@link ng.$cacheFactory `$cacheFactory`}
   * that will provide the cache for all requests who set their `cache` property to `true`.
   * If you set the `defaults.cache = false` then only requests that specify their own custom
   * cache object will be cached. See {@link $http#caching $http Caching} for more information.
   *
   * - **`defaults.xsrfCookieName`** - {string} - Name of cookie containing the XSRF token.
   * Defaults value is `'XSRF-TOKEN'`.
   *
   * - **`defaults.xsrfHeaderName`** - {string} - Name of HTTP header to populate with the
   * XSRF token. Defaults value is `'X-XSRF-TOKEN'`.
   *
   * - **`defaults.headers`** - {Object} - Default headers for all $http requests.
   * Refer to {@link ng.$http#setting-http-headers $http} for documentation on
   * setting default headers.
   *     - **`defaults.headers.common`**
   *     - **`defaults.headers.post`**
   *     - **`defaults.headers.put`**
   *     - **`defaults.headers.patch`**
   *
   *
   * - **`defaults.paramSerializer`** - `{string|function(Object<string,string>):string}` - A function
   *  used to the prepare string representation of request parameters (specified as an object).
   *  If specified as string, it is interpreted as a function registered with the {@link auto.$injector $injector}.
   *  Defaults to {@link ng.$httpParamSerializer $httpParamSerializer}.
   *
   **/
  var defaults = this.defaults = {
    // transform incoming response data
    transformResponse: [defaultHttpResponseTransform],

    // transform outgoing request data
    transformRequest: [function(d) {
      return isObject(d) && !isFile(d) && !isBlob(d) && !isFormData(d) ? toJson(d) : d;
    }],

    // default headers
    headers: {
      common: {
        'Accept': 'application/json, text/plain, */*'
      },
      post:   shallowCopy(CONTENT_TYPE_APPLICATION_JSON),
      put:    shallowCopy(CONTENT_TYPE_APPLICATION_JSON),
      patch:  shallowCopy(CONTENT_TYPE_APPLICATION_JSON)
    },

    xsrfCookieName: 'XSRF-TOKEN',
    xsrfHeaderName: 'X-XSRF-TOKEN',

    paramSerializer: '$httpParamSerializer'
  };

  var useApplyAsync = false;
  /**
   * @ngdoc method
   * @name $httpProvider#useApplyAsync
   * @description
   *
   * Configure $http service to combine processing of multiple http responses received at around
   * the same time via {@link ng.$rootScope.Scope#$applyAsync $rootScope.$applyAsync}. This can result in
   * significant performance improvement for bigger applications that make many HTTP requests
   * concurrently (common during application bootstrap).
   *
   * Defaults to false. If no value is specified, returns the current configured value.
   *
   * @param {boolean=} value If true, when requests are loaded, they will schedule a deferred
   *    "apply" on the next tick, giving time for subsequent requests in a roughly ~10ms window
   *    to load and share the same digest cycle.
   *
   * @returns {boolean|Object} If a value is specified, returns the $httpProvider for chaining.
   *    otherwise, returns the current configured value.
   **/
  this.useApplyAsync = function(value) {
    if (isDefined(value)) {
      useApplyAsync = !!value;
      return this;
    }
    return useApplyAsync;
  };

  var useLegacyPromise = true;
  /**
   * @ngdoc method
   * @name $httpProvider#useLegacyPromiseExtensions
   * @description
   *
   * Configure `$http` service to return promises without the shorthand methods `success` and `error`.
   * This should be used to make sure that applications work without these methods.
   *
   * Defaults to true. If no value is specified, returns the current configured value.
   *
   * @param {boolean=} value If true, `$http` will return a promise with the deprecated legacy `success` and `error` methods.
   *
   * @returns {boolean|Object} If a value is specified, returns the $httpProvider for chaining.
   *    otherwise, returns the current configured value.
   **/
  this.useLegacyPromiseExtensions = function(value) {
    if (isDefined(value)) {
      useLegacyPromise = !!value;
      return this;
    }
    return useLegacyPromise;
  };

  /**
   * @ngdoc property
   * @name $httpProvider#interceptors
   * @description
   *
   * Array containing service factories for all synchronous or asynchronous {@link ng.$http $http}
   * pre-processing of request or postprocessing of responses.
   *
   * These service factories are ordered by request, i.e. they are applied in the same order as the
   * array, on request, but reverse order, on response.
   *
   * {@link ng.$http#interceptors Interceptors detailed info}
   **/
  var interceptorFactories = this.interceptors = [];

  this.$get = ['$httpBackend', '$$cookieReader', '$cacheFactory', '$rootScope', '$q', '$injector',
      function($httpBackend, $$cookieReader, $cacheFactory, $rootScope, $q, $injector) {

    var defaultCache = $cacheFactory('$http');

    /**
     * Make sure that default param serializer is exposed as a function
     */
    defaults.paramSerializer = isString(defaults.paramSerializer) ?
      $injector.get(defaults.paramSerializer) : defaults.paramSerializer;

    /**
     * Interceptors stored in reverse order. Inner interceptors before outer interceptors.
     * The reversal is needed so that we can build up the interception chain around the
     * server request.
     */
    var reversedInterceptors = [];

    forEach(interceptorFactories, function(interceptorFactory) {
      reversedInterceptors.unshift(isString(interceptorFactory)
          ? $injector.get(interceptorFactory) : $injector.invoke(interceptorFactory));
    });

    /**
     * @ngdoc service
     * @kind function
     * @name $http
     * @requires ng.$httpBackend
     * @requires $cacheFactory
     * @requires $rootScope
     * @requires $q
     * @requires $injector
     *
     * @description
     * The `$http` service is a core Angular service that facilitates communication with the remote
     * HTTP servers via the browser's [XMLHttpRequest](https://developer.mozilla.org/en/xmlhttprequest)
     * object or via [JSONP](http://en.wikipedia.org/wiki/JSONP).
     *
     * For unit testing applications that use `$http` service, see
     * {@link ngMock.$httpBackend $httpBackend mock}.
     *
     * For a higher level of abstraction, please check out the {@link ngResource.$resource
     * $resource} service.
     *
     * The $http API is based on the {@link ng.$q deferred/promise APIs} exposed by
     * the $q service. While for simple usage patterns this doesn't matter much, for advanced usage
     * it is important to familiarize yourself with these APIs and the guarantees they provide.
     *
     *
     * ## General usage
     * The `$http` service is a function which takes a single argument â€” a {@link $http#usage configuration object} â€”
     * that is used to generate an HTTP request and returns  a {@link ng.$q promise}.
     *
     * ```js
     *   // Simple GET request example:
     *   $http({
     *     method: 'GET',
     *     url: '/someUrl'
     *   }).then(function successCallback(response) {
     *       // this callback will be called asynchronously
     *       // when the response is available
     *     }, function errorCallback(response) {
     *       // called asynchronously if an error occurs
     *       // or server returns response with an error status.
     *     });
     * ```
     *
     * The response object has these properties:
     *
     *   - **data** â€“ `{string|Object}` â€“ The response body transformed with the transform
     *     functions.
     *   - **status** â€“ `{number}` â€“ HTTP status code of the response.
     *   - **headers** â€“ `{function([headerName])}` â€“ Header getter function.
     *   - **config** â€“ `{Object}` â€“ The configuration object that was used to generate the request.
     *   - **statusText** â€“ `{string}` â€“ HTTP status text of the response.
     *
     * A response status code between 200 and 299 is considered a success status and
     * will result in the success callback being called. Note that if the response is a redirect,
     * XMLHttpRequest will transparently follow it, meaning that the error callback will not be
     * called for such responses.
     *
     *
     * ## Shortcut methods
     *
     * Shortcut methods are also available. All shortcut methods require passing in the URL, and
     * request data must be passed in for POST/PUT requests. An optional config can be passed as the
     * last argument.
     *
     * ```js
     *   $http.get('/someUrl', config).then(successCallback, errorCallback);
     *   $http.post('/someUrl', data, config).then(successCallback, errorCallback);
     * ```
     *
     * Complete list of shortcut methods:
     *
     * - {@link ng.$http#get $http.get}
     * - {@link ng.$http#head $http.head}
     * - {@link ng.$http#post $http.post}
     * - {@link ng.$http#put $http.put}
     * - {@link ng.$http#delete $http.delete}
     * - {@link ng.$http#jsonp $http.jsonp}
     * - {@link ng.$http#patch $http.patch}
     *
     *
     * ## Writing Unit Tests that use $http
     * When unit testing (using {@link ngMock ngMock}), it is necessary to call
     * {@link ngMock.$httpBackend#flush $httpBackend.flush()} to flush each pending
     * request using trained responses.
     *
     * ```
     * $httpBackend.expectGET(...);
     * $http.get(...);
     * $httpBackend.flush();
     * ```
     *
     * ## Deprecation Notice
     * <div class="alert alert-danger">
     *   The `$http` legacy promise methods `success` and `error` have been deprecated.
     *   Use the standard `then` method instead.
     *   If {@link $httpProvider#useLegacyPromiseExtensions `$httpProvider.useLegacyPromiseExtensions`} is set to
     *   `false` then these methods will throw {@link $http:legacy `$http/legacy`} error.
     * </div>
     *
     * ## Setting HTTP Headers
     *
     * The $http service will automatically add certain HTTP headers to all requests. These defaults
     * can be fully configured by accessing the `$httpProvider.defaults.headers` configuration
     * object, which currently contains this default configuration:
     *
     * - `$httpProvider.defaults.headers.common` (headers that are common for all requests):
     *   - `Accept: application/json, text/plain, * / *`
     * - `$httpProvider.defaults.headers.post`: (header defaults for POST requests)
     *   - `Content-Type: application/json`
     * - `$httpProvider.defaults.headers.put` (header defaults for PUT requests)
     *   - `Content-Type: application/json`
     *
     * To add or overwrite these defaults, simply add or remove a property from these configuration
     * objects. To add headers for an HTTP method other than POST or PUT, simply add a new object
     * with the lowercased HTTP method name as the key, e.g.
     * `$httpProvider.defaults.headers.get = { 'My-Header' : 'value' }`.
     *
     * The defaults can also be set at runtime via the `$http.defaults` object in the same
     * fashion. For example:
     *
     * ```
     * module.run(function($http) {
     *   $http.defaults.headers.common.Authorization = 'Basic YmVlcDpib29w'
     * });
     * ```
     *
     * In addition, you can supply a `headers` property in the config object passed when
     * calling `$http(config)`, which overrides the defaults without changing them globally.
     *
     * To explicitly remove a header automatically added via $httpProvider.defaults.headers on a per request basis,
     * Use the `headers` property, setting the desired header to `undefined`. For example:
     *
     * ```js
     * var req = {
     *  method: 'POST',
     *  url: 'http://example.com',
     *  headers: {
     *    'Content-Type': undefined
     *  },
     *  data: { test: 'test' }
     * }
     *
     * $http(req).then(function(){...}, function(){...});
     * ```
     *
     * ## Transforming Requests and Responses
     *
     * Both requests and responses can be transformed using transformation functions: `transformRequest`
     * and `transformResponse`. These properties can be a single function that returns
     * the transformed value (`function(data, headersGetter, status)`) or an array of such transformation functions,
     * which allows you to `push` or `unshift` a new transformation function into the transformation chain.
     *
     * ### Default Transformations
     *
     * The `$httpProvider` provider and `$http` service expose `defaults.transformRequest` and
     * `defaults.transformResponse` properties. If a request does not provide its own transformations
     * then these will be applied.
     *
     * You can augment or replace the default transformations by modifying these properties by adding to or
     * replacing the array.
     *
     * Angular provides the following default transformations:
     *
     * Request transformations (`$httpProvider.defaults.transformRequest` and `$http.defaults.transformRequest`):
     *
     * - If the `data` property of the request configuration object contains an object, serialize it
     *   into JSON format.
     *
     * Response transformations (`$httpProvider.defaults.transformResponse` and `$http.defaults.transformResponse`):
     *
     *  - If XSRF prefix is detected, strip it (see Security Considerations section below).
     *  - If JSON response is detected, deserialize it using a JSON parser.
     *
     *
     * ### Overriding the Default Transformations Per Request
     *
     * If you wish override the request/response transformations only for a single request then provide
     * `transformRequest` and/or `transformResponse` properties on the configuration object passed
     * into `$http`.
     *
     * Note that if you provide these properties on the config object the default transformations will be
     * overwritten. If you wish to augment the default transformations then you must include them in your
     * local transformation array.
     *
     * The following code demonstrates adding a new response transformation to be run after the default response
     * transformations have been run.
     *
     * ```js
     * function appendTransform(defaults, transform) {
     *
     *   // We can't guarantee that the default transformation is an array
     *   defaults = angular.isArray(defaults) ? defaults : [defaults];
     *
     *   // Append the new transformation to the defaults
     *   return defaults.concat(transform);
     * }
     *
     * $http({
     *   url: '...',
     *   method: 'GET',
     *   transformResponse: appendTransform($http.defaults.transformResponse, function(value) {
     *     return doTransform(value);
     *   })
     * });
     * ```
     *
     *
     * ## Caching
     *
     * To enable caching, set the request configuration `cache` property to `true` (to use default
     * cache) or to a custom cache object (built with {@link ng.$cacheFactory `$cacheFactory`}).
     * When the cache is enabled, `$http` stores the response from the server in the specified
     * cache. The next time the same request is made, the response is served from the cache without
     * sending a request to the server.
     *
     * Note that even if the response is served from cache, delivery of the data is asynchronous in
     * the same way that real requests are.
     *
     * If there are multiple GET requests for the same URL that should be cached using the same
     * cache, but the cache is not populated yet, only one request to the server will be made and
     * the remaining requests will be fulfilled using the response from the first request.
     *
     * You can change the default cache to a new object (built with
     * {@link ng.$cacheFactory `$cacheFactory`}) by updating the
     * {@link ng.$http#defaults `$http.defaults.cache`} property. All requests who set
     * their `cache` property to `true` will now use this cache object.
     *
     * If you set the default cache to `false` then only requests that specify their own custom
     * cache object will be cached.
     *
     * ## Interceptors
     *
     * Before you start creating interceptors, be sure to understand the
     * {@link ng.$q $q and deferred/promise APIs}.
     *
     * For purposes of global error handling, authentication, or any kind of synchronous or
     * asynchronous pre-processing of request or postprocessing of responses, it is desirable to be
     * able to intercept requests before they are handed to the server and
     * responses before they are handed over to the application code that
     * initiated these requests. The interceptors leverage the {@link ng.$q
     * promise APIs} to fulfill this need for both synchronous and asynchronous pre-processing.
     *
     * The interceptors are service factories that are registered with the `$httpProvider` by
     * adding them to the `$httpProvider.interceptors` array. The factory is called and
     * injected with dependencies (if specified) and returns the interceptor.
     *
     * There are two kinds of interceptors (and two kinds of rejection interceptors):
     *
     *   * `request`: interceptors get called with a http {@link $http#usage config} object. The function is free to
     *     modify the `config` object or create a new one. The function needs to return the `config`
     *     object directly, or a promise containing the `config` or a new `config` object.
     *   * `requestError`: interceptor gets called when a previous interceptor threw an error or
     *     resolved with a rejection.
     *   * `response`: interceptors get called with http `response` object. The function is free to
     *     modify the `response` object or create a new one. The function needs to return the `response`
     *     object directly, or as a promise containing the `response` or a new `response` object.
     *   * `responseError`: interceptor gets called when a previous interceptor threw an error or
     *     resolved with a rejection.
     *
     *
     * ```js
     *   // register the interceptor as a service
     *   $provide.factory('myHttpInterceptor', function($q, dependency1, dependency2) {
     *     return {
     *       // optional method
     *       'request': function(config) {
     *         // do something on success
     *         return config;
     *       },
     *
     *       // optional method
     *      'requestError': function(rejection) {
     *         // do something on error
     *         if (canRecover(rejection)) {
     *           return responseOrNewPromise
     *         }
     *         return $q.reject(rejection);
     *       },
     *
     *
     *
     *       // optional method
     *       'response': function(response) {
     *         // do something on success
     *         return response;
     *       },
     *
     *       // optional method
     *      'responseError': function(rejection) {
     *         // do something on error
     *         if (canRecover(rejection)) {
     *           return responseOrNewPromise
     *         }
     *         return $q.reject(rejection);
     *       }
     *     };
     *   });
     *
     *   $httpProvider.interceptors.push('myHttpInterceptor');
     *
     *
     *   // alternatively, register the interceptor via an anonymous factory
     *   $httpProvider.interceptors.push(function($q, dependency1, dependency2) {
     *     return {
     *      'request': function(config) {
     *          // same as above
     *       },
     *
     *       'response': function(response) {
     *          // same as above
     *       }
     *     };
     *   });
     * ```
     *
     * ## Security Considerations
     *
     * When designing web applications, consider security threats from:
     *
     * - [JSON vulnerability](http://haacked.com/archive/2008/11/20/anatomy-of-a-subtle-json-vulnerability.aspx)
     * - [XSRF](http://en.wikipedia.org/wiki/Cross-site_request_forgery)
     *
     * Both server and the client must cooperate in order to eliminate these threats. Angular comes
     * pre-configured with strategies that address these issues, but for this to work backend server
     * cooperation is required.
     *
     * ### JSON Vulnerability Protection
     *
     * A [JSON vulnerability](http://haacked.com/archive/2008/11/20/anatomy-of-a-subtle-json-vulnerability.aspx)
     * allows third party website to turn your JSON resource URL into
     * [JSONP](http://en.wikipedia.org/wiki/JSONP) request under some conditions. To
     * counter this your server can prefix all JSON requests with following string `")]}',\n"`.
     * Angular will automatically strip the prefix before processing it as JSON.
     *
     * For example if your server needs to return:
     * ```js
     * ['one','two']
     * ```
     *
     * which is vulnerable to attack, your server can return:
     * ```js
     * )]}',
     * ['one','two']
     * ```
     *
     * Angular will strip the prefix, before processing the JSON.
     *
     *
     * ### Cross Site Request Forgery (XSRF) Protection
     *
     * [XSRF](http://en.wikipedia.org/wiki/Cross-site_request_forgery) is a technique by which
     * an unauthorized site can gain your user's private data. Angular provides a mechanism
     * to counter XSRF. When performing XHR requests, the $http service reads a token from a cookie
     * (by default, `XSRF-TOKEN`) and sets it as an HTTP header (`X-XSRF-TOKEN`). Since only
     * JavaScript that runs on your domain could read the cookie, your server can be assured that
     * the XHR came from JavaScript running on your domain. The header will not be set for
     * cross-domain requests.
     *
     * To take advantage of this, your server needs to set a token in a JavaScript readable session
     * cookie called `XSRF-TOKEN` on the first HTTP GET request. On subsequent XHR requests the
     * server can verify that the cookie matches `X-XSRF-TOKEN` HTTP header, and therefore be sure
     * that only JavaScript running on your domain could have sent the request. The token must be
     * unique for each user and must be verifiable by the server (to prevent the JavaScript from
     * making up its own tokens). We recommend that the token is a digest of your site's
     * authentication cookie with a [salt](https://en.wikipedia.org/wiki/Salt_(cryptography&#41;)
     * for added security.
     *
     * The name of the headers can be specified using the xsrfHeaderName and xsrfCookieName
     * properties of either $httpProvider.defaults at config-time, $http.defaults at run-time,
     * or the per-request config object.
     *
     * In order to prevent collisions in environments where multiple Angular apps share the
     * same domain or subdomain, we recommend that each application uses unique cookie name.
     *
     * @param {object} config Object describing the request to be made and how it should be
     *    processed. The object has following properties:
     *
     *    - **method** â€“ `{string}` â€“ HTTP method (e.g. 'GET', 'POST', etc)
     *    - **url** â€“ `{string}` â€“ Absolute or relative URL of the resource that is being requested.
     *    - **params** â€“ `{Object.<string|Object>}` â€“ Map of strings or objects which will be serialized
     *      with the `paramSerializer` and appended as GET parameters.
     *    - **data** â€“ `{string|Object}` â€“ Data to be sent as the request message data.
     *    - **headers** â€“ `{Object}` â€“ Map of strings or functions which return strings representing
     *      HTTP headers to send to the server. If the return value of a function is null, the
     *      header will not be sent. Functions accept a config object as an argument.
     *    - **xsrfHeaderName** â€“ `{string}` â€“ Name of HTTP header to populate with the XSRF token.
     *    - **xsrfCookieName** â€“ `{string}` â€“ Name of cookie containing the XSRF token.
     *    - **transformRequest** â€“
     *      `{function(data, headersGetter)|Array.<function(data, headersGetter)>}` â€“
     *      transform function or an array of such functions. The transform function takes the http
     *      request body and headers and returns its transformed (typically serialized) version.
     *      See {@link ng.$http#overriding-the-default-transformations-per-request
     *      Overriding the Default Transformations}
     *    - **transformResponse** â€“
     *      `{function(data, headersGetter, status)|Array.<function(data, headersGetter, status)>}` â€“
     *      transform function or an array of such functions. The transform function takes the http
     *      response body, headers and status and returns its transformed (typically deserialized) version.
     *      See {@link ng.$http#overriding-the-default-transformations-per-request
     *      Overriding the Default TransformationjqLiks}
     *    - **paramSerializer** - `{string|function(Object<string,string>):string}` - A function used to
     *      prepare the string representation of request parameters (specified as an object).
     *      If specified as string, it is interpreted as function registered with the
     *      {@link $injector $injector}, which means you can create your own serializer
     *      by registering it as a {@link auto.$provide#service service}.
     *      The default serializer is the {@link $httpParamSerializer $httpParamSerializer};
     *      alternatively, you can use the {@link $httpParamSerializerJQLike $httpParamSerializerJQLike}
     *    - **cache** â€“ `{boolean|Cache}` â€“ If true, a default $http cache will be used to cache the
     *      GET request, otherwise if a cache instance built with
     *      {@link ng.$cacheFactory $cacheFactory}, this cache will be used for
     *      caching.
     *    - **timeout** â€“ `{number|Promise}` â€“ timeout in milliseconds, or {@link ng.$q promise}
     *      that should abort the request when resolved.
     *    - **withCredentials** - `{boolean}` - whether to set the `withCredentials` flag on the
     *      XHR object. See [requests with credentials](https://developer.mozilla.org/docs/Web/HTTP/Access_control_CORS#Requests_with_credentials)
     *      for more information.
     *    - **responseType** - `{string}` - see
     *      [XMLHttpRequest.responseType](https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest#xmlhttprequest-responsetype).
     *
     * @returns {HttpPromise} Returns a {@link ng.$q `Promise}` that will be resolved to a response object
     *                        when the request succeeds or fails.
     *
     *
     * @property {Array.<Object>} pendingRequests Array of config objects for currently pending
     *   requests. This is primarily meant to be used for debugging purposes.
     *
     *
     * @example
<example module="httpExample">
<file name="index.html">
  <div ng-controller="FetchController">
    <select ng-model="method" aria-label="Request method">
      <option>GET</option>
      <option>JSONP</option>
    </select>
    <input type="text" ng-model="url" size="80" aria-label="URL" />
    <button id="fetchbtn" ng-click="fetch()">fetch</button><br>
    <button id="samplegetbtn" ng-click="updateModel('GET', 'http-hello.html')">Sample GET</button>
    <button id="samplejsonpbtn"
      ng-click="updateModel('JSONP',
                    'https://angularjs.org/greet.php?callback=JSON_CALLBACK&name=Super%20Hero')">
      Sample JSONP
    </button>
    <button id="invalidjsonpbtn"
      ng-click="updateModel('JSONP', 'https://angularjs.org/doesntexist&callback=JSON_CALLBACK')">
        Invalid JSONP
      </button>
    <pre>http status code: {{status}}</pre>
    <pre>http response data: {{data}}</pre>
  </div>
</file>
<file name="script.js">
  angular.module('httpExample', [])
    .controller('FetchController', ['$scope', '$http', '$templateCache',
      function($scope, $http, $templateCache) {
        $scope.method = 'GET';
        $scope.url = 'http-hello.html';

        $scope.fetch = function() {
          $scope.code = null;
          $scope.response = null;

          $http({method: $scope.method, url: $scope.url, cache: $templateCache}).
            then(function(response) {
              $scope.status = response.status;
              $scope.data = response.data;
            }, function(response) {
              $scope.data = response.data || "Request failed";
              $scope.status = response.status;
          });
        };

        $scope.updateModel = function(method, url) {
          $scope.method = method;
          $scope.url = url;
        };
      }]);
</file>
<file name="http-hello.html">
  Hello, $http!
</file>
<file name="protractor.js" type="protractor">
  var status = element(by.binding('status'));
  var data = element(by.binding('data'));
  var fetchBtn = element(by.id('fetchbtn'));
  var sampleGetBtn = element(by.id('samplegetbtn'));
  var sampleJsonpBtn = element(by.id('samplejsonpbtn'));
  var invalidJsonpBtn = element(by.id('invalidjsonpbtn'));

  it('should make an xhr GET request', function() {
    sampleGetBtn.click();
    fetchBtn.click();
    expect(status.getText()).toMatch('200');
    expect(data.getText()).toMatch(/Hello, \$http!/);
  });

// Commented out due to flakes. See https://github.com/angular/angular.js/issues/9185
// it('should make a JSONP request to angularjs.org', function() {
//   sampleJsonpBtn.click();
//   fetchBtn.click();
//   expect(status.getText()).toMatch('200');
//   expect(data.getText()).toMatch(/Super Hero!/);
// });

  it('should make JSONP request to invalid URL and invoke the error handler',
      function() {
    invalidJsonpBtn.click();
    fetchBtn.click();
    expect(status.getText()).toMatch('0');
    expect(data.getText()).toMatch('Request failed');
  });
</file>
</example>
     */
    function $http(requestConfig) {

      if (!angular.isObject(requestConfig)) {
        throw minErr('$http')('badreq', 'Http request configuration must be an object.  Received: {0}', requestConfig);
      }

      var config = extend({
        method: 'get',
        transformRequest: defaults.transformRequest,
        transformResponse: defaults.transformResponse,
        paramSerializer: defaults.paramSerializer
      }, requestConfig);

      config.headers = mergeHeaders(requestConfig);
      config.method = uppercase(config.method);
      config.paramSerializer = isString(config.paramSerializer) ?
        $injector.get(config.paramSerializer) : config.paramSerializer;

      var serverRequest = function(config) {
        var headers = config.headers;
        var reqData = transformData(config.data, headersGetter(headers), undefined, config.transformRequest);

        // strip content-type if data is undefined
        if (isUndefined(reqData)) {
          forEach(headers, function(value, header) {
            if (lowercase(header) === 'content-type') {
                delete headers[header];
            }
          });
        }

        if (isUndefined(config.withCredentials) && !isUndefined(defaults.withCredentials)) {
          config.withCredentials = defaults.withCredentials;
        }

        // send request
        return sendReq(config, reqData).then(transformResponse, transformResponse);
      };

      var chain = [serverRequest, undefined];
      var promise = $q.when(config);

      // apply interceptors
      forEach(reversedInterceptors, function(interceptor) {
        if (interceptor.request || interceptor.requestError) {
          chain.unshift(interceptor.request, interceptor.requestError);
        }
        if (interceptor.response || interceptor.responseError) {
          chain.push(interceptor.response, interceptor.responseError);
        }
      });

      while (chain.length) {
        var thenFn = chain.shift();
        var rejectFn = chain.shift();

        promise = promise.then(thenFn, rejectFn);
      }

      if (useLegacyPromise) {
        promise.success = function(fn) {
          assertArgFn(fn, 'fn');

          promise.then(function(response) {
            fn(response.data, response.status, response.headers, config);
          });
          return promise;
        };

        promise.error = function(fn) {
          assertArgFn(fn, 'fn');

          promise.then(null, function(response) {
            fn(response.data, response.status, response.headers, config);
          });
          return promise;
        };
      } else {
        promise.success = $httpMinErrLegacyFn('success');
        promise.error = $httpMinErrLegacyFn('error');
      }

      return promise;

      function transformResponse(response) {
        // make a copy since the response must be cacheable
        var resp = extend({}, response);
        resp.data = transformData(response.data, response.headers, response.status,
                                  config.transformResponse);
        return (isSuccess(response.status))
          ? resp
          : $q.reject(resp);
      }

      function executeHeaderFns(headers, config) {
        var headerContent, processedHeaders = {};

        forEach(headers, function(headerFn, header) {
          if (isFunction(headerFn)) {
            headerContent = headerFn(config);
            if (headerContent != null) {
              processedHeaders[header] = headerContent;
            }
          } else {
            processedHeaders[header] = headerFn;
          }
        });

        return processedHeaders;
      }

      function mergeHeaders(config) {
        var defHeaders = defaults.headers,
            reqHeaders = extend({}, config.headers),
            defHeaderName, lowercaseDefHeaderName, reqHeaderName;

        defHeaders = extend({}, defHeaders.common, defHeaders[lowercase(config.method)]);

        // using for-in instead of forEach to avoid unecessary iteration after header has been found
        defaultHeadersIteration:
        for (defHeaderName in defHeaders) {
          lowercaseDefHeaderName = lowercase(defHeaderName);

          for (reqHeaderName in reqHeaders) {
            if (lowercase(reqHeaderName) === lowercaseDefHeaderName) {
              continue defaultHeadersIteration;
            }
          }

          reqHeaders[defHeaderName] = defHeaders[defHeaderName];
        }

        // execute if header value is a function for merged headers
        return executeHeaderFns(reqHeaders, shallowCopy(config));
      }
    }

    $http.pendingRequests = [];

    /**
     * @ngdoc method
     * @name $http#get
     *
     * @description
     * Shortcut method to perform `GET` request.
     *
     * @param {string} url Relative or absolute URL specifying the destination of the request
     * @param {Object=} config Optional configuration object
     * @returns {HttpPromise} Future object
     */

    /**
     * @ngdoc method
     * @name $http#delete
     *
     * @description
     * Shortcut method to perform `DELETE` request.
     *
     * @param {string} url Relative or absolute URL specifying the destination of the request
     * @param {Object=} config Optional configuration object
     * @returns {HttpPromise} Future object
     */

    /**
     * @ngdoc method
     * @name $http#head
     *
     * @description
     * Shortcut method to perform `HEAD` request.
     *
     * @param {string} url Relative or absolute URL specifying the destination of the request
     * @param {Object=} config Optional configuration object
     * @returns {HttpPromise} Future object
     */

    /**
     * @ngdoc method
     * @name $http#jsonp
     *
     * @description
     * Shortcut method to perform `JSONP` request.
     *
     * @param {string} url Relative or absolute URL specifying the destination of the request.
     *                     The name of the callback should be the string `JSON_CALLBACK`.
     * @param {Object=} config Optional configuration object
     * @returns {HttpPromise} Future object
     */
    createShortMethods('get', 'delete', 'head', 'jsonp');

    /**
     * @ngdoc method
     * @name $http#post
     *
     * @description
     * Shortcut method to perform `POST` request.
     *
     * @param {string} url Relative or absolute URL specifying the destination of the request
     * @param {*} data Request content
     * @param {Object=} config Optional configuration object
     * @returns {HttpPromise} Future object
     */

    /**
     * @ngdoc method
     * @name $http#put
     *
     * @description
     * Shortcut method to perform `PUT` request.
     *
     * @param {string} url Relative or absolute URL specifying the destination of the request
     * @param {*} data Request content
     * @param {Object=} config Optional configuration object
     * @returns {HttpPromise} Future object
     */

     /**
      * @ngdoc method
      * @name $http#patch
      *
      * @description
      * Shortcut method to perform `PATCH` request.
      *
      * @param {string} url Relative or absolute URL specifying the destination of the request
      * @param {*} data Request content
      * @param {Object=} config Optional configuration object
      * @returns {HttpPromise} Future object
      */
    createShortMethodsWithData('post', 'put', 'patch');

        /**
         * @ngdoc property
         * @name $http#defaults
         *
         * @description
         * Runtime equivalent of the `$httpProvider.defaults` property. Allows configuration of
         * default headers, withCredentials as well as request and response transformations.
         *
         * See "Setting HTTP Headers" and "Transforming Requests and Responses" sections above.
         */
    $http.defaults = defaults;


    return $http;


    function createShortMethods(names) {
      forEach(arguments, function(name) {
        $http[name] = function(url, config) {
          return $http(extend({}, config || {}, {
            method: name,
            url: url
          }));
        };
      });
    }


    function createShortMethodsWithData(name) {
      forEach(arguments, function(name) {
        $http[name] = function(url, data, config) {
          return $http(extend({}, config || {}, {
            method: name,
            url: url,
            data: data
          }));
        };
      });
    }


    /**
     * Makes the request.
     *
     * !!! ACCESSES CLOSURE VARS:
     * $httpBackend, defaults, $log, $rootScope, defaultCache, $http.pendingRequests
     */
    function sendReq(config, reqData) {
      var deferred = $q.defer(),
          promise = deferred.promise,
          cache,
          cachedResp,
          reqHeaders = config.headers,
          url = buildUrl(config.url, config.paramSerializer(config.params));

      $http.pendingRequests.push(config);
      promise.then(removePendingReq, removePendingReq);


      if ((config.cache || defaults.cache) && config.cache !== false &&
          (config.method === 'GET' || config.method === 'JSONP')) {
        cache = isObject(config.cache) ? config.cache
              : isObject(defaults.cache) ? defaults.cache
              : defaultCache;
      }

      if (cache) {
        cachedResp = cache.get(url);
        if (isDefined(cachedResp)) {
          if (isPromiseLike(cachedResp)) {
            // cached request has already been sent, but there is no response yet
            cachedResp.then(resolvePromiseWithResult, resolvePromiseWithResult);
          } else {
            // serving from cache
            if (isArray(cachedResp)) {
              resolvePromise(cachedResp[1], cachedResp[0], shallowCopy(cachedResp[2]), cachedResp[3]);
            } else {
              resolvePromise(cachedResp, 200, {}, 'OK');
            }
          }
        } else {
          // put the promise for the non-transformed response into cache as a placeholder
          cache.put(url, promise);
        }
      }


      // if we won't have the response in cache, set the xsrf headers and
      // send the request to the backend
      if (isUndefined(cachedResp)) {
        var xsrfValue = urlIsSameOrigin(config.url)
            ? $$cookieReader()[config.xsrfCookieName || defaults.xsrfCookieName]
            : undefined;
        if (xsrfValue) {
          reqHeaders[(config.xsrfHeaderName || defaults.xsrfHeaderName)] = xsrfValue;
        }

        $httpBackend(config.method, url, reqData, done, reqHeaders, config.timeout,
            config.withCredentials, config.responseType);
      }

      return promise;


      /**
       * Callback registered to $httpBackend():
       *  - caches the response if desired
       *  - resolves the raw $http promise
       *  - calls $apply
       */
      function done(status, response, headersString, statusText) {
        if (cache) {
          if (isSuccess(status)) {
            cache.put(url, [status, response, parseHeaders(headersString), statusText]);
          } else {
            // remove promise from the cache
            cache.remove(url);
          }
        }

        function resolveHttpPromise() {
          resolvePromise(response, status, headersString, statusText);
        }

        if (useApplyAsync) {
          $rootScope.$applyAsync(resolveHttpPromise);
        } else {
          resolveHttpPromise();
          if (!$rootScope.$$phase) $rootScope.$apply();
        }
      }


      /**
       * Resolves the raw $http promise.
       */
      function resolvePromise(response, status, headers, statusText) {
        //status: HTTP response status code, 0, -1 (aborted by timeout / promise)
        status = status >= -1 ? status : 0;

        (isSuccess(status) ? deferred.resolve : deferred.reject)({
          data: response,
          status: status,
          headers: headersGetter(headers),
          config: config,
          statusText: statusText
        });
      }

      function resolvePromiseWithResult(result) {
        resolvePromise(result.data, result.status, shallowCopy(result.headers()), result.statusText);
      }

      function removePendingReq() {
        var idx = $http.pendingRequests.indexOf(config);
        if (idx !== -1) $http.pendingRequests.splice(idx, 1);
      }
    }


    function buildUrl(url, serializedParams) {
      if (serializedParams.length > 0) {
        url += ((url.indexOf('?') == -1) ? '?' : '&') + serializedParams;
      }
      return url;
    }
  }];
}

/**
 * @ngdoc service
 * @name $xhrFactory
 *
 * @description
 * Factory function used to create XMLHttpRequest objects.
 *
 * Replace or decorate this service to create your own custom XMLHttpRequest objects.
 *
 * ```
 * angular.module('myApp', [])
 * .factory('$xhrFactory', function() {
 *   return function createXhr(method, url) {
 *     return new window.XMLHttpRequest({mozSystem: true});
 *   };
 * });
 * ```
 *
 * @param {string} method HTTP method of the request (GET, POST, PUT, ..)
 * @param {string} url URL of the request.
 */
function $xhrFactoryProvider() {
  this.$get = function() {
    return function createXhr() {
      return new window.XMLHttpRequest();
    };
  };
}

/**
 * @ngdoc service
 * @name $httpBackend
 * @requires $window
 * @requires $document
 * @requires $xhrFactory
 *
 * @description
 * HTTP backend used by the {@link ng.$http service} that delegates to
 * XMLHttpRequest object or JSONP and deals with browser incompatibilities.
 *
 * You should never need to use this service directly, instead use the higher-level abstractions:
 * {@link ng.$http $http} or {@link ngResource.$resource $resource}.
 *
 * During testing this implementation is swapped with {@link ngMock.$httpBackend mock
 * $httpBackend} which can be trained with responses.
 */
function $HttpBackendProvider() {
  this.$get = ['$browser', '$window', '$document', '$xhrFactory', function($browser, $window, $document, $xhrFactory) {
    return createHttpBackend($browser, $xhrFactory, $browser.defer, $window.angular.callbacks, $document[0]);
  }];
}

function createHttpBackend($browser, createXhr, $browserDefer, callbacks, rawDocument) {
  // TODO(vojta): fix the signature
  return function(method, url, post, callback, headers, timeout, withCredentials, responseType) {
    $browser.$$incOutstandingRequestCount();
    url = url || $browser.url();

    if (lowercase(method) == 'jsonp') {
      var callbackId = '_' + (callbacks.counter++).toString(36);
      callbacks[callbackId] = function(data) {
        callbacks[callbackId].data = data;
        callbacks[callbackId].called = true;
      };

      var jsonpDone = jsonpReq(url.replace('JSON_CALLBACK', 'angular.callbacks.' + callbackId),
          callbackId, function(status, text) {
        completeRequest(callback, status, callbacks[callbackId].data, "", text);
        callbacks[callbackId] = noop;
      });
    } else {

      var xhr = createXhr(method, url);

      xhr.open(method, url, true);
      forEach(headers, function(value, key) {
        if (isDefined(value)) {
            xhr.setRequestHeader(key, value);
        }
      });

      xhr.onload = function requestLoaded() {
        var statusText = xhr.statusText || '';

        // responseText is the old-school way of retrieving response (supported by IE9)
        // response/responseType properties were introduced in XHR Level2 spec (supported by IE10)
        var response = ('response' in xhr) ? xhr.response : xhr.responseText;

        // normalize IE9 bug (http://bugs.jquery.com/ticket/1450)
        var status = xhr.status === 1223 ? 204 : xhr.status;

        // fix status code when it is 0 (0 status is undocumented).
        // Occurs when accessing file resources or on Android 4.1 stock browser
        // while retrieving files from application cache.
        if (status === 0) {
          status = response ? 200 : urlResolve(url).protocol == 'file' ? 404 : 0;
        }

        completeRequest(callback,
            status,
            response,
            xhr.getAllResponseHeaders(),
            statusText);
      };

      var requestError = function() {
        // The response is always empty
        // See https://xhr.spec.whatwg.org/#request-error-steps and https://fetch.spec.whatwg.org/#concept-network-error
        completeRequest(callback, -1, null, null, '');
      };

      xhr.onerror = requestError;
      xhr.onabort = requestError;

      if (withCredentials) {
        xhr.withCredentials = true;
      }

      if (responseType) {
        try {
          xhr.responseType = responseType;
        } catch (e) {
          // WebKit added support for the json responseType value on 09/03/2013
          // https://bugs.webkit.org/show_bug.cgi?id=73648. Versions of Safari prior to 7 are
          // known to throw when setting the value "json" as the response type. Other older
          // browsers implementing the responseType
          //
          // The json response type can be ignored if not supported, because JSON payloads are
          // parsed on the client-side regardless.
          if (responseType !== 'json') {
            throw e;
          }
        }
      }

      xhr.send(isUndefined(post) ? null : post);
    }

    if (timeout > 0) {
      var timeoutId = $browserDefer(timeoutRequest, timeout);
    } else if (isPromiseLike(timeout)) {
      timeout.then(timeoutRequest);
    }


    function timeoutRequest() {
      jsonpDone && jsonpDone();
      xhr && xhr.abort();
    }

    function completeRequest(callback, status, response, headersString, statusText) {
      // cancel timeout and subsequent timeout promise resolution
      if (isDefined(timeoutId)) {
        $browserDefer.cancel(timeoutId);
      }
      jsonpDone = xhr = null;

      callback(status, response, headersString, statusText);
      $browser.$$completeOutstandingRequest(noop);
    }
  };

  function jsonpReq(url, callbackId, done) {
    // we can't use jQuery/jqLite here because jQuery does crazy stuff with script elements, e.g.:
    // - fetches local scripts via XHR and evals them
    // - adds and immediately removes script elements from the document
    var script = rawDocument.createElement('script'), callback = null;
    script.type = "text/javascript";
    script.src = url;
    script.async = true;

    callback = function(event) {
      removeEventListenerFn(script, "load", callback);
      removeEventListenerFn(script, "error", callback);
      rawDocument.body.removeChild(script);
      script = null;
      var status = -1;
      var text = "unknown";

      if (event) {
        if (event.type === "load" && !callbacks[callbackId].called) {
          event = { type: "error" };
        }
        text = event.type;
        status = event.type === "error" ? 404 : 200;
      }

      if (done) {
        done(status, text);
      }
    };

    addEventListenerFn(script, "load", callback);
    addEventListenerFn(script, "error", callback);
    rawDocument.body.appendChild(script);
    return callback;
  }
}

var $interpolateMinErr = angular.$interpolateMinErr = minErr('$interpolate');
$interpolateMinErr.throwNoconcat = function(text) {
  throw $interpolateMinErr('noconcat',
      "Error while interpolating: {0}\nStrict Contextual Escaping disallows " +
      "interpolations that concatenate multiple expressions when a trusted value is " +
      "required.  See http://docs.angularjs.org/api/ng.$sce", text);
};

$interpolateMinErr.interr = function(text, err) {
  return $interpolateMinErr('interr', "Can't interpolate: {0}\n{1}", text, err.toString());
};

/**
 * @ngdoc provider
 * @name $interpolateProvider
 *
 * @description
 *
 * Used for configuring the interpolation markup. Defaults to `{{` and `}}`.
 *
 * @example
<example module="customInterpolationApp">
<file name="index.html">
<script>
  var customInterpolationApp = angular.module('customInterpolationApp', []);

  customInterpolationApp.config(function($interpolateProvider) {
    $interpolateProvider.startSymbol('//');
    $interpolateProvider.endSymbol('//');
  });


  customInterpolationApp.controller('DemoController', function() {
      this.label = "This binding is brought you by // interpolation symbols.";
  });
</script>
<div ng-app="App" ng-controller="DemoController as demo">
    //demo.label//
</div>
</file>
<file name="protractor.js" type="protractor">
  it('should interpolate binding with custom symbols', function() {
    expect(element(by.binding('demo.label')).getText()).toBe('This binding is brought you by // interpolation symbols.');
  });
</file>
</example>
 */
function $InterpolateProvider() {
  var startSymbol = '{{';
  var endSymbol = '}}';

  /**
   * @ngdoc method
   * @name $interpolateProvider#startSymbol
   * @description
   * Symbol to denote start of expression in the interpolated string. Defaults to `{{`.
   *
   * @param {string=} value new value to set the starting symbol to.
   * @returns {string|self} Returns the symbol when used as getter and self if used as setter.
   */
  this.startSymbol = function(value) {
    if (value) {
      startSymbol = value;
      return this;
    } else {
      return startSymbol;
    }
  };

  /**
   * @ngdoc method
   * @name $interpolateProvider#endSymbol
   * @description
   * Symbol to denote the end of expression in the interpolated string. Defaults to `}}`.
   *
   * @param {string=} value new value to set the ending symbol to.
   * @returns {string|self} Returns the symbol when used as getter and self if used as setter.
   */
  this.endSymbol = function(value) {
    if (value) {
      endSymbol = value;
      return this;
    } else {
      return endSymbol;
    }
  };


  this.$get = ['$parse', '$exceptionHandler', '$sce', function($parse, $exceptionHandler, $sce) {
    var startSymbolLength = startSymbol.length,
        endSymbolLength = endSymbol.length,
        escapedStartRegexp = new RegExp(startSymbol.replace(/./g, escape), 'g'),
        escapedEndRegexp = new RegExp(endSymbol.replace(/./g, escape), 'g');

    function escape(ch) {
      return '\\\\\\' + ch;
    }

    function unescapeText(text) {
      return text.replace(escapedStartRegexp, startSymbol).
        replace(escapedEndRegexp, endSymbol);
    }

    function stringify(value) {
      if (value == null) { // null || undefined
        return '';
      }
      switch (typeof value) {
        case 'string':
          break;
        case 'number':
          value = '' + value;
          break;
        default:
          value = toJson(value);
      }

      return value;
    }

    /**
     * @ngdoc service
     * @name $interpolate
     * @kind function
     *
     * @requires $parse
     * @requires $sce
     *
     * @description
     *
     * Compiles a string with markup into an interpolation function. This service is used by the
     * HTML {@link ng.$compile $compile} service for data binding. See
     * {@link ng.$interpolateProvider $interpolateProvider} for configuring the
     * interpolation markup.
     *
     *
     * ```js
     *   var $interpolate = ...; // injected
     *   var exp = $interpolate('Hello {{name | uppercase}}!');
     *   expect(exp({name:'Angular'})).toEqual('Hello ANGULAR!');
     * ```
     *
     * `$interpolate` takes an optional fourth argument, `allOrNothing`. If `allOrNothing` is
     * `true`, the interpolation function will return `undefined` unless all embedded expressions
     * evaluate to a value other than `undefined`.
     *
     * ```js
     *   var $interpolate = ...; // injected
     *   var context = {greeting: 'Hello', name: undefined };
     *
     *   // default "forgiving" mode
     *   var exp = $interpolate('{{greeting}} {{name}}!');
     *   expect(exp(context)).toEqual('Hello !');
     *
     *   // "allOrNothing" mode
     *   exp = $interpolate('{{greeting}} {{name}}!', false, null, true);
     *   expect(exp(context)).toBeUndefined();
     *   context.name = 'Angular';
     *   expect(exp(context)).toEqual('Hello Angular!');
     * ```
     *
     * `allOrNothing` is useful for interpolating URLs. `ngSrc` and `ngSrcset` use this behavior.
     *
     * ####Escaped Interpolation
     * $interpolate provides a mechanism for escaping interpolation markers. Start and end markers
     * can be escaped by preceding each of their characters with a REVERSE SOLIDUS U+005C (backslash).
     * It will be rendered as a regular start/end marker, and will not be interpreted as an expression
     * or binding.
     *
     * This enables web-servers to prevent script injection attacks and defacing attacks, to some
     * degree, while also enabling code examples to work without relying on the
     * {@link ng.directive:ngNonBindable ngNonBindable} directive.
     *
     * **For security purposes, it is strongly encouraged that web servers escape user-supplied data,
     * replacing angle brackets (&lt;, &gt;) with &amp;lt; and &amp;gt; respectively, and replacing all
     * interpolation start/end markers with their escaped counterparts.**
     *
     * Escaped interpolation markers are only replaced with the actual interpolation markers in rendered
     * output when the $interpolate service processes the text. So, for HTML elements interpolated
     * by {@link ng.$compile $compile}, or otherwise interpolated with the `mustHaveExpression` parameter
     * set to `true`, the interpolated text must contain an unescaped interpolation expression. As such,
     * this is typically useful only when user-data is used in rendering a template from the server, or
     * when otherwise untrusted data is used by a directive.
     *
     * <example>
     *  <file name="index.html">
     *    <div ng-init="username='A user'">
     *      <p ng-init="apptitle='Escaping demo'">{{apptitle}}: \{\{ username = "defaced value"; \}\}
     *        </p>
     *      <p><strong>{{username}}</strong> attempts to inject code which will deface the
     *        application, but fails to accomplish their task, because the server has correctly
     *        escaped the interpolation start/end markers with REVERSE SOLIDUS U+005C (backslash)
     *        characters.</p>
     *      <p>Instead, the result of the attempted script injection is visible, and can be removed
     *        from the database by an administrator.</p>
     *    </div>
     *  </file>
     * </example>
     *
     * @param {string} text The text with markup to interpolate.
     * @param {boolean=} mustHaveExpression if set to true then the interpolation string must have
     *    embedded expression in order to return an interpolation function. Strings with no
     *    embedded expression will return null for the interpolation function.
     * @param {string=} trustedContext when provided, the returned function passes the interpolated
     *    result through {@link ng.$sce#getTrusted $sce.getTrusted(interpolatedResult,
     *    trustedContext)} before returning it.  Refer to the {@link ng.$sce $sce} service that
     *    provides Strict Contextual Escaping for details.
     * @param {boolean=} allOrNothing if `true`, then the returned function returns undefined
     *    unless all embedded expressions evaluate to a value other than `undefined`.
     * @returns {function(context)} an interpolation function which is used to compute the
     *    interpolated string. The function has these parameters:
     *
     * - `context`: evaluation context for all expressions embedded in the interpolated text
     */
    function $interpolate(text, mustHaveExpression, trustedContext, allOrNothing) {
      allOrNothing = !!allOrNothing;
      var startIndex,
          endIndex,
          index = 0,
          expressions = [],
          parseFns = [],
          textLength = text.length,
          exp,
          concat = [],
          expressionPositions = [];

      while (index < textLength) {
        if (((startIndex = text.indexOf(startSymbol, index)) != -1) &&
             ((endIndex = text.indexOf(endSymbol, startIndex + startSymbolLength)) != -1)) {
          if (index !== startIndex) {
            concat.push(unescapeText(text.substring(index, startIndex)));
          }
          exp = text.substring(startIndex + startSymbolLength, endIndex);
          expressions.push(exp);
          parseFns.push($parse(exp, parseStringifyInterceptor));
          index = endIndex + endSymbolLength;
          expressionPositions.push(concat.length);
          concat.push('');
        } else {
          // we did not find an interpolation, so we have to add the remainder to the separators array
          if (index !== textLength) {
            concat.push(unescapeText(text.substring(index)));
          }
          break;
        }
      }

      // Concatenating expressions makes it hard to reason about whether some combination of
      // concatenated values are unsafe to use and could easily lead to XSS.  By requiring that a
      // single expression be used for iframe[src], object[src], etc., we ensure that the value
      // that's used is assigned or constructed by some JS code somewhere that is more testable or
      // make it obvious that you bound the value to some user controlled value.  This helps reduce
      // the load when auditing for XSS issues.
      if (trustedContext && concat.length > 1) {
          $interpolateMinErr.throwNoconcat(text);
      }

      if (!mustHaveExpression || expressions.length) {
        var compute = function(values) {
          for (var i = 0, ii = expressions.length; i < ii; i++) {
            if (allOrNothing && isUndefined(values[i])) return;
            concat[expressionPositions[i]] = values[i];
          }
          return concat.join('');
        };

        var getValue = function(value) {
          return trustedContext ?
            $sce.getTrusted(trustedContext, value) :
            $sce.valueOf(value);
        };

        return extend(function interpolationFn(context) {
            var i = 0;
            var ii = expressions.length;
            var values = new Array(ii);

            try {
              for (; i < ii; i++) {
                values[i] = parseFns[i](context);
              }

              return compute(values);
            } catch (err) {
              $exceptionHandler($interpolateMinErr.interr(text, err));
            }

          }, {
          // all of these properties are undocumented for now
          exp: text, //just for compatibility with regular watchers created via $watch
          expressions: expressions,
          $$watchDelegate: function(scope, listener) {
            var lastValue;
            return scope.$watchGroup(parseFns, function interpolateFnWatcher(values, oldValues) {
              var currValue = compute(values);
              if (isFunction(listener)) {
                listener.call(this, currValue, values !== oldValues ? lastValue : currValue, scope);
              }
              lastValue = currValue;
            });
          }
        });
      }

      function parseStringifyInterceptor(value) {
        try {
          value = getValue(value);
          return allOrNothing && !isDefined(value) ? value : stringify(value);
        } catch (err) {
          $exceptionHandler($interpolateMinErr.interr(text, err));
        }
      }
    }


    /**
     * @ngdoc method
     * @name $interpolate#startSymbol
     * @description
     * Symbol to denote the start of expression in the interpolated string. Defaults to `{{`.
     *
     * Use {@link ng.$interpolateProvider#startSymbol `$interpolateProvider.startSymbol`} to change
     * the symbol.
     *
     * @returns {string} start symbol.
     */
    $interpolate.startSymbol = function() {
      return startSymbol;
    };


    /**
     * @ngdoc method
     * @name $interpolate#endSymbol
     * @description
     * Symbol to denote the end of expression in the interpolated string. Defaults to `}}`.
     *
     * Use {@link ng.$interpolateProvider#endSymbol `$interpolateProvider.endSymbol`} to change
     * the symbol.
     *
     * @returns {string} end symbol.
     */
    $interpolate.endSymbol = function() {
      return endSymbol;
    };

    return $interpolate;
  }];
}

function $IntervalProvider() {
  this.$get = ['$rootScope', '$window', '$q', '$$q',
       function($rootScope,   $window,   $q,   $$q) {
    var intervals = {};


     /**
      * @ngdoc service
      * @name $interval
      *
      * @description
      * Angular's wrapper for `window.setInterval`. The `fn` function is executed every `delay`
      * milliseconds.
      *
      * The return value of registering an interval function is a promise. This promise will be
      * notified upon each tick of the interval, and will be resolved after `count` iterations, or
      * run indefinitely if `count` is not defined. The value of the notification will be the
      * number of iterations that have run.
      * To cancel an interval, call `$interval.cancel(promise)`.
      *
      * In tests you can use {@link ngMock.$interval#flush `$interval.flush(millis)`} to
      * move forward by `millis` milliseconds and trigger any functions scheduled to run in that
      * time.
      *
      * <div class="alert alert-warning">
      * **Note**: Intervals created by this service must be explicitly destroyed when you are finished
      * with them.  In particular they are not automatically destroyed when a controller's scope or a
      * directive's element are destroyed.
      * You should take this into consideration and make sure to always cancel the interval at the
      * appropriate moment.  See the example below for more details on how and when to do this.
      * </div>
      *
      * @param {function()} fn A function that should be called repeatedly.
      * @param {number} delay Number of milliseconds between each function call.
      * @param {number=} [count=0] Number of times to repeat. If not set, or 0, will repeat
      *   indefinitely.
      * @param {boolean=} [invokeApply=true] If set to `false` skips model dirty checking, otherwise
      *   will invoke `fn` within the {@link ng.$rootScope.Scope#$apply $apply} block.
      * @param {...*=} Pass additional parameters to the executed function.
      * @returns {promise} A promise which will be notified on each iteration.
      *
      * @example
      * <example module="intervalExample">
      * <file name="index.html">
      *   <script>
      *     angular.module('intervalExample', [])
      *       .controller('ExampleController', ['$scope', '$interval',
      *         function($scope, $interval) {
      *           $scope.format = 'M/d/yy h:mm:ss a';
      *           $scope.blood_1 = 100;
      *           $scope.blood_2 = 120;
      *
      *           var stop;
      *           $scope.fight = function() {
      *             // Don't start a new fight if we are already fighting
      *             if ( angular.isDefined(stop) ) return;
      *
      *             stop = $interval(function() {
      *               if ($scope.blood_1 > 0 && $scope.blood_2 > 0) {
      *                 $scope.blood_1 = $scope.blood_1 - 3;
      *                 $scope.blood_2 = $scope.blood_2 - 4;
      *               } else {
      *                 $scope.stopFight();
      *               }
      *             }, 100);
      *           };
      *
      *           $scope.stopFight = function() {
      *             if (angular.isDefined(stop)) {
      *               $interval.cancel(stop);
      *               stop = undefined;
      *             }
      *           };
      *
      *           $scope.resetFight = function() {
      *             $scope.blood_1 = 100;
      *             $scope.blood_2 = 120;
      *           };
      *
      *           $scope.$on('$destroy', function() {
      *             // Make sure that the interval is destroyed too
      *             $scope.stopFight();
      *           });
      *         }])
      *       // Register the 'myCurrentTime' directive factory method.
      *       // We inject $interval and dateFilter service since the factory method is DI.
      *       .directive('myCurrentTime', ['$interval', 'dateFilter',
      *         function($interval, dateFilter) {
      *           // return the directive link function. (compile function not needed)
      *           return function(scope, element, attrs) {
      *             var format,  // date format
      *                 stopTime; // so that we can cancel the time updates
      *
      *             // used to update the UI
      *             function updateTime() {
      *               element.text(dateFilter(new Date(), format));
      *             }
      *
      *             // watch the expression, and update the UI on change.
      *             scope.$watch(attrs.myCurrentTime, function(value) {
      *               format = value;
      *               updateTime();
      *             });
      *
      *             stopTime = $interval(updateTime, 1000);
      *
      *             // listen on DOM destroy (removal) event, and cancel the next UI update
      *             // to prevent updating time after the DOM element was removed.
      *             element.on('$destroy', function() {
      *               $interval.cancel(stopTime);
      *             });
      *           }
      *         }]);
      *   </script>
      *
      *   <div>
      *     <div ng-controller="ExampleController">
      *       <label>Date format: <input ng-model="format"></label> <hr/>
      *       Current time is: <span my-current-time="format"></span>
      *       <hr/>
      *       Blood 1 : <font color='red'>{{blood_1}}</font>
      *       Blood 2 : <font color='red'>{{blood_2}}</font>
      *       <button type="button" data-ng-click="fight()">Fight</button>
      *       <button type="button" data-ng-click="stopFight()">StopFight</button>
      *       <button type="button" data-ng-click="resetFight()">resetFight</button>
      *     </div>
      *   </div>
      *
      * </file>
      * </example>
      */
    function interval(fn, delay, count, invokeApply) {
      var hasParams = arguments.length > 4,
          args = hasParams ? sliceArgs(arguments, 4) : [],
          setInterval = $window.setInterval,
          clearInterval = $window.clearInterval,
          iteration = 0,
          skipApply = (isDefined(invokeApply) && !invokeApply),
          deferred = (skipApply ? $$q : $q).defer(),
          promise = deferred.promise;

      count = isDefined(count) ? count : 0;

      promise.then(null, null, (!hasParams) ? fn : function() {
        fn.apply(null, args);
      });

      promise.$$intervalId = setInterval(function tick() {
        deferred.notify(iteration++);

        if (count > 0 && iteration >= count) {
          deferred.resolve(iteration);
          clearInterval(promise.$$intervalId);
          delete intervals[promise.$$intervalId];
        }

        if (!skipApply) $rootScope.$apply();

      }, delay);

      intervals[promise.$$intervalId] = deferred;

      return promise;
    }


     /**
      * @ngdoc method
      * @name $interval#cancel
      *
      * @description
      * Cancels a task associated with the `promise`.
      *
      * @param {Promise=} promise returned by the `$interval` function.
      * @returns {boolean} Returns `true` if the task was successfully canceled.
      */
    interval.cancel = function(promise) {
      if (promise && promise.$$intervalId in intervals) {
        intervals[promise.$$intervalId].reject('canceled');
        $window.clearInterval(promise.$$intervalId);
        delete intervals[promise.$$intervalId];
        return true;
      }
      return false;
    };

    return interval;
  }];
}

/**
 * @ngdoc service
 * @name $locale
 *
 * @description
 * $locale service provides localization rules for various Angular components. As of right now the
 * only public api is:
 *
 * * `id` â€“ `{string}` â€“ locale id formatted as `languageId-countryId` (e.g. `en-us`)
 */

var PATH_MATCH = /^([^\?#]*)(\?([^#]*))?(#(.*))?$/,
    DEFAULT_PORTS = {'http': 80, 'https': 443, 'ftp': 21};
var $locationMinErr = minErr('$location');


/**
 * Encode path using encodeUriSegment, ignoring forward slashes
 *
 * @param {string} path Path to encode
 * @returns {string}
 */
function encodePath(path) {
  var segments = path.split('/'),
      i = segments.length;

  while (i--) {
    segments[i] = encodeUriSegment(segments[i]);
  }

  return segments.join('/');
}

function parseAbsoluteUrl(absoluteUrl, locationObj) {
  var parsedUrl = urlResolve(absoluteUrl);

  locationObj.$$protocol = parsedUrl.protocol;
  locationObj.$$host = parsedUrl.hostname;
  locationObj.$$port = toInt(parsedUrl.port) || DEFAULT_PORTS[parsedUrl.protocol] || null;
}


function parseAppUrl(relativeUrl, locationObj) {
  var prefixed = (relativeUrl.charAt(0) !== '/');
  if (prefixed) {
    relativeUrl = '/' + relativeUrl;
  }
  var match = urlResolve(relativeUrl);
  locationObj.$$path = decodeURIComponent(prefixed && match.pathname.charAt(0) === '/' ?
      match.pathname.substring(1) : match.pathname);
  locationObj.$$search = parseKeyValue(match.search);
  locationObj.$$hash = decodeURIComponent(match.hash);

  // make sure path starts with '/';
  if (locationObj.$$path && locationObj.$$path.charAt(0) != '/') {
    locationObj.$$path = '/' + locationObj.$$path;
  }
}


/**
 *
 * @param {string} begin
 * @param {string} whole
 * @returns {string} returns text from whole after begin or undefined if it does not begin with
 *                   expected string.
 */
function beginsWith(begin, whole) {
  if (whole.indexOf(begin) === 0) {
    return whole.substr(begin.length);
  }
}


function stripHash(url) {
  var index = url.indexOf('#');
  return index == -1 ? url : url.substr(0, index);
}

function trimEmptyHash(url) {
  return url.replace(/(#.+)|#$/, '$1');
}


function stripFile(url) {
  return url.substr(0, stripHash(url).lastIndexOf('/') + 1);
}

/* return the server only (scheme://host:port) */
function serverBase(url) {
  return url.substring(0, url.indexOf('/', url.indexOf('//') + 2));
}


/**
 * LocationHtml5Url represents an url
 * This object is exposed as $location service when HTML5 mode is enabled and supported
 *
 * @constructor
 * @param {string} appBase application base URL
 * @param {string} appBaseNoFile application base URL stripped of any filename
 * @param {string} basePrefix url path prefix
 */
function LocationHtml5Url(appBase, appBaseNoFile, basePrefix) {
  this.$$html5 = true;
  basePrefix = basePrefix || '';
  parseAbsoluteUrl(appBase, this);


  /**
   * Parse given html5 (regular) url string into properties
   * @param {string} url HTML5 url
   * @private
   */
  this.$$parse = function(url) {
    var pathUrl = beginsWith(appBaseNoFile, url);
    if (!isString(pathUrl)) {
      throw $locationMinErr('ipthprfx', 'Invalid url "{0}", missing path prefix "{1}".', url,
          appBaseNoFile);
    }

    parseAppUrl(pathUrl, this);

    if (!this.$$path) {
      this.$$path = '/';
    }

    this.$$compose();
  };

  /**
   * Compose url and update `absUrl` property
   * @private
   */
  this.$$compose = function() {
    var search = toKeyValue(this.$$search),
        hash = this.$$hash ? '#' + encodeUriSegment(this.$$hash) : '';

    this.$$url = encodePath(this.$$path) + (search ? '?' + search : '') + hash;
    this.$$absUrl = appBaseNoFile + this.$$url.substr(1); // first char is always '/'
  };

  this.$$parseLinkUrl = function(url, relHref) {
    if (relHref && relHref[0] === '#') {
      // special case for links to hash fragments:
      // keep the old url and only replace the hash fragment
      this.hash(relHref.slice(1));
      return true;
    }
    var appUrl, prevAppUrl;
    var rewrittenUrl;

    if (isDefined(appUrl = beginsWith(appBase, url))) {
      prevAppUrl = appUrl;
      if (isDefined(appUrl = beginsWith(basePrefix, appUrl))) {
        rewrittenUrl = appBaseNoFile + (beginsWith('/', appUrl) || appUrl);
      } else {
        rewrittenUrl = appBase + prevAppUrl;
      }
    } else if (isDefined(appUrl = beginsWith(appBaseNoFile, url))) {
      rewrittenUrl = appBaseNoFile + appUrl;
    } else if (appBaseNoFile == url + '/') {
      rewrittenUrl = appBaseNoFile;
    }
    if (rewrittenUrl) {
      this.$$parse(rewrittenUrl);
    }
    return !!rewrittenUrl;
  };
}


/**
 * LocationHashbangUrl represents url
 * This object is exposed as $location service when developer doesn't opt into html5 mode.
 * It also serves as the base class for html5 mode fallback on legacy browsers.
 *
 * @constructor
 * @param {string} appBase application base URL
 * @param {string} appBaseNoFile application base URL stripped of any filename
 * @param {string} hashPrefix hashbang prefix
 */
function LocationHashbangUrl(appBase, appBaseNoFile, hashPrefix) {

  parseAbsoluteUrl(appBase, this);


  /**
   * Parse given hashbang url into properties
   * @param {string} url Hashbang url
   * @private
   */
  this.$$parse = function(url) {
    var withoutBaseUrl = beginsWith(appBase, url) || beginsWith(appBaseNoFile, url);
    var withoutHashUrl;

    if (!isUndefined(withoutBaseUrl) && withoutBaseUrl.charAt(0) === '#') {

      // The rest of the url starts with a hash so we have
      // got either a hashbang path or a plain hash fragment
      withoutHashUrl = beginsWith(hashPrefix, withoutBaseUrl);
      if (isUndefined(withoutHashUrl)) {
        // There was no hashbang prefix so we just have a hash fragment
        withoutHashUrl = withoutBaseUrl;
      }

    } else {
      // There was no hashbang path nor hash fragment:
      // If we are in HTML5 mode we use what is left as the path;
      // Otherwise we ignore what is left
      if (this.$$html5) {
        withoutHashUrl = withoutBaseUrl;
      } else {
        withoutHashUrl = '';
        if (isUndefined(withoutBaseUrl)) {
          appBase = url;
          this.replace();
        }
      }
    }

    parseAppUrl(withoutHashUrl, this);

    this.$$path = removeWindowsDriveName(this.$$path, withoutHashUrl, appBase);

    this.$$compose();

    /*
     * In Windows, on an anchor node on documents loaded from
     * the filesystem, the browser will return a pathname
     * prefixed with the drive name ('/C:/path') when a
     * pathname without a drive is set:
     *  * a.setAttribute('href', '/foo')
     *   * a.pathname === '/C:/foo' //true
     *
     * Inside of Angular, we're always using pathnames that
     * do not include drive names for routing.
     */
    function removeWindowsDriveName(path, url, base) {
      /*
      Matches paths for file protocol on windows,
      such as /C:/foo/bar, and captures only /foo/bar.
      */
      var windowsFilePathExp = /^\/[A-Z]:(\/.*)/;

      var firstPathSegmentMatch;

      //Get the relative path from the input URL.
      if (url.indexOf(base) === 0) {
        url = url.replace(base, '');
      }

      // The input URL intentionally contains a first path segment that ends with a colon.
      if (windowsFilePathExp.exec(url)) {
        return path;
      }

      firstPathSegmentMatch = windowsFilePathExp.exec(path);
      return firstPathSegmentMatch ? firstPathSegmentMatch[1] : path;
    }
  };

  /**
   * Compose hashbang url and update `absUrl` property
   * @private
   */
  this.$$compose = function() {
    var search = toKeyValue(this.$$search),
        hash = this.$$hash ? '#' + encodeUriSegment(this.$$hash) : '';

    this.$$url = encodePath(this.$$path) + (search ? '?' + search : '') + hash;
    this.$$absUrl = appBase + (this.$$url ? hashPrefix + this.$$url : '');
  };

  this.$$parseLinkUrl = function(url, relHref) {
    if (stripHash(appBase) == stripHash(url)) {
      this.$$parse(url);
      return true;
    }
    return false;
  };
}


/**
 * LocationHashbangUrl represents url
 * This object is exposed as $location service when html5 history api is enabled but the browser
 * does not support it.
 *
 * @constructor
 * @param {string} appBase application base URL
 * @param {string} appBaseNoFile application base URL stripped of any filename
 * @param {string} hashPrefix hashbang prefix
 */
function LocationHashbangInHtml5Url(appBase, appBaseNoFile, hashPrefix) {
  this.$$html5 = true;
  LocationHashbangUrl.apply(this, arguments);

  this.$$parseLinkUrl = function(url, relHref) {
    if (relHref && relHref[0] === '#') {
      // special case for links to hash fragments:
      // keep the old url and only replace the hash fragment
      this.hash(relHref.slice(1));
      return true;
    }

    var rewrittenUrl;
    var appUrl;

    if (appBase == stripHash(url)) {
      rewrittenUrl = url;
    } else if ((appUrl = beginsWith(appBaseNoFile, url))) {
      rewrittenUrl = appBase + hashPrefix + appUrl;
    } else if (appBaseNoFile === url + '/') {
      rewrittenUrl = appBaseNoFile;
    }
    if (rewrittenUrl) {
      this.$$parse(rewrittenUrl);
    }
    return !!rewrittenUrl;
  };

  this.$$compose = function() {
    var search = toKeyValue(this.$$search),
        hash = this.$$hash ? '#' + encodeUriSegment(this.$$hash) : '';

    this.$$url = encodePath(this.$$path) + (search ? '?' + search : '') + hash;
    // include hashPrefix in $$absUrl when $$url is empty so IE9 does not reload page because of removal of '#'
    this.$$absUrl = appBase + hashPrefix + this.$$url;
  };

}


var locationPrototype = {

  /**
   * Are we in html5 mode?
   * @private
   */
  $$html5: false,

  /**
   * Has any change been replacing?
   * @private
   */
  $$replace: false,

  /**
   * @ngdoc method
   * @name $location#absUrl
   *
   * @description
   * This method is getter only.
   *
   * Return full url representation with all segments encoded according to rules specified in
   * [RFC 3986](http://www.ietf.org/rfc/rfc3986.txt).
   *
   *
   * ```js
   * // given url http://example.com/#/some/path?foo=bar&baz=xoxo
   * var absUrl = $location.absUrl();
   * // => "http://example.com/#/some/path?foo=bar&baz=xoxo"
   * ```
   *
   * @return {string} full url
   */
  absUrl: locationGetter('$$absUrl'),

  /**
   * @ngdoc method
   * @name $location#url
   *
   * @description
   * This method is getter / setter.
   *
   * Return url (e.g. `/path?a=b#hash`) when called without any parameter.
   *
   * Change path, search and hash, when called with parameter and return `$location`.
   *
   *
   * ```js
   * // given url http://example.com/#/some/path?foo=bar&baz=xoxo
   * var url = $location.url();
   * // => "/some/path?foo=bar&baz=xoxo"
   * ```
   *
   * @param {string=} url New url without base prefix (e.g. `/path?a=b#hash`)
   * @return {string} url
   */
  url: function(url) {
    if (isUndefined(url)) {
      return this.$$url;
    }

    var match = PATH_MATCH.exec(url);
    if (match[1] || url === '') this.path(decodeURIComponent(match[1]));
    if (match[2] || match[1] || url === '') this.search(match[3] || '');
    this.hash(match[5] || '');

    return this;
  },

  /**
   * @ngdoc method
   * @name $location#protocol
   *
   * @description
   * This method is getter only.
   *
   * Return protocol of current url.
   *
   *
   * ```js
   * // given url http://example.com/#/some/path?foo=bar&baz=xoxo
   * var protocol = $location.protocol();
   * // => "http"
   * ```
   *
   * @return {string} protocol of current url
   */
  protocol: locationGetter('$$protocol'),

  /**
   * @ngdoc method
   * @name $location#host
   *
   * @description
   * This method is getter only.
   *
   * Return host of current url.
   *
   * Note: compared to the non-angular version `location.host` which returns `hostname:port`, this returns the `hostname` portion only.
   *
   *
   * ```js
   * // given url http://example.com/#/some/path?foo=bar&baz=xoxo
   * var host = $location.host();
   * // => "example.com"
   *
   * // given url http://user:password@example.com:8080/#/some/path?foo=bar&baz=xoxo
   * host = $location.host();
   * // => "example.com"
   * host = location.host;
   * // => "example.com:8080"
   * ```
   *
   * @return {string} host of current url.
   */
  host: locationGetter('$$host'),

  /**
   * @ngdoc method
   * @name $location#port
   *
   * @description
   * This method is getter only.
   *
   * Return port of current url.
   *
   *
   * ```js
   * // given url http://example.com/#/some/path?foo=bar&baz=xoxo
   * var port = $location.port();
   * // => 80
   * ```
   *
   * @return {Number} port
   */
  port: locationGetter('$$port'),

  /**
   * @ngdoc method
   * @name $location#path
   *
   * @description
   * This method is getter / setter.
   *
   * Return path of current url when called without any parameter.
   *
   * Change path when called with parameter and return `$location`.
   *
   * Note: Path should always begin with forward slash (/), this method will add the forward slash
   * if it is missing.
   *
   *
   * ```js
   * // given url http://example.com/#/some/path?foo=bar&baz=xoxo
   * var path = $location.path();
   * // => "/some/path"
   * ```
   *
   * @param {(string|number)=} path New path
   * @return {string} path
   */
  path: locationGetterSetter('$$path', function(path) {
    path = path !== null ? path.toString() : '';
    return path.charAt(0) == '/' ? path : '/' + path;
  }),

  /**
   * @ngdoc method
   * @name $location#search
   *
   * @description
   * This method is getter / setter.
   *
   * Return search part (as object) of current url when called without any parameter.
   *
   * Change search part when called with parameter and return `$location`.
   *
   *
   * ```js
   * // given url http://example.com/#/some/path?foo=bar&baz=xoxo
   * var searchObject = $location.search();
   * // => {foo: 'bar', baz: 'xoxo'}
   *
   * // set foo to 'yipee'
   * $location.search('foo', 'yipee');
   * // $location.search() => {foo: 'yipee', baz: 'xoxo'}
   * ```
   *
   * @param {string|Object.<string>|Object.<Array.<string>>} search New search params - string or
   * hash object.
   *
   * When called with a single argument the method acts as a setter, setting the `search` component
   * of `$location` to the specified value.
   *
   * If the argument is a hash object containing an array of values, these values will be encoded
   * as duplicate search parameters in the url.
   *
   * @param {(string|Number|Array<string>|boolean)=} paramValue If `search` is a string or number, then `paramValue`
   * will override only a single search property.
   *
   * If `paramValue` is an array, it will override the property of the `search` component of
   * `$location` specified via the first argument.
   *
   * If `paramValue` is `null`, the property specified via the first argument will be deleted.
   *
   * If `paramValue` is `true`, the property specified via the first argument will be added with no
   * value nor trailing equal sign.
   *
   * @return {Object} If called with no arguments returns the parsed `search` object. If called with
   * one or more arguments returns `$location` object itself.
   */
  search: function(search, paramValue) {
    switch (arguments.length) {
      case 0:
        return this.$$search;
      case 1:
        if (isString(search) || isNumber(search)) {
          search = search.toString();
          this.$$search = parseKeyValue(search);
        } else if (isObject(search)) {
          search = copy(search, {});
          // remove object undefined or null properties
          forEach(search, function(value, key) {
            if (value == null) delete search[key];
          });

          this.$$search = search;
        } else {
          throw $locationMinErr('isrcharg',
              'The first argument of the `$location#search()` call must be a string or an object.');
        }
        break;
      default:
        if (isUndefined(paramValue) || paramValue === null) {
          delete this.$$search[search];
        } else {
          this.$$search[search] = paramValue;
        }
    }

    this.$$compose();
    return this;
  },

  /**
   * @ngdoc method
   * @name $location#hash
   *
   * @description
   * This method is getter / setter.
   *
   * Returns the hash fragment when called without any parameters.
   *
   * Changes the hash fragment when called with a parameter and returns `$location`.
   *
   *
   * ```js
   * // given url http://example.com/#/some/path?foo=bar&baz=xoxo#hashValue
   * var hash = $location.hash();
   * // => "hashValue"
   * ```
   *
   * @param {(string|number)=} hash New hash fragment
   * @return {string} hash
   */
  hash: locationGetterSetter('$$hash', function(hash) {
    return hash !== null ? hash.toString() : '';
  }),

  /**
   * @ngdoc method
   * @name $location#replace
   *
   * @description
   * If called, all changes to $location during the current `$digest` will replace the current history
   * record, instead of adding a new one.
   */
  replace: function() {
    this.$$replace = true;
    return this;
  }
};

forEach([LocationHashbangInHtml5Url, LocationHashbangUrl, LocationHtml5Url], function(Location) {
  Location.prototype = Object.create(locationPrototype);

  /**
   * @ngdoc method
   * @name $location#state
   *
   * @description
   * This method is getter / setter.
   *
   * Return the history state object when called without any parameter.
   *
   * Change the history state object when called with one parameter and return `$location`.
   * The state object is later passed to `pushState` or `replaceState`.
   *
   * NOTE: This method is supported only in HTML5 mode and only in browsers supporting
   * the HTML5 History API (i.e. methods `pushState` and `replaceState`). If you need to support
   * older browsers (like IE9 or Android < 4.0), don't use this method.
   *
   * @param {object=} state State object for pushState or replaceState
   * @return {object} state
   */
  Location.prototype.state = function(state) {
    if (!arguments.length) {
      return this.$$state;
    }

    if (Location !== LocationHtml5Url || !this.$$html5) {
      throw $locationMinErr('nostate', 'History API state support is available only ' +
        'in HTML5 mode and only in browsers supporting HTML5 History API');
    }
    // The user might modify `stateObject` after invoking `$location.state(stateObject)`
    // but we're changing the $$state reference to $browser.state() during the $digest
    // so the modification window is narrow.
    this.$$state = isUndefined(state) ? null : state;

    return this;
  };
});


function locationGetter(property) {
  return function() {
    return this[property];
  };
}


function locationGetterSetter(property, preprocess) {
  return function(value) {
    if (isUndefined(value)) {
      return this[property];
    }

    this[property] = preprocess(value);
    this.$$compose();

    return this;
  };
}


/**
 * @ngdoc service
 * @name $location
 *
 * @requires $rootElement
 *
 * @description
 * The $location service parses the URL in the browser address bar (based on the
 * [window.location](https://developer.mozilla.org/en/window.location)) and makes the URL
 * available to your application. Changes to the URL in the address bar are reflected into
 * $location service and changes to $location are reflected into the browser address bar.
 *
 * **The $location service:**
 *
 * - Exposes the current URL in the browser address bar, so you can
 *   - Watch and observe the URL.
 *   - Change the URL.
 * - Synchronizes the URL with the browser when the user
 *   - Changes the address bar.
 *   - Clicks the back or forward button (or clicks a History link).
 *   - Clicks on a link.
 * - Represents the URL object as a set of methods (protocol, host, port, path, search, hash).
 *
 * For more information see {@link guide/$location Developer Guide: Using $location}
 */

/**
 * @ngdoc provider
 * @name $locationProvider
 * @description
 * Use the `$locationProvider` to configure how the application deep linking paths are stored.
 */
function $LocationProvider() {
  var hashPrefix = '',
      html5Mode = {
        enabled: false,
        requireBase: true,
        rewriteLinks: true
      };

  /**
   * @ngdoc method
   * @name $locationProvider#hashPrefix
   * @description
   * @param {string=} prefix Prefix for hash part (containing path and search)
   * @returns {*} current value if used as getter or itself (chaining) if used as setter
   */
  this.hashPrefix = function(prefix) {
    if (isDefined(prefix)) {
      hashPrefix = prefix;
      return this;
    } else {
      return hashPrefix;
    }
  };

  /**
   * @ngdoc method
   * @name $locationProvider#html5Mode
   * @description
   * @param {(boolean|Object)=} mode If boolean, sets `html5Mode.enabled` to value.
   *   If object, sets `enabled`, `requireBase` and `rewriteLinks` to respective values. Supported
   *   properties:
   *   - **enabled** â€“ `{boolean}` â€“ (default: false) If true, will rely on `history.pushState` to
   *     change urls where supported. Will fall back to hash-prefixed paths in browsers that do not
   *     support `pushState`.
   *   - **requireBase** - `{boolean}` - (default: `true`) When html5Mode is enabled, specifies
   *     whether or not a <base> tag is required to be present. If `enabled` and `requireBase` are
   *     true, and a base tag is not present, an error will be thrown when `$location` is injected.
   *     See the {@link guide/$location $location guide for more information}
   *   - **rewriteLinks** - `{boolean}` - (default: `true`) When html5Mode is enabled,
   *     enables/disables url rewriting for relative links.
   *
   * @returns {Object} html5Mode object if used as getter or itself (chaining) if used as setter
   */
  this.html5Mode = function(mode) {
    if (isBoolean(mode)) {
      html5Mode.enabled = mode;
      return this;
    } else if (isObject(mode)) {

      if (isBoolean(mode.enabled)) {
        html5Mode.enabled = mode.enabled;
      }

      if (isBoolean(mode.requireBase)) {
        html5Mode.requireBase = mode.requireBase;
      }

      if (isBoolean(mode.rewriteLinks)) {
        html5Mode.rewriteLinks = mode.rewriteLinks;
      }

      return this;
    } else {
      return html5Mode;
    }
  };

  /**
   * @ngdoc event
   * @name $location#$locationChangeStart
   * @eventType broadcast on root scope
   * @description
   * Broadcasted before a URL will change.
   *
   * This change can be prevented by calling
   * `preventDefault` method of the event. See {@link ng.$rootScope.Scope#$on} for more
   * details about event object. Upon successful change
   * {@link ng.$location#$locationChangeSuccess $locationChangeSuccess} is fired.
   *
   * The `newState` and `oldState` parameters may be defined only in HTML5 mode and when
   * the browser supports the HTML5 History API.
   *
   * @param {Object} angularEvent Synthetic event object.
   * @param {string} newUrl New URL
   * @param {string=} oldUrl URL that was before it was changed.
   * @param {string=} newState New history state object
   * @param {string=} oldState History state object that was before it was changed.
   */

  /**
   * @ngdoc event
   * @name $location#$locationChangeSuccess
   * @eventType broadcast on root scope
   * @description
   * Broadcasted after a URL was changed.
   *
   * The `newState` and `oldState` parameters may be defined only in HTML5 mode and when
   * the browser supports the HTML5 History API.
   *
   * @param {Object} angularEvent Synthetic event object.
   * @param {string} newUrl New URL
   * @param {string=} oldUrl URL that was before it was changed.
   * @param {string=} newState New history state object
   * @param {string=} oldState History state object that was before it was changed.
   */

  this.$get = ['$rootScope', '$browser', '$sniffer', '$rootElement', '$window',
      function($rootScope, $browser, $sniffer, $rootElement, $window) {
    var $location,
        LocationMode,
        baseHref = $browser.baseHref(), // if base[href] is undefined, it defaults to ''
        initialUrl = $browser.url(),
        appBase;

    if (html5Mode.enabled) {
      if (!baseHref && html5Mode.requireBase) {
        throw $locationMinErr('nobase',
          "$location in HTML5 mode requires a <base> tag to be present!");
      }
      appBase = serverBase(initialUrl) + (baseHref || '/');
      LocationMode = $sniffer.history ? LocationHtml5Url : LocationHashbangInHtml5Url;
    } else {
      appBase = stripHash(initialUrl);
      LocationMode = LocationHashbangUrl;
    }
    var appBaseNoFile = stripFile(appBase);

    $location = new LocationMode(appBase, appBaseNoFile, '#' + hashPrefix);
    $location.$$parseLinkUrl(initialUrl, initialUrl);

    $location.$$state = $browser.state();

    var IGNORE_URI_REGEXP = /^\s*(javascript|mailto):/i;

    function setBrowserUrlWithFallback(url, replace, state) {
      var oldUrl = $location.url();
      var oldState = $location.$$state;
      try {
        $browser.url(url, replace, state);

        // Make sure $location.state() returns referentially identical (not just deeply equal)
        // state object; this makes possible quick checking if the state changed in the digest
        // loop. Checking deep equality would be too expensive.
        $location.$$state = $browser.state();
      } catch (e) {
        // Restore old values if pushState fails
        $location.url(oldUrl);
        $location.$$state = oldState;

        throw e;
      }
    }

    $rootElement.on('click', function(event) {
      // TODO(vojta): rewrite link when opening in new tab/window (in legacy browser)
      // currently we open nice url link and redirect then

      if (!html5Mode.rewriteLinks || event.ctrlKey || event.metaKey || event.shiftKey || event.which == 2 || event.button == 2) return;

      var elm = jqLite(event.target);

      // traverse the DOM up to find first A tag
      while (nodeName_(elm[0]) !== 'a') {
        // ignore rewriting if no A tag (reached root element, or no parent - removed from document)
        if (elm[0] === $rootElement[0] || !(elm = elm.parent())[0]) return;
      }

      var absHref = elm.prop('href');
      // get the actual href attribute - see
      // http://msdn.microsoft.com/en-us/library/ie/dd347148(v=vs.85).aspx
      var relHref = elm.attr('href') || elm.attr('xlink:href');

      if (isObject(absHref) && absHref.toString() === '[object SVGAnimatedString]') {
        // SVGAnimatedString.animVal should be identical to SVGAnimatedString.baseVal, unless during
        // an animation.
        absHref = urlResolve(absHref.animVal).href;
      }

      // Ignore when url is started with javascript: or mailto:
      if (IGNORE_URI_REGEXP.test(absHref)) return;

      if (absHref && !elm.attr('target') && !event.isDefaultPrevented()) {
        if ($location.$$parseLinkUrl(absHref, relHref)) {
          // We do a preventDefault for all urls that are part of the angular application,
          // in html5mode and also without, so that we are able to abort navigation without
          // getting double entries in the location history.
          event.preventDefault();
          // update location manually
          if ($location.absUrl() != $browser.url()) {
            $rootScope.$apply();
            // hack to work around FF6 bug 684208 when scenario runner clicks on links
            $window.angular['ff-684208-preventDefault'] = true;
          }
        }
      }
    });


    // rewrite hashbang url <> html5 url
    if (trimEmptyHash($location.absUrl()) != trimEmptyHash(initialUrl)) {
      $browser.url($location.absUrl(), true);
    }

    var initializing = true;

    // update $location when $browser url changes
    $browser.onUrlChange(function(newUrl, newState) {

      if (isUndefined(beginsWith(appBaseNoFile, newUrl))) {
        // If we are navigating outside of the app then force a reload
        $window.location.href = newUrl;
        return;
      }

      $rootScope.$evalAsync(function() {
        var oldUrl = $location.absUrl();
        var oldState = $location.$$state;
        var defaultPrevented;
        newUrl = trimEmptyHash(newUrl);
        $location.$$parse(newUrl);
        $location.$$state = newState;

        defaultPrevented = $rootScope.$broadcast('$locationChangeStart', newUrl, oldUrl,
            newState, oldState).defaultPrevented;

        // if the location was changed by a `$locationChangeStart` handler then stop
        // processing this location change
        if ($location.absUrl() !== newUrl) return;

        if (defaultPrevented) {
          $location.$$parse(oldUrl);
          $location.$$state = oldState;
          setBrowserUrlWithFallback(oldUrl, false, oldState);
        } else {
          initializing = false;
          afterLocationChange(oldUrl, oldState);
        }
      });
      if (!$rootScope.$$phase) $rootScope.$digest();
    });

    // update browser
    $rootScope.$watch(function $locationWatch() {
      var oldUrl = trimEmptyHash($browser.url());
      var newUrl = trimEmptyHash($location.absUrl());
      var oldState = $browser.state();
      var currentReplace = $location.$$replace;
      var urlOrStateChanged = oldUrl !== newUrl ||
        ($location.$$html5 && $sniffer.history && oldState !== $location.$$state);

      if (initializing || urlOrStateChanged) {
        initializing = false;

        $rootScope.$evalAsync(function() {
          var newUrl = $location.absUrl();
          var defaultPrevented = $rootScope.$broadcast('$locationChangeStart', newUrl, oldUrl,
              $location.$$state, oldState).defaultPrevented;

          // if the location was changed by a `$locationChangeStart` handler then stop
          // processing this location change
          if ($location.absUrl() !== newUrl) return;

          if (defaultPrevented) {
            $location.$$parse(oldUrl);
            $location.$$state = oldState;
          } else {
            if (urlOrStateChanged) {
              setBrowserUrlWithFallback(newUrl, currentReplace,
                                        oldState === $location.$$state ? null : $location.$$state);
            }
            afterLocationChange(oldUrl, oldState);
          }
        });
      }

      $location.$$replace = false;

      // we don't need to return anything because $evalAsync will make the digest loop dirty when
      // there is a change
    });

    return $location;

    function afterLocationChange(oldUrl, oldState) {
      $rootScope.$broadcast('$locationChangeSuccess', $location.absUrl(), oldUrl,
        $location.$$state, oldState);
    }
}];
}

/**
 * @ngdoc service
 * @name $log
 * @requires $window
 *
 * @description
 * Simple service for logging. Default implementation safely writes the message
 * into the browser's console (if present).
 *
 * The main purpose of this service is to simplify debugging and troubleshooting.
 *
 * The default is to log `debug` messages. You can use
 * {@link ng.$logProvider ng.$logProvider#debugEnabled} to change this.
 *
 * @example
   <example module="logExample">
     <file name="script.js">
       angular.module('logExample', [])
         .controller('LogController', ['$scope', '$log', function($scope, $log) {
           $scope.$log = $log;
           $scope.message = 'Hello World!';
         }]);
     </file>
     <file name="index.html">
       <div ng-controller="LogController">
         <p>Reload this page with open console, enter text and hit the log button...</p>
         <label>Message:
         <input type="text" ng-model="message" /></label>
         <button ng-click="$log.log(message)">log</button>
         <button ng-click="$log.warn(message)">warn</button>
         <button ng-click="$log.info(message)">info</button>
         <button ng-click="$log.error(message)">error</button>
         <button ng-click="$log.debug(message)">debug</button>
       </div>
     </file>
   </example>
 */

/**
 * @ngdoc provider
 * @name $logProvider
 * @description
 * Use the `$logProvider` to configure how the application logs messages
 */
function $LogProvider() {
  var debug = true,
      self = this;

  /**
   * @ngdoc method
   * @name $logProvider#debugEnabled
   * @description
   * @param {boolean=} flag enable or disable debug level messages
   * @returns {*} current value if used as getter or itself (chaining) if used as setter
   */
  this.debugEnabled = function(flag) {
    if (isDefined(flag)) {
      debug = flag;
    return this;
    } else {
      return debug;
    }
  };

  this.$get = ['$window', function($window) {
    return {
      /**
       * @ngdoc method
       * @name $log#log
       *
       * @description
       * Write a log message
       */
      log: consoleLog('log'),

      /**
       * @ngdoc method
       * @name $log#info
       *
       * @description
       * Write an information message
       */
      info: consoleLog('info'),

      /**
       * @ngdoc method
       * @name $log#warn
       *
       * @description
       * Write a warning message
       */
      warn: consoleLog('warn'),

      /**
       * @ngdoc method
       * @name $log#error
       *
       * @description
       * Write an error message
       */
      error: consoleLog('error'),

      /**
       * @ngdoc method
       * @name $log#debug
       *
       * @description
       * Write a debug message
       */
      debug: (function() {
        var fn = consoleLog('debug');

        return function() {
          if (debug) {
            fn.apply(self, arguments);
          }
        };
      }())
    };

    function formatError(arg) {
      if (arg instanceof Error) {
        if (arg.stack) {
          arg = (arg.message && arg.stack.indexOf(arg.message) === -1)
              ? 'Error: ' + arg.message + '\n' + arg.stack
              : arg.stack;
        } else if (arg.sourceURL) {
          arg = arg.message + '\n' + arg.sourceURL + ':' + arg.line;
        }
      }
      return arg;
    }

    function consoleLog(type) {
      var console = $window.console || {},
          logFn = console[type] || console.log || noop,
          hasApply = false;

      // Note: reading logFn.apply throws an error in IE11 in IE8 document mode.
      // The reason behind this is that console.log has type "object" in IE8...
      try {
        hasApply = !!logFn.apply;
      } catch (e) {}

      if (hasApply) {
        return function() {
          var args = [];
          forEach(arguments, function(arg) {
            args.push(formatError(arg));
          });
          return logFn.apply(console, args);
        };
      }

      // we are IE which either doesn't have window.console => this is noop and we do nothing,
      // or we are IE where console.log doesn't have apply so we log at least first 2 args
      return function(arg1, arg2) {
        logFn(arg1, arg2 == null ? '' : arg2);
      };
    }
  }];
}

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
 *     Any commits to this file should be reviewed with security in mind.  *
 *   Changes to this file can potentially create security vulnerabilities. *
 *          An approval from 2 Core members with history of modifying      *
 *                         this file is required.                          *
 *                                                                         *
 *  Does the change somehow allow for arbitrary javascript to be executed? *
 *    Or allows for someone to change the prototype of built-in objects?   *
 *     Or gives undesired access to variables likes document or window?    *
 * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */

var $parseMinErr = minErr('$parse');

// Sandboxing Angular Expressions
// ------------------------------
// Angular expressions are generally considered safe because these expressions only have direct
// access to `$scope` and locals. However, one can obtain the ability to execute arbitrary JS code by
// obtaining a reference to native JS functions such as the Function constructor.
//
// As an example, consider the following Angular expression:
//
//   {}.toString.constructor('alert("evil JS code")')
//
// This sandboxing technique is not perfect and doesn't aim to be. The goal is to prevent exploits
// against the expression language, but not to prevent exploits that were enabled by exposing
// sensitive JavaScript or browser APIs on Scope. Exposing such objects on a Scope is never a good
// practice and therefore we are not even trying to protect against interaction with an object
// explicitly exposed in this way.
//
// In general, it is not possible to access a Window object from an angular expression unless a
// window or some DOM object that has a reference to window is published onto a Scope.
// Similarly we prevent invocations of function known to be dangerous, as well as assignments to
// native objects.
//
// See https://docs.angularjs.org/guide/security


function ensureSafeMemberName(name, fullExpression) {
  if (name === "__defineGetter__" || name === "__defineSetter__"
      || name === "__lookupGetter__" || name === "__lookupSetter__"
      || name === "__proto__") {
    throw $parseMinErr('isecfld',
        'Attempting to access a disallowed field in Angular expressions! '
        + 'Expression: {0}', fullExpression);
  }
  return name;
}

function getStringValue(name, fullExpression) {
  // From the JavaScript docs:
  // Property names must be strings. This means that non-string objects cannot be used
  // as keys in an object. Any non-string object, including a number, is typecasted
  // into a string via the toString method.
  //
  // So, to ensure that we are checking the same `name` that JavaScript would use,
  // we cast it to a string, if possible.
  // Doing `name + ''` can cause a repl error if the result to `toString` is not a string,
  // this is, this will handle objects that misbehave.
  name = name + '';
  if (!isString(name)) {
    throw $parseMinErr('iseccst',
        'Cannot convert object to primitive value! '
        + 'Expression: {0}', fullExpression);
  }
  return name;
}

function ensureSafeObject(obj, fullExpression) {
  // nifty check if obj is Function that is fast and works across iframes and other contexts
  if (obj) {
    if (obj.constructor === obj) {
      throw $parseMinErr('isecfn',
          'Referencing Function in Angular expressions is disallowed! Expression: {0}',
          fullExpression);
    } else if (// isWindow(obj)
        obj.window === obj) {
      throw $parseMinErr('isecwindow',
          'Referencing the Window in Angular expressions is disallowed! Expression: {0}',
          fullExpression);
    } else if (// isElement(obj)
        obj.children && (obj.nodeName || (obj.prop && obj.attr && obj.find))) {
      throw $parseMinErr('isecdom',
          'Referencing DOM nodes in Angular expressions is disallowed! Expression: {0}',
          fullExpression);
    } else if (// block Object so that we can't get hold of dangerous Object.* methods
        obj === Object) {
      throw $parseMinErr('isecobj',
          'Referencing Object in Angular expressions is disallowed! Expression: {0}',
          fullExpression);
    }
  }
  return obj;
}

var CALL = Function.prototype.call;
var APPLY = Function.prototype.apply;
var BIND = Function.prototype.bind;

function ensureSafeFunction(obj, fullExpression) {
  if (obj) {
    if (obj.constructor === obj) {
      throw $parseMinErr('isecfn',
        'Referencing Function in Angular expressions is disallowed! Expression: {0}',
        fullExpression);
    } else if (obj === CALL || obj === APPLY || obj === BIND) {
      throw $parseMinErr('isecff',
        'Referencing call, apply or bind in Angular expressions is disallowed! Expression: {0}',
        fullExpression);
    }
  }
}

function ensureSafeAssignContext(obj, fullExpression) {
  if (obj) {
    if (obj === (0).constructor || obj === (false).constructor || obj === ''.constructor ||
        obj === {}.constructor || obj === [].constructor || obj === Function.constructor) {
      throw $parseMinErr('isecaf',
        'Assigning to a constructor is disallowed! Expression: {0}', fullExpression);
    }
  }
}

var OPERATORS = createMap();
forEach('+ - * / % === !== == != < > <= >= && || ! = |'.split(' '), function(operator) { OPERATORS[operator] = true; });
var ESCAPE = {"n":"\n", "f":"\f", "r":"\r", "t":"\t", "v":"\v", "'":"'", '"':'"'};


/////////////////////////////////////////


/**
 * @constructor
 */
var Lexer = function(options) {
  this.options = options;
};

Lexer.prototype = {
  constructor: Lexer,

  lex: function(text) {
    this.text = text;
    this.index = 0;
    this.tokens = [];

    while (this.index < this.text.length) {
      var ch = this.text.charAt(this.index);
      if (ch === '"' || ch === "'") {
        this.readString(ch);
      } else if (this.isNumber(ch) || ch === '.' && this.isNumber(this.peek())) {
        this.readNumber();
      } else if (this.isIdent(ch)) {
        this.readIdent();
      } else if (this.is(ch, '(){}[].,;:?')) {
        this.tokens.push({index: this.index, text: ch});
        this.index++;
      } else if (this.isWhitespace(ch)) {
        this.index++;
      } else {
        var ch2 = ch + this.peek();
        var ch3 = ch2 + this.peek(2);
        var op1 = OPERATORS[ch];
        var op2 = OPERATORS[ch2];
        var op3 = OPERATORS[ch3];
        if (op1 || op2 || op3) {
          var token = op3 ? ch3 : (op2 ? ch2 : ch);
          this.tokens.push({index: this.index, text: token, operator: true});
          this.index += token.length;
        } else {
          this.throwError('Unexpected next character ', this.index, this.index + 1);
        }
      }
    }
    return this.tokens;
  },

  is: function(ch, chars) {
    return chars.indexOf(ch) !== -1;
  },

  peek: function(i) {
    var num = i || 1;
    return (this.index + num < this.text.length) ? this.text.charAt(this.index + num) : false;
  },

  isNumber: function(ch) {
    return ('0' <= ch && ch <= '9') && typeof ch === "string";
  },

  isWhitespace: function(ch) {
    // IE treats non-breaking space as \u00A0
    return (ch === ' ' || ch === '\r' || ch === '\t' ||
            ch === '\n' || ch === '\v' || ch === '\u00A0');
  },

  isIdent: function(ch) {
    return ('a' <= ch && ch <= 'z' ||
            'A' <= ch && ch <= 'Z' ||
            '_' === ch || ch === '$');
  },

  isExpOperator: function(ch) {
    return (ch === '-' || ch === '+' || this.isNumber(ch));
  },

  throwError: function(error, start, end) {
    end = end || this.index;
    var colStr = (isDefined(start)
            ? 's ' + start +  '-' + this.index + ' [' + this.text.substring(start, end) + ']'
            : ' ' + end);
    throw $parseMinErr('lexerr', 'Lexer Error: {0} at column{1} in expression [{2}].',
        error, colStr, this.text);
  },

  readNumber: function() {
    var number = '';
    var start = this.index;
    while (this.index < this.text.length) {
      var ch = lowercase(this.text.charAt(this.index));
      if (ch == '.' || this.isNumber(ch)) {
        number += ch;
      } else {
        var peekCh = this.peek();
        if (ch == 'e' && this.isExpOperator(peekCh)) {
          number += ch;
        } else if (this.isExpOperator(ch) &&
            peekCh && this.isNumber(peekCh) &&
            number.charAt(number.length - 1) == 'e') {
          number += ch;
        } else if (this.isExpOperator(ch) &&
            (!peekCh || !this.isNumber(peekCh)) &&
            number.charAt(number.length - 1) == 'e') {
          this.throwError('Invalid exponent');
        } else {
          break;
        }
      }
      this.index++;
    }
    this.tokens.push({
      index: start,
      text: number,
      constant: true,
      value: Number(number)
    });
  },

  readIdent: function() {
    var start = this.index;
    while (this.index < this.text.length) {
      var ch = this.text.charAt(this.index);
      if (!(this.isIdent(ch) || this.isNumber(ch))) {
        break;
      }
      this.index++;
    }
    this.tokens.push({
      index: start,
      text: this.text.slice(start, this.index),
      identifier: true
    });
  },

  readString: function(quote) {
    var start = this.index;
    this.index++;
    var string = '';
    var rawString = quote;
    var escape = false;
    while (this.index < this.text.length) {
      var ch = this.text.charAt(this.index);
      rawString += ch;
      if (escape) {
        if (ch === 'u') {
          var hex = this.text.substring(this.index + 1, this.index + 5);
          if (!hex.match(/[\da-f]{4}/i)) {
            this.throwError('Invalid unicode escape [\\u' + hex + ']');
          }
          this.index += 4;
          string += String.fromCharCode(parseInt(hex, 16));
        } else {
          var rep = ESCAPE[ch];
          string = string + (rep || ch);
        }
        escape = false;
      } else if (ch === '\\') {
        escape = true;
      } else if (ch === quote) {
        this.index++;
        this.tokens.push({
          index: start,
          text: rawString,
          constant: true,
          value: string
        });
        return;
      } else {
        string += ch;
      }
      this.index++;
    }
    this.throwError('Unterminated quote', start);
  }
};

var AST = function(lexer, options) {
  this.lexer = lexer;
  this.options = options;
};

AST.Program = 'Program';
AST.ExpressionStatement = 'ExpressionStatement';
AST.AssignmentExpression = 'AssignmentExpression';
AST.ConditionalExpression = 'ConditionalExpression';
AST.LogicalExpression = 'LogicalExpression';
AST.BinaryExpression = 'BinaryExpression';
AST.UnaryExpression = 'UnaryExpression';
AST.CallExpression = 'CallExpression';
AST.MemberExpression = 'MemberExpression';
AST.Identifier = 'Identifier';
AST.Literal = 'Literal';
AST.ArrayExpression = 'ArrayExpression';
AST.Property = 'Property';
AST.ObjectExpression = 'ObjectExpression';
AST.ThisExpression = 'ThisExpression';

// Internal use only
AST.NGValueParameter = 'NGValueParameter';

AST.prototype = {
  ast: function(text) {
    this.text = text;
    this.tokens = this.lexer.lex(text);

    var value = this.program();

    if (this.tokens.length !== 0) {
      this.throwError('is an unexpected token', this.tokens[0]);
    }

    return value;
  },

  program: function() {
    var body = [];
    while (true) {
      if (this.tokens.length > 0 && !this.peek('}', ')', ';', ']'))
        body.push(this.expressionStatement());
      if (!this.expect(';')) {
        return { type: AST.Program, body: body};
      }
    }
  },

  expressionStatement: function() {
    return { type: AST.ExpressionStatement, expression: this.filterChain() };
  },

  filterChain: function() {
    var left = this.expression();
    var token;
    while ((token = this.expect('|'))) {
      left = this.filter(left);
    }
    return left;
  },

  expression: function() {
    return this.assignment();
  },

  assignment: function() {
    var result = this.ternary();
    if (this.expect('=')) {
      result = { type: AST.AssignmentExpression, left: result, right: this.assignment(), operator: '='};
    }
    return result;
  },

  ternary: function() {
    var test = this.logicalOR();
    var alternate;
    var consequent;
    if (this.expect('?')) {
      alternate = this.expression();
      if (this.consume(':')) {
        consequent = this.expression();
        return { type: AST.ConditionalExpression, test: test, alternate: alternate, consequent: consequent};
      }
    }
    return test;
  },

  logicalOR: function() {
    var left = this.logicalAND();
    while (this.expect('||')) {
      left = { type: AST.LogicalExpression, operator: '||', left: left, right: this.logicalAND() };
    }
    return left;
  },

  logicalAND: function() {
    var left = this.equality();
    while (this.expect('&&')) {
      left = { type: AST.LogicalExpression, operator: '&&', left: left, right: this.equality()};
    }
    return left;
  },

  equality: function() {
    var left = this.relational();
    var token;
    while ((token = this.expect('==','!=','===','!=='))) {
      left = { type: AST.BinaryExpression, operator: token.text, left: left, right: this.relational() };
    }
    return left;
  },

  relational: function() {
    var left = this.additive();
    var token;
    while ((token = this.expect('<', '>', '<=', '>='))) {
      left = { type: AST.BinaryExpression, operator: token.text, left: left, right: this.additive() };
    }
    return left;
  },

  additive: function() {
    var left = this.multiplicative();
    var token;
    while ((token = this.expect('+','-'))) {
      left = { type: AST.BinaryExpression, operator: token.text, left: left, right: this.multiplicative() };
    }
    return left;
  },

  multiplicative: function() {
    var left = this.unary();
    var token;
    while ((token = this.expect('*','/','%'))) {
      left = { type: AST.BinaryExpression, operator: token.text, left: left, right: this.unary() };
    }
    return left;
  },

  unary: function() {
    var token;
    if ((token = this.expect('+', '-', '!'))) {
      return { type: AST.UnaryExpression, operator: token.text, prefix: true, argument: this.unary() };
    } else {
      return this.primary();
    }
  },

  primary: function() {
    var primary;
    if (this.expect('(')) {
      primary = this.filterChain();
      this.consume(')');
    } else if (this.expect('[')) {
      primary = this.arrayDeclaration();
    } else if (this.expect('{')) {
      primary = this.object();
    } else if (this.constants.hasOwnProperty(this.peek().text)) {
      primary = copy(this.constants[this.consume().text]);
    } else if (this.peek().identifier) {
      primary = this.identifier();
    } else if (this.peek().constant) {
      primary = this.constant();
    } else {
      this.throwError('not a primary expression', this.peek());
    }

    var next;
    while ((next = this.expect('(', '[', '.'))) {
      if (next.text === '(') {
        primary = {type: AST.CallExpression, callee: primary, arguments: this.parseArguments() };
        this.consume(')');
      } else if (next.text === '[') {
        primary = { type: AST.MemberExpression, object: primary, property: this.expression(), computed: true };
        this.consume(']');
      } else if (next.text === '.') {
        primary = { type: AST.MemberExpression, object: primary, property: this.identifier(), computed: false };
      } else {
        this.throwError('IMPOSSIBLE');
      }
    }
    return primary;
  },

  filter: function(baseExpression) {
    var args = [baseExpression];
    var result = {type: AST.CallExpression, callee: this.identifier(), arguments: args, filter: true};

    while (this.expect(':')) {
      args.push(this.expression());
    }

    return result;
  },

  parseArguments: function() {
    var args = [];
    if (this.peekToken().text !== ')') {
      do {
        args.push(this.expression());
      } while (this.expect(','));
    }
    return args;
  },

  identifier: function() {
    var token = this.consume();
    if (!token.identifier) {
      this.throwError('is not a valid identifier', token);
    }
    return { type: AST.Identifier, name: token.text };
  },

  constant: function() {
    // TODO check that it is a constant
    return { type: AST.Literal, value: this.consume().value };
  },

  arrayDeclaration: function() {
    var elements = [];
    if (this.peekToken().text !== ']') {
      do {
        if (this.peek(']')) {
          // Support trailing commas per ES5.1.
          break;
        }
        elements.push(this.expression());
      } while (this.expect(','));
    }
    this.consume(']');

    return { type: AST.ArrayExpression, elements: elements };
  },

  object: function() {
    var properties = [], property;
    if (this.peekToken().text !== '}') {
      do {
        if (this.peek('}')) {
          // Support trailing commas per ES5.1.
          break;
        }
        property = {type: AST.Property, kind: 'init'};
        if (this.peek().constant) {
          property.key = this.constant();
        } else if (this.peek().identifier) {
          property.key = this.identifier();
        } else {
          this.throwError("invalid key", this.peek());
        }
        this.consume(':');
        property.value = this.expression();
        properties.push(property);
      } while (this.expect(','));
    }
    this.consume('}');

    return {type: AST.ObjectExpression, properties: properties };
  },

  throwError: function(msg, token) {
    throw $parseMinErr('syntax',
        'Syntax Error: Token \'{0}\' {1} at column {2} of the expression [{3}] starting at [{4}].',
          token.text, msg, (token.index + 1), this.text, this.text.substring(token.index));
  },

  consume: function(e1) {
    if (this.tokens.length === 0) {
      throw $parseMinErr('ueoe', 'Unexpected end of expression: {0}', this.text);
    }

    var token = this.expect(e1);
    if (!token) {
      this.throwError('is unexpected, expecting [' + e1 + ']', this.peek());
    }
    return token;
  },

  peekToken: function() {
    if (this.tokens.length === 0) {
      throw $parseMinErr('ueoe', 'Unexpected end of expression: {0}', this.text);
    }
    return this.tokens[0];
  },

  peek: function(e1, e2, e3, e4) {
    return this.peekAhead(0, e1, e2, e3, e4);
  },

  peekAhead: function(i, e1, e2, e3, e4) {
    if (this.tokens.length > i) {
      var token = this.tokens[i];
      var t = token.text;
      if (t === e1 || t === e2 || t === e3 || t === e4 ||
          (!e1 && !e2 && !e3 && !e4)) {
        return token;
      }
    }
    return false;
  },

  expect: function(e1, e2, e3, e4) {
    var token = this.peek(e1, e2, e3, e4);
    if (token) {
      this.tokens.shift();
      return token;
    }
    return false;
  },


  /* `undefined` is not a constant, it is an identifier,
   * but using it as an identifier is not supported
   */
  constants: {
    'true': { type: AST.Literal, value: true },
    'false': { type: AST.Literal, value: false },
    'null': { type: AST.Literal, value: null },
    'undefined': {type: AST.Literal, value: undefined },
    'this': {type: AST.ThisExpression }
  }
};

function ifDefined(v, d) {
  return typeof v !== 'undefined' ? v : d;
}

function plusFn(l, r) {
  if (typeof l === 'undefined') return r;
  if (typeof r === 'undefined') return l;
  return l + r;
}

function isStateless($filter, filterName) {
  var fn = $filter(filterName);
  return !fn.$stateful;
}

function findConstantAndWatchExpressions(ast, $filter) {
  var allConstants;
  var argsToWatch;
  switch (ast.type) {
  case AST.Program:
    allConstants = true;
    forEach(ast.body, function(expr) {
      findConstantAndWatchExpressions(expr.expression, $filter);
      allConstants = allConstants && expr.expression.constant;
    });
    ast.constant = allConstants;
    break;
  case AST.Literal:
    ast.constant = true;
    ast.toWatch = [];
    break;
  case AST.UnaryExpression:
    findConstantAndWatchExpressions(ast.argument, $filter);
    ast.constant = ast.argument.constant;
    ast.toWatch = ast.argument.toWatch;
    break;
  case AST.BinaryExpression:
    findConstantAndWatchExpressions(ast.left, $filter);
    findConstantAndWatchExpressions(ast.right, $filter);
    ast.constant = ast.left.constant && ast.right.constant;
    ast.toWatch = ast.left.toWatch.concat(ast.right.toWatch);
    break;
  case AST.LogicalExpression:
    findConstantAndWatchExpressions(ast.left, $filter);
    findConstantAndWatchExpressions(ast.right, $filter);
    ast.constant = ast.left.constant && ast.right.constant;
    ast.toWatch = ast.constant ? [] : [ast];
    break;
  case AST.ConditionalExpression:
    findConstantAndWatchExpressions(ast.test, $filter);
    findConstantAndWatchExpressions(ast.alternate, $filter);
    findConstantAndWatchExpressions(ast.consequent, $filter);
    ast.constant = ast.test.constant && ast.alternate.constant && ast.consequent.constant;
    ast.toWatch = ast.constant ? [] : [ast];
    break;
  case AST.Identifier:
    ast.constant = false;
    ast.toWatch = [ast];
    break;
  case AST.MemberExpression:
    findConstantAndWatchExpressions(ast.object, $filter);
    if (ast.computed) {
      findConstantAndWatchExpressions(ast.property, $filter);
    }
    ast.constant = ast.object.constant && (!ast.computed || ast.property.constant);
    ast.toWatch = [ast];
    break;
  case AST.CallExpression:
    allConstants = ast.filter ? isStateless($filter, ast.callee.name) : false;
    argsToWatch = [];
    forEach(ast.arguments, function(expr) {
      findConstantAndWatchExpressions(expr, $filter);
      allConstants = allConstants && expr.constant;
      if (!expr.constant) {
        argsToWatch.push.apply(argsToWatch, expr.toWatch);
      }
    });
    ast.constant = allConstants;
    ast.toWatch = ast.filter && isStateless($filter, ast.callee.name) ? argsToWatch : [ast];
    break;
  case AST.AssignmentExpression:
    findConstantAndWatchExpressions(ast.left, $filter);
    findConstantAndWatchExpressions(ast.right, $filter);
    ast.constant = ast.left.constant && ast.right.constant;
    ast.toWatch = [ast];
    break;
  case AST.ArrayExpression:
    allConstants = true;
    argsToWatch = [];
    forEach(ast.elements, function(expr) {
      findConstantAndWatchExpressions(expr, $filter);
      allConstants = allConstants && expr.constant;
      if (!expr.constant) {
        argsToWatch.push.apply(argsToWatch, expr.toWatch);
      }
    });
    ast.constant = allConstants;
    ast.toWatch = argsToWatch;
    break;
  case AST.ObjectExpression:
    allConstants = true;
    argsToWatch = [];
    forEach(ast.properties, function(property) {
      findConstantAndWatchExpressions(property.value, $filter);
      allConstants = allConstants && property.value.constant;
      if (!property.value.constant) {
        argsToWatch.push.apply(argsToWatch, property.value.toWatch);
      }
    });
    ast.constant = allConstants;
    ast.toWatch = argsToWatch;
    break;
  case AST.ThisExpression:
    ast.constant = false;
    ast.toWatch = [];
    break;
  }
}

function getInputs(body) {
  if (body.length != 1) return;
  var lastExpression = body[0].expression;
  var candidate = lastExpression.toWatch;
  if (candidate.length !== 1) return candidate;
  return candidate[0] !== lastExpression ? candidate : undefined;
}

function isAssignable(ast) {
  return ast.type === AST.Identifier || ast.type === AST.MemberExpression;
}

function assignableAST(ast) {
  if (ast.body.length === 1 && isAssignable(ast.body[0].expression)) {
    return {type: AST.AssignmentExpression, left: ast.body[0].expression, right: {type: AST.NGValueParameter}, operator: '='};
  }
}

function isLiteral(ast) {
  return ast.body.length === 0 ||
      ast.body.length === 1 && (
      ast.body[0].expression.type === AST.Literal ||
      ast.body[0].expression.type === AST.ArrayExpression ||
      ast.body[0].expression.type === AST.ObjectExpression);
}

function isConstant(ast) {
  return ast.constant;
}

function ASTCompiler(astBuilder, $filter) {
  this.astBuilder = astBuilder;
  this.$filter = $filter;
}

ASTCompiler.prototype = {
  compile: function(expression, expensiveChecks) {
    var self = this;
    var ast = this.astBuilder.ast(expression);
    this.state = {
      nextId: 0,
      filters: {},
      expensiveChecks: expensiveChecks,
      fn: {vars: [], body: [], own: {}},
      assign: {vars: [], body: [], own: {}},
      inputs: []
    };
    findConstantAndWatchExpressions(ast, self.$filter);
    var extra = '';
    var assignable;
    this.stage = 'assign';
    if ((assignable = assignableAST(ast))) {
      this.state.computing = 'assign';
      var result = this.nextId();
      this.recurse(assignable, result);
      this.return_(result);
      extra = 'fn.assign=' + this.generateFunction('assign', 's,v,l');
    }
    var toWatch = getInputs(ast.body);
    self.stage = 'inputs';
    forEach(toWatch, function(watch, key) {
      var fnKey = 'fn' + key;
      self.state[fnKey] = {vars: [], body: [], own: {}};
      self.state.computing = fnKey;
      var intoId = self.nextId();
      self.recurse(watch, intoId);
      self.return_(intoId);
      self.state.inputs.push(fnKey);
      watch.watchId = key;
    });
    this.state.computing = 'fn';
    this.stage = 'main';
    this.recurse(ast);
    var fnString =
      // The build and minification steps remove the string "use strict" from the code, but this is done using a regex.
      // This is a workaround for this until we do a better job at only removing the prefix only when we should.
      '"' + this.USE + ' ' + this.STRICT + '";\n' +
      this.filterPrefix() +
      'var fn=' + this.generateFunction('fn', 's,l,a,i') +
      extra +
      this.watchFns() +
      'return fn;';

    /* jshint -W054 */
    var fn = (new Function('$filter',
        'ensureSafeMemberName',
        'ensureSafeObject',
        'ensureSafeFunction',
        'getStringValue',
        'ensureSafeAssignContext',
        'ifDefined',
        'plus',
        'text',
        fnString))(
          this.$filter,
          ensureSafeMemberName,
          ensureSafeObject,
          ensureSafeFunction,
          getStringValue,
          ensureSafeAssignContext,
          ifDefined,
          plusFn,
          expression);
    /* jshint +W054 */
    this.state = this.stage = undefined;
    fn.literal = isLiteral(ast);
    fn.constant = isConstant(ast);
    return fn;
  },

  USE: 'use',

  STRICT: 'strict',

  watchFns: function() {
    var result = [];
    var fns = this.state.inputs;
    var self = this;
    forEach(fns, function(name) {
      result.push('var ' + name + '=' + self.generateFunction(name, 's'));
    });
    if (fns.length) {
      result.push('fn.inputs=[' + fns.join(',') + '];');
    }
    return result.join('');
  },

  generateFunction: function(name, params) {
    return 'function(' + params + '){' +
        this.varsPrefix(name) +
        this.body(name) +
        '};';
  },

  filterPrefix: function() {
    var parts = [];
    var self = this;
    forEach(this.state.filters, function(id, filter) {
      parts.push(id + '=$filter(' + self.escape(filter) + ')');
    });
    if (parts.length) return 'var ' + parts.join(',') + ';';
    return '';
  },

  varsPrefix: function(section) {
    return this.state[section].vars.length ? 'var ' + this.state[section].vars.join(',') + ';' : '';
  },

  body: function(section) {
    return this.state[section].body.join('');
  },

  recurse: function(ast, intoId, nameId, recursionFn, create, skipWatchIdCheck) {
    var left, right, self = this, args, expression;
    recursionFn = recursionFn || noop;
    if (!skipWatchIdCheck && isDefined(ast.watchId)) {
      intoId = intoId || this.nextId();
      this.if_('i',
        this.lazyAssign(intoId, this.computedMember('i', ast.watchId)),
        this.lazyRecurse(ast, intoId, nameId, recursionFn, create, true)
      );
      return;
    }
    switch (ast.type) {
    case AST.Program:
      forEach(ast.body, function(expression, pos) {
        self.recurse(expression.expression, undefined, undefined, function(expr) { right = expr; });
        if (pos !== ast.body.length - 1) {
          self.current().body.push(right, ';');
        } else {
          self.return_(right);
        }
      });
      break;
    case AST.Literal:
      expression = this.escape(ast.value);
      this.assign(intoId, expression);
      recursionFn(expression);
      break;
    case AST.UnaryExpression:
      this.recurse(ast.argument, undefined, undefined, function(expr) { right = expr; });
      expression = ast.operator + '(' + this.ifDefined(right, 0) + ')';
      this.assign(intoId, expression);
      recursionFn(expression);
      break;
    case AST.BinaryExpression:
      this.recurse(ast.left, undefined, undefined, function(expr) { left = expr; });
      this.recurse(ast.right, undefined, undefined, function(expr) { right = expr; });
      if (ast.operator === '+') {
        expression = this.plus(left, right);
      } else if (ast.operator === '-') {
        expression = this.ifDefined(left, 0) + ast.operator + this.ifDefined(right, 0);
      } else {
        expression = '(' + left + ')' + ast.operator + '(' + right + ')';
      }
      this.assign(intoId, expression);
      recursionFn(expression);
      break;
    case AST.LogicalExpression:
      intoId = intoId || this.nextId();
      self.recurse(ast.left, intoId);
      self.if_(ast.operator === '&&' ? intoId : self.not(intoId), self.lazyRecurse(ast.right, intoId));
      recursionFn(intoId);
      break;
    case AST.ConditionalExpression:
      intoId = intoId || this.nextId();
      self.recurse(ast.test, intoId);
      self.if_(intoId, self.lazyRecurse(ast.alternate, intoId), self.lazyRecurse(ast.consequent, intoId));
      recursionFn(intoId);
      break;
    case AST.Identifier:
      intoId = intoId || this.nextId();
      if (nameId) {
        nameId.context = self.stage === 'inputs' ? 's' : this.assign(this.nextId(), this.getHasOwnProperty('l', ast.name) + '?l:s');
        nameId.computed = false;
        nameId.name = ast.name;
      }
      ensureSafeMemberName(ast.name);
      self.if_(self.stage === 'inputs' || self.not(self.getHasOwnProperty('l', ast.name)),
        function() {
          self.if_(self.stage === 'inputs' || 's', function() {
            if (create && create !== 1) {
              self.if_(
                self.not(self.nonComputedMember('s', ast.name)),
                self.lazyAssign(self.nonComputedMember('s', ast.name), '{}'));
            }
            self.assign(intoId, self.nonComputedMember('s', ast.name));
          });
        }, intoId && self.lazyAssign(intoId, self.nonComputedMember('l', ast.name))
        );
      if (self.state.expensiveChecks || isPossiblyDangerousMemberName(ast.name)) {
        self.addEnsureSafeObject(intoId);
      }
      recursionFn(intoId);
      break;
    case AST.MemberExpression:
      left = nameId && (nameId.context = this.nextId()) || this.nextId();
      intoId = intoId || this.nextId();
      self.recurse(ast.object, left, undefined, function() {
        self.if_(self.notNull(left), function() {
          if (ast.computed) {
            right = self.nextId();
            self.recurse(ast.property, right);
            self.getStringValue(right);
            self.addEnsureSafeMemberName(right);
            if (create && create !== 1) {
              self.if_(self.not(self.computedMember(left, right)), self.lazyAssign(self.computedMember(left, right), '{}'));
            }
            expression = self.ensureSafeObject(self.computedMember(left, right));
            self.assign(intoId, expression);
            if (nameId) {
              nameId.computed = true;
              nameId.name = right;
            }
          } else {
            ensureSafeMemberName(ast.property.name);
            if (create && create !== 1) {
              self.if_(self.not(self.nonComputedMember(left, ast.property.name)), self.lazyAssign(self.nonComputedMember(left, ast.property.name), '{}'));
            }
            expression = self.nonComputedMember(left, ast.property.name);
            if (self.state.expensiveChecks || isPossiblyDangerousMemberName(ast.property.name)) {
              expression = self.ensureSafeObject(expression);
            }
            self.assign(intoId, expression);
            if (nameId) {
              nameId.computed = false;
              nameId.name = ast.property.name;
            }
          }
        }, function() {
          self.assign(intoId, 'undefined');
        });
        recursionFn(intoId);
      }, !!create);
      break;
    case AST.CallExpression:
      intoId = intoId || this.nextId();
      if (ast.filter) {
        right = self.filter(ast.callee.name);
        args = [];
        forEach(ast.arguments, function(expr) {
          var argument = self.nextId();
          self.recurse(expr, argument);
          args.push(argument);
        });
        expression = right + '(' + args.join(',') + ')';
        self.assign(intoId, expression);
        recursionFn(intoId);
      } else {
        right = self.nextId();
        left = {};
        args = [];
        self.recurse(ast.callee, right, left, function() {
          self.if_(self.notNull(right), function() {
            self.addEnsureSafeFunction(right);
            forEach(ast.arguments, function(expr) {
              self.recurse(expr, self.nextId(), undefined, function(argument) {
                args.push(self.ensureSafeObject(argument));
              });
            });
            if (left.name) {
              if (!self.state.expensiveChecks) {
                self.addEnsureSafeObject(left.context);
              }
              expression = self.member(left.context, left.name, left.computed) + '(' + args.join(',') + ')';
            } else {
              expression = right + '(' + args.join(',') + ')';
            }
            expression = self.ensureSafeObject(expression);
            self.assign(intoId, expression);
          }, function() {
            self.assign(intoId, 'undefined');
          });
          recursionFn(intoId);
        });
      }
      break;
    case AST.AssignmentExpression:
      right = this.nextId();
      left = {};
      if (!isAssignable(ast.left)) {
        throw $parseMinErr('lval', 'Trying to assing a value to a non l-value');
      }
      this.recurse(ast.left, undefined, left, function() {
        self.if_(self.notNull(left.context), function() {
          self.recurse(ast.right, right);
          self.addEnsureSafeObject(self.member(left.context, left.name, left.computed));
          self.addEnsureSafeAssignContext(left.context);
          expression = self.member(left.context, left.name, left.computed) + ast.operator + right;
          self.assign(intoId, expression);
          recursionFn(intoId || expression);
        });
      }, 1);
      break;
    case AST.ArrayExpression:
      args = [];
      forEach(ast.elements, function(expr) {
        self.recurse(expr, self.nextId(), undefined, function(argument) {
          args.push(argument);
        });
      });
      expression = '[' + args.join(',') + ']';
      this.assign(intoId, expression);
      recursionFn(expression);
      break;
    case AST.ObjectExpression:
      args = [];
      forEach(ast.properties, function(property) {
        self.recurse(property.value, self.nextId(), undefined, function(expr) {
          args.push(self.escape(
              property.key.type === AST.Identifier ? property.key.name :
                ('' + property.key.value)) +
              ':' + expr);
        });
      });
      expression = '{' + args.join(',') + '}';
      this.assign(intoId, expression);
      recursionFn(expression);
      break;
    case AST.ThisExpression:
      this.assign(intoId, 's');
      recursionFn('s');
      break;
    case AST.NGValueParameter:
      this.assign(intoId, 'v');
      recursionFn('v');
      break;
    }
  },

  getHasOwnProperty: function(element, property) {
    var key = element + '.' + property;
    var own = this.current().own;
    if (!own.hasOwnProperty(key)) {
      own[key] = this.nextId(false, element + '&&(' + this.escape(property) + ' in ' + element + ')');
    }
    return own[key];
  },

  assign: function(id, value) {
    if (!id) return;
    this.current().body.push(id, '=', value, ';');
    return id;
  },

  filter: function(filterName) {
    if (!this.state.filters.hasOwnProperty(filterName)) {
      this.state.filters[filterName] = this.nextId(true);
    }
    return this.state.filters[filterName];
  },

  ifDefined: function(id, defaultValue) {
    return 'ifDefined(' + id + ',' + this.escape(defaultValue) + ')';
  },

  plus: function(left, right) {
    return 'plus(' + left + ',' + right + ')';
  },

  return_: function(id) {
    this.current().body.push('return ', id, ';');
  },

  if_: function(test, alternate, consequent) {
    if (test === true) {
      alternate();
    } else {
      var body = this.current().body;
      body.push('if(', test, '){');
      alternate();
      body.push('}');
      if (consequent) {
        body.push('else{');
        consequent();
        body.push('}');
      }
    }
  },

  not: function(expression) {
    return '!(' + expression + ')';
  },

  notNull: function(expression) {
    return expression + '!=null';
  },

  nonComputedMember: function(left, right) {
    return left + '.' + right;
  },

  computedMember: function(left, right) {
    return left + '[' + right + ']';
  },

  member: function(left, right, computed) {
    if (computed) return this.computedMember(left, right);
    return this.nonComputedMember(left, right);
  },

  addEnsureSafeObject: function(item) {
    this.current().body.push(this.ensureSafeObject(item), ';');
  },

  addEnsureSafeMemberName: function(item) {
    this.current().body.push(this.ensureSafeMemberName(item), ';');
  },

  addEnsureSafeFunction: function(item) {
    this.current().body.push(this.ensureSafeFunction(item), ';');
  },

  addEnsureSafeAssignContext: function(item) {
    this.current().body.push(this.ensureSafeAssignContext(item), ';');
  },

  ensureSafeObject: function(item) {
    return 'ensureSafeObject(' + item + ',text)';
  },

  ensureSafeMemberName: function(item) {
    return 'ensureSafeMemberName(' + item + ',text)';
  },

  ensureSafeFunction: function(item) {
    return 'ensureSafeFunction(' + item + ',text)';
  },

  getStringValue: function(item) {
    this.assign(item, 'getStringValue(' + item + ',text)');
  },

  ensureSafeAssignContext: function(item) {
    return 'ensureSafeAssignContext(' + item + ',text)';
  },

  lazyRecurse: function(ast, intoId, nameId, recursionFn, create, skipWatchIdCheck) {
    var self = this;
    return function() {
      self.recurse(ast, intoId, nameId, recursionFn, create, skipWatchIdCheck);
    };
  },

  lazyAssign: function(id, value) {
    var self = this;
    return function() {
      self.assign(id, value);
    };
  },

  stringEscapeRegex: /[^ a-zA-Z0-9]/g,

  stringEscapeFn: function(c) {
    return '\\u' + ('0000' + c.charCodeAt(0).toString(16)).slice(-4);
  },

  escape: function(value) {
    if (isString(value)) return "'" + value.replace(this.stringEscapeRegex, this.stringEscapeFn) + "'";
    if (isNumber(value)) return value.toString();
    if (value === true) return 'true';
    if (value === false) return 'false';
    if (value === null) return 'null';
    if (typeof value === 'undefined') return 'undefined';

    throw $parseMinErr('esc', 'IMPOSSIBLE');
  },

  nextId: function(skip, init) {
    var id = 'v' + (this.state.nextId++);
    if (!skip) {
      this.current().vars.push(id + (init ? '=' + init : ''));
    }
    return id;
  },

  current: function() {
    return this.state[this.state.computing];
  }
};


function ASTInterpreter(astBuilder, $filter) {
  this.astBuilder = astBuilder;
  this.$filter = $filter;
}

ASTInterpreter.prototype = {
  compile: function(expression, expensiveChecks) {
    var self = this;
    var ast = this.astBuilder.ast(expression);
    this.expression = expression;
    this.expensiveChecks = expensiveChecks;
    findConstantAndWatchExpressions(ast, self.$filter);
    var assignable;
    var assign;
    if ((assignable = assignableAST(ast))) {
      assign = this.recurse(assignable);
    }
    var toWatch = getInputs(ast.body);
    var inputs;
    if (toWatch) {
      inputs = [];
      forEach(toWatch, function(watch, key) {
        var input = self.recurse(watch);
        watch.input = input;
        inputs.push(input);
        watch.watchId = key;
      });
    }
    var expressions = [];
    forEach(ast.body, function(expression) {
      expressions.push(self.recurse(expression.expression));
    });
    var fn = ast.body.length === 0 ? function() {} :
             ast.body.length === 1 ? expressions[0] :
             function(scope, locals) {
               var lastValue;
               forEach(expressions, function(exp) {
                 lastValue = exp(scope, locals);
               });
               return lastValue;
             };
    if (assign) {
      fn.assign = function(scope, value, locals) {
        return assign(scope, locals, value);
      };
    }
    if (inputs) {
      fn.inputs = inputs;
    }
    fn.literal = isLiteral(ast);
    fn.constant = isConstant(ast);
    return fn;
  },

  recurse: function(ast, context, create) {
    var left, right, self = this, args, expression;
    if (ast.input) {
      return this.inputs(ast.input, ast.watchId);
    }
    switch (ast.type) {
    case AST.Literal:
      return this.value(ast.value, context);
    case AST.UnaryExpression:
      right = this.recurse(ast.argument);
      return this['unary' + ast.operator](right, context);
    case AST.BinaryExpression:
      left = this.recurse(ast.left);
      right = this.recurse(ast.right);
      return this['binary' + ast.operator](left, right, context);
    case AST.LogicalExpression:
      left = this.recurse(ast.left);
      right = this.recurse(ast.right);
      return this['binary' + ast.operator](left, right, context);
    case AST.ConditionalExpression:
      return this['ternary?:'](
        this.recurse(ast.test),
        this.recurse(ast.alternate),
        this.recurse(ast.consequent),
        context
      );
    case AST.Identifier:
      ensureSafeMemberName(ast.name, self.expression);
      return self.identifier(ast.name,
                             self.expensiveChecks || isPossiblyDangerousMemberName(ast.name),
                             context, create, self.expression);
    case AST.MemberExpression:
      left = this.recurse(ast.object, false, !!create);
      if (!ast.computed) {
        ensureSafeMemberName(ast.property.name, self.expression);
        right = ast.property.name;
      }
      if (ast.computed) right = this.recurse(ast.property);
      return ast.computed ?
        this.computedMember(left, right, context, create, self.expression) :
        this.nonComputedMember(left, right, self.expensiveChecks, context, create, self.expression);
    case AST.CallExpression:
      args = [];
      forEach(ast.arguments, function(expr) {
        args.push(self.recurse(expr));
      });
      if (ast.filter) right = this.$filter(ast.callee.name);
      if (!ast.filter) right = this.recurse(ast.callee, true);
      return ast.filter ?
        function(scope, locals, assign, inputs) {
          var values = [];
          for (var i = 0; i < args.length; ++i) {
            values.push(args[i](scope, locals, assign, inputs));
          }
          var value = right.apply(undefined, values, inputs);
          return context ? {context: undefined, name: undefined, value: value} : value;
        } :
        function(scope, locals, assign, inputs) {
          var rhs = right(scope, locals, assign, inputs);
          var value;
          if (rhs.value != null) {
            ensureSafeObject(rhs.context, self.expression);
            ensureSafeFunction(rhs.value, self.expression);
            var values = [];
            for (var i = 0; i < args.length; ++i) {
              values.push(ensureSafeObject(args[i](scope, locals, assign, inputs), self.expression));
            }
            value = ensureSafeObject(rhs.value.apply(rhs.context, values), self.expression);
          }
          return context ? {value: value} : value;
        };
    case AST.AssignmentExpression:
      left = this.recurse(ast.left, true, 1);
      right = this.recurse(ast.right);
      return function(scope, locals, assign, inputs) {
        var lhs = left(scope, locals, assign, inputs);
        var rhs = right(scope, locals, assign, inputs);
        ensureSafeObject(lhs.value, self.expression);
        ensureSafeAssignContext(lhs.context);
        lhs.context[lhs.name] = rhs;
        return context ? {value: rhs} : rhs;
      };
    case AST.ArrayExpression:
      args = [];
      forEach(ast.elements, function(expr) {
        args.push(self.recurse(expr));
      });
      return function(scope, locals, assign, inputs) {
        var value = [];
        for (var i = 0; i < args.length; ++i) {
          value.push(args[i](scope, locals, assign, inputs));
        }
        return context ? {value: value} : value;
      };
    case AST.ObjectExpression:
      args = [];
      forEach(ast.properties, function(property) {
        args.push({key: property.key.type === AST.Identifier ?
                        property.key.name :
                        ('' + property.key.value),
                   value: self.recurse(property.value)
        });
      });
      return function(scope, locals, assign, inputs) {
        var value = {};
        for (var i = 0; i < args.length; ++i) {
          value[args[i].key] = args[i].value(scope, locals, assign, inputs);
        }
        return context ? {value: value} : value;
      };
    case AST.ThisExpression:
      return function(scope) {
        return context ? {value: scope} : scope;
      };
    case AST.NGValueParameter:
      return function(scope, locals, assign, inputs) {
        return context ? {value: assign} : assign;
      };
    }
  },

  'unary+': function(argument, context) {
    return function(scope, locals, assign, inputs) {
      var arg = argument(scope, locals, assign, inputs);
      if (isDefined(arg)) {
        arg = +arg;
      } else {
        arg = 0;
      }
      return context ? {value: arg} : arg;
    };
  },
  'unary-': function(argument, context) {
    return function(scope, locals, assign, inputs) {
      var arg = argument(scope, locals, assign, inputs);
      if (isDefined(arg)) {
        arg = -arg;
      } else {
        arg = 0;
      }
      return context ? {value: arg} : arg;
    };
  },
  'unary!': function(argument, context) {
    return function(scope, locals, assign, inputs) {
      var arg = !argument(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary+': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var lhs = left(scope, locals, assign, inputs);
      var rhs = right(scope, locals, assign, inputs);
      var arg = plusFn(lhs, rhs);
      return context ? {value: arg} : arg;
    };
  },
  'binary-': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var lhs = left(scope, locals, assign, inputs);
      var rhs = right(scope, locals, assign, inputs);
      var arg = (isDefined(lhs) ? lhs : 0) - (isDefined(rhs) ? rhs : 0);
      return context ? {value: arg} : arg;
    };
  },
  'binary*': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) * right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary/': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) / right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary%': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) % right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary===': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) === right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary!==': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) !== right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary==': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) == right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary!=': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) != right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary<': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) < right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary>': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) > right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary<=': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) <= right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary>=': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) >= right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary&&': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) && right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'binary||': function(left, right, context) {
    return function(scope, locals, assign, inputs) {
      var arg = left(scope, locals, assign, inputs) || right(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  'ternary?:': function(test, alternate, consequent, context) {
    return function(scope, locals, assign, inputs) {
      var arg = test(scope, locals, assign, inputs) ? alternate(scope, locals, assign, inputs) : consequent(scope, locals, assign, inputs);
      return context ? {value: arg} : arg;
    };
  },
  value: function(value, context) {
    return function() { return context ? {context: undefined, name: undefined, value: value} : value; };
  },
  identifier: function(name, expensiveChecks, context, create, expression) {
    return function(scope, locals, assign, inputs) {
      var base = locals && (name in locals) ? locals : scope;
      if (create && create !== 1 && base && !(base[name])) {
        base[name] = {};
      }
      var value = base ? base[name] : undefined;
      if (expensiveChecks) {
        ensureSafeObject(value, expression);
      }
      if (context) {
        return {context: base, name: name, value: value};
      } else {
        return value;
      }
    };
  },
  computedMember: function(left, right, context, create, expression) {
    return function(scope, locals, assign, inputs) {
      var lhs = left(scope, locals, assign, inputs);
      var rhs;
      var value;
      if (lhs != null) {
        rhs = right(scope, locals, assign, inputs);
        rhs = getStringValue(rhs);
        ensureSafeMemberName(rhs, expression);
        if (create && create !== 1 && lhs && !(lhs[rhs])) {
          lhs[rhs] = {};
        }
        value = lhs[rhs];
        ensureSafeObject(value, expression);
      }
      if (context) {
        return {context: lhs, name: rhs, value: value};
      } else {
        return value;
      }
    };
  },
  nonComputedMember: function(left, right, expensiveChecks, context, create, expression) {
    return function(scope, locals, assign, inputs) {
      var lhs = left(scope, locals, assign, inputs);
      if (create && create !== 1 && lhs && !(lhs[right])) {
        lhs[right] = {};
      }
      var value = lhs != null ? lhs[right] : undefined;
      if (expensiveChecks || isPossiblyDangerousMemberName(right)) {
        ensureSafeObject(value, expression);
      }
      if (context) {
        return {context: lhs, name: right, value: value};
      } else {
        return value;
      }
    };
  },
  inputs: function(input, watchId) {
    return function(scope, value, locals, inputs) {
      if (inputs) return inputs[watchId];
      return input(scope, value, locals);
    };
  }
};

/**
 * @constructor
 */
var Parser = function(lexer, $filter, options) {
  this.lexer = lexer;
  this.$filter = $filter;
  this.options = options;
  this.ast = new AST(this.lexer);
  this.astCompiler = options.csp ? new ASTInterpreter(this.ast, $filter) :
                                   new ASTCompiler(this.ast, $filter);
};

Parser.prototype = {
  constructor: Parser,

  parse: function(text) {
    return this.astCompiler.compile(text, this.options.expensiveChecks);
  }
};

var getterFnCacheDefault = createMap();
var getterFnCacheExpensive = createMap();

function isPossiblyDangerousMemberName(name) {
  return name == 'constructor';
}

var objectValueOf = Object.prototype.valueOf;

function getValueOf(value) {
  return isFunction(value.valueOf) ? value.valueOf() : objectValueOf.call(value);
}

///////////////////////////////////

/**
 * @ngdoc service
 * @name $parse
 * @kind function
 *
 * @description
 *
 * Converts Angular {@link guide/expression expression} into a function.
 *
 * ```js
 *   var getter = $parse('user.name');
 *   var setter = getter.assign;
 *   var context = {user:{name:'angular'}};
 *   var locals = {user:{name:'local'}};
 *
 *   expect(getter(context)).toEqual('angular');
 *   setter(context, 'newValue');
 *   expect(context.user.name).toEqual('newValue');
 *   expect(getter(context, locals)).toEqual('local');
 * ```
 *
 *
 * @param {string} expression String expression to compile.
 * @returns {function(context, locals)} a function which represents the compiled expression:
 *
 *    * `context` â€“ `{object}` â€“ an object against which any expressions embedded in the strings
 *      are evaluated against (typically a scope object).
 *    * `locals` â€“ `{object=}` â€“ local variables context object, useful for overriding values in
 *      `context`.
 *
 *    The returned function also has the following properties:
 *      * `literal` â€“ `{boolean}` â€“ whether the expression's top-level node is a JavaScript
 *        literal.
 *      * `constant` â€“ `{boolean}` â€“ whether the expression is made entirely of JavaScript
 *        constant literals.
 *      * `assign` â€“ `{?function(context, value)}` â€“ if the expression is assignable, this will be
 *        set to a function to change its value on the given context.
 *
 */


/**
 * @ngdoc provider
 * @name $parseProvider
 *
 * @description
 * `$parseProvider` can be used for configuring the default behavior of the {@link ng.$parse $parse}
 *  service.
 */
function $ParseProvider() {
  var cacheDefault = createMap();
  var cacheExpensive = createMap();

  this.$get = ['$filter', function($filter) {
    var noUnsafeEval = csp().noUnsafeEval;
    var $parseOptions = {
          csp: noUnsafeEval,
          expensiveChecks: false
        },
        $parseOptionsExpensive = {
          csp: noUnsafeEval,
          expensiveChecks: true
        };

    return function $parse(exp, interceptorFn, expensiveChecks) {
      var parsedExpression, oneTime, cacheKey;

      switch (typeof exp) {
        case 'string':
          exp = exp.trim();
          cacheKey = exp;

          var cache = (expensiveChecks ? cacheExpensive : cacheDefault);
          parsedExpression = cache[cacheKey];

          if (!parsedExpression) {
            if (exp.charAt(0) === ':' && exp.charAt(1) === ':') {
              oneTime = true;
              exp = exp.substring(2);
            }
            var parseOptions = expensiveChecks ? $parseOptionsExpensive : $parseOptions;
            var lexer = new Lexer(parseOptions);
            var parser = new Parser(lexer, $filter, parseOptions);
            parsedExpression = parser.parse(exp);
            if (parsedExpression.constant) {
              parsedExpression.$$watchDelegate = constantWatchDelegate;
            } else if (oneTime) {
              parsedExpression.$$watchDelegate = parsedExpression.literal ?
                  oneTimeLiteralWatchDelegate : oneTimeWatchDelegate;
            } else if (parsedExpression.inputs) {
              parsedExpression.$$watchDelegate = inputsWatchDelegate;
            }
            cache[cacheKey] = parsedExpression;
          }
          return addInterceptor(parsedExpression, interceptorFn);

        case 'function':
          return addInterceptor(exp, interceptorFn);

        default:
          return noop;
      }
    };

    function expressionInputDirtyCheck(newValue, oldValueOfValue) {

      if (newValue == null || oldValueOfValue == null) { // null/undefined
        return newValue === oldValueOfValue;
      }

      if (typeof newValue === 'object') {

        // attempt to convert the value to a primitive type
        // TODO(docs): add a note to docs that by implementing valueOf even objects and arrays can
        //             be cheaply dirty-checked
        newValue = getValueOf(newValue);

        if (typeof newValue === 'object') {
          // objects/arrays are not supported - deep-watching them would be too expensive
          return false;
        }

        // fall-through to the primitive equality check
      }

      //Primitive or NaN
      return newValue === oldValueOfValue || (newValue !== newValue && oldValueOfValue !== oldValueOfValue);
    }

    function inputsWatchDelegate(scope, listener, objectEquality, parsedExpression, prettyPrintExpression) {
      var inputExpressions = parsedExpression.inputs;
      var lastResult;

      if (inputExpressions.length === 1) {
        var oldInputValueOf = expressionInputDirtyCheck; // init to something unique so that equals check fails
        inputExpressions = inputExpressions[0];
        return scope.$watch(function expressionInputWatch(scope) {
          var newInputValue = inputExpressions(scope);
          if (!expressionInputDirtyCheck(newInputValue, oldInputValueOf)) {
            lastResult = parsedExpression(scope, undefined, undefined, [newInputValue]);
            oldInputValueOf = newInputValue && getValueOf(newInputValue);
          }
          return lastResult;
        }, listener, objectEquality, prettyPrintExpression);
      }

      var oldInputValueOfValues = [];
      var oldInputValues = [];
      for (var i = 0, ii = inputExpressions.length; i < ii; i++) {
        oldInputValueOfValues[i] = expressionInputDirtyCheck; // init to something unique so that equals check fails
        oldInputValues[i] = null;
      }

      return scope.$watch(function expressionInputsWatch(scope) {
        var changed = false;

        for (var i = 0, ii = inputExpressions.length; i < ii; i++) {
          var newInputValue = inputExpressions[i](scope);
          if (changed || (changed = !expressionInputDirtyCheck(newInputValue, oldInputValueOfValues[i]))) {
            oldInputValues[i] = newInputValue;
            oldInputValueOfValues[i] = newInputValue && getValueOf(newInputValue);
          }
        }

        if (changed) {
          lastResult = parsedExpression(scope, undefined, undefined, oldInputValues);
        }

        return lastResult;
      }, listener, objectEquality, prettyPrintExpression);
    }

    function oneTimeWatchDelegate(scope, listener, objectEquality, parsedExpression) {
      var unwatch, lastValue;
      return unwatch = scope.$watch(function oneTimeWatch(scope) {
        return parsedExpression(scope);
      }, function oneTimeListener(value, old, scope) {
        lastValue = value;
        if (isFunction(listener)) {
          listener.apply(this, arguments);
        }
        if (isDefined(value)) {
          scope.$$postDigest(function() {
            if (isDefined(lastValue)) {
              unwatch();
            }
          });
        }
      }, objectEquality);
    }

    function oneTimeLiteralWatchDelegate(scope, listener, objectEquality, parsedExpression) {
      var unwatch, lastValue;
      return unwatch = scope.$watch(function oneTimeWatch(scope) {
        return parsedExpression(scope);
      }, function oneTimeListener(value, old, scope) {
        lastValue = value;
        if (isFunction(listener)) {
          listener.call(this, value, old, scope);
        }
        if (isAllDefined(value)) {
          scope.$$postDigest(function() {
            if (isAllDefined(lastValue)) unwatch();
          });
        }
      }, objectEquality);

      function isAllDefined(value) {
        var allDefined = true;
        forEach(value, function(val) {
          if (!isDefined(val)) allDefined = false;
        });
        return allDefined;
      }
    }

    function constantWatchDelegate(scope, listener, objectEquality, parsedExpression) {
      var unwatch;
      return unwatch = scope.$watch(function constantWatch(scope) {
        return parsedExpression(scope);
      }, function constantListener(value, old, scope) {
        if (isFunction(listener)) {
          listener.apply(this, arguments);
        }
        unwatch();
      }, objectEquality);
    }

    function addInterceptor(parsedExpression, interceptorFn) {
      if (!interceptorFn) return parsedExpression;
      var watchDelegate = parsedExpression.$$watchDelegate;
      var useInputs = false;

      var regularWatch =
          watchDelegate !== oneTimeLiteralWatchDelegate &&
          watchDelegate !== oneTimeWatchDelegate;

      var fn = regularWatch ? function regularInterceptedExpression(scope, locals, assign, inputs) {
        var value = useInputs && inputs ? inputs[0] : parsedExpression(scope, locals, assign, inputs);
        return interceptorFn(value, scope, locals);
      } : function oneTimeInterceptedExpression(scope, locals, assign, inputs) {
        var value = parsedExpression(scope, locals, assign, inputs);
        var result = interceptorFn(value, scope, locals);
        // we only return the interceptor's result if the
        // initial value is defined (for bind-once)
        return isDefined(value) ? result : value;
      };

      // Propagate $$watchDelegates other then inputsWatchDelegate
      if (parsedExpression.$$watchDelegate &&
          parsedExpression.$$watchDelegate !== inputsWatchDelegate) {
        fn.$$watchDelegate = parsedExpression.$$watchDelegate;
      } else if (!interceptorFn.$stateful) {
        // If there is an interceptor, but no watchDelegate then treat the interceptor like
        // we treat filters - it is assumed to be a pure function unless flagged with $stateful
        fn.$$watchDelegate = inputsWatchDelegate;
        useInputs = !parsedExpression.inputs;
        fn.inputs = parsedExpression.inputs ? parsedExpression.inputs : [parsedExpression];
      }

      return fn;
    }
  }];
}

/**
 * @ngdoc service
 * @name $q
 * @requires $rootScope
 *
 * @description
 * A service that helps you run functions asynchronously, and use their return values (or exceptions)
 * when they are done processing.
 *
 * This is an implementation of promises/deferred objects inspired by
 * [Kris Kowal's Q](https://github.com/kriskowal/q).
 *
 * $q can be used in two fashions --- one which is more similar to Kris Kowal's Q or jQuery's Deferred
 * implementations, and the other which resembles ES6 promises to some degree.
 *
 * # $q constructor
 *
 * The streamlined ES6 style promise is essentially just using $q as a constructor which takes a `resolver`
 * function as the first argument. This is similar to the native Promise implementation from ES6 Harmony,
 * see [MDN](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise).
 *
 * While the constructor-style use is supported, not all of the supporting methods from ES6 Harmony promises are
 * available yet.
 *
 * It can be used like so:
 *
 * ```js
 *   // for the purpose of this example let's assume that variables `$q` and `okToGreet`
 *   // are available in the current lexical scope (they could have been injected or passed in).
 *
 *   function asyncGreet(name) {
 *     // perform some asynchronous operation, resolve or reject the promise when appropriate.
 *     return $q(function(resolve, reject) {
 *       setTimeout(function() {
 *         if (okToGreet(name)) {
 *           resolve('Hello, ' + name + '!');
 *         } else {
 *           reject('Greeting ' + name + ' is not allowed.');
 *         }
 *       }, 1000);
 *     });
 *   }
 *
 *   var promise = asyncGreet('Robin Hood');
 *   promise.then(function(greeting) {
 *     alert('Success: ' + greeting);
 *   }, function(reason) {
 *     alert('Failed: ' + reason);
 *   });
 * ```
 *
 * Note: progress/notify callbacks are not currently supported via the ES6-style interface.
 *
 * Note: unlike ES6 behaviour, an exception thrown in the constructor function will NOT implicitly reject the promise.
 *
 * However, the more traditional CommonJS-style usage is still available, and documented below.
 *
 * [The CommonJS Promise proposal](http://wiki.commonjs.org/wiki/Promises) describes a promise as an
 * interface for interacting with an object that represents the result of an action that is
 * performed asynchronously, and may or may not be finished at any given point in time.
 *
 * From the perspective of dealing with error handling, deferred and promise APIs are to
 * asynchronous programming what `try`, `catch` and `throw` keywords are to synchronous programming.
 *
 * ```js
 *   // for the purpose of this example let's assume that variables `$q` and `okToGreet`
 *   // are available in the current lexical scope (they could have been injected or passed in).
 *
 *   function asyncGreet(name) {
 *     var deferred = $q.defer();
 *
 *     setTimeout(function() {
 *       deferred.notify('About to greet ' + name + '.');
 *
 *       if (okToGreet(name)) {
 *         deferred.resolve('Hello, ' + name + '!');
 *       } else {
 *         deferred.reject('Greeting ' + name + ' is not allowed.');
 *       }
 *     }, 1000);
 *
 *     return deferred.promise;
 *   }
 *
 *   var promise = asyncGreet('Robin Hood');
 *   promise.then(function(greeting) {
 *     alert('Success: ' + greeting);
 *   }, function(reason) {
 *     alert('Failed: ' + reason);
 *   }, function(update) {
 *     alert('Got notification: ' + update);
 *   });
 * ```
 *
 * At first it might not be obvious why this extra complexity is worth the trouble. The payoff
 * comes in the way of guarantees that promise and deferred APIs make, see
 * https://github.com/kriskowal/uncommonjs/blob/master/promises/specification.md.
 *
 * Additionally the promise api allows for composition that is very hard to do with the
 * traditional callback ([CPS](http://en.wikipedia.org/wiki/Continuation-passing_style)) approach.
 * For more on this please see the [Q documentation](https://github.com/kriskowal/q) especially the
 * section on serial or parallel joining of promises.
 *
 * # The Deferred API
 *
 * A new instance of deferred is constructed by calling `$q.defer()`.
 *
 * The purpose of the deferred object is to expose the associated Promise instance as well as APIs
 * that can be used for signaling the successful or unsuccessful completion, as well as the status
 * of the task.
 *
 * **Methods**
 *
 * - `resolve(value)` â€“ resolves the derived promise with the `value`. If the value is a rejection
 *   constructed via `$q.reject`, the promise will be rejected instead.
 * - `reject(reason)` â€“ rejects the derived promise with the `reason`. This is equivalent to
 *   resolving it with a rejection constructed via `$q.reject`.
 * - `notify(value)` - provides updates on the status of the promise's execution. This may be called
 *   multiple times before the promise is either resolved or rejected.
 *
 * **Properties**
 *
 * - promise â€“ `{Promise}` â€“ promise object associated with this deferred.
 *
 *
 * # The Promise API
 *
 * A new promise instance is created when a deferred instance is created and can be retrieved by
 * calling `deferred.promise`.
 *
 * The purpose of the promise object is to allow for interested parties to get access to the result
 * of the deferred task when it completes.
 *
 * **Methods**
 *
 * - `then(successCallback, errorCallback, notifyCallback)` â€“ regardless of when the promise was or
 *   will be resolved or rejected, `then` calls one of the success or error callbacks asynchronously
 *   as soon as the result is available. The callbacks are called with a single argument: the result
 *   or rejection reason. Additionally, the notify callback may be called zero or more times to
 *   provide a progress indication, before the promise is resolved or rejected.
 *
 *   This method *returns a new promise* which is resolved or rejected via the return value of the
 *   `successCallback`, `errorCallback` (unless that value is a promise, in which case it is resolved
 *   with the value which is resolved in that promise using
 *   [promise chaining](http://www.html5rocks.com/en/tutorials/es6/promises/#toc-promises-queues)).
 *   It also notifies via the return value of the `notifyCallback` method. The promise cannot be
 *   resolved or rejected from the notifyCallback method.
 *
 * - `catch(errorCallback)` â€“ shorthand for `promise.then(null, errorCallback)`
 *
 * - `finally(callback, notifyCallback)` â€“ allows you to observe either the fulfillment or rejection of a promise,
 *   but to do so without modifying the final value. This is useful to release resources or do some
 *   clean-up that needs to be done whether the promise was rejected or resolved. See the [full
 *   specification](https://github.com/kriskowal/q/wiki/API-Reference#promisefinallycallback) for
 *   more information.
 *
 * # Chaining promises
 *
 * Because calling the `then` method of a promise returns a new derived promise, it is easily
 * possible to create a chain of promises:
 *
 * ```js
 *   promiseB = promiseA.then(function(result) {
 *     return result + 1;
 *   });
 *
 *   // promiseB will be resolved immediately after promiseA is resolved and its value
 *   // will be the result of promiseA incremented by 1
 * ```
 *
 * It is possible to create chains of any length and since a promise can be resolved with another
 * promise (which will defer its resolution further), it is possible to pause/defer resolution of
 * the promises at any point in the chain. This makes it possible to implement powerful APIs like
 * $http's response interceptors.
 *
 *
 * # Differences between Kris Kowal's Q and $q
 *
 *  There are two main differences:
 *
 * - $q is integrated with the {@link ng.$rootScope.Scope} Scope model observation
 *   mechanism in angular, which means faster propagation of resolution or rejection into your
 *   models and avoiding unnecessary browser repaints, which would result in flickering UI.
 * - Q has many more features than $q, but that comes at a cost of bytes. $q is tiny, but contains
 *   all the important functionality needed for common async tasks.
 *
 *  # Testing
 *
 *  ```js
 *    it('should simulate promise', inject(function($q, $rootScope) {
 *      var deferred = $q.defer();
 *      var promise = deferred.promise;
 *      var resolvedValue;
 *
 *      promise.then(function(value) { resolvedValue = value; });
 *      expect(resolvedValue).toBeUndefined();
 *
 *      // Simulate resolving of promise
 *      deferred.resolve(123);
 *      // Note that the 'then' function does not get called synchronously.
 *      // This is because we want the promise API to always be async, whether or not
 *      // it got called synchronously or asynchronously.
 *      expect(resolvedValue).toBeUndefined();
 *
 *      // Propagate promise resolution to 'then' functions using $apply().
 *      $rootScope.$apply();
 *      expect(resolvedValue).toEqual(123);
 *    }));
 *  ```
 *
 * @param {function(function, function)} resolver Function which is responsible for resolving or
 *   rejecting the newly created promise. The first parameter is a function which resolves the
 *   promise, the second parameter is a function which rejects the promise.
 *
 * @returns {Promise} The newly created promise.
 */
function $QProvider() {

  this.$get = ['$rootScope', '$exceptionHandler', function($rootScope, $exceptionHandler) {
    return qFactory(function(callback) {
      $rootScope.$evalAsync(callback);
    }, $exceptionHandler);
  }];
}

function $$QProvider() {
  this.$get = ['$browser', '$exceptionHandler', function($browser, $exceptionHandler) {
    return qFactory(function(callback) {
      $browser.defer(callback);
    }, $exceptionHandler);
  }];
}

/**
 * Constructs a promise manager.
 *
 * @param {function(function)} nextTick Function for executing functions in the next turn.
 * @param {function(...*)} exceptionHandler Function into which unexpected exceptions are passed for
 *     debugging purposes.
 * @returns {object} Promise manager.
 */
function qFactory(nextTick, exceptionHandler) {
  var $qMinErr = minErr('$q', TypeError);
  function callOnce(self, resolveFn, rejectFn) {
    var called = false;
    function wrap(fn) {
      return function(value) {
        if (called) return;
        called = true;
        fn.call(self, value);
      };
    }

    return [wrap(resolveFn), wrap(rejectFn)];
  }

  /**
   * @ngdoc method
   * @name ng.$q#defer
   * @kind function
   *
   * @description
   * Creates a `Deferred` object which represents a task which will finish in the future.
   *
   * @returns {Deferred} Returns a new instance of deferred.
   */
  var defer = function() {
    return new Deferred();
  };

  function Promise() {
    this.$$state = { status: 0 };
  }

  extend(Promise.prototype, {
    then: function(onFulfilled, onRejected, progressBack) {
      if (isUndefined(onFulfilled) && isUndefined(onRejected) && isUndefined(progressBack)) {
        return this;
      }
      var result = new Deferred();

      this.$$state.pending = this.$$state.pending || [];
      this.$$state.pending.push([result, onFulfilled, onRejected, progressBack]);
      if (this.$$state.status > 0) scheduleProcessQueue(this.$$state);

      return result.promise;
    },

    "catch": function(callback) {
      return this.then(null, callback);
    },

    "finally": function(callback, progressBack) {
      return this.then(function(value) {
        return handleCallback(value, true, callback);
      }, function(error) {
        return handleCallback(error, false, callback);
      }, progressBack);
    }
  });

  //Faster, more basic than angular.bind http://jsperf.com/angular-bind-vs-custom-vs-native
  function simpleBind(context, fn) {
    return function(value) {
      fn.call(context, value);
    };
  }

  function processQueue(state) {
    var fn, deferred, pending;

    pending = state.pending;
    state.processScheduled = false;
    state.pending = undefined;
    for (var i = 0, ii = pending.length; i < ii; ++i) {
      deferred = pending[i][0];
      fn = pending[i][state.status];
      try {
        if (isFunction(fn)) {
          deferred.resolve(fn(state.value));
        } else if (state.status === 1) {
          deferred.resolve(state.value);
        } else {
          deferred.reject(state.value);
        }
      } catch (e) {
        deferred.reject(e);
        exceptionHandler(e);
      }
    }
  }

  function scheduleProcessQueue(state) {
    if (state.processScheduled || !state.pending) return;
    state.processScheduled = true;
    nextTick(function() { processQueue(state); });
  }

  function Deferred() {
    this.promise = new Promise();
    //Necessary to support unbound execution :/
    this.resolve = simpleBind(this, this.resolve);
    this.reject = simpleBind(this, this.reject);
    this.notify = simpleBind(this, this.notify);
  }

  extend(Deferred.prototype, {
    resolve: function(val) {
      if (this.promise.$$state.status) return;
      if (val === this.promise) {
        this.$$reject($qMinErr(
          'qcycle',
          "Expected promise to be resolved with value other than itself '{0}'",
          val));
      } else {
        this.$$resolve(val);
      }

    },

    $$resolve: function(val) {
      var then, fns;

      fns = callOnce(this, this.$$resolve, this.$$reject);
      try {
        if ((isObject(val) || isFunction(val))) then = val && val.then;
        if (isFunction(then)) {
          this.promise.$$state.status = -1;
          then.call(val, fns[0], fns[1], this.notify);
        } else {
          this.promise.$$state.value = val;
          this.promise.$$state.status = 1;
          scheduleProcessQueue(this.promise.$$state);
        }
      } catch (e) {
        fns[1](e);
        exceptionHandler(e);
      }
    },

    reject: function(reason) {
      if (this.promise.$$state.status) return;
      this.$$reject(reason);
    },

    $$reject: function(reason) {
      this.promise.$$state.value = reason;
      this.promise.$$state.status = 2;
      scheduleProcessQueue(this.promise.$$state);
    },

    notify: function(progress) {
      var callbacks = this.promise.$$state.pending;

      if ((this.promise.$$state.status <= 0) && callbacks && callbacks.length) {
        nextTick(function() {
          var callback, result;
          for (var i = 0, ii = callbacks.length; i < ii; i++) {
            result = callbacks[i][0];
            callback = callbacks[i][3];
            try {
              result.notify(isFunction(callback) ? callback(progress) : progress);
            } catch (e) {
              exceptionHandler(e);
            }
          }
        });
      }
    }
  });

  /**
   * @ngdoc method
   * @name $q#reject
   * @kind function
   *
   * @description
   * Creates a promise that is resolved as rejected with the specified `reason`. This api should be
   * used to forward rejection in a chain of promises. If you are dealing with the last promise in
   * a promise chain, you don't need to worry about it.
   *
   * When comparing deferreds/promises to the familiar behavior of try/catch/throw, think of
   * `reject` as the `throw` keyword in JavaScript. This also means that if you "catch" an error via
   * a promise error callback and you want to forward the error to the promise derived from the
   * current promise, you have to "rethrow" the error by returning a rejection constructed via
   * `reject`.
   *
   * ```js
   *   promiseB = promiseA.then(function(result) {
   *     // success: do something and resolve promiseB
   *     //          with the old or a new result
   *     return result;
   *   }, function(reason) {
   *     // error: handle the error if possible and
   *     //        resolve promiseB with newPromiseOrValue,
   *     //        otherwise forward the rejection to promiseB
   *     if (canHandle(reason)) {
   *      // handle the error and recover
   *      return newPromiseOrValue;
   *     }
   *     return $q.reject(reason);
   *   });
   * ```
   *
   * @param {*} reason Constant, message, exception or an object representing the rejection reason.
   * @returns {Promise} Returns a promise that was already resolved as rejected with the `reason`.
   */
  var reject = function(reason) {
    var result = new Deferred();
    result.reject(reason);
    return result.promise;
  };

  var makePromise = function makePromise(value, resolved) {
    var result = new Deferred();
    if (resolved) {
      result.resolve(value);
    } else {
      result.reject(value);
    }
    return result.promise;
  };

  var handleCallback = function handleCallback(value, isResolved, callback) {
    var callbackOutput = null;
    try {
      if (isFunction(callback)) callbackOutput = callback();
    } catch (e) {
      return makePromise(e, false);
    }
    if (isPromiseLike(callbackOutput)) {
      return callbackOutput.then(function() {
        return makePromise(value, isResolved);
      }, function(error) {
        return makePromise(error, false);
      });
    } else {
      return makePromise(value, isResolved);
    }
  };

  /**
   * @ngdoc method
   * @name $q#when
   * @kind function
   *
   * @description
   * Wraps an object that might be a value or a (3rd party) then-able promise into a $q promise.
   * This is useful when you are dealing with an object that might or might not be a promise, or if
   * the promise comes from a source that can't be trusted.
   *
   * @param {*} value Value or a promise
   * @param {Function=} successCallback
   * @param {Function=} errorCallback
   * @param {Function=} progressCallback
   * @returns {Promise} Returns a promise of the passed value or promise
   */


  var when = function(value, callback, errback, progressBack) {
    var result = new Deferred();
    result.resolve(value);
    return result.promise.then(callback, errback, progressBack);
  };

  /**
   * @ngdoc method
   * @name $q#resolve
   * @kind function
   *
   * @description
   * Alias of {@link ng.$q#when when} to maintain naming consistency with ES6.
   *
   * @param {*} value Value or a promise
   * @param {Function=} successCallback
   * @param {Function=} errorCallback
   * @param {Function=} progressCallback
   * @returns {Promise} Returns a promise of the passed value or promise
   */
  var resolve = when;

  /**
   * @ngdoc method
   * @name $q#all
   * @kind function
   *
   * @description
   * Combines multiple promises into a single promise that is resolved when all of the input
   * promises are resolved.
   *
   * @param {Array.<Promise>|Object.<Promise>} promises An array or hash of promises.
   * @returns {Promise} Returns a single promise that will be resolved with an array/hash of values,
   *   each value corresponding to the promise at the same index/key in the `promises` array/hash.
   *   If any of the promises is resolved with a rejection, this resulting promise will be rejected
   *   with the same rejection value.
   */

  function all(promises) {
    var deferred = new Deferred(),
        counter = 0,
        results = isArray(promises) ? [] : {};

    forEach(promises, function(promise, key) {
      counter++;
      when(promise).then(function(value) {
        if (results.hasOwnProperty(key)) return;
        results[key] = value;
        if (!(--counter)) deferred.resolve(results);
      }, function(reason) {
        if (results.hasOwnProperty(key)) return;
        deferred.reject(reason);
      });
    });

    if (counter === 0) {
      deferred.resolve(results);
    }

    return deferred.promise;
  }

  var $Q = function Q(resolver) {
    if (!isFunction(resolver)) {
      throw $qMinErr('norslvr', "Expected resolverFn, got '{0}'", resolver);
    }

    if (!(this instanceof Q)) {
      // More useful when $Q is the Promise itself.
      return new Q(resolver);
    }

    var deferred = new Deferred();

    function resolveFn(value) {
      deferred.resolve(value);
    }

    function rejectFn(reason) {
      deferred.reject(reason);
    }

    resolver(resolveFn, rejectFn);

    return deferred.promise;
  };

  $Q.defer = defer;
  $Q.reject = reject;
  $Q.when = when;
  $Q.resolve = resolve;
  $Q.all = all;

  return $Q;
}

function $$RAFProvider() { //rAF
  this.$get = ['$window', '$timeout', function($window, $timeout) {
    var requestAnimationFrame = $window.requestAnimationFrame ||
                                $window.webkitRequestAnimationFrame;

    var cancelAnimationFrame = $window.cancelAnimationFrame ||
                               $window.webkitCancelAnimationFrame ||
                               $window.webkitCancelRequestAnimationFrame;

    var rafSupported = !!requestAnimationFrame;
    var raf = rafSupported
      ? function(fn) {
          var id = requestAnimationFrame(fn);
          return function() {
            cancelAnimationFrame(id);
          };
        }
      : function(fn) {
          var timer = $timeout(fn, 16.66, false); // 1000 / 60 = 16.666
          return function() {
            $timeout.cancel(timer);
          };
        };

    raf.supported = rafSupported;

    return raf;
  }];
}

/**
 * DESIGN NOTES
 *
 * The design decisions behind the scope are heavily favored for speed and memory consumption.
 *
 * The typical use of scope is to watch the expressions, which most of the time return the same
 * value as last time so we optimize the operation.
 *
 * Closures construction is expensive in terms of speed as well as memory:
 *   - No closures, instead use prototypical inheritance for API
 *   - Internal state needs to be stored on scope directly, which means that private state is
 *     exposed as $$____ properties
 *
 * Loop operations are optimized by using while(count--) { ... }
 *   - This means that in order to keep the same order of execution as addition we have to add
 *     items to the array at the beginning (unshift) instead of at the end (push)
 *
 * Child scopes are created and removed often
 *   - Using an array would be slow since inserts in the middle are expensive; so we use linked lists
 *
 * There are fewer watches than observers. This is why you don't want the observer to be implemented
 * in the same way as watch. Watch requires return of the initialization function which is expensive
 * to construct.
 */


/**
 * @ngdoc provider
 * @name $rootScopeProvider
 * @description
 *
 * Provider for the $rootScope service.
 */

/**
 * @ngdoc method
 * @name $rootScopeProvider#digestTtl
 * @description
 *
 * Sets the number of `$digest` iterations the scope should attempt to execute before giving up and
 * assuming that the model is unstable.
 *
 * The current default is 10 iterations.
 *
 * In complex applications it's possible that the dependencies between `$watch`s will result in
 * several digest iterations. However if an application needs more than the default 10 digest
 * iterations for its model to stabilize then you should investigate what is causing the model to
 * continuously change during the digest.
 *
 * Increasing the TTL could have performance implications, so you should not change it without
 * proper justification.
 *
 * @param {number} limit The number of digest iterations.
 */


/**
 * @ngdoc service
 * @name $rootScope
 * @description
 *
 * Every application has a single root {@link ng.$rootScope.Scope scope}.
 * All other scopes are descendant scopes of the root scope. Scopes provide separation
 * between the model and the view, via a mechanism for watching the model for changes.
 * They also provide event emission/broadcast and subscription facility. See the
 * {@link guide/scope developer guide on scopes}.
 */
function $RootScopeProvider() {
  var TTL = 10;
  var $rootScopeMinErr = minErr('$rootScope');
  var lastDirtyWatch = null;
  var applyAsyncId = null;

  this.digestTtl = function(value) {
    if (arguments.length) {
      TTL = value;
    }
    return TTL;
  };

  function createChildScopeClass(parent) {
    function ChildScope() {
      this.$$watchers = this.$$nextSibling =
          this.$$childHead = this.$$childTail = null;
      this.$$listeners = {};
      this.$$listenerCount = {};
      this.$$watchersCount = 0;
      this.$id = nextUid();
      this.$$ChildScope = null;
    }
    ChildScope.prototype = parent;
    return ChildScope;
  }

  this.$get = ['$injector', '$exceptionHandler', '$parse', '$browser',
      function($injector, $exceptionHandler, $parse, $browser) {

    function destroyChildScope($event) {
        $event.currentScope.$$destroyed = true;
    }

    function cleanUpScope($scope) {

      if (msie === 9) {
        // There is a memory leak in IE9 if all child scopes are not disconnected
        // completely when a scope is destroyed. So this code will recurse up through
        // all this scopes children
        //
        // See issue https://github.com/angular/angular.js/issues/10706
        $scope.$$childHead && cleanUpScope($scope.$$childHead);
        $scope.$$nextSibling && cleanUpScope($scope.$$nextSibling);
      }

      // The code below works around IE9 and V8's memory leaks
      //
      // See:
      // - https://code.google.com/p/v8/issues/detail?id=2073#c26
      // - https://github.com/angular/angular.js/issues/6794#issuecomment-38648909
      // - https://github.com/angular/angular.js/issues/1313#issuecomment-10378451

      $scope.$parent = $scope.$$nextSibling = $scope.$$prevSibling = $scope.$$childHead =
          $scope.$$childTail = $scope.$root = $scope.$$watchers = null;
    }

    /**
     * @ngdoc type
     * @name $rootScope.Scope
     *
     * @description
     * A root scope can be retrieved using the {@link ng.$rootScope $rootScope} key from the
     * {@link auto.$injector $injector}. Child scopes are created using the
     * {@link ng.$rootScope.Scope#$new $new()} method. (Most scopes are created automatically when
     * compiled HTML template is executed.) See also the {@link guide/scope Scopes guide} for
     * an in-depth introduction and usage examples.
     *
     *
     * # Inheritance
     * A scope can inherit from a parent scope, as in this example:
     * ```js
         var parent = $rootScope;
         var child = parent.$new();

         parent.salutation = "Hello";
         expect(child.salutation).toEqual('Hello');

         child.salutation = "Welcome";
         expect(child.salutation).toEqual('Welcome');
         expect(parent.salutation).toEqual('Hello');
     * ```
     *
     * When interacting with `Scope` in tests, additional helper methods are available on the
     * instances of `Scope` type. See {@link ngMock.$rootScope.Scope ngMock Scope} for additional
     * details.
     *
     *
     * @param {Object.<string, function()>=} providers Map of service factory which need to be
     *                                       provided for the current scope. Defaults to {@link ng}.
     * @param {Object.<string, *>=} instanceCache Provides pre-instantiated services which should
     *                              append/override services provided by `providers`. This is handy
     *                              when unit-testing and having the need to override a default
     *                              service.
     * @returns {Object} Newly created scope.
     *
     */
    function Scope() {
      this.$id = nextUid();
      this.$$phase = this.$parent = this.$$watchers =
                     this.$$nextSibling = this.$$prevSibling =
                     this.$$childHead = this.$$childTail = null;
      this.$root = this;
      this.$$destroyed = false;
      this.$$listeners = {};
      this.$$listenerCount = {};
      this.$$watchersCount = 0;
      this.$$isolateBindings = null;
    }

    /**
     * @ngdoc property
     * @name $rootScope.Scope#$id
     *
     * @description
     * Unique scope ID (monotonically increasing) useful for debugging.
     */

     /**
      * @ngdoc property
      * @name $rootScope.Scope#$parent
      *
      * @description
      * Reference to the parent scope.
      */

      /**
       * @ngdoc property
       * @name $rootScope.Scope#$root
       *
       * @description
       * Reference to the root scope.
       */

    Scope.prototype = {
      constructor: Scope,
      /**
       * @ngdoc method
       * @name $rootScope.Scope#$new
       * @kind function
       *
       * @description
       * Creates a new child {@link ng.$rootScope.Scope scope}.
       *
       * The parent scope will propagate the {@link ng.$rootScope.Scope#$digest $digest()} event.
       * The scope can be removed from the scope hierarchy using {@link ng.$rootScope.Scope#$destroy $destroy()}.
       *
       * {@link ng.$rootScope.Scope#$destroy $destroy()} must be called on a scope when it is
       * desired for the scope and its child scopes to be permanently detached from the parent and
       * thus stop participating in model change detection and listener notification by invoking.
       *
       * @param {boolean} isolate If true, then the scope does not prototypically inherit from the
       *         parent scope. The scope is isolated, as it can not see parent scope properties.
       *         When creating widgets, it is useful for the widget to not accidentally read parent
       *         state.
       *
       * @param {Scope} [parent=this] The {@link ng.$rootScope.Scope `Scope`} that will be the `$parent`
       *                              of the newly created scope. Defaults to `this` scope if not provided.
       *                              This is used when creating a transclude scope to correctly place it
       *                              in the scope hierarchy while maintaining the correct prototypical
       *                              inheritance.
       *
       * @returns {Object} The newly created child scope.
       *
       */
      $new: function(isolate, parent) {
        var child;

        parent = parent || this;

        if (isolate) {
          child = new Scope();
          child.$root = this.$root;
        } else {
          // Only create a child scope class if somebody asks for one,
          // but cache it to allow the VM to optimize lookups.
          if (!this.$$ChildScope) {
            this.$$ChildScope = createChildScopeClass(this);
          }
          child = new this.$$ChildScope();
        }
        child.$parent = parent;
        child.$$prevSibling = parent.$$childTail;
        if (parent.$$childHead) {
          parent.$$childTail.$$nextSibling = child;
          parent.$$childTail = child;
        } else {
          parent.$$childHead = parent.$$childTail = child;
        }

        // When the new scope is not isolated or we inherit from `this`, and
        // the parent scope is destroyed, the property `$$destroyed` is inherited
        // prototypically. In all other cases, this property needs to be set
        // when the parent scope is destroyed.
        // The listener needs to be added after the parent is set
        if (isolate || parent != this) child.$on('$destroy', destroyChildScope);

        return child;
      },

      /**
       * @ngdoc method
       * @name $rootScope.Scope#$watch
       * @kind function
       *
       * @description
       * Registers a `listener` callback to be executed whenever the `watchExpression` changes.
       *
       * - The `watchExpression` is called on every call to {@link ng.$rootScope.Scope#$digest
       *   $digest()} and should return the value that will be watched. (`watchExpression` should not change
       *   its value when executed multiple times with the same input because it may be executed multiple
       *   times by {@link ng.$rootScope.Scope#$digest $digest()}. That is, `watchExpression` should be
       *   [idempotent](http://en.wikipedia.org/wiki/Idempotence).
       * - The `listener` is called only when the value from the current `watchExpression` and the
       *   previous call to `watchExpression` are not equal (with the exception of the initial run,
       *   see below). Inequality is determined according to reference inequality,
       *   [strict comparison](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Comparison_Operators)
       *    via the `!==` Javascript operator, unless `objectEquality == true`
       *   (see next point)
       * - When `objectEquality == true`, inequality of the `watchExpression` is determined
       *   according to the {@link angular.equals} function. To save the value of the object for
       *   later comparison, the {@link angular.copy} function is used. This therefore means that
       *   watching complex objects will have adverse memory and performance implications.
       * - The watch `listener` may change the model, which may trigger other `listener`s to fire.
       *   This is achieved by rerunning the watchers until no changes are detected. The rerun
       *   iteration limit is 10 to prevent an infinite loop deadlock.
       *
       *
       * If you want to be notified whenever {@link ng.$rootScope.Scope#$digest $digest} is called,
       * you can register a `watchExpression` function with no `listener`. (Be prepared for
       * multiple calls to your `watchExpression` because it will execute multiple times in a
       * single {@link ng.$rootScope.Scope#$digest $digest} cycle if a change is detected.)
       *
       * After a watcher is registered with the scope, the `listener` fn is called asynchronously
       * (via {@link ng.$rootScope.Scope#$evalAsync $evalAsync}) to initialize the
       * watcher. In rare cases, this is undesirable because the listener is called when the result
       * of `watchExpression` didn't change. To detect this scenario within the `listener` fn, you
       * can compare the `newVal` and `oldVal`. If these two values are identical (`===`) then the
       * listener was called due to initialization.
       *
       *
       *
       * # Example
       * ```js
           // let's assume that scope was dependency injected as the $rootScope
           var scope = $rootScope;
           scope.name = 'misko';
           scope.counter = 0;

           expect(scope.counter).toEqual(0);
           scope.$watch('name', function(newValue, oldValue) {
             scope.counter = scope.counter + 1;
           });
           expect(scope.counter).toEqual(0);

           scope.$digest();
           // the listener is always called during the first $digest loop after it was registered
           expect(scope.counter).toEqual(1);

           scope.$digest();
           // but now it will not be called unless the value changes
           expect(scope.counter).toEqual(1);

           scope.name = 'adam';
           scope.$digest();
           expect(scope.counter).toEqual(2);



           // Using a function as a watchExpression
           var food;
           scope.foodCounter = 0;
           expect(scope.foodCounter).toEqual(0);
           scope.$watch(
             // This function returns the value being watched. It is called for each turn of the $digest loop
             function() { return food; },
             // This is the change listener, called when the value returned from the above function changes
             function(newValue, oldValue) {
               if ( newValue !== oldValue ) {
                 // Only increment the counter if the value changed
                 scope.foodCounter = scope.foodCounter + 1;
               }
             }
           );
           // No digest has been run so the counter will be zero
           expect(scope.foodCounter).toEqual(0);

           // Run the digest but since food has not changed count will still be zero
           scope.$digest();
           expect(scope.foodCounter).toEqual(0);

           // Update food and run digest.  Now the counter will increment
           food = 'cheeseburger';
           scope.$digest();
           expect(scope.foodCounter).toEqual(1);

       * ```
       *
       *
       *
       * @param {(function()|string)} watchExpression Expression that is evaluated on each
       *    {@link ng.$rootScope.Scope#$digest $digest} cycle. A change in the return value triggers
       *    a call to the `listener`.
       *
       *    - `string`: Evaluated as {@link guide/expression expression}
       *    - `function(scope)`: called with current `scope` as a parameter.
       * @param {function(newVal, oldVal, scope)} listener Callback called whenever the value
       *    of `watchExpression` changes.
       *
       *    - `newVal` contains the current value of the `watchExpression`
       *    - `oldVal` contains the previous value of the `watchExpression`
       *    - `scope` refers to the current scope
       * @param {boolean=} objectEquality Compare for object equality using {@link angular.equals} instead of
       *     comparing for reference equality.
       * @returns {function()} Returns a deregistration function for this listener.
       */
      $watch: function(watchExp, listener, objectEquality, prettyPrintExpression) {
        var get = $parse(watchExp);

        if (get.$$watchDelegate) {
          return get.$$watchDelegate(this, listener, objectEquality, get, watchExp);
        }
        var scope = this,
            array = scope.$$watchers,
            watcher = {
              fn: listener,
              last: initWatchVal,
              get: get,
              exp: prettyPrintExpression || watchExp,
              eq: !!objectEquality
            };

        lastDirtyWatch = null;

        if (!isFunction(listener)) {
          watcher.fn = noop;
        }

        if (!array) {
          array = scope.$$watchers = [];
        }
        // we use unshift since we use a while loop in $digest for speed.
        // the while loop reads in reverse order.
        array.unshift(watcher);
        incrementWatchersCount(this, 1);

        return function deregisterWatch() {
          if (arrayRemove(array, watcher) >= 0) {
            incrementWatchersCount(scope, -1);
          }
          lastDirtyWatch = null;
        };
      },

      /**
       * @ngdoc method
       * @name $rootScope.Scope#$watchGroup
       * @kind function
       *
       * @description
       * A variant of {@link ng.$rootScope.Scope#$watch $watch()} where it watches an array of `watchExpressions`.
       * If any one expression in the collection changes the `listener` is executed.
       *
       * - The items in the `watchExpressions` array are observed via standard $watch operation and are examined on every
       *   call to $digest() to see if any items changes.
       * - The `listener` is called whenever any expression in the `watchExpressions` array changes.
       *
       * @param {Array.<string|Function(scope)>} watchExpressions Array of expressions that will be individually
       * watched using {@link ng.$rootScope.Scope#$watch $watch()}
       *
       * @param {function(newValues, oldValues, scope)} listener Callback called whenever the return value of any
       *    expression in `watchExpressions` changes
       *    The `newValues` array contains the current values of the `watchExpressions`, with the indexes matching
       *    those of `watchExpression`
       *    and the `oldValues` array contains the previous values of the `watchExpressions`, with the indexes matching
       *    those of `watchExpression`
       *    The `scope` refers to the current scope.
       * @returns {function()} Returns a de-registration function for all listeners.
       */
      $watchGroup: function(watchExpressions, listener) {
        var oldValues = new Array(watchExpressions.length);
        var newValues = new Array(watchExpressions.length);
        var deregisterFns = [];
        var self = this;
        var changeReactionScheduled = false;
        var firstRun = true;

        if (!watchExpressions.length) {
          // No expressions means we call the listener ASAP
          var shouldCall = true;
          self.$evalAsync(function() {
            if (shouldCall) listener(newValues, newValues, self);
          });
          return function deregisterWatchGroup() {
            shouldCall = false;
          };
        }

        if (watchExpressions.length === 1) {
          // Special case size of one
          return this.$watch(watchExpressions[0], function watchGroupAction(value, oldValue, scope) {
            newValues[0] = value;
            oldValues[0] = oldValue;
            listener(newValues, (value === oldValue) ? newValues : oldValues, scope);
          });
        }

        forEach(watchExpressions, function(expr, i) {
          var unwatchFn = self.$watch(expr, function watchGroupSubAction(value, oldValue) {
            newValues[i] = value;
            oldValues[i] = oldValue;
            if (!changeReactionScheduled) {
              changeReactionScheduled = true;
              self.$evalAsync(watchGroupAction);
            }
          });
          deregisterFns.push(unwatchFn);
        });

        function watchGroupAction() {
          changeReactionScheduled = false;

          if (firstRun) {
            firstRun = false;
            listener(newValues, newValues, self);
          } else {
            listener(newValues, oldValues, self);
          }
        }

        return function deregisterWatchGroup() {
          while (deregisterFns.length) {
            deregisterFns.shift()();
          }
        };
      },


      /**
       * @ngdoc method
       * @name $rootScope.Scope#$watchCollection
       * @kind function
       *
       * @description
       * Shallow watches the properties of an object and fires whenever any of the properties change
       * (for arrays, this implies watching the array items; for object maps, this implies watching
       * the properties). If a change is detected, the `listener` callback is fired.
       *
       * - The `obj` collection is observed via standard $watch operation and is examined on every
       *   call to $digest() to see if any items have been added, removed, or moved.
       * - The `listener` is called whenever anything within the `obj` has changed. Examples include
       *   adding, removing, and moving items belonging to an object or array.
       *
       *
       * # Example
       * ```js
          $scope.names = ['igor', 'matias', 'misko', 'james'];
          $scope.dataCount = 4;

          $scope.$watchCollection('names', function(newNames, oldNames) {
            $scope.dataCount = newNames.length;
          });

          expect($scope.dataCount).toEqual(4);
          $scope.$digest();

          //still at 4 ... no changes
          expect($scope.dataCount).toEqual(4);

          $scope.names.pop();
          $scope.$digest();

          //now there's been a change
          expect($scope.dataCount).toEqual(3);
       * ```
       *
       *
       * @param {string|function(scope)} obj Evaluated as {@link guide/expression expression}. The
       *    expression value should evaluate to an object or an array which is observed on each
       *    {@link ng.$rootScope.Scope#$digest $digest} cycle. Any shallow change within the
       *    collection will trigger a call to the `listener`.
       *
       * @param {function(newCollection, oldCollection, scope)} listener a callback function called
       *    when a change is detected.
       *    - The `newCollection` object is the newly modified data obtained from the `obj` expression
       *    - The `oldCollection` object is a copy of the former collection data.
       *      Due to performance considerations, the`oldCollection` value is computed only if the
       *      `listener` function declares two or more arguments.
       *    - The `scope` argument refers to the current scope.
       *
       * @returns {function()} Returns a de-registration function for this listener. When the
       *    de-registration function is executed, the internal watch operation is terminated.
       */
      $watchCollection: function(obj, listener) {
        $watchCollectionInterceptor.$stateful = true;

        var self = this;
        // the current value, updated on each dirty-check run
        var newValue;
        // a shallow copy of the newValue from the last dirty-check run,
        // updated to match newValue during dirty-check run
        var oldValue;
        // a shallow copy of the newValue from when the last change happened
        var veryOldValue;
        // only track veryOldValue if the listener is asking for it
        var trackVeryOldValue = (listener.length > 1);
        var changeDetected = 0;
        var changeDetector = $parse(obj, $watchCollectionInterceptor);
        var internalArray = [];
        var internalObject = {};
        var initRun = true;
        var oldLength = 0;

        function $watchCollectionInterceptor(_value) {
          newValue = _value;
          var newLength, key, bothNaN, newItem, oldItem;

          // If the new value is undefined, then return undefined as the watch may be a one-time watch
          if (isUndefined(newValue)) return;

          if (!isObject(newValue)) { // if primitive
            if (oldValue !== newValue) {
              oldValue = newValue;
              changeDetected++;
            }
          } else if (isArrayLike(newValue)) {
            if (oldValue !== internalArray) {
              // we are transitioning from something which was not an array into array.
              oldValue = internalArray;
              oldLength = oldValue.length = 0;
              changeDetected++;
            }

            newLength = newValue.length;

            if (oldLength !== newLength) {
              // if lengths do not match we need to trigger change notification
              changeDetected++;
              oldValue.length = oldLength = newLength;
            }
            // copy the items to oldValue and look for changes.
            for (var i = 0; i < newLength; i++) {
              oldItem = oldValue[i];
              newItem = newValue[i];

              bothNaN = (oldItem !== oldItem) && (newItem !== newItem);
              if (!bothNaN && (oldItem !== newItem)) {
                changeDetected++;
                oldValue[i] = newItem;
              }
            }
          } else {
            if (oldValue !== internalObject) {
              // we are transitioning from something which was not an object into object.
              oldValue = internalObject = {};
              oldLength = 0;
              changeDetected++;
            }
            // copy the items to oldValue and look for changes.
            newLength = 0;
            for (key in newValue) {
              if (hasOwnProperty.call(newValue, key)) {
                newLength++;
                newItem = newValue[key];
                oldItem = oldValue[key];

                if (key in oldValue) {
                  bothNaN = (oldItem !== oldItem) && (newItem !== newItem);
                  if (!bothNaN && (oldItem !== newItem)) {
                    changeDetected++;
                    oldValue[key] = newItem;
                  }
                } else {
                  oldLength++;
                  oldValue[key] = newItem;
                  changeDetected++;
                }
              }
            }
            if (oldLength > newLength) {
              // we used to have more keys, need to find them and destroy them.
              changeDetected++;
              for (key in oldValue) {
                if (!hasOwnProperty.call(newValue, key)) {
                  oldLength--;
                  delete oldValue[key];
                }
              }
            }
          }
          return changeDetected;
        }

        function $watchCollectionAction() {
          if (initRun) {
            initRun = false;
            listener(newValue, newValue, self);
          } else {
            listener(newValue, veryOldValue, self);
          }

          // make a copy for the next time a collection is changed
          if (trackVeryOldValue) {
            if (!isObject(newValue)) {
              //primitive
              veryOldValue = newValue;
            } else if (isArrayLike(newValue)) {
              veryOldValue = new Array(newValue.length);
              for (var i = 0; i < newValue.length; i++) {
                veryOldValue[i] = newValue[i];
              }
            } else { // if object
              veryOldValue = {};
              for (var key in newValue) {
                if (hasOwnProperty.call(newValue, key)) {
                  veryOldValue[key] = newValue[key];
                }
              }
            }
          }
        }

        return this.$watch(changeDetector, $watchCollectionAction);
      },

      /**
       * @ngdoc method
       * @name $rootScope.Scope#$digest
       * @kind function
       *
       * @description
       * Processes all of the {@link ng.$rootScope.Scope#$watch watchers} of the current scope and
       * its children. Because a {@link ng.$rootScope.Scope#$watch watcher}'s listener can change
       * the model, the `$digest()` keeps calling the {@link ng.$rootScope.Scope#$watch watchers}
       * until no more listeners are firing. This means that it is possible to get into an infinite
       * loop. This function will throw `'Maximum iteration limit exceeded.'` if the number of
       * iterations exceeds 10.
       *
       * Usually, you don't call `$digest()` directly in
       * {@link ng.directive:ngController controllers} or in
       * {@link ng.$compileProvider#directive directives}.
       * Instead, you should call {@link ng.$rootScope.Scope#$apply $apply()} (typically from within
       * a {@link ng.$compileProvider#directive directive}), which will force a `$digest()`.
       *
       * If you want to be notified whenever `$digest()` is called,
       * you can register a `watchExpression` function with
       * {@link ng.$rootScope.Scope#$watch $watch()} with no `listener`.
       *
       * In unit tests, you may need to call `$digest()` to simulate the scope life cycle.
       *
       * # Example
       * ```js
           var scope = ...;
           scope.name = 'misko';
           scope.counter = 0;

           expect(scope.counter).toEqual(0);
           scope.$watch('name', function(newValue, oldValue) {
             scope.counter = scope.counter + 1;
           });
           expect(scope.counter).toEqual(0);

           scope.$digest();
           // the listener is always called during the first $digest loop after it was registered
           expect(scope.counter).toEqual(1);

           scope.$digest();
           // but now it will not be called unless the value changes
           expect(scope.counter).toEqual(1);

           scope.name = 'adam';
           scope.$digest();
           expect(scope.counter).toEqual(2);
       * ```
       *
       */
      $digest: function() {
        var watch, value, last,
            watchers,
            length,
            dirty, ttl = TTL,
            next, current, target = this,
            watchLog = [],
            logIdx, logMsg, asyncTask;

        beginPhase('$digest');
        // Check for changes to browser url that happened in sync before the call to $digest
        $browser.$$checkUrlChange();

        if (this === $rootScope && applyAsyncId !== null) {
          // If this is the root scope, and $applyAsync has scheduled a deferred $apply(), then
          // cancel the scheduled $apply and flush the queue of expressions to be evaluated.
          $browser.defer.cancel(applyAsyncId);
          flushApplyAsync();
        }

        lastDirtyWatch = null;

        do { // "while dirty" loop
          dirty = false;
          current = target;

          while (asyncQueue.length) {
            try {
              asyncTask = asyncQueue.shift();
              asyncTask.scope.$eval(asyncTask.expression, asyncTask.locals);
            } catch (e) {
              $exceptionHandler(e);
            }
            lastDirtyWatch = null;
          }

          traverseScopesLoop:
          do { // "traverse the scopes" loop
            if ((watchers = current.$$watchers)) {
              // process our watches
              length = watchers.length;
              while (length--) {
                try {
                  watch = watchers[length];
                  // Most common watches are on primitives, in which case we can short
                  // circuit it with === operator, only when === fails do we use .equals
                  if (watch) {
                    if ((value = watch.get(current)) !== (last = watch.last) &&
                        !(watch.eq
                            ? equals(value, last)
                            : (typeof value === 'number' && typeof last === 'number'
                               && isNaN(value) && isNaN(last)))) {
                      dirty = true;
                      lastDirtyWatch = watch;
                      watch.last = watch.eq ? copy(value, null) : value;
                      watch.fn(value, ((last === initWatchVal) ? value : last), current);
                      if (ttl < 5) {
                        logIdx = 4 - ttl;
                        if (!watchLog[logIdx]) watchLog[logIdx] = [];
                        watchLog[logIdx].push({
                          msg: isFunction(watch.exp) ? 'fn: ' + (watch.exp.name || watch.exp.toString()) : watch.exp,
                          newVal: value,
                          oldVal: last
                        });
                      }
                    } else if (watch === lastDirtyWatch) {
                      // If the most recently dirty watcher is now clean, short circuit since the remaining watchers
                      // have already been tested.
                      dirty = false;
                      break traverseScopesLoop;
                    }
                  }
                } catch (e) {
                  $exceptionHandler(e);
                }
              }
            }

            // Insanity Warning: scope depth-first traversal
            // yes, this code is a bit crazy, but it works and we have tests to prove it!
            // this piece should be kept in sync with the traversal in $broadcast
            if (!(next = ((current.$$watchersCount && current.$$childHead) ||
                (current !== target && current.$$nextSibling)))) {
              while (current !== target && !(next = current.$$nextSibling)) {
                current = current.$parent;
              }
            }
          } while ((current = next));

          // `break traverseScopesLoop;` takes us to here

          if ((dirty || asyncQueue.length) && !(ttl--)) {
            clearPhase();
            throw $rootScopeMinErr('infdig',
                '{0} $digest() iterations reached. Aborting!\n' +
                'Watchers fired in the last 5 iterations: {1}',
                TTL, watchLog);
          }

        } while (dirty || asyncQueue.length);

        clearPhase();

        while (postDigestQueue.length) {
          try {
            postDigestQueue.shift()();
          } catch (e) {
            $exceptionHandler(e);
          }
        }
      },


      /**
       * @ngdoc event
       * @name $rootScope.Scope#$destroy
       * @eventType broadcast on scope being destroyed
       *
       * @description
       * Broadcasted when a scope and its children are being destroyed.
       *
       * Note that, in AngularJS, there is also a `$destroy` jQuery event, which can be used to
       * clean up DOM bindings before an element is removed from the DOM.
       */

      /**
       * @ngdoc method
       * @name $rootScope.Scope#$destroy
       * @kind function
       *
       * @description
       * Removes the current scope (and all of its children) from the parent scope. Removal implies
       * that calls to {@link ng.$rootScope.Scope#$digest $digest()} will no longer
       * propagate to the current scope and its children. Removal also implies that the current
       * scope is eligible for garbage collection.
       *
       * The `$destroy()` is usually used by directives such as
       * {@link ng.directive:ngRepeat ngRepeat} for managing the
       * unrolling of the loop.
       *
       * Just before a scope is destroyed, a `$destroy` event is broadcasted on this scope.
       * Application code can register a `$destroy` event handler that will give it a chance to
       * perform any necessary cleanup.
       *
       * Note that, in AngularJS, there is also a `$destroy` jQuery event, which can be used to
       * clean up DOM bindings before an element is removed from the DOM.
       */
      $destroy: function() {
        // We can't destroy a scope that has been already destroyed.
        if (this.$$destroyed) return;
        var parent = this.$parent;

        this.$broadcast('$destroy');
        this.$$destroyed = true;

        if (this === $rootScope) {
          //Remove handlers attached to window when $rootScope is removed
          $browser.$$applicationDestroyed();
        }

        incrementWatchersCount(this, -this.$$watchersCount);
        for (var eventName in this.$$listenerCount) {
          decrementListenerCount(this, this.$$listenerCount[eventName], eventName);
        }

        // sever all the references to parent scopes (after this cleanup, the current scope should
        // not be retained by any of our references and should be eligible for garbage collection)
        if (parent && parent.$$childHead == this) parent.$$childHead = this.$$nextSibling;
        if (parent && parent.$$childTail == this) parent.$$childTail = this.$$prevSibling;
        if (this.$$prevSibling) this.$$prevSibling.$$nextSibling = this.$$nextSibling;
        if (this.$$nextSibling) this.$$nextSibling.$$prevSibling = this.$$prevSibling;

        // Disable listeners, watchers and apply/digest methods
        this.$destroy = this.$digest = this.$apply = this.$evalAsync = this.$applyAsync = noop;
        this.$on = this.$watch = this.$watchGroup = function() { return noop; };
        this.$$listeners = {};

        // Disconnect the next sibling to prevent `cleanUpScope` destroying those too
        this.$$nextSibling = null;
        cleanUpScope(this);
      },

      /**
       * @ngdoc method
       * @name $rootScope.Scope#$eval
       * @kind function
       *
       * @description
       * Executes the `expression` on the current scope and returns the result. Any exceptions in
       * the expression are propagated (uncaught). This is useful when evaluating Angular
       * expressions.
       *
       * # Example
       * ```js
           var scope = ng.$rootScope.Scope();
           scope.a = 1;
           scope.b = 2;

           expect(scope.$eval('a+b')).toEqual(3);
           expect(scope.$eval(function(scope){ return scope.a + scope.b; })).toEqual(3);
       * ```
       *
       * @param {(string|function())=} expression An angular expression to be executed.
       *
       *    - `string`: execute using the rules as defined in  {@link guide/expression expression}.
       *    - `function(scope)`: execute the function with the current `scope` parameter.
       *
       * @param {(object)=} locals Local variables object, useful for overriding values in scope.
       * @returns {*} The result of evaluating the expression.
       */
      $eval: function(expr, locals) {
        return $parse(expr)(this, locals);
      },

      /**
       * @ngdoc method
       * @name $rootScope.Scope#$evalAsync
       * @kind function
       *
       * @description
       * Executes the expression on the current scope at a later point in time.
       *
       * The `$evalAsync` makes no guarantees as to when the `expression` will be executed, only
       * that:
       *
       *   - it will execute after the function that scheduled the evaluation (preferably before DOM
       *     rendering).
       *   - at least one {@link ng.$rootScope.Scope#$digest $digest cycle} will be performed after
       *     `expression` execution.
       *
       * Any exceptions from the execution of the expression are forwarded to the
       * {@link ng.$exceptionHandler $exceptionHandler} service.
       *
       * __Note:__ if this function is called outside of a `$digest` cycle, a new `$digest` cycle
       * will be scheduled. However, it is encouraged to always call code that changes the model
       * from within an `$apply` call. That includes code evaluated via `$evalAsync`.
       *
       * @param {(string|function())=} expression An angular expression to be executed.
       *
       *    - `string`: execute using the rules as defined in {@link guide/expression expression}.
       *    - `function(scope)`: execute the function with the current `scope` parameter.
       *
       * @param {(object)=} locals Local variables object, useful for overriding values in scope.
       */
      $evalAsync: function(expr, locals) {
        // if we are outside of an $digest loop and this is the first time we are scheduling async
        // task also schedule async auto-flush
        if (!$rootScope.$$phase && !asyncQueue.length) {
          $browser.defer(function() {
            if (asyncQueue.length) {
              $rootScope.$digest();
            }
          });
        }

        asyncQueue.push({scope: this, expression: expr, locals: locals});
      },

      $$postDigest: function(fn) {
        postDigestQueue.push(fn);
      },

      /**
       * @ngdoc method
       * @name $rootScope.Scope#$apply
       * @kind function
       *
       * @description
       * `$apply()` is used to execute an expression in angular from outside of the angular
       * framework. (For example from browser DOM events, setTimeout, XHR or third party libraries).
       * Because we are calling into the angular framework we need to perform proper scope life
       * cycle of {@link ng.$exceptionHandler exception handling},
       * {@link ng.$rootScope.Scope#$digest executing watches}.
       *
       * ## Life cycle
       *
       * # Pseudo-Code of `$apply()`
       * ```js
           function $apply(expr) {
             try {
               return $eval(expr);
             } catch (e) {
               $exceptionHandler(e);
             } finally {
               $root.$digest();
             }
           }
       * ```
       *
       *
       * Scope's `$apply()` method transitions through the following stages:
       *
       * 1. The {@link guide/expression expression} is executed using the
       *    {@link ng.$rootScope.Scope#$eval $eval()} method.
       * 2. Any exceptions from the execution of the expression are forwarded to the
       *    {@link ng.$exceptionHandler $exceptionHandler} service.
       * 3. The {@link ng.$rootScope.Scope#$watch watch} listeners are fired immediately after the
       *    expression was executed using the {@link ng.$rootScope.Scope#$digest $digest()} method.
       *
       *
       * @param {(string|function())=} exp An angular expression to be executed.
       *
       *    - `string`: execute using the rules as defined in {@link guide/expression expression}.
       *    - `function(scope)`: execute the function with current `scope` parameter.
       *
       * @returns {*} The result of evaluating the expression.
       */
      $apply: function(expr) {
        try {
          beginPhase('$apply');
          try {
            return this.$eval(expr);
          } finally {
            clearPhase();
          }
        } catch (e) {
          $exceptionHandler(e);
        } finally {
          try {
            $rootScope.$digest();
          } catch (e) {
            $exceptionHandler(e);
            throw e;
          }
        }
      },

      /**
       * @ngdoc method
       * @name $rootScope.Scope#$applyAsync
       * @kind function
       *
       * @description
       * Schedule the invocation of $apply to occur at a later time. The actual time difference
       * varies across browsers, but is typically around ~10 milliseconds.
       *
       * This can be used to queue up multiple expressions which need to be evaluated in the same
       * digest.
       *
       * @param {(string|function())=} exp An angular expression to be executed.
       *
       *    - `string`: execute using the rules as defined in {@link guide/expression expression}.
       *    - `function(scope)`: execute the function with current `scope` parameter.
       */
      $applyAsync: function(expr) {
        var scope = this;
        expr && applyAsyncQueue.push($applyAsyncExpression);
        scheduleApplyAsync();

        function $applyAsyncExpression() {
          scope.$eval(expr);
        }
      },

      /**
       * @ngdoc method
       * @name $rootScope.Scope#$on
       * @kind function
       *
       * @description
       * Listens on events of a given type. See {@link ng.$rootScope.Scope#$emit $emit} for
       * discussion of event life cycle.
       *
       * The event listener function format is: `function(event, args...)`. The `event` object
       * passed into the listener has the following attributes:
       *
       *   - `targetScope` - `{Scope}`: the scope on which the event was `$emit`-ed or
       *     `$broadcast`-ed.
       *   - `currentScope` - `{Scope}`: the scope that is currently handling the event. Once the
       *     event propagates through the scope hierarchy, this property is set to null.
       *   - `name` - `{string}`: name of the event.
       *   - `stopPropagation` - `{function=}`: calling `stopPropagation` function will cancel
       *     further event propagation (available only for events that were `$emit`-ed).
       *   - `preventDefault` - `{function}`: calling `preventDefault` sets `defaultPrevented` flag
       *     to true.
       *   - `defaultPrevented` - `{boolean}`: true if `preventDefault` was called.
       *
       * @param {string} name Event name to listen on.
       * @param {function(event, ...args)} listener Function to call when the event is emitted.
       * @returns {function()} Returns a deregistration function for this listener.
       */
      $on: function(name, listener) {
        var namedListeners = this.$$listeners[name];
        if (!namedListeners) {
          this.$$listeners[name] = namedListeners = [];
        }
        namedListeners.push(listener);

        var current = this;
        do {
          if (!current.$$listenerCount[name]) {
            current.$$listenerCount[name] = 0;
          }
          current.$$listenerCount[name]++;
        } while ((current = current.$parent));

        var self = this;
        return function() {
          var indexOfListener = namedListeners.indexOf(listener);
          if (indexOfListener !== -1) {
            namedListeners[indexOfListener] = null;
            decrementListenerCount(self, 1, name);
          }
        };
      },


      /**
       * @ngdoc method
       * @name $rootScope.Scope#$emit
       * @kind function
       *
       * @description
       * Dispatches an event `name` upwards through the scope hierarchy notifying the
       * registered {@link ng.$rootScope.Scope#$on} listeners.
       *
       * The event life cycle starts at the scope on which `$emit` was called. All
       * {@link ng.$rootScope.Scope#$on listeners} listening for `name` event on this scope get
       * notified. Afterwards, the event traverses upwards toward the root scope and calls all
       * registered listeners along the way. The event will stop propagating if one of the listeners
       * cancels it.
       *
       * Any exception emitted from the {@link ng.$rootScope.Scope#$on listeners} will be passed
       * onto the {@link ng.$exceptionHandler $exceptionHandler} service.
       *
       * @param {string} name Event name to emit.
       * @param {...*} args Optional one or more arguments which will be passed onto the event listeners.
       * @return {Object} Event object (see {@link ng.$rootScope.Scope#$on}).
       */
      $emit: function(name, args) {
        var empty = [],
            namedListeners,
            scope = this,
            stopPropagation = false,
            event = {
              name: name,
              targetScope: scope,
              stopPropagation: function() {stopPropagation = true;},
              preventDefault: function() {
                event.defaultPrevented = true;
              },
              defaultPrevented: false
            },
            listenerArgs = concat([event], arguments, 1),
            i, length;

        do {
          namedListeners = scope.$$listeners[name] || empty;
          event.currentScope = scope;
          for (i = 0, length = namedListeners.length; i < length; i++) {

            // if listeners were deregistered, defragment the array
            if (!namedListeners[i]) {
              namedListeners.splice(i, 1);
              i--;
              length--;
              continue;
            }
            try {
              //allow all listeners attached to the current scope to run
              namedListeners[i].apply(null, listenerArgs);
            } catch (e) {
              $exceptionHandler(e);
            }
          }
          //if any listener on the current scope stops propagation, prevent bubbling
          if (stopPropagation) {
            event.currentScope = null;
            return event;
          }
          //traverse upwards
          scope = scope.$parent;
        } while (scope);

        event.currentScope = null;

        return event;
      },


      /**
       * @ngdoc method
       * @name $rootScope.Scope#$broadcast
       * @kind function
       *
       * @description
       * Dispatches an event `name` downwards to all child scopes (and their children) notifying the
       * registered {@link ng.$rootScope.Scope#$on} listeners.
       *
       * The event life cycle starts at the scope on which `$broadcast` was called. All
       * {@link ng.$rootScope.Scope#$on listeners} listening for `name` event on this scope get
       * notified. Afterwards, the event propagates to all direct and indirect scopes of the current
       * scope and calls all registered listeners along the way. The event cannot be canceled.
       *
       * Any exception emitted from the {@link ng.$rootScope.Scope#$on listeners} will be passed
       * onto the {@link ng.$exceptionHandler $exceptionHandler} service.
       *
       * @param {string} name Event name to broadcast.
       * @param {...*} args Optional one or more arguments which will be passed onto the event listeners.
       * @return {Object} Event object, see {@link ng.$rootScope.Scope#$on}
       */
      $broadcast: function(name, args) {
        var target = this,
            current = target,
            next = target,
            event = {
              name: name,
              targetScope: target,
              preventDefault: function() {
                event.defaultPrevented = true;
              },
              defaultPrevented: false
            };

        if (!target.$$listenerCount[name]) return event;

        var listenerArgs = concat([event], arguments, 1),
            listeners, i, length;

        //down while you can, then up and next sibling or up and next sibling until back at root
        while ((current = next)) {
          event.currentScope = current;
          listeners = current.$$listeners[name] || [];
          for (i = 0, length = listeners.length; i < length; i++) {
            // if listeners were deregistered, defragment the array
            if (!listeners[i]) {
              listeners.splice(i, 1);
              i--;
              length--;
              continue;
            }

            try {
              listeners[i].apply(null, listenerArgs);
            } catch (e) {
              $exceptionHandler(e);
            }
          }

          // Insanity Warning: scope depth-first traversal
          // yes, this code is a bit crazy, but it works and we have tests to prove it!
          // this piece should be kept in sync with the traversal in $digest
          // (though it differs due to having the extra check for $$listenerCount)
          if (!(next = ((current.$$listenerCount[name] && current.$$childHead) ||
              (current !== target && current.$$nextSibling)))) {
            while (current !== target && !(next = current.$$nextSibling)) {
              current = current.$parent;
            }
          }
        }

        event.currentScope = null;
        return event;
      }
    };

    var $rootScope = new Scope();

    //The internal queues. Expose them on the $rootScope for debugging/testing purposes.
    var asyncQueue = $rootScope.$$asyncQueue = [];
    var postDigestQueue = $rootScope.$$postDigestQueue = [];
    var applyAsyncQueue = $rootScope.$$applyAsyncQueue = [];

    return $rootScope;


    function beginPhase(phase) {
      if ($rootScope.$$phase) {
        throw $rootScopeMinErr('inprog', '{0} already in progress', $rootScope.$$phase);
      }

      $rootScope.$$phase = phase;
    }

    function clearPhase() {
      $rootScope.$$phase = null;
    }

    function incrementWatchersCount(current, count) {
      do {
        current.$$watchersCount += count;
      } while ((current = current.$parent));
    }

    function decrementListenerCount(current, count, name) {
      do {
        current.$$listenerCount[name] -= count;

        if (current.$$listenerCount[name] === 0) {
          delete current.$$listenerCount[name];
        }
      } while ((current = current.$parent));
    }

    /**
     * function used as an initial value for watchers.
     * because it's unique we can easily tell it apart from other values
     */
    function initWatchVal() {}

    function flushApplyAsync() {
      while (applyAsyncQueue.length) {
        try {
          applyAsyncQueue.shift()();
        } catch (e) {
          $exceptionHandler(e);
        }
      }
      applyAsyncId = null;
    }

    function scheduleApplyAsync() {
      if (applyAsyncId === null) {
        applyAsyncId = $browser.defer(function() {
          $rootScope.$apply(flushApplyAsync);
        });
      }
    }
  }];
}

/**
 * @description
 * Private service to sanitize uris for links and images. Used by $compile and $sanitize.
 */
function $$SanitizeUriProvider() {
  var aHrefSanitizationWhitelist = /^\s*(https?|ftp|mailto|tel|file):/,
    imgSrcSanitizationWhitelist = /^\s*((https?|ftp|file|blob):|data:image\/)/;

  /**
   * @description
   * Retrieves or overrides the default regular expression that is used for whitelisting of safe
   * urls during a[href] sanitization.
   *
   * The sanitization is a security measure aimed at prevent XSS attacks via html links.
   *
   * Any url about to be assigned to a[href] via data-binding is first normalized and turned into
   * an absolute url. Afterwards, the url is matched against the `aHrefSanitizationWhitelist`
   * regular expression. If a match is found, the original url is written into the dom. Otherwise,
   * the absolute url is prefixed with `'unsafe:'` string and only then is it written into the DOM.
   *
   * @param {RegExp=} regexp New regexp to whitelist urls with.
   * @returns {RegExp|ng.$compileProvider} Current RegExp if called without value or self for
   *    chaining otherwise.
   */
  this.aHrefSanitizationWhitelist = function(regexp) {
    if (isDefined(regexp)) {
      aHrefSanitizationWhitelist = regexp;
      return this;
    }
    return aHrefSanitizationWhitelist;
  };


  /**
   * @description
   * Retrieves or overrides the default regular expression that is used for whitelisting of safe
   * urls during img[src] sanitization.
   *
   * The sanitization is a security measure aimed at prevent XSS attacks via html links.
   *
   * Any url about to be assigned to img[src] via data-binding is first normalized and turned into
   * an absolute url. Afterwards, the url is matched against the `imgSrcSanitizationWhitelist`
   * regular expression. If a match is found, the original url is written into the dom. Otherwise,
   * the absolute url is prefixed with `'unsafe:'` string and only then is it written into the DOM.
   *
   * @param {RegExp=} regexp New regexp to whitelist urls with.
   * @returns {RegExp|ng.$compileProvider} Current RegExp if called without value or self for
   *    chaining otherwise.
   */
  this.imgSrcSanitizationWhitelist = function(regexp) {
    if (isDefined(regexp)) {
      imgSrcSanitizationWhitelist = regexp;
      return this;
    }
    return imgSrcSanitizationWhitelist;
  };

  this.$get = function() {
    return function sanitizeUri(uri, isImage) {
      var regex = isImage ? imgSrcSanitizationWhitelist : aHrefSanitizationWhitelist;
      var normalizedVal;
      normalizedVal = urlResolve(uri).href;
      if (normalizedVal !== '' && !normalizedVal.match(regex)) {
        return 'unsafe:' + normalizedVal;
      }
      return uri;
    };
  };
}

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
 *     Any commits to this file should be reviewed with security in mind.  *
 *   Changes to this file can potentially create security vulnerabilities. *
 *          An approval from 2 Core members with history of modifying      *
 *                         this file is required.                          *
 *                                                                         *
 *  Does the change somehow allow for arbitrary javascript to be executed? *
 *    Or allows for someone to change the prototype of built-in objects?   *
 *     Or gives undesired access to variables likes document or window?    *
 * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */

var $sceMinErr = minErr('$sce');

var SCE_CONTEXTS = {
  HTML: 'html',
  CSS: 'css',
  URL: 'url',
  // RESOURCE_URL is a subtype of URL used in contexts where a privileged resource is sourced from a
  // url.  (e.g. ng-include, script src, templateUrl)
  RESOURCE_URL: 'resourceUrl',
  JS: 'js'
};

// Helper functions follow.

function adjustMatcher(matcher) {
  if (matcher === 'self') {
    return matcher;
  } else if (isString(matcher)) {
    // Strings match exactly except for 2 wildcards - '*' and '**'.
    // '*' matches any character except those from the set ':/.?&'.
    // '**' matches any character (like .* in a RegExp).
    // More than 2 *'s raises an error as it's ill defined.
    if (matcher.indexOf('***') > -1) {
      throw $sceMinErr('iwcard',
          'Illegal sequence *** in string matcher.  String: {0}', matcher);
    }
    matcher = escapeForRegexp(matcher).
                  replace('\\*\\*', '.*').
                  replace('\\*', '[^:/.?&;]*');
    return new RegExp('^' + matcher + '$');
  } else if (isRegExp(matcher)) {
    // The only other type of matcher allowed is a Regexp.
    // Match entire URL / disallow partial matches.
    // Flags are reset (i.e. no global, ignoreCase or multiline)
    return new RegExp('^' + matcher.source + '$');
  } else {
    throw $sceMinErr('imatcher',
        'Matchers may only be "self", string patterns or RegExp objects');
  }
}


function adjustMatchers(matchers) {
  var adjustedMatchers = [];
  if (isDefined(matchers)) {
    forEach(matchers, function(matcher) {
      adjustedMatchers.push(adjustMatcher(matcher));
    });
  }
  return adjustedMatchers;
}


/**
 * @ngdoc service
 * @name $sceDelegate
 * @kind function
 *
 * @description
 *
 * `$sceDelegate` is a service that is used by the `$sce` service to provide {@link ng.$sce Strict
 * Contextual Escaping (SCE)} services to AngularJS.
 *
 * Typically, you would configure or override the {@link ng.$sceDelegate $sceDelegate} instead of
 * the `$sce` service to customize the way Strict Contextual Escaping works in AngularJS.  This is
 * because, while the `$sce` provides numerous shorthand methods, etc., you really only need to
 * override 3 core functions (`trustAs`, `getTrusted` and `valueOf`) to replace the way things
 * work because `$sce` delegates to `$sceDelegate` for these operations.
 *
 * Refer {@link ng.$sceDelegateProvider $sceDelegateProvider} to configure this service.
 *
 * The default instance of `$sceDelegate` should work out of the box with little pain.  While you
 * can override it completely to change the behavior of `$sce`, the common case would
 * involve configuring the {@link ng.$sceDelegateProvider $sceDelegateProvider} instead by setting
 * your own whitelists and blacklists for trusting URLs used for loading AngularJS resources such as
 * templates.  Refer {@link ng.$sceDelegateProvider#resourceUrlWhitelist
 * $sceDelegateProvider.resourceUrlWhitelist} and {@link
 * ng.$sceDelegateProvider#resourceUrlBlacklist $sceDelegateProvider.resourceUrlBlacklist}
 */

/**
 * @ngdoc provider
 * @name $sceDelegateProvider
 * @description
 *
 * The `$sceDelegateProvider` provider allows developers to configure the {@link ng.$sceDelegate
 * $sceDelegate} service.  This allows one to get/set the whitelists and blacklists used to ensure
 * that the URLs used for sourcing Angular templates are safe.  Refer {@link
 * ng.$sceDelegateProvider#resourceUrlWhitelist $sceDelegateProvider.resourceUrlWhitelist} and
 * {@link ng.$sceDelegateProvider#resourceUrlBlacklist $sceDelegateProvider.resourceUrlBlacklist}
 *
 * For the general details about this service in Angular, read the main page for {@link ng.$sce
 * Strict Contextual Escaping (SCE)}.
 *
 * **Example**:  Consider the following case. <a name="example"></a>
 *
 * - your app is hosted at url `http://myapp.example.com/`
 * - but some of your templates are hosted on other domains you control such as
 *   `http://srv01.assets.example.com/`,Â  `http://srv02.assets.example.com/`, etc.
 * - and you have an open redirect at `http://myapp.example.com/clickThru?...`.
 *
 * Here is what a secure configuration for this scenario might look like:
 *
 * ```
 *  angular.module('myApp', []).config(function($sceDelegateProvider) {
 *    $sceDelegateProvider.resourceUrlWhitelist([
 *      // Allow same origin resource loads.
 *      'self',
 *      // Allow loading from our assets domain.  Notice the difference between * and **.
 *      'http://srv*.assets.example.com/**'
 *    ]);
 *
 *    // The blacklist overrides the whitelist so the open redirect here is blocked.
 *    $sceDelegateProvider.resourceUrlBlacklist([
 *      'http://myapp.example.com/clickThru**'
 *    ]);
 *  });
 * ```
 */

function $SceDelegateProvider() {
  this.SCE_CONTEXTS = SCE_CONTEXTS;

  // Resource URLs can also be trusted by policy.
  var resourceUrlWhitelist = ['self'],
      resourceUrlBlacklist = [];

  /**
   * @ngdoc method
   * @name $sceDelegateProvider#resourceUrlWhitelist
   * @kind function
   *
   * @param {Array=} whitelist When provided, replaces the resourceUrlWhitelist with the value
   *     provided.  This must be an array or null.  A snapshot of this array is used so further
   *     changes to the array are ignored.
   *
   *     Follow {@link ng.$sce#resourceUrlPatternItem this link} for a description of the items
   *     allowed in this array.
   *
   *     Note: **an empty whitelist array will block all URLs**!
   *
   * @return {Array} the currently set whitelist array.
   *
   * The **default value** when no whitelist has been explicitly set is `['self']` allowing only
   * same origin resource requests.
   *
   * @description
   * Sets/Gets the whitelist of trusted resource URLs.
   */
  this.resourceUrlWhitelist = function(value) {
    if (arguments.length) {
      resourceUrlWhitelist = adjustMatchers(value);
    }
    return resourceUrlWhitelist;
  };

  /**
   * @ngdoc method
   * @name $sceDelegateProvider#resourceUrlBlacklist
   * @kind function
   *
   * @param {Array=} blacklist When provided, replaces the resourceUrlBlacklist with the value
   *     provided.  This must be an array or null.  A snapshot of this array is used so further
   *     changes to the array are ignored.
   *
   *     Follow {@link ng.$sce#resourceUrlPatternItem this link} for a description of the items
   *     allowed in this array.
   *
   *     The typical usage for the blacklist is to **block
   *     [open redirects](http://cwe.mitre.org/data/definitions/601.html)** served by your domain as
   *     these would otherwise be trusted but actually return content from the redirected domain.
   *
   *     Finally, **the blacklist overrides the whitelist** and has the final say.
   *
   * @return {Array} the currently set blacklist array.
   *
   * The **default value** when no whitelist has been explicitly set is the empty array (i.e. there
   * is no blacklist.)
   *
   * @description
   * Sets/Gets the blacklist of trusted resource URLs.
   */

  this.resourceUrlBlacklist = function(value) {
    if (arguments.length) {
      resourceUrlBlacklist = adjustMatchers(value);
    }
    return resourceUrlBlacklist;
  };

  this.$get = ['$injector', function($injector) {

    var htmlSanitizer = function htmlSanitizer(html) {
      throw $sceMinErr('unsafe', 'Attempting to use an unsafe value in a safe context.');
    };

    if ($injector.has('$sanitize')) {
      htmlSanitizer = $injector.get('$sanitize');
    }


    function matchUrl(matcher, parsedUrl) {
      if (matcher === 'self') {
        return urlIsSameOrigin(parsedUrl);
      } else {
        // definitely a regex.  See adjustMatchers()
        return !!matcher.exec(parsedUrl.href);
      }
    }

    function isResourceUrlAllowedByPolicy(url) {
      var parsedUrl = urlResolve(url.toString());
      var i, n, allowed = false;
      // Ensure that at least one item from the whitelist allows this url.
      for (i = 0, n = resourceUrlWhitelist.length; i < n; i++) {
        if (matchUrl(resourceUrlWhitelist[i], parsedUrl)) {
          allowed = true;
          break;
        }
      }
      if (allowed) {
        // Ensure that no item from the blacklist blocked this url.
        for (i = 0, n = resourceUrlBlacklist.length; i < n; i++) {
          if (matchUrl(resourceUrlBlacklist[i], parsedUrl)) {
            allowed = false;
            break;
          }
        }
      }
      return allowed;
    }

    function generateHolderType(Base) {
      var holderType = function TrustedValueHolderType(trustedValue) {
        this.$$unwrapTrustedValue = function() {
          return trustedValue;
        };
      };
      if (Base) {
        holderType.prototype = new Base();
      }
      holderType.prototype.valueOf = function sceValueOf() {
        return this.$$unwrapTrustedValue();
      };
      holderType.prototype.toString = function sceToString() {
        return this.$$unwrapTrustedValue().toString();
      };
      return holderType;
    }

    var trustedValueHolderBase = generateHolderType(),
        byType = {};

    byType[SCE_CONTEXTS.HTML] = generateHolderType(trustedValueHolderBase);
    byType[SCE_CONTEXTS.CSS] = generateHolderType(trustedValueHolderBase);
    byType[SCE_CONTEXTS.URL] = generateHolderType(trustedValueHolderBase);
    byType[SCE_CONTEXTS.JS] = generateHolderType(trustedValueHolderBase);
    byType[SCE_CONTEXTS.RESOURCE_URL] = generateHolderType(byType[SCE_CONTEXTS.URL]);

    /**
     * @ngdoc method
     * @name $sceDelegate#trustAs
     *
     * @description
     * Returns an object that is trusted by angular for use in specified strict
     * contextual escaping contexts (such as ng-bind-html, ng-include, any src
     * attribute interpolation, any dom event binding attribute interpolation
     * such as for onclick,  etc.) that uses the provided value.
     * See {@link ng.$sce $sce} for enabling strict contextual escaping.
     *
     * @param {string} type The kind of context in which this value is safe for use.  e.g. url,
     *   resourceUrl, html, js and css.
     * @param {*} value The value that that should be considered trusted/safe.
     * @returns {*} A value that can be used to stand in for the provided `value` in places
     * where Angular expects a $sce.trustAs() return value.
     */
    function trustAs(type, trustedValue) {
      var Constructor = (byType.hasOwnProperty(type) ? byType[type] : null);
      if (!Constructor) {
        throw $sceMinErr('icontext',
            'Attempted to trust a value in invalid context. Context: {0}; Value: {1}',
            type, trustedValue);
      }
      if (trustedValue === null || isUndefined(trustedValue) || trustedValue === '') {
        return trustedValue;
      }
      // All the current contexts in SCE_CONTEXTS happen to be strings.  In order to avoid trusting
      // mutable objects, we ensure here that the value passed in is actually a string.
      if (typeof trustedValue !== 'string') {
        throw $sceMinErr('itype',
            'Attempted to trust a non-string value in a content requiring a string: Context: {0}',
            type);
      }
      return new Constructor(trustedValue);
    }

    /**
     * @ngdoc method
     * @name $sceDelegate#valueOf
     *
     * @description
     * If the passed parameter had been returned by a prior call to {@link ng.$sceDelegate#trustAs
     * `$sceDelegate.trustAs`}, returns the value that had been passed to {@link
     * ng.$sceDelegate#trustAs `$sceDelegate.trustAs`}.
     *
     * If the passed parameter is not a value that had been returned by {@link
     * ng.$sceDelegate#trustAs `$sceDelegate.trustAs`}, returns it as-is.
     *
     * @param {*} value The result of a prior {@link ng.$sceDelegate#trustAs `$sceDelegate.trustAs`}
     *      call or anything else.
     * @returns {*} The `value` that was originally provided to {@link ng.$sceDelegate#trustAs
     *     `$sceDelegate.trustAs`} if `value` is the result of such a call.  Otherwise, returns
     *     `value` unchanged.
     */
    function valueOf(maybeTrusted) {
      if (maybeTrusted instanceof trustedValueHolderBase) {
        return maybeTrusted.$$unwrapTrustedValue();
      } else {
        return maybeTrusted;
      }
    }

    /**
     * @ngdoc method
     * @name $sceDelegate#getTrusted
     *
     * @description
     * Takes the result of a {@link ng.$sceDelegate#trustAs `$sceDelegate.trustAs`} call and
     * returns the originally supplied value if the queried context type is a supertype of the
     * created type.  If this condition isn't satisfied, throws an exception.
     *
     * @param {string} type The kind of context in which this value is to be used.
     * @param {*} maybeTrusted The result of a prior {@link ng.$sceDelegate#trustAs
     *     `$sceDelegate.trustAs`} call.
     * @returns {*} The value the was originally provided to {@link ng.$sceDelegate#trustAs
     *     `$sceDelegate.trustAs`} if valid in this context.  Otherwise, throws an exception.
     */
    function getTrusted(type, maybeTrusted) {
      if (maybeTrusted === null || isUndefined(maybeTrusted) || maybeTrusted === '') {
        return maybeTrusted;
      }
      var constructor = (byType.hasOwnProperty(type) ? byType[type] : null);
      if (constructor && maybeTrusted instanceof constructor) {
        return maybeTrusted.$$unwrapTrustedValue();
      }
      // If we get here, then we may only take one of two actions.
      // 1. sanitize the value for the requested type, or
      // 2. throw an exception.
      if (type === SCE_CONTEXTS.RESOURCE_URL) {
        if (isResourceUrlAllowedByPolicy(maybeTrusted)) {
          return maybeTrusted;
        } else {
          throw $sceMinErr('insecurl',
              'Blocked loading resource from url not allowed by $sceDelegate policy.  URL: {0}',
              maybeTrusted.toString());
        }
      } else if (type === SCE_CONTEXTS.HTML) {
        return htmlSanitizer(maybeTrusted);
      }
      throw $sceMinErr('unsafe', 'Attempting to use an unsafe value in a safe context.');
    }

    return { trustAs: trustAs,
             getTrusted: getTrusted,
             valueOf: valueOf };
  }];
}


/**
 * @ngdoc provider
 * @name $sceProvider
 * @description
 *
 * The $sceProvider provider allows developers to configure the {@link ng.$sce $sce} service.
 * -   enable/disable Strict Contextual Escaping (SCE) in a module
 * -   override the default implementation with a custom delegate
 *
 * Read more about {@link ng.$sce Strict Contextual Escaping (SCE)}.
 */

/* jshint maxlen: false*/

/**
 * @ngdoc service
 * @name $sce
 * @kind function
 *
 * @description
 *
 * `$sce` is a service that provides Strict Contextual Escaping services to AngularJS.
 *
 * # Strict Contextual Escaping
 *
 * Strict Contextual Escaping (SCE) is a mode in which AngularJS requires bindings in certain
 * contexts to result in a value that is marked as safe to use for that context.  One example of
 * such a context is binding arbitrary html controlled by the user via `ng-bind-html`.  We refer
 * to these contexts as privileged or SCE contexts.
 *
 * As of version 1.2, Angular ships with SCE enabled by default.
 *
 * Note:  When enabled (the default), IE<11 in quirks mode is not supported.  In this mode, IE<11 allow
 * one to execute arbitrary javascript by the use of the expression() syntax.  Refer
 * <http://blogs.msdn.com/b/ie/archive/2008/10/16/ending-expressions.aspx> to learn more about them.
 * You can ensure your document is in standards mode and not quirks mode by adding `<!doctype html>`
 * to the top of your HTML document.
 *
 * SCE assists in writing code in way that (a) is secure by default and (b) makes auditing for
 * security vulnerabilities such as XSS, clickjacking, etc. a lot easier.
 *
 * Here's an example of a binding in a privileged context:
 *
 * ```
 * <input ng-model="userHtml" aria-label="User input">
 * <div ng-bind-html="userHtml"></div>
 * ```
 *
 * Notice that `ng-bind-html` is bound to `userHtml` controlled by the user.  With SCE
 * disabled, this application allows the user to render arbitrary HTML into the DIV.
 * In a more realistic example, one may be rendering user comments, blog articles, etc. via
 * bindings.  (HTML is just one example of a context where rendering user controlled input creates
 * security vulnerabilities.)
 *
 * For the case of HTML, you might use a library, either on the client side, or on the server side,
 * to sanitize unsafe HTML before binding to the value and rendering it in the document.
 *
 * How would you ensure that every place that used these types of bindings was bound to a value that
 * was sanitized by your library (or returned as safe for rendering by your server?)  How can you
 * ensure that you didn't accidentally delete the line that sanitized the value, or renamed some
 * properties/fields and forgot to update the binding to the sanitized value?
 *
 * To be secure by default, you want to ensure that any such bindings are disallowed unless you can
 * determine that something explicitly says it's safe to use a value for binding in that
 * context.  You can then audit your code (a simple grep would do) to ensure that this is only done
 * for those values that you can easily tell are safe - because they were received from your server,
 * sanitized by your library, etc.  You can organize your codebase to help with this - perhaps
 * allowing only the files in a specific directory to do this.  Ensuring that the internal API
 * exposed by that code doesn't markup arbitrary values as safe then becomes a more manageable task.
 *
 * In the case of AngularJS' SCE service, one uses {@link ng.$sce#trustAs $sce.trustAs}
 * (and shorthand methods such as {@link ng.$sce#trustAsHtml $sce.trustAsHtml}, etc.) to
 * obtain values that will be accepted by SCE / privileged contexts.
 *
 *
 * ## How does it work?
 *
 * In privileged contexts, directives and code will bind to the result of {@link ng.$sce#getTrusted
 * $sce.getTrusted(context, value)} rather than to the value directly.  Directives use {@link
 * ng.$sce#parseAs $sce.parseAs} rather than `$parse` to watch attribute bindings, which performs the
 * {@link ng.$sce#getTrusted $sce.getTrusted} behind the scenes on non-constant literals.
 *
 * As an example, {@link ng.directive:ngBindHtml ngBindHtml} uses {@link
 * ng.$sce#parseAsHtml $sce.parseAsHtml(binding expression)}.  Here's the actual code (slightly
 * simplified):
 *
 * ```
 * var ngBindHtmlDirective = ['$sce', function($sce) {
 *   return function(scope, element, attr) {
 *     scope.$watch($sce.parseAsHtml(attr.ngBindHtml), function(value) {
 *       element.html(value || '');
 *     });
 *   };
 * }];
 * ```
 *
 * ## Impact on loading templates
 *
 * This applies both to the {@link ng.directive:ngInclude `ng-include`} directive as well as
 * `templateUrl`'s specified by {@link guide/directive directives}.
 *
 * By default, Angular only loads templates from the same domain and protocol as the application
 * document.  This is done by calling {@link ng.$sce#getTrustedResourceUrl
 * $sce.getTrustedResourceUrl} on the template URL.  To load templates from other domains and/or
 * protocols, you may either {@link ng.$sceDelegateProvider#resourceUrlWhitelist whitelist
 * them} or {@link ng.$sce#trustAsResourceUrl wrap it} into a trusted value.
 *
 * *Please note*:
 * The browser's
 * [Same Origin Policy](https://code.google.com/p/browsersec/wiki/Part2#Same-origin_policy_for_XMLHttpRequest)
 * and [Cross-Origin Resource Sharing (CORS)](http://www.w3.org/TR/cors/)
 * policy apply in addition to this and may further restrict whether the template is successfully
 * loaded.  This means that without the right CORS policy, loading templates from a different domain
 * won't work on all browsers.  Also, loading templates from `file://` URL does not work on some
 * browsers.
 *
 * ## This feels like too much overhead
 *
 * It's important to remember that SCE only applies to interpolation expressions.
 *
 * If your expressions are constant literals, they're automatically trusted and you don't need to
 * call `$sce.trustAs` on them (remember to include the `ngSanitize` module) (e.g.
 * `<div ng-bind-html="'<b>implicitly trusted</b>'"></div>`) just works.
 *
 * Additionally, `a[href]` and `img[src]` automatically sanitize their URLs and do not pass them
 * through {@link ng.$sce#getTrusted $sce.getTrusted}.  SCE doesn't play a role here.
 *
 * The included {@link ng.$sceDelegate $sceDelegate} comes with sane defaults to allow you to load
 * templates in `ng-include` from your application's domain without having to even know about SCE.
 * It blocks loading templates from other domains or loading templates over http from an https
 * served document.  You can change these by setting your own custom {@link
 * ng.$sceDelegateProvider#resourceUrlWhitelist whitelists} and {@link
 * ng.$sceDelegateProvider#resourceUrlBlacklist blacklists} for matching such URLs.
 *
 * This significantly reduces the overhead.  It is far easier to pay the small overhead and have an
 * application that's secure and can be audited to verify that with much more ease than bolting
 * security onto an application later.
 *
 * <a name="contexts"></a>
 * ## What trusted context types are supported?
 *
 * | Context             | Notes          |
 * |---------------------|----------------|
 * | `$sce.HTML`         | For HTML that's safe to source into the application.  The {@link ng.directive:ngBindHtml ngBindHtml} directive uses this context for bindings. If an unsafe value is encountered and the {@link ngSanitize $sanitize} module is present this will sanitize the value instead of throwing an error. |
 * | `$sce.CSS`          | For CSS that's safe to source into the application.  Currently unused.  Feel free to use it in your own directives. |
 * | `$sce.URL`          | For URLs that are safe to follow as links.  Currently unused (`<a href=` and `<img src=` sanitize their urls and don't constitute an SCE context. |
 * | `$sce.RESOURCE_URL` | For URLs that are not only safe to follow as links, but whose contents are also safe to include in your application.  Examples include `ng-include`, `src` / `ngSrc` bindings for tags other than `IMG` (e.g. `IFRAME`, `OBJECT`, etc.)  <br><br>Note that `$sce.RESOURCE_URL` makes a stronger statement about the URL than `$sce.URL` does and therefore contexts requiring values trusted for `$sce.RESOURCE_URL` can be used anywhere that values trusted for `$sce.URL` are required. |
 * | `$sce.JS`           | For JavaScript that is safe to execute in your application's context.  Currently unused.  Feel free to use it in your own directives. |
 *
 * ## Format of items in {@link ng.$sceDelegateProvider#resourceUrlWhitelist resourceUrlWhitelist}/{@link ng.$sceDelegateProvider#resourceUrlBlacklist Blacklist} <a name="resourceUrlPatternItem"></a>
 *
 *  Each element in these arrays must be one of the following:
 *
 *  - **'self'**
 *    - The special **string**, `'self'`, can be used to match against all URLs of the **same
 *      domain** as the application document using the **same protocol**.
 *  - **String** (except the special value `'self'`)
 *    - The string is matched against the full *normalized / absolute URL* of the resource
 *      being tested (substring matches are not good enough.)
 *    - There are exactly **two wildcard sequences** - `*` and `**`.  All other characters
 *      match themselves.
 *    - `*`: matches zero or more occurrences of any character other than one of the following 6
 *      characters: '`:`', '`/`', '`.`', '`?`', '`&`' and '`;`'.  It's a useful wildcard for use
 *      in a whitelist.
 *    - `**`: matches zero or more occurrences of *any* character.  As such, it's not
 *      appropriate for use in a scheme, domain, etc. as it would match too much.  (e.g.
 *      http://**.example.com/ would match http://evil.com/?ignore=.example.com/ and that might
 *      not have been the intention.)  Its usage at the very end of the path is ok.  (e.g.
 *      http://foo.example.com/templates/**).
 *  - **RegExp** (*see caveat below*)
 *    - *Caveat*:  While regular expressions are powerful and offer great flexibility,  their syntax
 *      (and all the inevitable escaping) makes them *harder to maintain*.  It's easy to
 *      accidentally introduce a bug when one updates a complex expression (imho, all regexes should
 *      have good test coverage).  For instance, the use of `.` in the regex is correct only in a
 *      small number of cases.  A `.` character in the regex used when matching the scheme or a
 *      subdomain could be matched against a `:` or literal `.` that was likely not intended.   It
 *      is highly recommended to use the string patterns and only fall back to regular expressions
 *      as a last resort.
 *    - The regular expression must be an instance of RegExp (i.e. not a string.)  It is
 *      matched against the **entire** *normalized / absolute URL* of the resource being tested
 *      (even when the RegExp did not have the `^` and `$` codes.)  In addition, any flags
 *      present on the RegExp (such as multiline, global, ignoreCase) are ignored.
 *    - If you are generating your JavaScript from some other templating engine (not
 *      recommended, e.g. in issue [#4006](https://github.com/angular/angular.js/issues/4006)),
 *      remember to escape your regular expression (and be aware that you might need more than
 *      one level of escaping depending on your templating engine and the way you interpolated
 *      the value.)  Do make use of your platform's escaping mechanism as it might be good
 *      enough before coding your own.  E.g. Ruby has
 *      [Regexp.escape(str)](http://www.ruby-doc.org/core-2.0.0/Regexp.html#method-c-escape)
 *      and Python has [re.escape](http://docs.python.org/library/re.html#re.escape).
 *      Javascript lacks a similar built in function for escaping.  Take a look at Google
 *      Closure library's [goog.string.regExpEscape(s)](
 *      http://docs.closure-library.googlecode.com/git/closure_goog_string_string.js.source.html#line962).
 *
 * Refer {@link ng.$sceDelegateProvider $sceDelegateProvider} for an example.
 *
 * ## Show me an example using SCE.
 *
 * <example module="mySceApp" deps="angular-sanitize.js">
 * <file name="index.html">
 *   <div ng-controller="AppController as myCtrl">
 *     <i ng-bind-html="myCtrl.explicitlyTrustedHtml" id="explicitlyTrustedHtml"></i><br><br>
 *     <b>User comments</b><br>
 *     By default, HTML that isn't explicitly trusted (e.g. Alice's comment) is sanitized when
 *     $sanitize is available.  If $sanitize isn't available, this results in an error instead of an
 *     exploit.
 *     <div class="well">
 *       <div ng-repeat="userComment in myCtrl.userComments">
 *         <b>{{userComment.name}}</b>:
 *         <span ng-bind-html="userComment.htmlComment" class="htmlComment"></span>
 *         <br>
 *       </div>
 *     </div>
 *   </div>
 * </file>
 *
 * <file name="script.js">
 *   angular.module('mySceApp', ['ngSanitize'])
 *     .controller('AppController', ['$http', '$templateCache', '$sce',
 *       function($http, $templateCache, $sce) {
 *         var self = this;
 *         $http.get("test_data.json", {cache: $templateCache}).success(function(userComments) {
 *           self.userComments = userComments;
 *         });
 *         self.explicitlyTrustedHtml = $sce.trustAsHtml(
 *             '<span onmouseover="this.textContent=&quot;Explicitly trusted HTML bypasses ' +
 *             'sanitization.&quot;">Hover over this text.</span>');
 *       }]);
 * </file>
 *
 * <file name="test_data.json">
 * [
 *   { "name": "Alice",
 *     "htmlComment":
 *         "<span onmouseover='this.textContent=\"PWN3D!\"'>Is <i>anyone</i> reading this?</span>"
 *   },
 *   { "name": "Bob",
 *     "htmlComment": "<i>Yes!</i>  Am I the only other one?"
 *   }
 * ]
 * </file>
 *
 * <file name="protractor.js" type="protractor">
 *   describe('SCE doc demo', function() {
 *     it('should sanitize untrusted values', function() {
 *       expect(element.all(by.css('.htmlComment')).first().getInnerHtml())
 *           .toBe('<span>Is <i>anyone</i> reading this?</span>');
 *     });
 *
 *     it('should NOT sanitize explicitly trusted values', function() {
 *       expect(element(by.id('explicitlyTrustedHtml')).getInnerHtml()).toBe(
 *           '<span onmouseover="this.textContent=&quot;Explicitly trusted HTML bypasses ' +
 *           'sanitization.&quot;">Hover over this text.</span>');
 *     });
 *   });
 * </file>
 * </example>
 *
 *
 *
 * ## Can I disable SCE completely?
 *
 * Yes, you can.  However, this is strongly discouraged.  SCE gives you a lot of security benefits
 * for little coding overhead.  It will be much harder to take an SCE disabled application and
 * either secure it on your own or enable SCE at a later stage.  It might make sense to disable SCE
 * for cases where you have a lot of existing code that was written before SCE was introduced and
 * you're migrating them a module at a time.
 *
 * That said, here's how you can completely disable SCE:
 *
 * ```
 * angular.module('myAppWithSceDisabledmyApp', []).config(function($sceProvider) {
 *   // Completely disable SCE.  For demonstration purposes only!
 *   // Do not use in new projects.
 *   $sceProvider.enabled(false);
 * });
 * ```
 *
 */
/* jshint maxlen: 100 */

function $SceProvider() {
  var enabled = true;

  /**
   * @ngdoc method
   * @name $sceProvider#enabled
   * @kind function
   *
   * @param {boolean=} value If provided, then enables/disables SCE.
   * @return {boolean} true if SCE is enabled, false otherwise.
   *
   * @description
   * Enables/disables SCE and returns the current value.
   */
  this.enabled = function(value) {
    if (arguments.length) {
      enabled = !!value;
    }
    return enabled;
  };


  /* Design notes on the default implementation for SCE.
   *
   * The API contract for the SCE delegate
   * -------------------------------------
   * The SCE delegate object must provide the following 3 methods:
   *
   * - trustAs(contextEnum, value)
   *     This method is used to tell the SCE service that the provided value is OK to use in the
   *     contexts specified by contextEnum.  It must return an object that will be accepted by
   *     getTrusted() for a compatible contextEnum and return this value.
   *
   * - valueOf(value)
   *     For values that were not produced by trustAs(), return them as is.  For values that were
   *     produced by trustAs(), return the corresponding input value to trustAs.  Basically, if
   *     trustAs is wrapping the given values into some type, this operation unwraps it when given
   *     such a value.
   *
   * - getTrusted(contextEnum, value)
   *     This function should return the a value that is safe to use in the context specified by
   *     contextEnum or throw and exception otherwise.
   *
   * NOTE: This contract deliberately does NOT state that values returned by trustAs() must be
   * opaque or wrapped in some holder object.  That happens to be an implementation detail.  For
   * instance, an implementation could maintain a registry of all trusted objects by context.  In
   * such a case, trustAs() would return the same object that was passed in.  getTrusted() would
   * return the same object passed in if it was found in the registry under a compatible context or
   * throw an exception otherwise.  An implementation might only wrap values some of the time based
   * on some criteria.  getTrusted() might return a value and not throw an exception for special
   * constants or objects even if not wrapped.  All such implementations fulfill this contract.
   *
   *
   * A note on the inheritance model for SCE contexts
   * ------------------------------------------------
   * I've used inheritance and made RESOURCE_URL wrapped types a subtype of URL wrapped types.  This
   * is purely an implementation details.
   *
   * The contract is simply this:
   *
   *     getTrusted($sce.RESOURCE_URL, value) succeeding implies that getTrusted($sce.URL, value)
   *     will also succeed.
   *
   * Inheritance happens to capture this in a natural way.  In some future, we
   * may not use inheritance anymore.  That is OK because no code outside of
   * sce.js and sceSpecs.js would need to be aware of this detail.
   */

  this.$get = ['$parse', '$sceDelegate', function(
                $parse,   $sceDelegate) {
    // Prereq: Ensure that we're not running in IE<11 quirks mode.  In that mode, IE < 11 allow
    // the "expression(javascript expression)" syntax which is insecure.
    if (enabled && msie < 8) {
      throw $sceMinErr('iequirks',
        'Strict Contextual Escaping does not support Internet Explorer version < 11 in quirks ' +
        'mode.  You can fix this by adding the text <!doctype html> to the top of your HTML ' +
        'document.  See http://docs.angularjs.org/api/ng.$sce for more information.');
    }

    var sce = shallowCopy(SCE_CONTEXTS);

    /**
     * @ngdoc method
     * @name $sce#isEnabled
     * @kind function
     *
     * @return {Boolean} true if SCE is enabled, false otherwise.  If you want to set the value, you
     * have to do it at module config time on {@link ng.$sceProvider $sceProvider}.
     *
     * @description
     * Returns a boolean indicating if SCE is enabled.
     */
    sce.isEnabled = function() {
      return enabled;
    };
    sce.trustAs = $sceDelegate.trustAs;
    sce.getTrusted = $sceDelegate.getTrusted;
    sce.valueOf = $sceDelegate.valueOf;

    if (!enabled) {
      sce.trustAs = sce.getTrusted = function(type, value) { return value; };
      sce.valueOf = identity;
    }

    /**
     * @ngdoc method
     * @name $sce#parseAs
     *
     * @description
     * Converts Angular {@link guide/expression expression} into a function.  This is like {@link
     * ng.$parse $parse} and is identical when the expression is a literal constant.  Otherwise, it
     * wraps the expression in a call to {@link ng.$sce#getTrusted $sce.getTrusted(*type*,
     * *result*)}
     *
     * @param {string} type The kind of SCE context in which this result will be used.
     * @param {string} expression String expression to compile.
     * @returns {function(context, locals)} a function which represents the compiled expression:
     *
     *    * `context` â€“ `{object}` â€“ an object against which any expressions embedded in the strings
     *      are evaluated against (typically a scope object).
     *    * `locals` â€“ `{object=}` â€“ local variables context object, useful for overriding values in
     *      `context`.
     */
    sce.parseAs = function sceParseAs(type, expr) {
      var parsed = $parse(expr);
      if (parsed.literal && parsed.constant) {
        return parsed;
      } else {
        return $parse(expr, function(value) {
          return sce.getTrusted(type, value);
        });
      }
    };

    /**
     * @ngdoc method
     * @name $sce#trustAs
     *
     * @description
     * Delegates to {@link ng.$sceDelegate#trustAs `$sceDelegate.trustAs`}.  As such,
     * returns an object that is trusted by angular for use in specified strict contextual
     * escaping contexts (such as ng-bind-html, ng-include, any src attribute
     * interpolation, any dom event binding attribute interpolation such as for onclick,  etc.)
     * that uses the provided value.  See * {@link ng.$sce $sce} for enabling strict contextual
     * escaping.
     *
     * @param {string} type The kind of context in which this value is safe for use.  e.g. url,
     *   resourceUrl, html, js and css.
     * @param {*} value The value that that should be considered trusted/safe.
     * @returns {*} A value that can be used to stand in for the provided `value` in places
     * where Angular expects a $sce.trustAs() return value.
     */

    /**
     * @ngdoc method
     * @name $sce#trustAsHtml
     *
     * @description
     * Shorthand method.  `$sce.trustAsHtml(value)` â†’
     *     {@link ng.$sceDelegate#trustAs `$sceDelegate.trustAs($sce.HTML, value)`}
     *
     * @param {*} value The value to trustAs.
     * @returns {*} An object that can be passed to {@link ng.$sce#getTrustedHtml
     *     $sce.getTrustedHtml(value)} to obtain the original value.  (privileged directives
     *     only accept expressions that are either literal constants or are the
     *     return value of {@link ng.$sce#trustAs $sce.trustAs}.)
     */

    /**
     * @ngdoc method
     * @name $sce#trustAsUrl
     *
     * @description
     * Shorthand method.  `$sce.trustAsUrl(value)` â†’
     *     {@link ng.$sceDelegate#trustAs `$sceDelegate.trustAs($sce.URL, value)`}
     *
     * @param {*} value The value to trustAs.
     * @returns {*} An object that can be passed to {@link ng.$sce#getTrustedUrl
     *     $sce.getTrustedUrl(value)} to obtain the original value.  (privileged directives
     *     only accept expressions that are either literal constants or are the
     *     return value of {@link ng.$sce#trustAs $sce.trustAs}.)
     */

    /**
     * @ngdoc method
     * @name $sce#trustAsResourceUrl
     *
     * @description
     * Shorthand method.  `$sce.trustAsResourceUrl(value)` â†’
     *     {@link ng.$sceDelegate#trustAs `$sceDelegate.trustAs($sce.RESOURCE_URL, value)`}
     *
     * @param {*} value The value to trustAs.
     * @returns {*} An object that can be passed to {@link ng.$sce#getTrustedResourceUrl
     *     $sce.getTrustedResourceUrl(value)} to obtain the original value.  (privileged directives
     *     only accept expressions that are either literal constants or are the return
     *     value of {@link ng.$sce#trustAs $sce.trustAs}.)
     */

    /**
     * @ngdoc method
     * @name $sce#trustAsJs
     *
     * @description
     * Shorthand method.  `$sce.trustAsJs(value)` â†’
     *     {@link ng.$sceDelegate#trustAs `$sceDelegate.trustAs($sce.JS, value)`}
     *
     * @param {*} value The value to trustAs.
     * @returns {*} An object that can be passed to {@link ng.$sce#getTrustedJs
     *     $sce.getTrustedJs(value)} to obtain the original value.  (privileged directives
     *     only accept expressions that are either literal constants or are the
     *     return value of {@link ng.$sce#trustAs $sce.trustAs}.)
     */

    /**
     * @ngdoc method
     * @name $sce#getTrusted
     *
     * @description
     * Delegates to {@link ng.$sceDelegate#getTrusted `$sceDelegate.getTrusted`}.  As such,
     * takes the result of a {@link ng.$sce#trustAs `$sce.trustAs`}() call and returns the
     * originally supplied value if the queried context type is a supertype of the created type.
     * If this condition isn't satisfied, throws an exception.
     *
     * @param {string} type The kind of context in which this value is to be used.
     * @param {*} maybeTrusted The result of a prior {@link ng.$sce#trustAs `$sce.trustAs`}
     *                         call.
     * @returns {*} The value the was originally provided to
     *              {@link ng.$sce#trustAs `$sce.trustAs`} if valid in this context.
     *              Otherwise, throws an exception.
     */

    /**
     * @ngdoc method
     * @name $sce#getTrustedHtml
     *
     * @description
     * Shorthand method.  `$sce.getTrustedHtml(value)` â†’
     *     {@link ng.$sceDelegate#getTrusted `$sceDelegate.getTrusted($sce.HTML, value)`}
     *
     * @param {*} value The value to pass to `$sce.getTrusted`.
     * @returns {*} The return value of `$sce.getTrusted($sce.HTML, value)`
     */

    /**
     * @ngdoc method
     * @name $sce#getTrustedCss
     *
     * @description
     * Shorthand method.  `$sce.getTrustedCss(value)` â†’
     *     {@link ng.$sceDelegate#getTrusted `$sceDelegate.getTrusted($sce.CSS, value)`}
     *
     * @param {*} value The value to pass to `$sce.getTrusted`.
     * @returns {*} The return value of `$sce.getTrusted($sce.CSS, value)`
     */

    /**
     * @ngdoc method
     * @name $sce#getTrustedUrl
     *
     * @description
     * Shorthand method.  `$sce.getTrustedUrl(value)` â†’
     *     {@link ng.$sceDelegate#getTrusted `$sceDelegate.getTrusted($sce.URL, value)`}
     *
     * @param {*} value The value to pass to `$sce.getTrusted`.
     * @returns {*} The return value of `$sce.getTrusted($sce.URL, value)`
     */

    /**
     * @ngdoc method
     * @name $sce#getTrustedResourceUrl
     *
     * @description
     * Shorthand method.  `$sce.getTrustedResourceUrl(value)` â†’
     *     {@link ng.$sceDelegate#getTrusted `$sceDelegate.getTrusted($sce.RESOURCE_URL, value)`}
     *
     * @param {*} value The value to pass to `$sceDelegate.getTrusted`.
     * @returns {*} The return value of `$sce.getTrusted($sce.RESOURCE_URL, value)`
     */

    /**
     * @ngdoc method
     * @name $sce#getTrustedJs
     *
     * @description
     * Shorthand method.  `$sce.getTrustedJs(value)` â†’
     *     {@link ng.$sceDelegate#getTrusted `$sceDelegate.getTrusted($sce.JS, value)`}
     *
     * @param {*} value The value to pass to `$sce.getTrusted`.
     * @returns {*} The return value of `$sce.getTrusted($sce.JS, value)`
     */

    /**
     * @ngdoc method
     * @name $sce#parseAsHtml
     *
     * @description
     * Shorthand method.  `$sce.parseAsHtml(expression string)` â†’
     *     {@link ng.$sce#parseAs `$sce.parseAs($sce.HTML, value)`}
     *
     * @param {string} expression String expression to compile.
     * @returns {function(context, locals)} a function which represents the compiled expression:
     *
     *    * `context` â€“ `{object}` â€“ an object against which any expressions embedded in the strings
     *      are evaluated against (typically a scope object).
     *    * `locals` â€“ `{object=}` â€“ local variables context object, useful for overriding values in
     *      `context`.
     */

    /**
     * @ngdoc method
     * @name $sce#parseAsCss
     *
     * @description
     * Shorthand method.  `$sce.parseAsCss(value)` â†’
     *     {@link ng.$sce#parseAs `$sce.parseAs($sce.CSS, value)`}
     *
     * @param {string} expression String expression to compile.
     * @returns {function(context, locals)} a function which represents the compiled expression:
     *
     *    * `context` â€“ `{object}` â€“ an object against which any expressions embedded in the strings
     *      are evaluated against (typically a scope object).
     *    * `locals` â€“ `{object=}` â€“ local variables context object, useful for overriding values in
     *      `context`.
     */

    /**
     * @ngdoc method
     * @name $sce#parseAsUrl
     *
     * @description
     * Shorthand method.  `$sce.parseAsUrl(value)` â†’
     *     {@link ng.$sce#parseAs `$sce.parseAs($sce.URL, value)`}
     *
     * @param {string} expression String expression to compile.
     * @returns {function(context, locals)} a function which represents the compiled expression:
     *
     *    * `context` â€“ `{object}` â€“ an object against which any expressions embedded in the strings
     *      are evaluated against (typically a scope object).
     *    * `locals` â€“ `{object=}` â€“ local variables context object, useful for overriding values in
     *      `context`.
     */

    /**
     * @ngdoc method
     * @name $sce#parseAsResourceUrl
     *
     * @description
     * Shorthand method.  `$sce.parseAsResourceUrl(value)` â†’
     *     {@link ng.$sce#parseAs `$sce.parseAs($sce.RESOURCE_URL, value)`}
     *
     * @param {string} expression String expression to compile.
     * @returns {function(context, locals)} a function which represents the compiled expression:
     *
     *    * `context` â€“ `{object}` â€“ an object against which any expressions embedded in the strings
     *      are evaluated against (typically a scope object).
     *    * `locals` â€“ `{object=}` â€“ local variables context object, useful for overriding values in
     *      `context`.
     */

    /**
     * @ngdoc method
     * @name $sce#parseAsJs
     *
     * @description
     * Shorthand method.  `$sce.parseAsJs(value)` â†’
     *     {@link ng.$sce#parseAs `$sce.parseAs($sce.JS, value)`}
     *
     * @param {string} expression String expression to compile.
     * @returns {function(context, locals)} a function which represents the compiled expression:
     *
     *    * `context` â€“ `{object}` â€“ an object against which any expressions embedded in the strings
     *      are evaluated against (typically a scope object).
     *    * `locals` â€“ `{object=}` â€“ local variables context object, useful for overriding values in
     *      `context`.
     */

    // Shorthand delegations.
    var parse = sce.parseAs,
        getTrusted = sce.getTrusted,
        trustAs = sce.trustAs;

    forEach(SCE_CONTEXTS, function(enumValue, name) {
      var lName = lowercase(name);
      sce[camelCase("parse_as_" + lName)] = function(expr) {
        return parse(enumValue, expr);
      };
      sce[camelCase("get_trusted_" + lName)] = function(value) {
        return getTrusted(enumValue, value);
      };
      sce[camelCase("trust_as_" + lName)] = function(value) {
        return trustAs(enumValue, value);
      };
    });

    return sce;
  }];
}

/**
 * !!! This is an undocumented "private" service !!!
 *
 * @name $sniffer
 * @requires $window
 * @requires $document
 *
 * @property {boolean} history Does the browser support html5 history api ?
 * @property {boolean} transitions Does the browser support CSS transition events ?
 * @property {boolean} animations Does the browser support CSS animation events ?
 *
 * @description
 * This is very simple implementation of testing browser's features.
 */
function $SnifferProvider() {
  this.$get = ['$window', '$document', function($window, $document) {
    var eventSupport = {},
        android =
          toInt((/android (\d+)/.exec(lowercase(($window.navigator || {}).userAgent)) || [])[1]),
        boxee = /Boxee/i.test(($window.navigator || {}).userAgent),
        document = $document[0] || {},
        vendorPrefix,
        vendorRegex = /^(Moz|webkit|ms)(?=[A-Z])/,
        bodyStyle = document.body && document.body.style,
        transitions = false,
        animations = false,
        match;

    if (bodyStyle) {
      for (var prop in bodyStyle) {
        if (match = vendorRegex.exec(prop)) {
          vendorPrefix = match[0];
          vendorPrefix = vendorPrefix.substr(0, 1).toUpperCase() + vendorPrefix.substr(1);
          break;
        }
      }

      if (!vendorPrefix) {
        vendorPrefix = ('WebkitOpacity' in bodyStyle) && 'webkit';
      }

      transitions = !!(('transition' in bodyStyle) || (vendorPrefix + 'Transition' in bodyStyle));
      animations  = !!(('animation' in bodyStyle) || (vendorPrefix + 'Animation' in bodyStyle));

      if (android && (!transitions ||  !animations)) {
        transitions = isString(bodyStyle.webkitTransition);
        animations = isString(bodyStyle.webkitAnimation);
      }
    }


    return {
      // Android has history.pushState, but it does not update location correctly
      // so let's not use the history API at all.
      // http://code.google.com/p/android/issues/detail?id=17471
      // https://github.com/angular/angular.js/issues/904

      // older webkit browser (533.9) on Boxee box has exactly the same problem as Android has
      // so let's not use the history API also
      // We are purposefully using `!(android < 4)` to cover the case when `android` is undefined
      // jshint -W018
      history: !!($window.history && $window.history.pushState && !(android < 4) && !boxee),
      // jshint +W018
      hasEvent: function(event) {
        // IE9 implements 'input' event it's so fubared that we rather pretend that it doesn't have
        // it. In particular the event is not fired when backspace or delete key are pressed or
        // when cut operation is performed.
        // IE10+ implements 'input' event but it erroneously fires under various situations,
        // e.g. when placeholder changes, or a form is focused.
        if (event === 'input' && msie <= 11) return false;

        if (isUndefined(eventSupport[event])) {
          var divElm = document.createElement('div');
          eventSupport[event] = 'on' + event in divElm;
        }

        return eventSupport[event];
      },
      csp: csp(),
      vendorPrefix: vendorPrefix,
      transitions: transitions,
      animations: animations,
      android: android
    };
  }];
}

var $compileMinErr = minErr('$compile');

/**
 * @ngdoc service
 * @name $templateRequest
 *
 * @description
 * The `$templateRequest` service runs security checks then downloads the provided template using
 * `$http` and, upon success, stores the contents inside of `$templateCache`. If the HTTP request
 * fails or the response data of the HTTP request is empty, a `$compile` error will be thrown (the
 * exception can be thwarted by setting the 2nd parameter of the function to true). Note that the
 * contents of `$templateCache` are trusted, so the call to `$sce.getTrustedUrl(tpl)` is omitted
 * when `tpl` is of type string and `$templateCache` has the matching entry.
 *
 * @param {string|TrustedResourceUrl} tpl The HTTP request template URL
 * @param {boolean=} ignoreRequestError Whether or not to ignore the exception when the request fails or the template is empty
 *
 * @return {Promise} a promise for the HTTP response data of the given URL.
 *
 * @property {number} totalPendingRequests total amount of pending template requests being downloaded.
 */
function $TemplateRequestProvider() {
  this.$get = ['$templateCache', '$http', '$q', '$sce', function($templateCache, $http, $q, $sce) {
    function handleRequestFn(tpl, ignoreRequestError) {
      handleRequestFn.totalPendingRequests++;

      // We consider the template cache holds only trusted templates, so
      // there's no need to go through whitelisting again for keys that already
      // are included in there. This also makes Angular accept any script
      // directive, no matter its name. However, we still need to unwrap trusted
      // types.
      if (!isString(tpl) || !$templateCache.get(tpl)) {
        tpl = $sce.getTrustedResourceUrl(tpl);
      }

      var transformResponse = $http.defaults && $http.defaults.transformResponse;

      if (isArray(transformResponse)) {
        transformResponse = transformResponse.filter(function(transformer) {
          return transformer !== defaultHttpResponseTransform;
        });
      } else if (transformResponse === defaultHttpResponseTransform) {
        transformResponse = null;
      }

      var httpOptions = {
        cache: $templateCache,
        transformResponse: transformResponse
      };

      return $http.get(tpl, httpOptions)
        ['finally'](function() {
          handleRequestFn.totalPendingRequests--;
        })
        .then(function(response) {
          $templateCache.put(tpl, response.data);
          return response.data;
        }, handleError);

      function handleError(resp) {
        if (!ignoreRequestError) {
          throw $compileMinErr('tpload', 'Failed to load template: {0} (HTTP status: {1} {2})',
            tpl, resp.status, resp.statusText);
        }
        return $q.reject(resp);
      }
    }

    handleRequestFn.totalPendingRequests = 0;

    return handleRequestFn;
  }];
}

function $$TestabilityProvider() {
  this.$get = ['$rootScope', '$browser', '$location',
       function($rootScope,   $browser,   $location) {

    /**
     * @name $testability
     *
     * @description
     * The private $$testability service provides a collection of methods for use when debugging
     * or by automated test and debugging tools.
     */
    var testability = {};

    /**
     * @name $$testability#findBindings
     *
     * @description
     * Returns an array of elements that are bound (via ng-bind or {{}})
     * to expressions matching the input.
     *
     * @param {Element} element The element root to search from.
     * @param {string} expression The binding expression to match.
     * @param {boolean} opt_exactMatch If true, only returns exact matches
     *     for the expression. Filters and whitespace are ignored.
     */
    testability.findBindings = function(element, expression, opt_exactMatch) {
      var bindings = element.getElementsByClassName('ng-binding');
      var matches = [];
      forEach(bindings, function(binding) {
        var dataBinding = angular.element(binding).data('$binding');
        if (dataBinding) {
          forEach(dataBinding, function(bindingName) {
            if (opt_exactMatch) {
              var matcher = new RegExp('(^|\\s)' + escapeForRegexp(expression) + '(\\s|\\||$)');
              if (matcher.test(bindingName)) {
                matches.push(binding);
              }
            } else {
              if (bindingName.indexOf(expression) != -1) {
                matches.push(binding);
              }
            }
          });
        }
      });
      return matches;
    };

    /**
     * @name $$testability#findModels
     *
     * @description
     * Returns an array of elements that are two-way found via ng-model to
     * expressions matching the input.
     *
     * @param {Element} element The element root to search from.
     * @param {string} expression The model expression to match.
     * @param {boolean} opt_exactMatch If true, only returns exact matches
     *     for the expression.
     */
    testability.findModels = function(element, expression, opt_exactMatch) {
      var prefixes = ['ng-', 'data-ng-', 'ng\\:'];
      for (var p = 0; p < prefixes.length; ++p) {
        var attributeEquals = opt_exactMatch ? '=' : '*=';
        var selector = '[' + prefixes[p] + 'model' + attributeEquals + '"' + expression + '"]';
        var elements = element.querySelectorAll(selector);
        if (elements.length) {
          return elements;
        }
      }
    };

    /**
     * @name $$testability#getLocation
     *
     * @description
     * Shortcut for getting the location in a browser agnostic way. Returns
     *     the path, search, and hash. (e.g. /path?a=b#hash)
     */
    testability.getLocation = function() {
      return $location.url();
    };

    /**
     * @name $$testability#setLocation
     *
     * @description
     * Shortcut for navigating to a location without doing a full page reload.
     *
     * @param {string} url The location url (path, search and hash,
     *     e.g. /path?a=b#hash) to go to.
     */
    testability.setLocation = function(url) {
      if (url !== $location.url()) {
        $location.url(url);
        $rootScope.$digest();
      }
    };

    /**
     * @name $$testability#whenStable
     *
     * @description
     * Calls the callback when $timeout and $http requests are completed.
     *
     * @param {function} callback
     */
    testability.whenStable = function(callback) {
      $browser.notifyWhenNoOutstandingRequests(callback);
    };

    return testability;
  }];
}

function $TimeoutProvider() {
  this.$get = ['$rootScope', '$browser', '$q', '$$q', '$exceptionHandler',
       function($rootScope,   $browser,   $q,   $$q,   $exceptionHandler) {

    var deferreds = {};


     /**
      * @ngdoc service
      * @name $timeout
      *
      * @description
      * Angular's wrapper for `window.setTimeout`. The `fn` function is wrapped into a try/catch
      * block and delegates any exceptions to
      * {@link ng.$exceptionHandler $exceptionHandler} service.
      *
      * The return value of calling `$timeout` is a promise, which will be resolved when
      * the delay has passed and the timeout function, if provided, is executed.
      *
      * To cancel a timeout request, call `$timeout.cancel(promise)`.
      *
      * In tests you can use {@link ngMock.$timeout `$timeout.flush()`} to
      * synchronously flush the queue of deferred functions.
      *
      * If you only want a promise that will be resolved after some specified delay
      * then you can call `$timeout` without the `fn` function.
      *
      * @param {function()=} fn A function, whose execution should be delayed.
      * @param {number=} [delay=0] Delay in milliseconds.
      * @param {boolean=} [invokeApply=true] If set to `false` skips model dirty checking, otherwise
      *   will invoke `fn` within the {@link ng.$rootScope.Scope#$apply $apply} block.
      * @param {...*=} Pass additional parameters to the executed function.
      * @returns {Promise} Promise that will be resolved when the timeout is reached. The value this
      *   promise will be resolved with is the return value of the `fn` function.
      *
      */
    function timeout(fn, delay, invokeApply) {
      if (!isFunction(fn)) {
        invokeApply = delay;
        delay = fn;
        fn = noop;
      }

      var args = sliceArgs(arguments, 3),
          skipApply = (isDefined(invokeApply) && !invokeApply),
          deferred = (skipApply ? $$q : $q).defer(),
          promise = deferred.promise,
          timeoutId;

      timeoutId = $browser.defer(function() {
        try {
          deferred.resolve(fn.apply(null, args));
        } catch (e) {
          deferred.reject(e);
          $exceptionHandler(e);
        }
        finally {
          delete deferreds[promise.$$timeoutId];
        }

        if (!skipApply) $rootScope.$apply();
      }, delay);

      promise.$$timeoutId = timeoutId;
      deferreds[timeoutId] = deferred;

      return promise;
    }


     /**
      * @ngdoc method
      * @name $timeout#cancel
      *
      * @description
      * Cancels a task associated with the `promise`. As a result of this, the promise will be
      * resolved with a rejection.
      *
      * @param {Promise=} promise Promise returned by the `$timeout` function.
      * @returns {boolean} Returns `true` if the task hasn't executed yet and was successfully
      *   canceled.
      */
    timeout.cancel = function(promise) {
      if (promise && promise.$$timeoutId in deferreds) {
        deferreds[promise.$$timeoutId].reject('canceled');
        delete deferreds[promise.$$timeoutId];
        return $browser.defer.cancel(promise.$$timeoutId);
      }
      return false;
    };

    return timeout;
  }];
}

// NOTE:  The usage of window and document instead of $window and $document here is
// deliberate.  This service depends on the specific behavior of anchor nodes created by the
// browser (resolving and parsing URLs) that is unlikely to be provided by mock objects and
// cause us to break tests.  In addition, when the browser resolves a URL for XHR, it
// doesn't know about mocked locations and resolves URLs to the real document - which is
// exactly the behavior needed here.  There is little value is mocking these out for this
// service.
var urlParsingNode = document.createElement("a");
var originUrl = urlResolve(window.location.href);


/**
 *
 * Implementation Notes for non-IE browsers
 * ----------------------------------------
 * Assigning a URL to the href property of an anchor DOM node, even one attached to the DOM,
 * results both in the normalizing and parsing of the URL.  Normalizing means that a relative
 * URL will be resolved into an absolute URL in the context of the application document.
 * Parsing means that the anchor node's host, hostname, protocol, port, pathname and related
 * properties are all populated to reflect the normalized URL.  This approach has wide
 * compatibility - Safari 1+, Mozilla 1+, Opera 7+,e etc.  See
 * http://www.aptana.com/reference/html/api/HTMLAnchorElement.html
 *
 * Implementation Notes for IE
 * ---------------------------
 * IE <= 10 normalizes the URL when assigned to the anchor node similar to the other
 * browsers.  However, the parsed components will not be set if the URL assigned did not specify
 * them.  (e.g. if you assign a.href = "foo", then a.protocol, a.host, etc. will be empty.)  We
 * work around that by performing the parsing in a 2nd step by taking a previously normalized
 * URL (e.g. by assigning to a.href) and assigning it a.href again.  This correctly populates the
 * properties such as protocol, hostname, port, etc.
 *
 * References:
 *   http://developer.mozilla.org/en-US/docs/Web/API/HTMLAnchorElement
 *   http://www.aptana.com/reference/html/api/HTMLAnchorElement.html
 *   http://url.spec.whatwg.org/#urlutils
 *   https://github.com/angular/angular.js/pull/2902
 *   http://james.padolsey.com/javascript/parsing-urls-with-the-dom/
 *
 * @kind function
 * @param {string} url The URL to be parsed.
 * @description Normalizes and parses a URL.
 * @returns {object} Returns the normalized URL as a dictionary.
 *
 *   | member name   | Description    |
 *   |---------------|----------------|
 *   | href          | A normalized version of the provided URL if it was not an absolute URL |
 *   | protocol      | The protocol including the trailing colon                              |
 *   | host          | The host and port (if the port is non-default) of the normalizedUrl    |
 *   | search        | The search params, minus the question mark                             |
 *   | hash          | The hash string, minus the hash symbol
 *   | hostname      | The hostname
 *   | port          | The port, without ":"
 *   | pathname      | The pathname, beginning with "/"
 *
 */
function urlResolve(url) {
  var href = url;

  if (msie) {
    // Normalize before parse.  Refer Implementation Notes on why this is
    // done in two steps on IE.
    urlParsingNode.setAttribute("href", href);
    href = urlParsingNode.href;
  }

  urlParsingNode.setAttribute('href', href);

  // urlParsingNode provides the UrlUtils interface - http://url.spec.whatwg.org/#urlutils
  return {
    href: urlParsingNode.href,
    protocol: urlParsingNode.protocol ? urlParsingNode.protocol.replace(/:$/, '') : '',
    host: urlParsingNode.host,
    search: urlParsingNode.search ? urlParsingNode.search.replace(/^\?/, '') : '',
    hash: urlParsingNode.hash ? urlParsingNode.hash.replace(/^#/, '') : '',
    hostname: urlParsingNode.hostname,
    port: urlParsingNode.port,
    pathname: (urlParsingNode.pathname.charAt(0) === '/')
      ? urlParsingNode.pathname
      : '/' + urlParsingNode.pathname
  };
}

/**
 * Parse a request URL and determine whether this is a same-origin request as the application document.
 *
 * @param {string|object} requestUrl The url of the request as a string that will be resolved
 * or a parsed URL object.
 * @returns {boolean} Whether the request is for the same origin as the application document.
 */
function urlIsSameOrigin(requestUrl) {
  var parsed = (isString(requestUrl)) ? urlResolve(requestUrl) : requestUrl;
  return (parsed.protocol === originUrl.protocol &&
          parsed.host === originUrl.host);
}

/**
 * @ngdoc service
 * @name $window
 *
 * @description
 * A reference to the browser's `window` object. While `window`
 * is globally available in JavaScript, it causes testability problems, because
 * it is a global variable. In angular we always refer to it through the
 * `$window` service, so it may be overridden, removed or mocked for testing.
 *
 * Expressions, like the one defined for the `ngClick` directive in the example
 * below, are evaluated with respect to the current scope.  Therefore, there is
 * no risk of inadvertently coding in a dependency on a global value in such an
 * expression.
 *
 * @example
   <example module="windowExample">
     <file name="index.html">
       <script>
         angular.module('windowExample', [])
           .controller('ExampleController', ['$scope', '$window', function($scope, $window) {
             $scope.greeting = 'Hello, World!';
             $scope.doGreeting = function(greeting) {
               $window.alert(greeting);
             };
           }]);
       </script>
       <div ng-controller="ExampleController">
         <input type="text" ng-model="greeting" aria-label="greeting" />
         <button ng-click="doGreeting(greeting)">ALERT</button>
       </div>
     </file>
     <file name="protractor.js" type="protractor">
      it('should display the greeting in the input box', function() {
       element(by.model('greeting')).sendKeys('Hello, E2E Tests');
       // If we click the button it will block the test runner
       // element(':button').click();
      });
     </file>
   </example>
 */
function $WindowProvider() {
  this.$get = valueFn(window);
}

/**
 * @name $$cookieReader
 * @requires $document
 *
 * @description
 * This is a private service for reading cookies used by $http and ngCookies
 *
 * @return {Object} a key/value map of the current cookies
 */
function $$CookieReader($document) {
  var rawDocument = $document[0] || {};
  var lastCookies = {};
  var lastCookieString = '';

  function safeDecodeURIComponent(str) {
    try {
      return decodeURIComponent(str);
    } catch (e) {
      return str;
    }
  }

  return function() {
    var cookieArray, cookie, i, index, name;
    var currentCookieString = rawDocument.cookie || '';

    if (currentCookieString !== lastCookieString) {
      lastCookieString = currentCookieString;
      cookieArray = lastCookieString.split('; ');
      lastCookies = {};

      for (i = 0; i < cookieArray.length; i++) {
        cookie = cookieArray[i];
        index = cookie.indexOf('=');
        if (index > 0) { //ignore nameless cookies
          name = safeDecodeURIComponent(cookie.substring(0, index));
          // the first value that is seen for a cookie is the most
          // specific one.  values for the same cookie name that
          // follow are for less specific paths.
          if (isUndefined(lastCookies[name])) {
            lastCookies[name] = safeDecodeURIComponent(cookie.substring(index + 1));
          }
        }
      }
    }
    return lastCookies;
  };
}

$$CookieReader.$inject = ['$document'];

function $$CookieReaderProvider() {
  this.$get = $$CookieReader;
}

/* global currencyFilter: true,
 dateFilter: true,
 filterFilter: true,
 jsonFilter: true,
 limitToFilter: true,
 lowercaseFilter: true,
 numberFilter: true,
 orderByFilter: true,
 uppercaseFilter: true,
 */

/**
 * @ngdoc provider
 * @name $filterProvider
 * @description
 *
 * Filters are just functions which transform input to an output. However filters need to be
 * Dependency Injected. To achieve this a filter definition consists of a factory function which is
 * annotated with dependencies and is responsible for creating a filter function.
 *
 * <div class="alert alert-warning">
 * **Note:** Filter names must be valid angular {@link expression} identifiers, such as `uppercase` or `orderBy`.
 * Names with special characters, such as hyphens and dots, are not allowed. If you wish to namespace
 * your filters, then you can use capitalization (`myappSubsectionFilterx`) or underscores
 * (`myapp_subsection_filterx`).
 * </div>
 *
 * ```js
 *   // Filter registration
 *   function MyModule($provide, $filterProvider) {
 *     // create a service to demonstrate injection (not always needed)
 *     $provide.value('greet', function(name){
 *       return 'Hello ' + name + '!';
 *     });
 *
 *     // register a filter factory which uses the
 *     // greet service to demonstrate DI.
 *     $filterProvider.register('greet', function(greet){
 *       // return the filter function which uses the greet service
 *       // to generate salutation
 *       return function(text) {
 *         // filters need to be forgiving so check input validity
 *         return text && greet(text) || text;
 *       };
 *     });
 *   }
 * ```
 *
 * The filter function is registered with the `$injector` under the filter name suffix with
 * `Filter`.
 *
 * ```js
 *   it('should be the same instance', inject(
 *     function($filterProvider) {
 *       $filterProvider.register('reverse', function(){
 *         return ...;
 *       });
 *     },
 *     function($filter, reverseFilter) {
 *       expect($filter('reverse')).toBe(reverseFilter);
 *     });
 * ```
 *
 *
 * For more information about how angular filters work, and how to create your own filters, see
 * {@link guide/filter Filters} in the Angular Developer Guide.
 */

/**
 * @ngdoc service
 * @name $filter
 * @kind function
 * @description
 * Filters are used for formatting data displayed to the user.
 *
 * The general syntax in templates is as follows:
 *
 *         {{ expression [| filter_name[:parameter_value] ... ] }}
 *
 * @param {String} name Name of the filter function to retrieve
 * @return {Function} the filter function
 * @example
   <example name="$filter" module="filterExample">
     <file name="index.html">
       <div ng-controller="MainCtrl">
        <h3>{{ originalText }}</h3>
        <h3>{{ filteredText }}</h3>
       </div>
     </file>

     <file name="script.js">
      angular.module('filterExample', [])
      .controller('MainCtrl', function($scope, $filter) {
        $scope.originalText = 'hello';
        $scope.filteredText = $filter('uppercase')($scope.originalText);
      });
     </file>
   </example>
  */
$FilterProvider.$inject = ['$provide'];
function $FilterProvider($provide) {
  var suffix = 'Filter';

  /**
   * @ngdoc method
   * @name $filterProvider#register
   * @param {string|Object} name Name of the filter function, or an object map of filters where
   *    the keys are the filter names and the values are the filter factories.
   *
   *    <div class="alert alert-warning">
   *    **Note:** Filter names must be valid angular {@link expression} identifiers, such as `uppercase` or `orderBy`.
   *    Names with special characters, such as hyphens and dots, are not allowed. If you wish to namespace
   *    your filters, then you can use capitalization (`myappSubsectionFilterx`) or underscores
   *    (`myapp_subsection_filterx`).
   *    </div>
    * @param {Function} factory If the first argument was a string, a factory function for the filter to be registered.
   * @returns {Object} Registered filter instance, or if a map of filters was provided then a map
   *    of the registered filter instances.
   */
  function register(name, factory) {
    if (isObject(name)) {
      var filters = {};
      forEach(name, function(filter, key) {
        filters[key] = register(key, filter);
      });
      return filters;
    } else {
      return $provide.factory(name + suffix, factory);
    }
  }
  this.register = register;

  this.$get = ['$injector', function($injector) {
    return function(name) {
      return $injector.get(name + suffix);
    };
  }];

  ////////////////////////////////////////

  /* global
    currencyFilter: false,
    dateFilter: false,
    filterFilter: false,
    jsonFilter: false,
    limitToFilter: false,
    lowercaseFilter: false,
    numberFilter: false,
    orderByFilter: false,
    uppercaseFilter: false,
  */

  register('currency', currencyFilter);
  register('date', dateFilter);
  register('filter', filterFilter);
  register('json', jsonFilter);
  register('limitTo', limitToFilter);
  register('lowercase', lowercaseFilter);
  register('number', numberFilter);
  register('orderBy', orderByFilter);
  register('uppercase', uppercaseFilter);
}

/**
 * @ngdoc filter
 * @name filter
 * @kind function
 *
 * @description
 * Selects a subset of items from `array` and returns it as a new array.
 *
 * @param {Array} array The source array.
 * @param {string|Object|function()} expression The predicate to be used for selecting items from
 *   `array`.
 *
 *   Can be one of:
 *
 *   - `string`: The string is used for matching against the contents of the `array`. All strings or
 *     objects with string properties in `array` that match this string will be returned. This also
 *     applies to nested object properties.
 *     The predicate can be negated by prefixing the string with `!`.
 *
 *   - `Object`: A pattern object can be used to filter specific properties on objects contained
 *     by `array`. For example `{name:"M", phone:"1"}` predicate will return an array of items
 *     which have property `name` containing "M" and property `phone` containing "1". A special
 *     property name `$` can be used (as in `{$:"text"}`) to accept a match against any
 *     property of the object or its nested object properties. That's equivalent to the simple
 *     substring match with a `string` as described above. The predicate can be negated by prefixing
 *     the string with `!`.
 *     For example `{name: "!M"}` predicate will return an array of items which have property `name`
 *     not containing "M".
 *
 *     Note that a named property will match properties on the same level only, while the special
 *     `$` property will match properties on the same level or deeper. E.g. an array item like
 *     `{name: {first: 'John', last: 'Doe'}}` will **not** be matched by `{name: 'John'}`, but
 *     **will** be matched by `{$: 'John'}`.
 *
 *   - `function(value, index, array)`: A predicate function can be used to write arbitrary filters.
 *     The function is called for each element of the array, with the element, its index, and
 *     the entire array itself as arguments.
 *
 *     The final result is an array of those elements that the predicate returned true for.
 *
 * @param {function(actual, expected)|true|undefined} comparator Comparator which is used in
 *     determining if the expected value (from the filter expression) and actual value (from
 *     the object in the array) should be considered a match.
 *
 *   Can be one of:
 *
 *   - `function(actual, expected)`:
 *     The function will be given the object value and the predicate value to compare and
 *     should return true if both values should be considered equal.
 *
 *   - `true`: A shorthand for `function(actual, expected) { return angular.equals(actual, expected)}`.
 *     This is essentially strict comparison of expected and actual.
 *
 *   - `false|undefined`: A short hand for a function which will look for a substring match in case
 *     insensitive way.
 *
 *     Primitive values are converted to strings. Objects are not compared against primitives,
 *     unless they have a custom `toString` method (e.g. `Date` objects).
 *
 * @example
   <example>
     <file name="index.html">
       <div ng-init="friends = [{name:'John', phone:'555-1276'},
                                {name:'Mary', phone:'800-BIG-MARY'},
                                {name:'Mike', phone:'555-4321'},
                                {name:'Adam', phone:'555-5678'},
                                {name:'Julie', phone:'555-8765'},
                                {name:'Juliette', phone:'555-5678'}]"></div>

       <label>Search: <input ng-model="searchText"></label>
       <table id="searchTextResults">
         <tr><th>Name</th><th>Phone</th></tr>
         <tr ng-repeat="friend in friends | filter:searchText">
           <td>{{friend.name}}</td>
           <td>{{friend.phone}}</td>
         </tr>
       </table>
       <hr>
       <label>Any: <input ng-model="search.$"></label> <br>
       <label>Name only <input ng-model="search.name"></label><br>
       <label>Phone only <input ng-model="search.phone"></label><br>
       <label>Equality <input type="checkbox" ng-model="strict"></label><br>
       <table id="searchObjResults">
         <tr><th>Name</th><th>Phone</th></tr>
         <tr ng-repeat="friendObj in friends | filter:search:strict">
           <td>{{friendObj.name}}</td>
           <td>{{friendObj.phone}}</td>
         </tr>
       </table>
     </file>
     <file name="protractor.js" type="protractor">
       var expectFriendNames = function(expectedNames, key) {
         element.all(by.repeater(key + ' in friends').column(key + '.name')).then(function(arr) {
           arr.forEach(function(wd, i) {
             expect(wd.getText()).toMatch(expectedNames[i]);
           });
         });
       };

       it('should search across all fields when filtering with a string', function() {
         var searchText = element(by.model('searchText'));
         searchText.clear();
         searchText.sendKeys('m');
         expectFriendNames(['Mary', 'Mike', 'Adam'], 'friend');

         searchText.clear();
         searchText.sendKeys('76');
         expectFriendNames(['John', 'Julie'], 'friend');
       });

       it('should search in specific fields when filtering with a predicate object', function() {
         var searchAny = element(by.model('search.$'));
         searchAny.clear();
         searchAny.sendKeys('i');
         expectFriendNames(['Mary', 'Mike', 'Julie', 'Juliette'], 'friendObj');
       });
       it('should use a equal comparison when comparator is true', function() {
         var searchName = element(by.model('search.name'));
         var strict = element(by.model('strict'));
         searchName.clear();
         searchName.sendKeys('Julie');
         strict.click();
         expectFriendNames(['Julie'], 'friendObj');
       });
     </file>
   </example>
 */
function filterFilter() {
  return function(array, expression, comparator) {
    if (!isArrayLike(array)) {
      if (array == null) {
        return array;
      } else {
        throw minErr('filter')('notarray', 'Expected array but received: {0}', array);
      }
    }

    var expressionType = getTypeForFilter(expression);
    var predicateFn;
    var matchAgainstAnyProp;

    switch (expressionType) {
      case 'function':
        predicateFn = expression;
        break;
      case 'boolean':
      case 'null':
      case 'number':
      case 'string':
        matchAgainstAnyProp = true;
        //jshint -W086
      case 'object':
        //jshint +W086
        predicateFn = createPredicateFn(expression, comparator, matchAgainstAnyProp);
        break;
      default:
        return array;
    }

    return Array.prototype.filter.call(array, predicateFn);
  };
}

// Helper functions for `filterFilter`
function createPredicateFn(expression, comparator, matchAgainstAnyProp) {
  var shouldMatchPrimitives = isObject(expression) && ('$' in expression);
  var predicateFn;

  if (comparator === true) {
    comparator = equals;
  } else if (!isFunction(comparator)) {
    comparator = function(actual, expected) {
      if (isUndefined(actual)) {
        // No substring matching against `undefined`
        return false;
      }
      if ((actual === null) || (expected === null)) {
        // No substring matching against `null`; only match against `null`
        return actual === expected;
      }
      if (isObject(expected) || (isObject(actual) && !hasCustomToString(actual))) {
        // Should not compare primitives against objects, unless they have custom `toString` method
        return false;
      }

      actual = lowercase('' + actual);
      expected = lowercase('' + expected);
      return actual.indexOf(expected) !== -1;
    };
  }

  predicateFn = function(item) {
    if (shouldMatchPrimitives && !isObject(item)) {
      return deepCompare(item, expression.$, comparator, false);
    }
    return deepCompare(item, expression, comparator, matchAgainstAnyProp);
  };

  return predicateFn;
}

function deepCompare(actual, expected, comparator, matchAgainstAnyProp, dontMatchWholeObject) {
  var actualType = getTypeForFilter(actual);
  var expectedType = getTypeForFilter(expected);

  if ((expectedType === 'string') && (expected.charAt(0) === '!')) {
    return !deepCompare(actual, expected.substring(1), comparator, matchAgainstAnyProp);
  } else if (isArray(actual)) {
    // In case `actual` is an array, consider it a match
    // if ANY of it's items matches `expected`
    return actual.some(function(item) {
      return deepCompare(item, expected, comparator, matchAgainstAnyProp);
    });
  }

  switch (actualType) {
    case 'object':
      var key;
      if (matchAgainstAnyProp) {
        for (key in actual) {
          if ((key.charAt(0) !== '$') && deepCompare(actual[key], expected, comparator, true)) {
            return true;
          }
        }
        return dontMatchWholeObject ? false : deepCompare(actual, expected, comparator, false);
      } else if (expectedType === 'object') {
        for (key in expected) {
          var expectedVal = expected[key];
          if (isFunction(expectedVal) || isUndefined(expectedVal)) {
            continue;
          }

          var matchAnyProperty = key === '$';
          var actualVal = matchAnyProperty ? actual : actual[key];
          if (!deepCompare(actualVal, expectedVal, comparator, matchAnyProperty, matchAnyProperty)) {
            return false;
          }
        }
        return true;
      } else {
        return comparator(actual, expected);
      }
      break;
    case 'function':
      return false;
    default:
      return comparator(actual, expected);
  }
}

// Used for easily differentiating between `null` and actual `object`
function getTypeForFilter(val) {
  return (val === null) ? 'null' : typeof val;
}

/**
 * @ngdoc filter
 * @name currency
 * @kind function
 *
 * @description
 * Formats a number as a currency (ie $1,234.56). When no currency symbol is provided, default
 * symbol for current locale is used.
 *
 * @param {number} amount Input to filter.
 * @param {string=} symbol Currency symbol or identifier to be displayed.
 * @param {number=} fractionSize Number of decimal places to round the amount to, defaults to default max fraction size for current locale
 * @returns {string} Formatted number.
 *
 *
 * @example
   <example module="currencyExample">
     <file name="index.html">
       <script>
         angular.module('currencyExample', [])
           .controller('ExampleController', ['$scope', function($scope) {
             $scope.amount = 1234.56;
           }]);
       </script>
       <div ng-controller="ExampleController">
         <input type="number" ng-model="amount" aria-label="amount"> <br>
         default currency symbol ($): <span id="currency-default">{{amount | currency}}</span><br>
         custom currency identifier (USD$): <span id="currency-custom">{{amount | currency:"USD$"}}</span>
         no fractions (0): <span id="currency-no-fractions">{{amount | currency:"USD$":0}}</span>
       </div>
     </file>
     <file name="protractor.js" type="protractor">
       it('should init with 1234.56', function() {
         expect(element(by.id('currency-default')).getText()).toBe('$1,234.56');
         expect(element(by.id('currency-custom')).getText()).toBe('USD$1,234.56');
         expect(element(by.id('currency-no-fractions')).getText()).toBe('USD$1,235');
       });
       it('should update', function() {
         if (browser.params.browser == 'safari') {
           // Safari does not understand the minus key. See
           // https://github.com/angular/protractor/issues/481
           return;
         }
         element(by.model('amount')).clear();
         element(by.model('amount')).sendKeys('-1234');
         expect(element(by.id('currency-default')).getText()).toBe('-$1,234.00');
         expect(element(by.id('currency-custom')).getText()).toBe('-USD$1,234.00');
         expect(element(by.id('currency-no-fractions')).getText()).toBe('-USD$1,234');
       });
     </file>
   </example>
 */
currencyFilter.$inject = ['$locale'];
function currencyFilter($locale) {
  var formats = $locale.NUMBER_FORMATS;
  return function(amount, currencySymbol, fractionSize) {
    if (isUndefined(currencySymbol)) {
      currencySymbol = formats.CURRENCY_SYM;
    }

    if (isUndefined(fractionSize)) {
      fractionSize = formats.PATTERNS[1].maxFrac;
    }

    // if null or undefined pass it through
    return (amount == null)
        ? amount
        : formatNumber(amount, formats.PATTERNS[1], formats.GROUP_SEP, formats.DECIMAL_SEP, fractionSize).
            replace(/\u00A4/g, currencySymbol);
  };
}

/**
 * @ngdoc filter
 * @name number
 * @kind function
 *
 * @description
 * Formats a number as text.
 *
 * If the input is null or undefined, it will just be returned.
 * If the input is infinite (Infinity/-Infinity) the Infinity symbol 'âˆž' is returned.
 * If the input is not a number an empty string is returned.
 *
 *
 * @param {number|string} number Number to format.
 * @param {(number|string)=} fractionSize Number of decimal places to round the number to.
 * If this is not provided then the fraction size is computed from the current locale's number
 * formatting pattern. In the case of the default locale, it will be 3.
 * @returns {string} Number rounded to decimalPlaces and places a â€œ,â€ after each third digit.
 *
 * @example
   <example module="numberFilterExample">
     <file name="index.html">
       <script>
         angular.module('numberFilterExample', [])
           .controller('ExampleController', ['$scope', function($scope) {
             $scope.val = 1234.56789;
           }]);
       </script>
       <div ng-controller="ExampleController">
         <label>Enter number: <input ng-model='val'></label><br>
         Default formatting: <span id='number-default'>{{val | number}}</span><br>
         No fractions: <span>{{val | number:0}}</span><br>
         Negative number: <span>{{-val | number:4}}</span>
       </div>
     </file>
     <file name="protractor.js" type="protractor">
       it('should format numbers', function() {
         expect(element(by.id('number-default')).getText()).toBe('1,234.568');
         expect(element(by.binding('val | number:0')).getText()).toBe('1,235');
         expect(element(by.binding('-val | number:4')).getText()).toBe('-1,234.5679');
       });

       it('should update', function() {
         element(by.model('val')).clear();
         element(by.model('val')).sendKeys('3374.333');
         expect(element(by.id('number-default')).getText()).toBe('3,374.333');
         expect(element(by.binding('val | number:0')).getText()).toBe('3,374');
         expect(element(by.binding('-val | number:4')).getText()).toBe('-3,374.3330');
      });
     </file>
   </example>
 */


numberFilter.$inject = ['$locale'];
function numberFilter($locale) {
  var formats = $locale.NUMBER_FORMATS;
  return function(number, fractionSize) {

    // if null or undefined pass it through
    return (number == null)
        ? number
        : formatNumber(number, formats.PATTERNS[0], formats.GROUP_SEP, formats.DECIMAL_SEP,
                       fractionSize);
  };
}

var DECIMAL_SEP = '.';
function formatNumber(number, pattern, groupSep, decimalSep, fractionSize) {
  if (isObject(number)) return '';

  var isNegative = number < 0;
  number = Math.abs(number);

  var isInfinity = number === Infinity;
  if (!isInfinity && !isFinite(number)) return '';

  var numStr = number + '',
      formatedText = '',
      hasExponent = false,
      parts = [];

  if (isInfinity) formatedText = '\u221e';

  if (!isInfinity && numStr.indexOf('e') !== -1) {
    var match = numStr.match(/([\d\.]+)e(-?)(\d+)/);
    if (match && match[2] == '-' && match[3] > fractionSize + 1) {
      number = 0;
    } else {
      formatedText = numStr;
      hasExponent = true;
    }
  }

  if (!isInfinity && !hasExponent) {
    var fractionLen = (numStr.split(DECIMAL_SEP)[1] || '').length;

    // determine fractionSize if it is not specified
    if (isUndefined(fractionSize)) {
      fractionSize = Math.min(Math.max(pattern.minFrac, fractionLen), pattern.maxFrac);
    }

    // safely round numbers in JS without hitting imprecisions of floating-point arithmetics
    // inspired by:
    // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Math/round
    number = +(Math.round(+(number.toString() + 'e' + fractionSize)).toString() + 'e' + -fractionSize);

    var fraction = ('' + number).split(DECIMAL_SEP);
    var whole = fraction[0];
    fraction = fraction[1] || '';

    var i, pos = 0,
        lgroup = pattern.lgSize,
        group = pattern.gSize;

    if (whole.length >= (lgroup + group)) {
      pos = whole.length - lgroup;
      for (i = 0; i < pos; i++) {
        if ((pos - i) % group === 0 && i !== 0) {
          formatedText += groupSep;
        }
        formatedText += whole.charAt(i);
      }
    }

    for (i = pos; i < whole.length; i++) {
      if ((whole.length - i) % lgroup === 0 && i !== 0) {
        formatedText += groupSep;
      }
      formatedText += whole.charAt(i);
    }

    // format fraction part.
    while (fraction.length < fractionSize) {
      fraction += '0';
    }

    if (fractionSize && fractionSize !== "0") formatedText += decimalSep + fraction.substr(0, fractionSize);
  } else {
    if (fractionSize > 0 && number < 1) {
      formatedText = number.toFixed(fractionSize);
      number = parseFloat(formatedText);
      formatedText = formatedText.replace(DECIMAL_SEP, decimalSep);
    }
  }

  if (number === 0) {
    isNegative = false;
  }

  parts.push(isNegative ? pattern.negPre : pattern.posPre,
             formatedText,
             isNegative ? pattern.negSuf : pattern.posSuf);
  return parts.join('');
}

function padNumber(num, digits, trim) {
  var neg = '';
  if (num < 0) {
    neg =  '-';
    num = -num;
  }
  num = '' + num;
  while (num.length < digits) num = '0' + num;
  if (trim) {
    num = num.substr(num.length - digits);
  }
  return neg + num;
}


function dateGetter(name, size, offset, trim) {
  offset = offset || 0;
  return function(date) {
    var value = date['get' + name]();
    if (offset > 0 || value > -offset) {
      value += offset;
    }
    if (value === 0 && offset == -12) value = 12;
    return padNumber(value, size, trim);
  };
}

function dateStrGetter(name, shortForm) {
  return function(date, formats) {
    var value = date['get' + name]();
    var get = uppercase(shortForm ? ('SHORT' + name) : name);

    return formats[get][value];
  };
}

function timeZoneGetter(date, formats, offset) {
  var zone = -1 * offset;
  var paddedZone = (zone >= 0) ? "+" : "";

  paddedZone += padNumber(Math[zone > 0 ? 'floor' : 'ceil'](zone / 60), 2) +
                padNumber(Math.abs(zone % 60), 2);

  return paddedZone;
}

function getFirstThursdayOfYear(year) {
    // 0 = index of January
    var dayOfWeekOnFirst = (new Date(year, 0, 1)).getDay();
    // 4 = index of Thursday (+1 to account for 1st = 5)
    // 11 = index of *next* Thursday (+1 account for 1st = 12)
    return new Date(year, 0, ((dayOfWeekOnFirst <= 4) ? 5 : 12) - dayOfWeekOnFirst);
}

function getThursdayThisWeek(datetime) {
    return new Date(datetime.getFullYear(), datetime.getMonth(),
      // 4 = index of Thursday
      datetime.getDate() + (4 - datetime.getDay()));
}

function weekGetter(size) {
   return function(date) {
      var firstThurs = getFirstThursdayOfYear(date.getFullYear()),
         thisThurs = getThursdayThisWeek(date);

      var diff = +thisThurs - +firstThurs,
         result = 1 + Math.round(diff / 6.048e8); // 6.048e8 ms per week

      return padNumber(result, size);
   };
}

function ampmGetter(date, formats) {
  return date.getHours() < 12 ? formats.AMPMS[0] : formats.AMPMS[1];
}

function eraGetter(date, formats) {
  return date.getFullYear() <= 0 ? formats.ERAS[0] : formats.ERAS[1];
}

function longEraGetter(date, formats) {
  return date.getFullYear() <= 0 ? formats.ERANAMES[0] : formats.ERANAMES[1];
}

var DATE_FORMATS = {
  yyyy: dateGetter('FullYear', 4),
    yy: dateGetter('FullYear', 2, 0, true),
     y: dateGetter('FullYear', 1),
  MMMM: dateStrGetter('Month'),
   MMM: dateStrGetter('Month', true),
    MM: dateGetter('Month', 2, 1),
     M: dateGetter('Month', 1, 1),
    dd: dateGetter('Date', 2),
     d: dateGetter('Date', 1),
    HH: dateGetter('Hours', 2),
     H: dateGetter('Hours', 1),
    hh: dateGetter('Hours', 2, -12),
     h: dateGetter('Hours', 1, -12),
    mm: dateGetter('Minutes', 2),
     m: dateGetter('Minutes', 1),
    ss: dateGetter('Seconds', 2),
     s: dateGetter('Seconds', 1),
     // while ISO 8601 requires fractions to be prefixed with `.` or `,`
     // we can be just safely rely on using `sss` since we currently don't support single or two digit fractions
   sss: dateGetter('Milliseconds', 3),
  EEEE: dateStrGetter('Day'),
   EEE: dateStrGetter('Day', true),
     a: ampmGetter,
     Z: timeZoneGetter,
    ww: weekGetter(2),
     w: weekGetter(1),
     G: eraGetter,
     GG: eraGetter,
     GGG: eraGetter,
     GGGG: longEraGetter
};

var DATE_FORMATS_SPLIT = /((?:[^yMdHhmsaZEwG']+)|(?:'(?:[^']|'')*')|(?:E+|y+|M+|d+|H+|h+|m+|s+|a|Z|G+|w+))(.*)/,
    NUMBER_STRING = /^\-?\d+$/;

/**
 * @ngdoc filter
 * @name date
 * @kind function
 *
 * @description
 *   Formats `date` to a string based on the requested `format`.
 *
 *   `format` string can be composed of the following elements:
 *
 *   * `'yyyy'`: 4 digit representation of year (e.g. AD 1 => 0001, AD 2010 => 2010)
 *   * `'yy'`: 2 digit representation of year, padded (00-99). (e.g. AD 2001 => 01, AD 2010 => 10)
 *   * `'y'`: 1 digit representation of year, e.g. (AD 1 => 1, AD 199 => 199)
 *   * `'MMMM'`: Month in year (January-December)
 *   * `'MMM'`: Month in year (Jan-Dec)
 *   * `'MM'`: Month in year, padded (01-12)
 *   * `'M'`: Month in year (1-12)
 *   * `'dd'`: Day in month, padded (01-31)
 *   * `'d'`: Day in month (1-31)
 *   * `'EEEE'`: Day in Week,(Sunday-Saturday)
 *   * `'EEE'`: Day in Week, (Sun-Sat)
 *   * `'HH'`: Hour in day, padded (00-23)
 *   * `'H'`: Hour in day (0-23)
 *   * `'hh'`: Hour in AM/PM, padded (01-12)
 *   * `'h'`: Hour in AM/PM, (1-12)
 *   * `'mm'`: Minute in hour, padded (00-59)
 *   * `'m'`: Minute in hour (0-59)
 *   * `'ss'`: Second in minute, padded (00-59)
 *   * `'s'`: Second in minute (0-59)
 *   * `'sss'`: Millisecond in second, padded (000-999)
 *   * `'a'`: AM/PM marker
 *   * `'Z'`: 4 digit (+sign) representation of the timezone offset (-1200-+1200)
 *   * `'ww'`: Week of year, padded (00-53). Week 01 is the week with the first Thursday of the year
 *   * `'w'`: Week of year (0-53). Week 1 is the week with the first Thursday of the year
 *   * `'G'`, `'GG'`, `'GGG'`: The abbreviated form of the era string (e.g. 'AD')
 *   * `'GGGG'`: The long form of the era string (e.g. 'Anno Domini')
 *
 *   `format` string can also be one of the following predefined
 *   {@link guide/i18n localizable formats}:
 *
 *   * `'medium'`: equivalent to `'MMM d, y h:mm:ss a'` for en_US locale
 *     (e.g. Sep 3, 2010 12:05:08 PM)
 *   * `'short'`: equivalent to `'M/d/yy h:mm a'` for en_US  locale (e.g. 9/3/10 12:05 PM)
 *   * `'fullDate'`: equivalent to `'EEEE, MMMM d, y'` for en_US  locale
 *     (e.g. Friday, September 3, 2010)
 *   * `'longDate'`: equivalent to `'MMMM d, y'` for en_US  locale (e.g. September 3, 2010)
 *   * `'mediumDate'`: equivalent to `'MMM d, y'` for en_US  locale (e.g. Sep 3, 2010)
 *   * `'shortDate'`: equivalent to `'M/d/yy'` for en_US locale (e.g. 9/3/10)
 *   * `'mediumTime'`: equivalent to `'h:mm:ss a'` for en_US locale (e.g. 12:05:08 PM)
 *   * `'shortTime'`: equivalent to `'h:mm a'` for en_US locale (e.g. 12:05 PM)
 *
 *   `format` string can contain literal values. These need to be escaped by surrounding with single quotes (e.g.
 *   `"h 'in the morning'"`). In order to output a single quote, escape it - i.e., two single quotes in a sequence
 *   (e.g. `"h 'o''clock'"`).
 *
 * @param {(Date|number|string)} date Date to format either as Date object, milliseconds (string or
 *    number) or various ISO 8601 datetime string formats (e.g. yyyy-MM-ddTHH:mm:ss.sssZ and its
 *    shorter versions like yyyy-MM-ddTHH:mmZ, yyyy-MM-dd or yyyyMMddTHHmmssZ). If no timezone is
 *    specified in the string input, the time is considered to be in the local timezone.
 * @param {string=} format Formatting rules (see Description). If not specified,
 *    `mediumDate` is used.
 * @param {string=} timezone Timezone to be used for formatting. It understands UTC/GMT and the
 *    continental US time zone abbreviations, but for general use, use a time zone offset, for
 *    example, `'+0430'` (4 hours, 30 minutes east of the Greenwich meridian)
 *    If not specified, the timezone of the browser will be used.
 * @returns {string} Formatted string or the input if input is not recognized as date/millis.
 *
 * @example
   <example>
     <file name="index.html">
       <span ng-non-bindable>{{1288323623006 | date:'medium'}}</span>:
           <span>{{1288323623006 | date:'medium'}}</span><br>
       <span ng-non-bindable>{{1288323623006 | date:'yyyy-MM-dd HH:mm:ss Z'}}</span>:
          <span>{{1288323623006 | date:'yyyy-MM-dd HH:mm:ss Z'}}</span><br>
       <span ng-non-bindable>{{1288323623006 | date:'MM/dd/yyyy @ h:mma'}}</span>:
          <span>{{'1288323623006' | date:'MM/dd/yyyy @ h:mma'}}</span><br>
       <span ng-non-bindable>{{1288323623006 | date:"MM/dd/yyyy 'at' h:mma"}}</span>:
          <span>{{'1288323623006' | date:"MM/dd/yyyy 'at' h:mma"}}</span><br>
     </file>
     <file name="protractor.js" type="protractor">
       it('should format date', function() {
         expect(element(by.binding("1288323623006 | date:'medium'")).getText()).
            toMatch(/Oct 2\d, 2010 \d{1,2}:\d{2}:\d{2} (AM|PM)/);
         expect(element(by.binding("1288323623006 | date:'yyyy-MM-dd HH:mm:ss Z'")).getText()).
            toMatch(/2010\-10\-2\d \d{2}:\d{2}:\d{2} (\-|\+)?\d{4}/);
         expect(element(by.binding("'1288323623006' | date:'MM/dd/yyyy @ h:mma'")).getText()).
            toMatch(/10\/2\d\/2010 @ \d{1,2}:\d{2}(AM|PM)/);
         expect(element(by.binding("'1288323623006' | date:\"MM/dd/yyyy 'at' h:mma\"")).getText()).
            toMatch(/10\/2\d\/2010 at \d{1,2}:\d{2}(AM|PM)/);
       });
     </file>
   </example>
 */
dateFilter.$inject = ['$locale'];
function dateFilter($locale) {


  var R_ISO8601_STR = /^(\d{4})-?(\d\d)-?(\d\d)(?:T(\d\d)(?::?(\d\d)(?::?(\d\d)(?:\.(\d+))?)?)?(Z|([+-])(\d\d):?(\d\d))?)?$/;
                     // 1        2       3         4          5          6          7          8  9     10      11
  function jsonStringToDate(string) {
    var match;
    if (match = string.match(R_ISO8601_STR)) {
      var date = new Date(0),
          tzHour = 0,
          tzMin  = 0,
          dateSetter = match[8] ? date.setUTCFullYear : date.setFullYear,
          timeSetter = match[8] ? date.setUTCHours : date.setHours;

      if (match[9]) {
        tzHour = toInt(match[9] + match[10]);
        tzMin = toInt(match[9] + match[11]);
      }
      dateSetter.call(date, toInt(match[1]), toInt(match[2]) - 1, toInt(match[3]));
      var h = toInt(match[4] || 0) - tzHour;
      var m = toInt(match[5] || 0) - tzMin;
      var s = toInt(match[6] || 0);
      var ms = Math.round(parseFloat('0.' + (match[7] || 0)) * 1000);
      timeSetter.call(date, h, m, s, ms);
      return date;
    }
    return string;
  }


  return function(date, format, timezone) {
    var text = '',
        parts = [],
        fn, match;

    format = format || 'mediumDate';
    format = $locale.DATETIME_FORMATS[format] || format;
    if (isString(date)) {
      date = NUMBER_STRING.test(date) ? toInt(date) : jsonStringToDate(date);
    }

    if (isNumber(date)) {
      date = new Date(date);
    }

    if (!isDate(date) || !isFinite(date.getTime())) {
      return date;
    }

    while (format) {
      match = DATE_FORMATS_SPLIT.exec(format);
      if (match) {
        parts = concat(parts, match, 1);
        format = parts.pop();
      } else {
        parts.push(format);
        format = null;
      }
    }

    var dateTimezoneOffset = date.getTimezoneOffset();
    if (timezone) {
      dateTimezoneOffset = timezoneToOffset(timezone, date.getTimezoneOffset());
      date = convertTimezoneToLocal(date, timezone, true);
    }
    forEach(parts, function(value) {
      fn = DATE_FORMATS[value];
      text += fn ? fn(date, $locale.DATETIME_FORMATS, dateTimezoneOffset)
                 : value.replace(/(^'|'$)/g, '').replace(/''/g, "'");
    });

    return text;
  };
}


/**
 * @ngdoc filter
 * @name json
 * @kind function
 *
 * @description
 *   Allows you to convert a JavaScript object into JSON string.
 *
 *   This filter is mostly useful for debugging. When using the double curly {{value}} notation
 *   the binding is automatically converted to JSON.
 *
 * @param {*} object Any JavaScript object (including arrays and primitive types) to filter.
 * @param {number=} spacing The number of spaces to use per indentation, defaults to 2.
 * @returns {string} JSON string.
 *
 *
 * @example
   <example>
     <file name="index.html">
       <pre id="default-spacing">{{ {'name':'value'} | json }}</pre>
       <pre id="custom-spacing">{{ {'name':'value'} | json:4 }}</pre>
     </file>
     <file name="protractor.js" type="protractor">
       it('should jsonify filtered objects', function() {
         expect(element(by.id('default-spacing')).getText()).toMatch(/\{\n  "name": ?"value"\n}/);
         expect(element(by.id('custom-spacing')).getText()).toMatch(/\{\n    "name": ?"value"\n}/);
       });
     </file>
   </example>
 *
 */
function jsonFilter() {
  return function(object, spacing) {
    if (isUndefined(spacing)) {
        spacing = 2;
    }
    return toJson(object, spacing);
  };
}


/**
 * @ngdoc filter
 * @name lowercase
 * @kind function
 * @description
 * Converts string to lowercase.
 * @see angular.lowercase
 */
var lowercaseFilter = valueFn(lowercase);


/**
 * @ngdoc filter
 * @name uppercase
 * @kind function
 * @description
 * Converts string to uppercase.
 * @see angular.uppercase
 */
var uppercaseFilter = valueFn(uppercase);

/**
 * @ngdoc filter
 * @name limitTo
 * @kind function
 *
 * @description
 * Creates a new array or string containing only a specified number of elements. The elements
 * are taken from either the beginning or the end of the source array, string or number, as specified by
 * the value and sign (positive or negative) of `limit`. If a number is used as input, it is
 * converted to a string.
 *
 * @param {Array|string|number} input Source array, string or number to be limited.
 * @param {string|number} limit The length of the returned array or string. If the `limit` number
 *     is positive, `limit` number of items from the beginning of the source array/string are copied.
 *     If the number is negative, `limit` number  of items from the end of the source array/string
 *     are copied. The `limit` will be trimmed if it exceeds `array.length`. If `limit` is undefined,
 *     the input will be returned unchanged.
 * @param {(string|number)=} begin Index at which to begin limitation. As a negative index, `begin`
 *     indicates an offset from the end of `input`. Defaults to `0`.
 * @returns {Array|string} A new sub-array or substring of length `limit` or less if input array
 *     had less than `limit` elements.
 *
 * @example
   <example module="limitToExample">
     <file name="index.html">
       <script>
         angular.module('limitToExample', [])
           .controller('ExampleController', ['$scope', function($scope) {
             $scope.numbers = [1,2,3,4,5,6,7,8,9];
             $scope.letters = "abcdefghi";
             $scope.longNumber = 2345432342;
             $scope.numLimit = 3;
             $scope.letterLimit = 3;
             $scope.longNumberLimit = 3;
           }]);
       </script>
       <div ng-controller="ExampleController">
         <label>
            Limit {{numbers}} to:
            <input type="number" step="1" ng-model="numLimit">
         </label>
         <p>Output numbers: {{ numbers | limitTo:numLimit }}</p>
         <label>
            Limit {{letters}} to:
            <input type="number" step="1" ng-model="letterLimit">
         </label>
         <p>Output letters: {{ letters | limitTo:letterLimit }}</p>
         <label>
            Limit {{longNumber}} to:
            <input type="number" step="1" ng-model="longNumberLimit">
         </label>
         <p>Output long number: {{ longNumber | limitTo:longNumberLimit }}</p>
       </div>
     </file>
     <file name="protractor.js" type="protractor">
       var numLimitInput = element(by.model('numLimit'));
       var letterLimitInput = element(by.model('letterLimit'));
       var longNumberLimitInput = element(by.model('longNumberLimit'));
       var limitedNumbers = element(by.binding('numbers | limitTo:numLimit'));
       var limitedLetters = element(by.binding('letters | limitTo:letterLimit'));
       var limitedLongNumber = element(by.binding('longNumber | limitTo:longNumberLimit'));

       it('should limit the number array to first three items', function() {
         expect(numLimitInput.getAttribute('value')).toBe('3');
         expect(letterLimitInput.getAttribute('value')).toBe('3');
         expect(longNumberLimitInput.getAttribute('value')).toBe('3');
         expect(limitedNumbers.getText()).toEqual('Output numbers: [1,2,3]');
         expect(limitedLetters.getText()).toEqual('Output letters: abc');
         expect(limitedLongNumber.getText()).toEqual('Output long number: 234');
       });

       // There is a bug in safari and protractor that doesn't like the minus key
       // it('should update the output when -3 is entered', function() {
       //   numLimitInput.clear();
       //   numLimitInput.sendKeys('-3');
       //   letterLimitInput.clear();
       //   letterLimitInput.sendKeys('-3');
       //   longNumberLimitInput.clear();
       //   longNumberLimitInput.sendKeys('-3');
       //   expect(limitedNumbers.getText()).toEqual('Output numbers: [7,8,9]');
       //   expect(limitedLetters.getText()).toEqual('Output letters: ghi');
       //   expect(limitedLongNumber.getText()).toEqual('Output long number: 342');
       // });

       it('should not exceed the maximum size of input array', function() {
         numLimitInput.clear();
         numLimitInput.sendKeys('100');
         letterLimitInput.clear();
         letterLimitInput.sendKeys('100');
         longNumberLimitInput.clear();
         longNumberLimitInput.sendKeys('100');
         expect(limitedNumbers.getText()).toEqual('Output numbers: [1,2,3,4,5,6,7,8,9]');
         expect(limitedLetters.getText()).toEqual('Output letters: abcdefghi');
         expect(limitedLongNumber.getText()).toEqual('Output long number: 2345432342');
       });
     </file>
   </example>
*/
function limitToFilter() {
  return function(input, limit, begin) {
    if (Math.abs(Number(limit)) === Infinity) {
      limit = Number(limit);
    } else {
      limit = toInt(limit);
    }
    if (isNaN(limit)) return input;

    if (isNumber(input)) input = input.toString();
    if (!isArray(input) && !isString(input)) return input;

    begin = (!begin || isNaN(begin)) ? 0 : toInt(begin);
    begin = (begin < 0) ? Math.max(0, input.length + begin) : begin;

    if (limit >= 0) {
      return input.slice(begin, begin + limit);
    } else {
      if (begin === 0) {
        return input.slice(limit, input.length);
      } else {
        return input.slice(Math.max(0, begin + limit), begin);
      }
    }
  };
}

/**
 * @ngdoc filter
 * @name orderBy
 * @kind function
 *
 * @description
 * Orders a specified `array` by the `expression` predicate. It is ordered alphabetically
 * for strings and numerically for numbers. Note: if you notice numbers are not being sorted
 * as expected, make sure they are actually being saved as numbers and not strings.
 *
 * @param {Array} array The array to sort.
 * @param {function(*)|string|Array.<(function(*)|string)>=} expression A predicate to be
 *    used by the comparator to determine the order of elements.
 *
 *    Can be one of:
 *
 *    - `function`: Getter function. The result of this function will be sorted using the
 *      `<`, `===`, `>` operator.
 *    - `string`: An Angular expression. The result of this expression is used to compare elements
 *      (for example `name` to sort by a property called `name` or `name.substr(0, 3)` to sort by
 *      3 first characters of a property called `name`). The result of a constant expression
 *      is interpreted as a property name to be used in comparisons (for example `"special name"`
 *      to sort object by the value of their `special name` property). An expression can be
 *      optionally prefixed with `+` or `-` to control ascending or descending sort order
 *      (for example, `+name` or `-name`). If no property is provided, (e.g. `'+'`) then the array
 *      element itself is used to compare where sorting.
 *    - `Array`: An array of function or string predicates. The first predicate in the array
 *      is used for sorting, but when two items are equivalent, the next predicate is used.
 *
 *    If the predicate is missing or empty then it defaults to `'+'`.
 *
 * @param {boolean=} reverse Reverse the order of the array.
 * @returns {Array} Sorted copy of the source array.
 *
 *
 * @example
 * The example below demonstrates a simple ngRepeat, where the data is sorted
 * by age in descending order (predicate is set to `'-age'`).
 * `reverse` is not set, which means it defaults to `false`.
   <example module="orderByExample">
     <file name="index.html">
       <script>
         angular.module('orderByExample', [])
           .controller('ExampleController', ['$scope', function($scope) {
             $scope.friends =
                 [{name:'John', phone:'555-1212', age:10},
                  {name:'Mary', phone:'555-9876', age:19},
                  {name:'Mike', phone:'555-4321', age:21},
                  {name:'Adam', phone:'555-5678', age:35},
                  {name:'Julie', phone:'555-8765', age:29}];
           }]);
       </script>
       <div ng-controller="ExampleController">
         <table class="friend">
           <tr>
             <th>Name</th>
             <th>Phone Number</th>
             <th>Age</th>
           </tr>
           <tr ng-repeat="friend in friends | orderBy:'-age'">
             <td>{{friend.name}}</td>
             <td>{{friend.phone}}</td>
             <td>{{friend.age}}</td>
           </tr>
         </table>
       </div>
     </file>
   </example>
 *
 * The predicate and reverse parameters can be controlled dynamically through scope properties,
 * as shown in the next example.
 * @example
   <example module="orderByExample">
     <file name="index.html">
       <script>
         angular.module('orderByExample', [])
           .controller('ExampleController', ['$scope', function($scope) {
             $scope.friends =
                 [{name:'John', phone:'555-1212', age:10},
                  {name:'Mary', phone:'555-9876', age:19},
                  {name:'Mike', phone:'555-4321', age:21},
                  {name:'Adam', phone:'555-5678', age:35},
                  {name:'Julie', phone:'555-8765', age:29}];
             $scope.predicate = 'age';
             $scope.reverse = true;
             $scope.order = function(predicate) {
               $scope.reverse = ($scope.predicate === predicate) ? !$scope.reverse : false;
               $scope.predicate = predicate;
             };
           }]);
       </script>
       <style type="text/css">
         .sortorder:after {
           content: '\25b2';
         }
         .sortorder.reverse:after {
           content: '\25bc';
         }
       </style>
       <div ng-controller="ExampleController">
         <pre>Sorting predicate = {{predicate}}; reverse = {{reverse}}</pre>
         <hr/>
         [ <a href="" ng-click="predicate=''">unsorted</a> ]
         <table class="friend">
           <tr>
             <th>
               <a href="" ng-click="order('name')">Name</a>
               <span class="sortorder" ng-show="predicate === 'name'" ng-class="{reverse:reverse}"></span>
             </th>
             <th>
               <a href="" ng-click="order('phone')">Phone Number</a>
               <span class="sortorder" ng-show="predicate === 'phone'" ng-class="{reverse:reverse}"></span>
             </th>
             <th>
               <a href="" ng-click="order('age')">Age</a>
               <span class="sortorder" ng-show="predicate === 'age'" ng-class="{reverse:reverse}"></span>
             </th>
           </tr>
           <tr ng-repeat="friend in friends | orderBy:predicate:reverse">
             <td>{{friend.name}}</td>
             <td>{{friend.phone}}</td>
             <td>{{friend.age}}</td>
           </tr>
         </table>
       </div>
     </file>
   </example>
 *
 * It's also possible to call the orderBy filter manually, by injecting `$filter`, retrieving the
 * filter routine with `$filter('orderBy')`, and calling the returned filter routine with the
 * desired parameters.
 *
 * Example:
 *
 * @example
  <example module="orderByExample">
    <file name="index.html">
      <div ng-controller="ExampleController">
        <table class="friend">
          <tr>
            <th><a href="" ng-click="reverse=false;order('name', false)">Name</a>
              (<a href="" ng-click="order('-name',false)">^</a>)</th>
            <th><a href="" ng-click="reverse=!reverse;order('phone', reverse)">Phone Number</a></th>
            <th><a href="" ng-click="reverse=!reverse;order('age',reverse)">Age</a></th>
          </tr>
          <tr ng-repeat="friend in friends">
            <td>{{friend.name}}</td>
            <td>{{friend.phone}}</td>
            <td>{{friend.age}}</td>
          </tr>
        </table>
      </div>
    </file>

    <file name="script.js">
      angular.module('orderByExample', [])
        .controller('ExampleController', ['$scope', '$filter', function($scope, $filter) {
          var orderBy = $filter('orderBy');
          $scope.friends = [
            { name: 'John',    phone: '555-1212',    age: 10 },
            { name: 'Mary',    phone: '555-9876',    age: 19 },
            { name: 'Mike',    phone: '555-4321',    age: 21 },
            { name: 'Adam',    phone: '555-5678',    age: 35 },
            { name: 'Julie',   phone: '555-8765',    age: 29 }
          ];
          $scope.order = function(predicate, reverse) {
            $scope.friends = orderBy($scope.friends, predicate, reverse);
          };
          $scope.order('-age',false);
        }]);
    </file>
</example>
 */
orderByFilter.$inject = ['$parse'];
function orderByFilter($parse) {
  return function(array, sortPredicate, reverseOrder) {

    if (!(isArrayLike(array))) return array;

    if (!isArray(sortPredicate)) { sortPredicate = [sortPredicate]; }
    if (sortPredicate.length === 0) { sortPredicate = ['+']; }

    var predicates = processPredicates(sortPredicate, reverseOrder);
    // Add a predicate at the end that evaluates to the element index. This makes the
    // sort stable as it works as a tie-breaker when all the input predicates cannot
    // distinguish between two elements.
    predicates.push({ get: function() { return {}; }, descending: reverseOrder ? -1 : 1});

    // The next three lines are a version of a Swartzian Transform idiom from Perl
    // (sometimes called the Decorate-Sort-Undecorate idiom)
    // See https://en.wikipedia.org/wiki/Schwartzian_transform
    var compareValues = Array.prototype.map.call(array, getComparisonObject);
    compareValues.sort(doComparison);
    array = compareValues.map(function(item) { return item.value; });

    return array;

    function getComparisonObject(value, index) {
      return {
        value: value,
        predicateValues: predicates.map(function(predicate) {
          return getPredicateValue(predicate.get(value), index);
        })
      };
    }

    function doComparison(v1, v2) {
      var result = 0;
      for (var index=0, length = predicates.length; index < length; ++index) {
        result = compare(v1.predicateValues[index], v2.predicateValues[index]) * predicates[index].descending;
        if (result) break;
      }
      return result;
    }
  };

  function processPredicates(sortPredicate, reverseOrder) {
    reverseOrder = reverseOrder ? -1 : 1;
    return sortPredicate.map(function(predicate) {
      var descending = 1, get = identity;

      if (isFunction(predicate)) {
        get = predicate;
      } else if (isString(predicate)) {
        if ((predicate.charAt(0) == '+' || predicate.charAt(0) == '-')) {
          descending = predicate.charAt(0) == '-' ? -1 : 1;
          predicate = predicate.substring(1);
        }
        if (predicate !== '') {
          get = $parse(predicate);
          if (get.constant) {
            var key = get();
            get = function(value) { return value[key]; };
          }
        }
      }
      return { get: get, descending: descending * reverseOrder };
    });
  }

  function isPrimitive(value) {
    switch (typeof value) {
      case 'number': /* falls through */
      case 'boolean': /* falls through */
      case 'string':
        return true;
      default:
        return false;
    }
  }

  function objectValue(value, index) {
    // If `valueOf` is a valid function use that
    if (typeof value.valueOf === 'function') {
      value = value.valueOf();
      if (isPrimitive(value)) return value;
    }
    // If `toString` is a valid function and not the one from `Object.prototype` use that
    if (hasCustomToString(value)) {
      value = value.toString();
      if (isPrimitive(value)) return value;
    }
    // We have a basic object so we use the position of the object in the collection
    return index;
  }

  function getPredicateValue(value, index) {
    var type = typeof value;
    if (value === null) {
      type = 'string';
      value = 'null';
    } else if (type === 'string') {
      value = value.toLowerCase();
    } else if (type === 'object') {
      value = objectValue(value, index);
    }
    return { value: value, type: type };
  }

  function compare(v1, v2) {
    var result = 0;
    if (v1.type === v2.type) {
      if (v1.value !== v2.value) {
        result = v1.value < v2.value ? -1 : 1;
      }
    } else {
      result = v1.type < v2.type ? -1 : 1;
    }
    return result;
  }
}

function ngDirective(directive) {
  if (isFunction(directive)) {
    directive = {
      link: directive
    };
  }
  directive.restrict = directive.restrict || 'AC';
  return valueFn(directive);
}

/**
 * @ngdoc directive
 * @name a
 * @restrict E
 *
 * @description
 * Modifies the default behavior of the html A tag so that the default action is prevented when
 * the href attribute is empty.
 *
 * This change permits the easy creation of action links with the `ngClick` directive
 * without changing the location or causing page reloads, e.g.:
 * `<a href="" ng-click="list.addItem()">Add Item</a>`
 */
var htmlAnchorDirective = valueFn({
  restrict: 'E',
  compile: function(element, attr) {
    if (!attr.href && !attr.xlinkHref) {
      return function(scope, element) {
        // If the linked element is not an anchor tag anymore, do nothing
        if (element[0].nodeName.toLowerCase() !== 'a') return;

        // SVGAElement does not use the href attribute, but rather the 'xlinkHref' attribute.
        var href = toString.call(element.prop('href')) === '[object SVGAnimatedString]' ?
                   'xlink:href' : 'href';
        element.on('click', function(event) {
          // if we have no href url, then don't navigate anywhere.
          if (!element.attr(href)) {
            event.preventDefault();
          }
        });
      };
    }
  }
});

/**
 * @ngdoc directive
 * @name ngHref
 * @restrict A
 * @priority 99
 *
 * @description
 * Using Angular markup like `{{hash}}` in an href attribute will
 * make the link go to the wrong URL if the user clicks it before
 * Angular has a chance to replace the `{{hash}}` markup with its
 * value. Until Angular replaces the markup the link will be broken
 * and will most likely return a 404 error. The `ngHref` directive
 * solves this problem.
 *
 * The wrong way to write it:
 * ```html
 * <a href="http://www.gravatar.com/avatar/{{hash}}">link1</a>
 * ```
 *
 * The correct way to write it:
 * ```html
 * <a ng-href="http://www.gravatar.com/avatar/{{hash}}">link1</a>
 * ```
 *
 * @element A
 * @param {template} ngHref any string which can contain `{{}}` markup.
 *
 * @example
 * This example shows various combinations of `href`, `ng-href` and `ng-click` attributes
 * in links and their different behaviors:
    <example>
      <file name="index.html">
        <input ng-model="value" /><br />
        <a id="link-1" href ng-click="value = 1">link 1</a> (link, don't reload)<br />
        <a id="link-2" href="" ng-click="value = 2">link 2</a> (link, don't reload)<br />
        <a id="link-3" ng-href="/{{'123'}}">link 3</a> (link, reload!)<br />
        <a id="link-4" href="" name="xx" ng-click="value = 4">anchor</a> (link, don't reload)<br />
        <a id="link-5" name="xxx" ng-click="value = 5">anchor</a> (no link)<br />
        <a id="link-6" ng-href="{{value}}">link</a> (link, change location)
      </file>
      <file name="protractor.js" type="protractor">
        it('should execute ng-click but not reload when href without value', function() {
          element(by.id('link-1')).click();
          expect(element(by.model('value')).getAttribute('value')).toEqual('1');
          expect(element(by.id('link-1')).getAttribute('href')).toBe('');
        });

        it('should execute ng-click but not reload when href empty string', function() {
          element(by.id('link-2')).click();
          expect(element(by.model('value')).getAttribute('value')).toEqual('2');
          expect(element(by.id('link-2')).getAttribute('href')).toBe('');
        });

        it('should execute ng-click and change url when ng-href specified', function() {
          expect(element(by.id('link-3')).getAttribute('href')).toMatch(/\/123$/);

          element(by.id('link-3')).click();

          // At this point, we navigate away from an Angular page, so we need
          // to use browser.driver to get the base webdriver.

          browser.wait(function() {
            return browser.driver.getCurrentUrl().then(function(url) {
              return url.match(/\/123$/);
            });
          }, 5000, 'page should navigate to /123');
        });

        it('should execute ng-click but not reload when href empty string and name specified', function() {
          element(by.id('link-4')).click();
          expect(element(by.model('value')).getAttribute('value')).toEqual('4');
          expect(element(by.id('link-4')).getAttribute('href')).toBe('');
        });

        it('should execute ng-click but not reload when no href but name specified', function() {
          element(by.id('link-5')).click();
          expect(element(by.model('value')).getAttribute('value')).toEqual('5');
          expect(element(by.id('link-5')).getAttribute('href')).toBe(null);
        });

        it('should only change url when only ng-href', function() {
          element(by.model('value')).clear();
          element(by.model('value')).sendKeys('6');
          expect(element(by.id('link-6')).getAttribute('href')).toMatch(/\/6$/);

          element(by.id('link-6')).click();

          // At this point, we navigate away from an Angular page, so we need
          // to use browser.driver to get the base webdriver.
          browser.wait(function() {
            return browser.driver.getCurrentUrl().then(function(url) {
              return url.match(/\/6$/);
            });
          }, 5000, 'page should navigate to /6');
        });
      </file>
    </example>
 */

/**
 * @ngdoc directive
 * @name ngSrc
 * @restrict A
 * @priority 99
 *
 * @description
 * Using Angular markup like `{{hash}}` in a `src` attribute doesn't
 * work right: The browser will fetch from the URL with the literal
 * text `{{hash}}` until Angular replaces the expression inside
 * `{{hash}}`. The `ngSrc` directive solves this problem.
 *
 * The buggy way to write it:
 * ```html
 * <img src="http://www.gravatar.com/avatar/{{hash}}" alt="Description"/>
 * ```
 *
 * The correct way to write it:
 * ```html
 * <img ng-src="http://www.gravatar.com/avatar/{{hash}}" alt="Description" />
 * ```
 *
 * @element IMG
 * @param {template} ngSrc any string which can contain `{{}}` markup.
 */

/**
 * @ngdoc directive
 * @name ngSrcset
 * @restrict A
 * @priority 99
 *
 * @description
 * Using Angular markup like `{{hash}}` in a `srcset` attribute doesn't
 * work right: The browser will fetch from the URL with the literal
 * text `{{hash}}` until Angular replaces the expression inside
 * `{{hash}}`. The `ngSrcset` directive solves this problem.
 *
 * The buggy way to write it:
 * ```html
 * <img srcset="http://www.gravatar.com/avatar/{{hash}} 2x" alt="Description"/>
 * ```
 *
 * The correct way to write it:
 * ```html
 * <img ng-srcset="http://www.gravatar.com/avatar/{{hash}} 2x" alt="Description" />
 * ```
 *
 * @element IMG
 * @param {template} ngSrcset any string which can contain `{{}}` markup.
 */

/**
 * @ngdoc directive
 * @name ngDisabled
 * @restrict A
 * @priority 100
 *
 * @description
 *
 * This directive sets the `disabled` attribute on the element if the
 * {@link guide/expression expression} inside `ngDisabled` evaluates to truthy.
 *
 * A special directive is necessary because we cannot use interpolation inside the `disabled`
 * attribute.  The following example would make the button enabled on Chrome/Firefox
 * but not on older IEs:
 *
 * ```html
 * <!-- See below for an example of ng-disabled being used correctly -->
 * <div ng-init="isDisabled = false">
 *  <button disabled="{{isDisabled}}">Disabled</button>
 * </div>
 * ```
 *
 * This is because the HTML specification does not require browsers to preserve the values of
 * boolean attributes such as `disabled` (Their presence means true and their absence means false.)
 * If we put an Angular interpolation expression into such an attribute then the
 * binding information would be lost when the browser removes the attribute.
 *
 * @example
    <example>
      <file name="index.html">
        <label>Click me to toggle: <input type="checkbox" ng-model="checked"></label><br/>
        <button ng-model="button" ng-disabled="checked">Button</button>
      </file>
      <file name="protractor.js" type="protractor">
        it('should toggle button', function() {
          expect(element(by.css('button')).getAttribute('disabled')).toBeFalsy();
          element(by.model('checked')).click();
          expect(element(by.css('button')).getAttribute('disabled')).toBeTruthy();
        });
      </file>
    </example>
 *
 * @element INPUT
 * @param {expression} ngDisabled If the {@link guide/expression expression} is truthy,
 *     then the `disabled` attribute will be set on the element
 */


/**
 * @ngdoc directive
 * @name ngChecked
 * @restrict A
 * @priority 100
 *
 * @description
 * Sets the `checked` attribute on the element, if the expression inside `ngChecked` is truthy.
 *
 * Note that this directive should not be used together with {@link ngModel `ngModel`},
 * as this can lead to unexpected behavior.
 *
 * ### Why do we need `ngChecked`?
 *
 * The HTML specification does not require browsers to preserve the values of boolean attributes
 * such as checked. (Their presence means true and their absence means false.)
 * If we put an Angular interpolation expression into such an attribute then the
 * binding information would be lost when the browser removes the attribute.
 * The `ngChecked` directive solves this problem for the `checked` attribute.
 * This complementary directive is not removed by the browser and so provides
 * a permanent reliable place to store the binding information.
 * @example
    <example>
      <file name="index.html">
        <label>Check me to check both: <input type="checkbox" ng-model="master"></label><br/>
        <input id="checkSlave" type="checkbox" ng-checked="master" aria-label="Slave input">
      </file>
      <file name="protractor.js" type="protractor">
        it('should check both checkBoxes', function() {
          expect(element(by.id('checkSlave')).getAttribute('checked')).toBeFalsy();
          element(by.model('master')).click();
          expect(element(by.id('checkSlave')).getAttribute('checked')).toBeTruthy();
        });
      </file>
    </example>
 *
 * @element INPUT
 * @param {expression} ngChecked If the {@link guide/expression expression} is truthy,
 *     then the `checked` attribute will be set on the element
 */


/**
 * @ngdoc directive
 * @name ngReadonly
 * @restrict A
 * @priority 100
 *
 * @description
 * The HTML specification does not require browsers to preserve the values of boolean attributes
 * such as readonly. (Their presence means true and their absence means false.)
 * If we put an Angular interpolation expression into such an attribute then the
 * binding information would be lost when the browser removes the attribute.
 * The `ngReadonly` directive solves this problem for the `readonly` attribute.
 * This complementary directive is not removed by the browser and so provides
 * a permanent reliable place to store the binding information.
 * @example
    <example>
      <file name="index.html">
        <label>Check me to make text readonly: <input type="checkbox" ng-model="checked"></label><br/>
        <input type="text" ng-readonly="checked" value="I'm Angular" aria-label="Readonly field" />
      </file>
      <file name="protractor.js" type="protractor">
        it('should toggle readonly attr', function() {
          expect(element(by.css('[type="text"]')).getAttribute('readonly')).toBeFalsy();
          element(by.model('checked')).click();
          expect(element(by.css('[type="text"]')).getAttribute('readonly')).toBeTruthy();
        });
      </file>
    </example>
 *
 * @element INPUT
 * @param {expression} ngReadonly If the {@link guide/expression expression} is truthy,
 *     then special attribute "readonly" will be set on the element
 */


/**
 * @ngdoc directive
 * @name ngSelected
 * @restrict A
 * @priority 100
 *
 * @description
 * The HTML specification does not require browsers to preserve the values of boolean attributes
 * such as selected. (Their presence means true and their absence means false.)
 * If we put an Angular interpolation expression into such an attribute then the
 * binding information would be lost when the browser removes the attribute.
 * The `ngSelected` directive solves this problem for the `selected` attribute.
 * This complementary directive is not removed by the browser and so provides
 * a permanent reliable place to store the binding information.
 *
 * @example
    <example>
      <file name="index.html">
        <label>Check me to select: <input type="checkbox" ng-model="selected"></label><br/>
        <select aria-label="ngSelected demo">
          <option>Hello!</option>
          <option id="greet" ng-selected="selected">Greetings!</option>
        </select>
      </file>
      <file name="protractor.js" type="protractor">
        it('should select Greetings!', function() {
          expect(element(by.id('greet')).getAttribute('selected')).toBeFalsy();
          element(by.model('selected')).click();
          expect(element(by.id('greet')).getAttribute('selected')).toBeTruthy();
        });
      </file>
    </example>
 *
 * @element OPTION
 * @param {expression} ngSelected If the {@link guide/expression expression} is truthy,
 *     then special attribute "selected" will be set on the element
 */

/**
 * @ngdoc directive
 * @name ngOpen
 * @restrict A
 * @priority 100
 *
 * @description
 * The HTML specification does not require browsers to preserve the values of boolean attributes
 * such as open. (Their presence means true and their absence means false.)
 * If we put an Angular interpolation expression into such an attribute then the
 * binding information would be lost when the browser removes the attribute.
 * The `ngOpen` directive solves this problem for the `open` attribute.
 * This complementary directive is not removed by the browser and so provides
 * a permanent reliable place to store the binding information.
 * @example
     <example>
       <file name="index.html">
         <label>Check me check multiple: <input type="checkbox" ng-model="open"></label><br/>
         <details id="details" ng-open="open">
            <summary>Show/Hide me</summary>
         </details>
       </file>
       <file name="protractor.js" type="protractor">
         it('should toggle open', function() {
           expect(element(by.id('details')).getAttribute('open')).toBeFalsy();
           element(by.model('open')).click();
           expect(element(by.id('details')).getAttribute('open')).toBeTruthy();
         });
       </file>
     </example>
 *
 * @element DETAILS
 * @param {expression} ngOpen If the {@link guide/expression expression} is truthy,
 *     then special attribute "open" will be set on the element
 */

var ngAttributeAliasDirectives = {};

// boolean attrs are evaluated
forEach(BOOLEAN_ATTR, function(propName, attrName) {
  // binding to multiple is not supported
  if (propName == "multiple") return;

  function defaultLinkFn(scope, element, attr) {
    scope.$watch(attr[normalized], function ngBooleanAttrWatchAction(value) {
      attr.$set(attrName, !!value);
    });
  }

  var normalized = directiveNormalize('ng-' + attrName);
  var linkFn = defaultLinkFn;

  if (propName === 'checked') {
    linkFn = function(scope, element, attr) {
      // ensuring ngChecked doesn't interfere with ngModel when both are set on the same input
      if (attr.ngModel !== attr[normalized]) {
        defaultLinkFn(scope, element, attr);
      }
    };
  }

  ngAttributeAliasDirectives[normalized] = function() {
    return {
      restrict: 'A',
      priority: 100,
      link: linkFn
    };
  };
});

// aliased input attrs are evaluated
forEach(ALIASED_ATTR, function(htmlAttr, ngAttr) {
  ngAttributeAliasDirectives[ngAttr] = function() {
    return {
      priority: 100,
      link: function(scope, element, attr) {
        //special case ngPattern when a literal regular expression value
        //is used as the expression (this way we don't have to watch anything).
        if (ngAttr === "ngPattern" && attr.ngPattern.charAt(0) == "/") {
          var match = attr.ngPattern.match(REGEX_STRING_REGEXP);
          if (match) {
            attr.$set("ngPattern", new RegExp(match[1], match[2]));
            return;
          }
        }

        scope.$watch(attr[ngAttr], function ngAttrAliasWatchAction(value) {
          attr.$set(ngAttr, value);
        });
      }
    };
  };
});

// ng-src, ng-srcset, ng-href are interpolated
forEach(['src', 'srcset', 'href'], function(attrName) {
  var normalized = directiveNormalize('ng-' + attrName);
  ngAttributeAliasDirectives[normalized] = function() {
    return {
      priority: 99, // it needs to run after the attributes are interpolated
      link: function(scope, element, attr) {
        var propName = attrName,
            name = attrName;

        if (attrName === 'href' &&
            toString.call(element.prop('href')) === '[object SVGAnimatedString]') {
          name = 'xlinkHref';
          attr.$attr[name] = 'xlink:href';
          propName = null;
        }

        attr.$observe(normalized, function(value) {
          if (!value) {
            if (attrName === 'href') {
              attr.$set(name, null);
            }
            return;
          }

          attr.$set(name, value);

          // on IE, if "ng:src" directive declaration is used and "src" attribute doesn't exist
          // then calling element.setAttribute('src', 'foo') doesn't do anything, so we need
          // to set the property as well to achieve the desired effect.
          // we use attr[attrName] value since $set can sanitize the url.
          if (msie && propName) element.prop(propName, attr[name]);
        });
      }
    };
  };
});

/* global -nullFormCtrl, -SUBMITTED_CLASS, addSetValidityMethod: true
 */
var nullFormCtrl = {
  $addControl: noop,
  $$renameControl: nullFormRenameControl,
  $removeControl: noop,
  $setValidity: noop,
  $setDirty: noop,
  $setPristine: noop,
  $setSubmitted: noop
},
SUBMITTED_CLASS = 'ng-submitted';

function nullFormRenameControl(control, name) {
  control.$name = name;
}

/**
 * @ngdoc type
 * @name form.FormController
 *
 * @property {boolean} $pristine True if user has not interacted with the form yet.
 * @property {boolean} $dirty True if user has already interacted with the form.
 * @property {boolean} $valid True if all of the containing forms and controls are valid.
 * @property {boolean} $invalid True if at least one containing control or form is invalid.
 * @property {boolean} $pending True if at least one containing control or form is pending.
 * @property {boolean} $submitted True if user has submitted the form even if its invalid.
 *
 * @property {Object} $error Is an object hash, containing references to controls or
 *  forms with failing validators, where:
 *
 *  - keys are validation tokens (error names),
 *  - values are arrays of controls or forms that have a failing validator for given error name.
 *
 *  Built-in validation tokens:
 *
 *  - `email`
 *  - `max`
 *  - `maxlength`
 *  - `min`
 *  - `minlength`
 *  - `number`
 *  - `pattern`
 *  - `required`
 *  - `url`
 *  - `date`
 *  - `datetimelocal`
 *  - `time`
 *  - `week`
 *  - `month`
 *
 * @description
 * `FormController` keeps track of all its controls and nested forms as well as the state of them,
 * such as being valid/invalid or dirty/pristine.
 *
 * Each {@link ng.directive:form form} directive creates an instance
 * of `FormController`.
 *
 */
//asks for $scope to fool the BC controller module
FormController.$inject = ['$element', '$attrs', '$scope', '$animate', '$interpolate'];
function FormController(element, attrs, $scope, $animate, $interpolate) {
  var form = this,
      controls = [];

  // init state
  form.$error = {};
  form.$$success = {};
  form.$pending = undefined;
  form.$name = $interpolate(attrs.name || attrs.ngForm || '')($scope);
  form.$dirty = false;
  form.$pristine = true;
  form.$valid = true;
  form.$invalid = false;
  form.$submitted = false;
  form.$$parentForm = nullFormCtrl;

  /**
   * @ngdoc method
   * @name form.FormController#$rollbackViewValue
   *
   * @description
   * Rollback all form controls pending updates to the `$modelValue`.
   *
   * Updates may be pending by a debounced event or because the input is waiting for a some future
   * event defined in `ng-model-options`. This method is typically needed by the reset button of
   * a form that uses `ng-model-options` to pend updates.
   */
  form.$rollbackViewValue = function() {
    forEach(controls, function(control) {
      control.$rollbackViewValue();
    });
  };

  /**
   * @ngdoc method
   * @name form.FormController#$commitViewValue
   *
   * @description
   * Commit all form controls pending updates to the `$modelValue`.
   *
   * Updates may be pending by a debounced event or because the input is waiting for a some future
   * event defined in `ng-model-options`. This method is rarely needed as `NgModelController`
   * usually handles calling this in response to input events.
   */
  form.$commitViewValue = function() {
    forEach(controls, function(control) {
      control.$commitViewValue();
    });
  };

  /**
   * @ngdoc method
   * @name form.FormController#$addControl
   * @param {object} control control object, either a {@link form.FormController} or an
   * {@link ngModel.NgModelController}
   *
   * @description
   * Register a control with the form. Input elements using ngModelController do this automatically
   * when they are linked.
   *
   * Note that the current state of the control will not be reflected on the new parent form. This
   * is not an issue with normal use, as freshly compiled and linked controls are in a `$pristine`
   * state.
   *
   * However, if the method is used programmatically, for example by adding dynamically created controls,
   * or controls that have been previously removed without destroying their corresponding DOM element,
   * it's the developers responsiblity to make sure the current state propagates to the parent form.
   *
   * For example, if an input control is added that is already `$dirty` and has `$error` properties,
   * calling `$setDirty()` and `$validate()` afterwards will propagate the state to the parent form.
   */
  form.$addControl = function(control) {
    // Breaking change - before, inputs whose name was "hasOwnProperty" were quietly ignored
    // and not added to the scope.  Now we throw an error.
    assertNotHasOwnProperty(control.$name, 'input');
    controls.push(control);

    if (control.$name) {
      form[control.$name] = control;
    }

    control.$$parentForm = form;
  };

  // Private API: rename a form control
  form.$$renameControl = function(control, newName) {
    var oldName = control.$name;

    if (form[oldName] === control) {
      delete form[oldName];
    }
    form[newName] = control;
    control.$name = newName;
  };

  /**
   * @ngdoc method
   * @name form.FormController#$removeControl
   * @param {object} control control object, either a {@link form.FormController} or an
   * {@link ngModel.NgModelController}
   *
   * @description
   * Deregister a control from the form.
   *
   * Input elements using ngModelController do this automatically when they are destroyed.
   *
   * Note that only the removed control's validation state (`$errors`etc.) will be removed from the
   * form. `$dirty`, `$submitted` states will not be changed, because the expected behavior can be
   * different from case to case. For example, removing the only `$dirty` control from a form may or
   * may not mean that the form is still `$dirty`.
   */
  form.$removeControl = function(control) {
    if (control.$name && form[control.$name] === control) {
      delete form[control.$name];
    }
    forEach(form.$pending, function(value, name) {
      form.$setValidity(name, null, control);
    });
    forEach(form.$error, function(value, name) {
      form.$setValidity(name, null, control);
    });
    forEach(form.$$success, function(value, name) {
      form.$setValidity(name, null, control);
    });

    arrayRemove(controls, control);
    control.$$parentForm = nullFormCtrl;
  };


  /**
   * @ngdoc method
   * @name form.FormController#$setValidity
   *
   * @description
   * Sets the validity of a form control.
   *
   * This method will also propagate to parent forms.
   */
  addSetValidityMethod({
    ctrl: this,
    $element: element,
    set: function(object, property, controller) {
      var list = object[property];
      if (!list) {
        object[property] = [controller];
      } else {
        var index = list.indexOf(controller);
        if (index === -1) {
          list.push(controller);
        }
      }
    },
    unset: function(object, property, controller) {
      var list = object[property];
      if (!list) {
        return;
      }
      arrayRemove(list, controller);
      if (list.length === 0) {
        delete object[property];
      }
    },
    $animate: $animate
  });

  /**
   * @ngdoc method
   * @name form.FormController#$setDirty
   *
   * @description
   * Sets the form to a dirty state.
   *
   * This method can be called to add the 'ng-dirty' class and set the form to a dirty
   * state (ng-dirty class). This method will also propagate to parent forms.
   */
  form.$setDirty = function() {
    $animate.removeClass(element, PRISTINE_CLASS);
    $animate.addClass(element, DIRTY_CLASS);
    form.$dirty = true;
    form.$pristine = false;
    form.$$parentForm.$setDirty();
  };

  /**
   * @ngdoc method
   * @name form.FormController#$setPristine
   *
   * @description
   * Sets the form to its pristine state.
   *
   * This method can be called to remove the 'ng-dirty' class and set the form to its pristine
   * state (ng-pristine class). This method will also propagate to all the controls contained
   * in this form.
   *
   * Setting a form back to a pristine state is often useful when we want to 'reuse' a form after
   * saving or resetting it.
   */
  form.$setPristine = function() {
    $animate.setClass(element, PRISTINE_CLASS, DIRTY_CLASS + ' ' + SUBMITTED_CLASS);
    form.$dirty = false;
    form.$pristine = true;
    form.$submitted = false;
    forEach(controls, function(control) {
      control.$setPristine();
    });
  };

  /**
   * @ngdoc method
   * @name form.FormController#$setUntouched
   *
   * @description
   * Sets the form to its untouched state.
   *
   * This method can be called to remove the 'ng-touched' class and set the form controls to their
   * untouched state (ng-untouched class).
   *
   * Setting a form controls back to their untouched state is often useful when setting the form
   * back to its pristine state.
   */
  form.$setUntouched = function() {
    forEach(controls, function(control) {
      control.$setUntouched();
    });
  };

  /**
   * @ngdoc method
   * @name form.FormController#$setSubmitted
   *
   * @description
   * Sets the form to its submitted state.
   */
  form.$setSubmitted = function() {
    $animate.addClass(element, SUBMITTED_CLASS);
    form.$submitted = true;
    form.$$parentForm.$setSubmitted();
  };
}

/**
 * @ngdoc directive
 * @name ngForm
 * @restrict EAC
 *
 * @description
 * Nestable alias of {@link ng.directive:form `form`} directive. HTML
 * does not allow nesting of form elements. It is useful to nest forms, for example if the validity of a
 * sub-group of controls needs to be determined.
 *
 * Note: the purpose of `ngForm` is to group controls,
 * but not to be a replacement for the `<form>` tag with all of its capabilities
 * (e.g. posting to the server, ...).
 *
 * @param {string=} ngForm|name Name of the form. If specified, the form controller will be published into
 *                       related scope, under this name.
 *
 */

 /**
 * @ngdoc directive
 * @name form
 * @restrict E
 *
 * @description
 * Directive that instantiates
 * {@link form.FormController FormController}.
 *
 * If the `name` attribute is specified, the form controller is published onto the current scope under
 * this name.
 *
 * # Alias: {@link ng.directive:ngForm `ngForm`}
 *
 * In Angular, forms can be nested. This means that the outer form is valid when all of the child
 * forms are valid as well. However, browsers do not allow nesting of `<form>` elements, so
 * Angular provides the {@link ng.directive:ngForm `ngForm`} directive which behaves identically to
 * `<form>` but can be nested.  This allows you to have nested forms, which is very useful when
 * using Angular validation directives in forms that are dynamically generated using the
 * {@link ng.directive:ngRepeat `ngRepeat`} directive. Since you cannot dynamically generate the `name`
 * attribute of input elements using interpolation, you have to wrap each set of repeated inputs in an
 * `ngForm` directive and nest these in an outer `form` element.
 *
 *
 * # CSS classes
 *  - `ng-valid` is set if the form is valid.
 *  - `ng-invalid` is set if the form is invalid.
 *  - `ng-pending` is set if the form is pending.
 *  - `ng-pristine` is set if the form is pristine.
 *  - `ng-dirty` is set if the form is dirty.
 *  - `ng-submitted` is set if the form was submitted.
 *
 * Keep in mind that ngAnimate can detect each of these classes when added and removed.
 *
 *
 * # Submitting a form and preventing the default action
 *
 * Since the role of forms in client-side Angular applications is different than in classical
 * roundtrip apps, it is desirable for the browser not to translate the form submission into a full
 * page reload that sends the data to the server. Instead some javascript logic should be triggered
 * to handle the form submission in an application-specific way.
 *
 * For this reason, Angular prevents the default action (form submission to the server) unless the
 * `<form>` element has an `action` attribute specified.
 *
 * You can use one of the following two ways to specify what javascript method should be called when
 * a form is submitted:
 *
 * - {@link ng.directive:ngSubmit ngSubmit} directive on the form element
 * - {@link ng.directive:ngClick ngClick} directive on the first
  *  button or input field of type submit (input[type=submit])
 *
 * To prevent double execution of the handler, use only one of the {@link ng.directive:ngSubmit ngSubmit}
 * or {@link ng.directive:ngClick ngClick} directives.
 * This is because of the following form submission rules in the HTML specification:
 *
 * - If a form has only one input field then hitting enter in this field triggers form submit
 * (`ngSubmit`)
 * - if a form has 2+ input fields and no buttons or input[type=submit] then hitting enter
 * doesn't trigger submit
 * - if a form has one or more input fields and one or more buttons or input[type=submit] then
 * hitting enter in any of the input fields will trigger the click handler on the *first* button or
 * input[type=submit] (`ngClick`) *and* a submit handler on the enclosing form (`ngSubmit`)
 *
 * Any pending `ngModelOptions` changes will take place immediately when an enclosing form is
 * submitted. Note that `ngClick` events will occur before the model is updated. Use `ngSubmit`
 * to have access to the updated model.
 *
 * ## Animation Hooks
 *
 * Animations in ngForm are triggered when any of the associated CSS classes are added and removed.
 * These classes are: `.ng-pristine`, `.ng-dirty`, `.ng-invalid` and `.ng-valid` as well as any
 * other validations that are performed within the form. Animations in ngForm are similar to how
 * they work in ngClass and animations can be hooked into using CSS transitions, keyframes as well
 * as JS animations.
 *
 * The following example shows a simple way to utilize CSS transitions to style a form element
 * that has been rendered as invalid after it has been validated:
 *
 * <pre>
 * //be sure to include ngAnimate as a module to hook into more
 * //advanced animations
 * .my-form {
 *   transition:0.5s linear all;
 *   background: white;
 * }
 * .my-form.ng-invalid {
 *   background: red;
 *   color:white;
 * }
 * </pre>
 *
 * @example
    <example deps="angular-animate.js" animations="true" fixBase="true" module="formExample">
      <file name="index.html">
       <script>
         angular.module('formExample', [])
           .controller('FormController', ['$scope', function($scope) {
             $scope.userType = 'guest';
           }]);
       </script>
       <style>
        .my-form {
          transition:all linear 0.5s;
          background: transparent;
        }
        .my-form.ng-invalid {
          background: red;
        }
       </style>
       <form name="myForm" ng-controller="FormController" class="my-form">
         userType: <input name="input" ng-model="userType" required>
         <span class="error" ng-show="myForm.input.$error.required">Required!</span><br>
         <code>userType = {{userType}}</code><br>
         <code>myForm.input.$valid = {{myForm.input.$valid}}</code><br>
         <code>myForm.input.$error = {{myForm.input.$error}}</code><br>
         <code>myForm.$valid = {{myForm.$valid}}</code><br>
         <code>myForm.$error.required = {{!!myForm.$error.required}}</code><br>
        </form>
      </file>
      <file name="protractor.js" type="protractor">
        it('should initialize to model', function() {
          var userType = element(by.binding('userType'));
          var valid = element(by.binding('myForm.input.$valid'));

          expect(userType.getText()).toContain('guest');
          expect(valid.getText()).toContain('true');
        });

        it('should be invalid if empty', function() {
          var userType = element(by.binding('userType'));
          var valid = element(by.binding('myForm.input.$valid'));
          var userInput = element(by.model('userType'));

          userInput.clear();
          userInput.sendKeys('');

          expect(userType.getText()).toEqual('userType =');
          expect(valid.getText()).toContain('false');
        });
      </file>
    </example>
 *
 * @param {string=} name Name of the form. If specified, the form controller will be published into
 *                       related scope, under this name.
 */
var formDirectiveFactory = function(isNgForm) {
  return ['$timeout', '$parse', function($timeout, $parse) {
    var formDirective = {
      name: 'form',
      restrict: isNgForm ? 'EAC' : 'E',
      require: ['form', '^^?form'], //first is the form's own ctrl, second is an optional parent form
      controller: FormController,
      compile: function ngFormCompile(formElement, attr) {
        // Setup initial state of the control
        formElement.addClass(PRISTINE_CLASS).addClass(VALID_CLASS);

        var nameAttr = attr.name ? 'name' : (isNgForm && attr.ngForm ? 'ngForm' : false);

        return {
          pre: function ngFormPreLink(scope, formElement, attr, ctrls) {
            var controller = ctrls[0];

            // if `action` attr is not present on the form, prevent the default action (submission)
            if (!('action' in attr)) {
              // we can't use jq events because if a form is destroyed during submission the default
              // action is not prevented. see #1238
              //
              // IE 9 is not affected because it doesn't fire a submit event and try to do a full
              // page reload if the form was destroyed by submission of the form via a click handler
              // on a button in the form. Looks like an IE9 specific bug.
              var handleFormSubmission = function(event) {
                scope.$apply(function() {
                  controller.$commitViewValue();
                  controller.$setSubmitted();
                });

                event.preventDefault();
              };

              addEventListenerFn(formElement[0], 'submit', handleFormSubmission);

              // unregister the preventDefault listener so that we don't not leak memory but in a
              // way that will achieve the prevention of the default action.
              formElement.on('$destroy', function() {
                $timeout(function() {
                  removeEventListenerFn(formElement[0], 'submit', handleFormSubmission);
                }, 0, false);
              });
            }

            var parentFormCtrl = ctrls[1] || controller.$$parentForm;
            parentFormCtrl.$addControl(controller);

            var setter = nameAttr ? getSetter(controller.$name) : noop;

            if (nameAttr) {
              setter(scope, controller);
              attr.$observe(nameAttr, function(newValue) {
                if (controller.$name === newValue) return;
                setter(scope, undefined);
                controller.$$parentForm.$$renameControl(controller, newValue);
                setter = getSetter(controller.$name);
                setter(scope, controller);
              });
            }
            formElement.on('$destroy', function() {
              controller.$$parentForm.$removeControl(controller);
              setter(scope, undefined);
              extend(controller, nullFormCtrl); //stop propagating child destruction handlers upwards
            });
          }
        };
      }
    };

    return formDirective;

    function getSetter(expression) {
      if (expression === '') {
        //create an assignable expression, so forms with an empty name can be renamed later
        return $parse('this[""]').assign;
      }
      return $parse(expression).assign || noop;
    }
  }];
};

var formDirective = formDirectiveFactory();
var ngFormDirective = formDirectiveFactory(true);

/* global VALID_CLASS: false,
  INVALID_CLASS: false,
  PRISTINE_CLASS: false,
  DIRTY_CLASS: false,
  UNTOUCHED_CLASS: false,
  TOUCHED_CLASS: false,
  ngModelMinErr: false,
*/

// Regex code is obtained from SO: https://stackoverflow.com/questions/3143070/javascript-regex-iso-datetime#answer-3143231
var ISO_DATE_REGEXP = /\d{4}-[01]\d-[0-3]\dT[0-2]\d:[0-5]\d:[0-5]\d\.\d+([+-][0-2]\d:[0-5]\d|Z)/;
// See valid URLs in RFC3987 (http://tools.ietf.org/html/rfc3987)
var URL_REGEXP = /^[A-Za-z][A-Za-z\d.+-]*:\/*(?:\w+(?::\w+)?@)?[^\s/]+(?::\d+)?(?:\/[\w#!:.?+=&%@\-/]*)?$/;
var EMAIL_REGEXP = /^[a-z0-9!#$%&'*+\/=?^_`{|}~.-]+@[a-z0-9]([a-z0-9-]*[a-z0-9])?(\.[a-z0-9]([a-z0-9-]*[a-z0-9])?)*$/i;
var NUMBER_REGEXP = /^\s*(\-|\+)?(\d+|(\d*(\.\d*)))([eE][+-]?\d+)?\s*$/;
var DATE_REGEXP = /^(\d{4})-(\d{2})-(\d{2})$/;
var DATETIMELOCAL_REGEXP = /^(\d{4})-(\d\d)-(\d\d)T(\d\d):(\d\d)(?::(\d\d)(\.\d{1,3})?)?$/;
var WEEK_REGEXP = /^(\d{4})-W(\d\d)$/;
var MONTH_REGEXP = /^(\d{4})-(\d\d)$/;
var TIME_REGEXP = /^(\d\d):(\d\d)(?::(\d\d)(\.\d{1,3})?)?$/;

var inputType = {

  /**
   * @ngdoc input
   * @name input[text]
   *
   * @description
   * Standard HTML text input with angular data binding, inherited by most of the `input` elements.
   *
   *
   * @param {string} ngModel Assignable angular expression to data-bind to.
   * @param {string=} name Property name of the form under which the control is published.
   * @param {string=} required Adds `required` validation error key if the value is not entered.
   * @param {string=} ngRequired Adds `required` attribute and `required` validation constraint to
   *    the element when the ngRequired expression evaluates to true. Use `ngRequired` instead of
   *    `required` when you want to data-bind to the `required` attribute.
   * @param {number=} ngMinlength Sets `minlength` validation error key if the value is shorter than
   *    minlength.
   * @param {number=} ngMaxlength Sets `maxlength` validation error key if the value is longer than
   *    maxlength. Setting the attribute to a negative or non-numeric value, allows view values of
   *    any length.
   * @param {string=} pattern Similar to `ngPattern` except that the attribute value is the actual string
   *    that contains the regular expression body that will be converted to a regular expression
   *    as in the ngPattern directive.
   * @param {string=} ngPattern Sets `pattern` validation error key if the ngModel value does not match
   *    a RegExp found by evaluating the Angular expression given in the attribute value.
   *    If the expression evaluates to a RegExp object, then this is used directly.
   *    If the expression evaluates to a string, then it will be converted to a RegExp
   *    after wrapping it in `^` and `$` characters. For instance, `"abc"` will be converted to
   *    `new RegExp('^abc$')`.<br />
   *    **Note:** Avoid using the `g` flag on the RegExp, as it will cause each successive search to
   *    start at the index of the last search's match, thus not taking the whole input value into
   *    account.
   * @param {string=} ngChange Angular expression to be executed when input changes due to user
   *    interaction with the input element.
   * @param {boolean=} [ngTrim=true] If set to false Angular will not automatically trim the input.
   *    This parameter is ignored for input[type=password] controls, which will never trim the
   *    input.
   *
   * @example
      <example name="text-input-directive" module="textInputExample">
        <file name="index.html">
         <script>
           angular.module('textInputExample', [])
             .controller('ExampleController', ['$scope', function($scope) {
               $scope.example = {
                 text: 'guest',
                 word: /^\s*\w*\s*$/
               };
             }]);
         </script>
         <form name="myForm" ng-controller="ExampleController">
           <label>Single word:
             <input type="text" name="input" ng-model="example.text"
                    ng-pattern="example.word" required ng-trim="false">
           </label>
           <div role="alert">
             <span class="error" ng-show="myForm.input.$error.required">
               Required!</span>
             <span class="error" ng-show="myForm.input.$error.pattern">
               Single word only!</span>
           </div>
           <tt>text = {{example.text}}</tt><br/>
           <tt>myForm.input.$valid = {{myForm.input.$valid}}</tt><br/>
           <tt>myForm.input.$error = {{myForm.input.$error}}</tt><br/>
           <tt>myForm.$valid = {{myForm.$valid}}</tt><br/>
           <tt>myForm.$error.required = {{!!myForm.$error.required}}</tt><br/>
          </form>
        </file>
        <file name="protractor.js" type="protractor">
          var text = element(by.binding('example.text'));
          var valid = element(by.binding('myForm.input.$valid'));
          var input = element(by.model('example.text'));

          it('should initialize to model', function() {
            expect(text.getText()).toContain('guest');
            expect(valid.getText()).toContain('true');
          });

          it('should be invalid if empty', function() {
            input.clear();
            input.sendKeys('');

            expect(text.getText()).toEqual('text =');
            expect(valid.getText()).toContain('false');
          });

          it('should be invalid if multi word', function() {
            input.clear();
            input.sendKeys('hello world');

            expect(valid.getText()).toContain('false');
          });
        </file>
      </example>
   */
  'text': textInputType,

    /**
     * @ngdoc input
     * @name input[date]
     *
     * @description
     * Input with date validation and transformation. In browsers that do not yet support
     * the HTML5 date input, a text element will be used. In that case, text must be entered in a valid ISO-8601
     * date format (yyyy-MM-dd), for example: `2009-01-06`. Since many
     * modern browsers do not yet support this input type, it is important to provide cues to users on the
     * expected input format via a placeholder or label.
     *
     * The model must always be a Date object, otherwise Angular will throw an error.
     * Invalid `Date` objects (dates whose `getTime()` is `NaN`) will be rendered as an empty string.
     *
     * The timezone to be used to read/write the `Date` instance in the model can be defined using
     * {@link ng.directive:ngModelOptions ngModelOptions}. By default, this is the timezone of the browser.
     *
     * @param {string} ngModel Assignable angular expression to data-bind to.
     * @param {string=} name Property name of the form under which the control is published.
     * @param {string=} min Sets the `min` validation error key if the value entered is less than `min`. This must be a
     *   valid ISO date string (yyyy-MM-dd). You can also use interpolation inside this attribute
     *   (e.g. `min="{{minDate | date:'yyyy-MM-dd'}}"`). Note that `min` will also add native HTML5
     *   constraint validation.
     * @param {string=} max Sets the `max` validation error key if the value entered is greater than `max`. This must be
     *   a valid ISO date string (yyyy-MM-dd). You can also use interpolation inside this attribute
     *   (e.g. `max="{{maxDate | date:'yyyy-MM-dd'}}"`). Note that `max` will also add native HTML5
     *   constraint validation.
     * @param {(date|string)=} ngMin Sets the `min` validation constraint to the Date / ISO date string
     *   the `ngMin` expression evaluates to. Note that it does not set the `min` attribute.
     * @param {(date|string)=} ngMax Sets the `max` validation constraint to the Date / ISO date string
     *   the `ngMax` expression evaluates to. Note that it does not set the `max` attribute.
     * @param {string=} required Sets `required` validation error key if the value is not entered.
     * @param {string=} ngRequired Adds `required` attribute and `required` validation constraint to
     *    the element when the ngRequired expression evaluates to true. Use `ngRequired` instead of
     *    `required` when you want to data-bind to the `required` attribute.
     * @param {string=} ngChange Angular expression to be executed when input changes due to user
     *    interaction with the input element.
     *
     * @example
     <example name="date-input-directive" module="dateInputExample">
     <file name="index.html">
       <script>
          angular.module('dateInputExample', [])
            .controller('DateController', ['$scope', function($scope) {
              $scope.example = {
                value: new Date(2013, 9, 22)
              };
            }]);
       </script>
       <form name="myForm" ng-controller="DateController as dateCtrl">
          <label for="exampleInput">Pick a date in 2013:</label>
          <input type="date" id="exampleInput" name="input" ng-model="example.value"
              placeholder="yyyy-MM-dd" min="2013-01-01" max="2013-12-31" required />
          <div role="alert">
            <span class="error" ng-show="myForm.input.$error.required">
                Required!</span>
            <span class="error" ng-show="myForm.input.$error.date">
                Not a valid date!</span>
           </div>
           <tt>value = {{example.value | date: "yyyy-MM-dd"}}</tt><br/>
           <tt>myForm.input.$valid = {{myForm.input.$valid}}</tt><br/>
           <tt>myForm.input.$error = {{myForm.input.$error}}</tt><br/>
           <tt>myForm.$valid = {{myForm.$valid}}</tt><br/>
           <tt>myForm.$error.required = {{!!myForm.$error.required}}</tt><br/>
       </form>
     </file>
     <file name="protractor.js" type="protractor">
        var value = element(by.binding('example.value | date: "yyyy-MM-dd"'));
        var valid = element(by.binding('myForm.input.$valid'));
        var input = element(by.model('example.value'));

        // currently protractor/webdriver does not support
        // sending keys to all known HTML5 input controls
        // for various browsers (see https://github.com/angular/protractor/issues/562).
        function setInput(val) {
          // set the value of the element and force validation.
          var scr = "var ipt = document.getElementById('exampleInput'); " +
          "ipt.value = '" + val + "';" +
          "angular.element(ipt).scope().$apply(function(s) { s.myForm[ipt.name].$setViewValue('" + val + "'); });";
          browser.executeScript(scr);
        }

        it('should initialize to model', function() {
          expect(value.getText()).toContain('2013-10-22');
          expect(valid.getText()).toContain('myForm.input.$valid = true');
        });

        it('should be invalid if empty', function() {
          setInput('');
          expect(value.getText()).toEqual('value =');
          expect(valid.getText()).toContain('myForm.input.$valid = false');
        });

        it('should be invalid if over max', function() {
          setInput('2015-01-01');
          expect(value.getText()).toContain('');
          expect(valid.getText()).toContain('myForm.input.$valid = false');
        });
     </file>
     </example>
     */
  'date': createDateInputType('date', DATE_REGEXP,
         createDateParser(DATE_REGEXP, ['yyyy', 'MM', 'dd']),
         'yyyy-MM-dd'),

   /**
    * @ngdoc input
    * @name input[datetime-local]
    *
    * @description
    * Input with datetime validation and transformation. In browsers that do not yet support
    * the HTML5 date input, a text element will be used. In that case, the text must be entered in a valid ISO-8601
    * local datetime format (yyyy-MM-ddTHH:mm:ss), for example: `2010-12-28T14:57:00`.
    *
    * The model must always be a Date object, otherwise Angular will throw an error.
    * Invalid `Date` objects (dates whose `getTime()` is `NaN`) will be rendered as an empty string.
    *
    * The timezone to be used to read/write the `Date` instance in the model can be defined using
    * {@link ng.directive:ngModelOptions ngModelOptions}. By default, this is the timezone of the browser.
    *
    * @param {string} ngModel Assignable angular expression to data-bind to.
    * @param {string=} name Property name of the form under which the control is published.
    * @param {string=} min Sets the `min` validation error key if the value entered is less than `min`.
    *   This must be a valid ISO datetime format (yyyy-MM-ddTHH:mm:ss). You can also use interpolation
    *   inside this attribute (e.g. `min="{{minDatetimeLocal | date:'yyyy-MM-ddTHH:mm:ss'}}"`).
    *   Note that `min` will also add native HTML5 constraint validation.
    * @param {string=} max Sets the `max` validation error key if the value entered is greater than `max`.
    *   This must be a valid ISO datetime format (yyyy-MM-ddTHH:mm:ss). You can also use interpolation
    *   inside this attribute (e.g. `max="{{maxDatetimeLocal | date:'yyyy-MM-ddTHH:mm:ss'}}"`).
    *   Note that `max` will also add native HTML5 constraint validation.
    * @param {(date|string)=} ngMin Sets the `min` validation error key to the Date / ISO datetime string
    *   the `ngMin` expression evaluates to. Note that it does not set the `min` attribute.
    * @param {(date|string)=} ngMax Sets the `max` validation error key to the Date / ISO datetime string
    *   the `ngMax` expression evaluates to. Note that it does not set the `max` attribute.
    * @param {string=} required Sets `required` validation error key if the value is not entered.
    * @param {string=} ngRequired Adds `required` attribute and `required` validation constraint to
    *    the element when the ngRequired expression evaluates to true. Use `ngRequired` instead of
    *    `required` when you want to data-bind to the `required` attribute.
    * @param {string=} ngChange Angular expression to be executed when input changes due to user
    *    interaction with the input element.
    *
    * @example
    <example name="datetimelocal-input-directive" module="dateExample">
    <file name="index.html">
      <script>
        angular.module('dateExample', [])
          .controller('DateController', ['$scope', function($scope) {
            $scope.example = {
              value: new Date(2010, 11, 28, 14, 57)
            };
          }]);
      </script>
      <form name="myForm" ng-controller="DateController as dateCtrl">
        <label for="exampleInput">Pick a date between in 2013:</label>
        <input type="datetime-local" id="exampleInput" name="input" ng-model="example.value"
            placeholder="yyyy-MM-ddTHH:mm:ss" min="2001-01-01T00:00:00" max="2013-12-31T00:00:00" required />
        <div role="alert">
          <span class="error" ng-show="myForm.input.$error.required">
              Required!</span>
          <span class="error" ng-show="myForm.input.$error.datetimelocal">
              Not a valid date!</span>
        </div>
        <tt>value = {{example.value | date: "yyyy-MM-ddTHH:mm:ss"}}</tt><br/>
        <tt>myForm.input.$valid = {{myForm.input.$valid}}</tt><br/>
        <tt>myForm.input.$error = {{myForm.input.$error}}</tt><br/>
        <tt>myForm.$valid = {{myForm.$valid}}</tt><br/>
        <tt>myForm.$error.required = {{!!myForm.$error.required}}</tt><br/>
      </form>
    </file>
    <file name="protractor.js" type="protractor">
      var value = element(by.binding('example.value | date: "yyyy-MM-ddTHH:mm:ss"'));
      var valid = element(by.binding('myForm.input.$valid'));
      var input = element(by.model('example.value'));

      // currently protractor/webdriver does not support
      // sending keys to all known HTML5 input controls
      // for various browsers (https://github.com/angular/protractor/issues/562).
      function setInput(val) {
        // set the value of the element and force validation.
        var scr = "var ipt = document.getElementById('exampleInput'); " +
        "ipt.value = '" + val + "';" +
        "angular.element(ipt).scope().$apply(function(s) { s.myForm[ipt.name].$setViewValue('" + val + "'); });";
        browser.executeScript(scr);
      }

      it('should initialize to model', function() {
        expect(value.getText()).toContain('2010-12-28T14:57:00');
        expect(valid.getText()).toContain('myForm.input.$valid = true');
      });

      it('should be invalid if empty', function() {
        setInput('');
        expect(value.getText()).toEqual('value =');
        expect(valid.getText()).toContain('myForm.input.$valid = false');
      });

      it('should be invalid if over max', function() {
        setInput('2015-01-01T23:59:00');
        expect(value.getText()).toContain('');
        expect(valid.getText()).toContain('myForm.input.$valid = false');
      });
    </file>
    </example>
    */
  'datetime-local': createDateInputType('datetimelocal', DATETIMELOCAL_REGEXP,
      createDateParser(DATETIMELOCAL_REGEXP, ['yyyy', 'MM', 'dd', 'HH', 'mm', 'ss', 'sss']),
      'yyyy-MM-ddTHH:mm:ss.sss'),

  /**
   * @ngdoc input
   * @name input[time]
   *
   * @description
   * Input with time validation and transformation. In browsers that do not yet support
   * the HTML5 date input, a text element will be used. In that case, the text must be entered in a valid ISO-8601
   * local time format (HH:mm:ss), for example: `14:57:00`. Model must be a Date object. This binding will always output a
   * Date object to the model of January 1, 1970, or local date `new Date(1970, 0, 1, HH, mm, ss)`.
   *
   * The model must always be a Date object, otherwise Angular will throw an error.
   * Invalid `Date` objects (dates whose `getTime()` is `NaN`) will be rendered as an empty string.
   *
   * The timezone to be used to read/write the `Date` instance in the model can be defined using
   * {@link ng.directive:ngModelOptions ngModelOptions}. By default, this is the timezone of the browser.
   *
   * @param {string} ngModel Assignable angular expression to data-bind to.
   * @param {string=} name Property name of the form under which the control is published.
   * @param {string=} min Sets the `min` validation error key if the value entered is less than `min`.
   *   This must be a valid ISO time format (HH:mm:ss). You can also use interpolation inside this
   *   attribute (e.g. `min="{{minTime | date:'HH:mm:ss'}}"`). Note that `min` will also add
   *   native HTML5 constraint validation.
   * @param {string=} max Sets the `max` validation error key if the value entered is greater than `max`.
   *   This must be a valid ISO time format (HH:mm:ss). You can also use interpolation inside this
   *   attribute (e.g. `max="{{maxTime | date:'HH:mm:ss'}}"`). Note that `max` will also add
   *   native HTML5 constraint validation.
   * @param {(date|string)=} ngMin Sets the `min` validation constraint to the Date / ISO time string the
   *   `ngMin` expression evaluates to. Note that it does not set the `min` attribute.
   * @param {(date|string)=} ngMax Sets the `max` validation constraint to the Date / ISO time string the
   *   `ngMax` expression evaluates to. Note that it does not set the `max` attribute.
   * @param {string=} required Sets `required` validation error key if the value is not entered.
   * @param {string=} ngRequired Adds `required` attribute and `required` validation constraint to
   *    the element when the ngRequired expression evaluates to true. Use `ngRequired` instead of
   *    `required` when you want to data-bind to the `required` attribute.
   * @param {string=} ngChange Angular expression to be executed when input changes due to user
   *    interaction with the input element.
   *
   * @example
   <example name="time-input-directive" module="timeExample">
   <file name="index.html">
     <script>
      angular.module('timeExample', [])
        .controller('DateController', ['$scope', function($scope) {
          $scope.example = {
            value: new Date(1970, 0, 1, 14, 57, 0)
          };
        }]);
     </script>
     <form name="myForm" ng-controller="DateController as dateCtrl">
        <label for="exampleInput">Pick a between 8am and 5pm:</label>
        <input type="time" id="exampleInput" name="input" ng-model="example.value"
            placeholder="HH:mm:ss" min="08:00:00" max="17:00:00" required />
        <div role="alert">
          <span class="error" ng-show="myForm.input.$error.required">
              Required!</span>
          <span class="error" ng-show="myForm.input.$error.time">
              Not a valid date!</span>
        </div>
        <tt>value = {{example.value | date: "HH:mm:ss"}}</tt><br/>
        <tt>myForm.input.$valid = {{myForm.input.$valid}}</tt><br/>
        <tt>myForm.input.$error = {{myForm.input.$error}}</tt><br/>
        <tt>myForm.$valid = {{myForm.$valid}}</tt><br/>
        <tt>myForm.$error.required = {{!!myForm.$error.required}}</tt><br/>
     </form>
   </file>
   <file name="protractor.js" type="protractor">
      var value = element(by.binding('example.value | date: "HH:mm:ss"'));
      var valid = element(by.binding('myForm.input.$valid'));
      var input = element(by.model('example.value'));

      // currently protractor/webdriver does not support
      // sending keys to all known HTML5 input controls
      // for various browsers (https://github.com/angular/protractor/issues/562).
      function setInput(val) {
        // set the value of the element and force validation.
        var scr = "var ipt = document.getElementById('exampleInput'); " +
        "ipt.value = '" + val + "';" +
        "angular.element(ipt).scope().$apply(function(s) { s.myForm[ipt.name].$setViewValue('" + val + "'); });";
        browser.executeScript(scr);
      }

      it('should initialize to model', function() {
        expect(value.getText()).toContain('14:57:00');
        expect(valid.getText()).toContain('myForm.input.$valid = true');
      });

      it('should be invalid if empty', function() {
        setInput('');
        expect(value.getText()).toEqual('value =');
        expect(valid.getText()).toContain('myForm.input.$valid = false');
      });

      it('should be invalid if over max', function() {
        setInput('23:59:00');
        expect(value.getText()).toContain('');
        expect(valid.getText()).toContain('myForm.input.$valid = false');
      });
   </file>
   </example>
   */
  'time': createDateInputType('time', TIME_REGEXP,
      createDateParser(TIME_REGEXP, ['HH', 'mm', 'ss', 'sss']),
     'HH:mm:ss.sss'),

   /**
    * @ngdoc input
    * @name input[week]
    *
    * @description
    * Input with week-of-the-year validation and transformation to Date. In browsers that do not yet support
    * the HTML5 week input, a text element will be used. In that case, the text must be entered in a valid ISO-8601
    * week format (yyyy-W##), for example: `2013-W02`.
    *
    * The model must always be a Date object, otherwise Angular will throw an error.
    * Invalid `Date` objects (dates whose `getTime()` is `NaN`) will be rendered as an empty string.
    *
    * The timezone to be used to read/write the `Date` instance in the model can be defined using
    * {@link ng.directive:ngModelOptions ngModelOptions}. By default, this is the timezone of the browser.
    *
    * @param {string} ngModel Assignable angular expression to data-bind to.
    * @param {string=} name Property name of the form under which the control is published.
    * @param {string=} min Sets the `min` validation error key if the value entered is less than `min`.
    *   This must be a valid ISO week format (yyyy-W##). You can also use interpolation inside this
    *   attribute (e.g. `min="{{minWeek | date:'yyyy-Www'}}"`). Note that `min` will also add
    *   native HTML5 constraint validation.
    * @param {string=} max Sets the `max` validation error key if the value entered is greater than `max`.
    *   This must be a valid ISO week format (yyyy-W##). You can also use interpolation inside this
    *   attribute (e.g. `max="{{maxWeek | date:'yyyy-Www'}}"`). Note that `max` will also add
    *   native HTML5 constraint validation.
    * @param {(date|string)=} ngMin Sets the `min` validation constraint to the Date / ISO week string
    *   the `ngMin` expression evaluates to. Note that it does not set the `min` attribute.
    * @param {(date|string)=} ngMax Sets the `max` validation constraint to the Date / ISO week string
    *   the `ngMax` expression evaluates to. Note that it does not set the `max` attribute.
    * @param {string=} required Sets `required` validation error key if the value is not entered.
    * @param {string=} ngRequired Adds `required` attribute and `required` validation constraint to
    *    the element when the ngRequired expression evaluates to true. Use `ngRequired` instead of
    *    `required` when you want to data-bind to the `required` attribute.
    * @param {string=} ngChange Angular expression to be executed when input changes due to user
    *    interaction with the input element.
    *
    * @example
    <example name="week-input-directive" module="weekExample">
    <file name="index.html">
      <script>
      angular.module('weekExample', [])
        .controller('DateController', ['$scope', function($scope) {
          $scope.example = {
            value: new Date(2013, 0, 3)
          };
        }]);
      </script>
      <form name="myForm" ng-controller="DateController as dateCtrl">
        <label>Pick a date between in 2013:
          <input id="exampleInput" type="week" name="input" ng-model="example.value"
                 placeholder="YYYY-W##" min="2012-W32"
                 max="2013-W52" required />
        </label>
        <div role="alert">
          <span class="error" ng-show="myForm.input.$error.required">
              Required!</span>
          <span class="error" ng-show="myForm.input.$error.week">
              Not a valid date!</span>
        </div>
        <tt>value = {{example.value | date: "yyyy-Www"}}</tt><br/>
        <tt>myForm.input.$valid = {{myForm.input.$valid}}</tt><br/>
        <tt>myForm.input.$error = {{myForm.input.$error}}</tt><br/>
        <tt>myForm.$valid = {{myForm.$valid}}</tt><br/>
        <tt>myForm.$error.required = {{!!myForm.$error.required}}</tt><br/>
      </form>
    </file>
    <file name="protractor.js" type="protractor">
      var value = element(by.binding('example.value | date: "yyyy-Www"'));
      var valid = element(by.binding('myForm.input.$valid'));
      var input = element(by.model('example.value'));

      // currently protractor/webdriver does not support
      // sending keys to all known HTML5 input controls
      // for various browsers (https://github.com/angular/protractor/issues/562).
      function setInput(val) {
        // set the value of the element and force validation.
        var scr = "var ipt = document.getElementById('exampleInput'); " +
        "ipt.value = '" + val + "';" +
        "angular.element(ipt).scope().$apply(function(s) { s.myForm[ipt.name].$setViewValue('" + val + "'); });";
        browser.executeScript(scr);
      }

      it('should initialize to model', function() {
        expect(value.getText()).toContain('2013-W01');
        expect(valid.getText()).toContain('myForm.input.$valid = true');
      });

      it('should be invalid if empty', function() {
        setInput('');
        expect(value.getText()).toEqual('value =');
        expect(valid.getText()).toContain('myForm.input.$valid = false');
      });

      it('should be invalid if over max', function() {
        setInput('2015-W01');
        expect(value.getText()).toContain('');
        expect(valid.getText()).toContain('myForm.input.$valid = false');
      });
    </file>
    </example>
    */
  'week': createDateInputType('week', WEEK_REGEXP, weekParser, 'yyyy-Www'),

  /**
   * @ngdoc input
   * @name input[month]
   *
   * @description
   * Input with month validation and transformation. In browsers that do not yet support
   * the HTML5 month input, a text element will be used. In that case, the text must be entered in a valid ISO-8601
   * month format (yyyy-MM), for example: `2009-01`.
   *
   * The model must always be a Date object, otherwise Angular will throw an error.
   * Invalid `Date` objects (dates whose `getTime()` is `NaN`) will be rendered as an empty string.
   * If the model is not set to the first of the month, the next view to model update will set it
   * to the first of the month.
   *
   * The timezone to be used to read/write the `Date` instance in the model can be defined using
   * {@link ng.directive:ngModelOptions ngModelOptions}. By default, this is the timezone of the browser.
   *
   * @param {string} ngModel Assignable angular expression to data-bind to.
   * @param {string=} name Property name of the form under which the control is published.
   * @param {string=} min Sets the `min` validation error key if the value entered is less than `min`.
   *   This must be a valid ISO month format (yyyy-MM). You can also use interpolation inside this
   *   attribute (e.g. `min="{{minMonth | date:'yyyy-MM'}}"`). Note that `min` will also add
   *   native HTML5 constraint validation.
   * @param {string=} max Sets the `max` validation error key if the value entered is greater than `max`.
   *   This must be a valid ISO month format (yyyy-MM). You can also use interpolation inside this
   *   attribute (e.g. `max="{{maxMonth | date:'yyyy-MM'}}"`). Note that `max` will also add
   *   native HTML5 constraint validation.
   * @param {(date|string)=} ngMin Sets the `min` validation constraint to the Date / ISO week string
   *   the `ngMin` expression evaluates to. Note that it does not set the `min` attribute.
   * @param {(date|string)=} ngMax Sets the `max` validation constraint to the Date / ISO week string
   *   the `ngMax` expression evaluates to. Note that it does not set the `max` attribute.

   * @param {string=} required Sets `required` validation error key if the value is not entered.
   * @param {string=} ngRequired Adds `required` attribute and `required` validation constraint to
   *    the element when the ngRequired expression evaluates to true. Use `ngRequired` instead of
   *    `required` when you want to data-bind to the `required` attribute.
   * @param {string=} ngChange Angular expression to be executed when input changes due to user
   *    interaction with the input element.
   *
   * @example
   <example name="month-input-directive" module="monthExample">
   <file name="index.html">
     <script>
      angular.module('monthExample', [])
        .controller('DateController', ['$scope', function($scope) {
          $scope.example = {
            value: new Date(2013, 9, 1)
          };
        }]);
     </script>
     <form name="myForm" ng-controller="DateController as dateCtrl">
       <label for="exampleInput">Pick a month in 2013:</label>
       <input id="exampleInput" type="month" name="input" ng-model="example.value"
          placeholder="yyyy-MM" min="2013-01" max="2013-12" required />
       <div role="alert">
         <span class="error" ng-show="myForm.input.$error.required">
            Required!</span>
         <span class="error" ng-show="myForm.input.$error.month">
            Not a valid month!</span>
       </div>
       <tt>value = {{example.value | date: "yyyy-MM"}}</tt><br/>
       <tt>myForm.input.$valid = {{myForm.input.$valid}}</tt><br/>
       <tt>myForm.input.$error = {{myForm.input.$error}}</tt><br/>
       <tt>myForm.$valid = {{myForm.$valid}}</tt><br/>
       <tt>myForm.$error.required = {{!!myForm.$error.required}}</tt><br/>
     </form>
   </file>
   <file name="protractor.js" type="protractor">
      var value = element(by.binding('example.value | date: "yyyy-MM"'));
      var valid = element(by.binding('myForm.input.$valid'));
      var input = element(by.model('example.value'));

      // currently protractor/webdriver does not support
      // sending keys to all known HTML5 input controls
      // for various browsers (https://github.com/angular/protractor/issues/562).
      function setInput(val) {
        // set the value of the element and force validation.
        var scr = "var ipt = document.getElementById('exampleInput'); " +
        "ipt.value = '" + val + "';" +
        "angular.element(ipt).scope().$apply(function(s) { s.myForm[ipt.name].$setViewValue('" + val + "'); });";
        browser.executeScript(scr);
      }

      it('should initialize to model', function() {
        expect(value.getText()).toContain('2013-10');
        expect(valid.getText()).toContain('myForm.input.$valid = true');
      });

      it('should be invalid if empty', function() {
        setInput('');
        expect(value.getText()).toEqual('value =');
        expect(valid.getText()).toContain('myForm.input.$valid = false');
      });

      it('should be invalid if over max', function() {
        setInput('2015-01');
        expect(value.getText()).toContain('');
        expect(valid.getText()).toContain('myForm.input.$valid = false');
      });
   </file>
   </example>
   */
  'month': createDateInputType('month', MONTH_REGEXP,
     createDateParser(MONTH_REGEXP, ['yyyy', 'MM']),
     'yyyy-MM'),

  /**
   * @ngdoc input
   * @name input[number]
   *
   * @description
   * Text input with number validation and transformation. Sets the `number` validation
   * error if not a valid number.
   *
   * <div class="alert alert-warning">
   * The model must always be of type `number` otherwise Angular will throw an error.
   * Be aware that a string containing a number is not enough. See the {@link ngModel:numfmt}
   * error docs for more information and an example of how to convert your model if necessary.
   * </div>
   *
   * ## Issues with HTML5 constraint validation
   *
   * In browsers that follow the
   * [HTML5 specification](https://html.spec.whatwg.org/multipage/forms.html#number-state-%28type=number%29),
   * `input[number]` does not work as expected with {@link ngModelOptions `ngModelOptions.allowInvalid`}.
   * If a non-number is entered in the input, the browser will report the value as an empty string,
   * which means the view / model values in `ngModel` and subsequently the scope value
   * will also be an empty string.
   *
   *
   * @param {string} ngModel Assignable angular expression to data-bind to.
   * @param {string=} name Property name of the form under which the control is published.
   * @param {string=} min Sets the `min` validation error key if the value entered is less than `min`.
   * @param {string=} max Sets the `max` validation error key if the value entered is greater than `max`.
   * @param {string=} required Sets `required` validation error key if the value is not entered.
   * @param {string=} ngRequired Adds `required` attribute and `required` validation constraint to
   *    the element when the ngRequired expression evaluates to true. Use `ngRequired` instead of
   *    `required` when you want to data-bind to the `required` attribute.
   * @param {number=} ngMinlength Sets `minlength` validation error key if the value is shorter than
   *    minlength.
   * @param {number=} ngMaxlength Sets `maxlength` validation error key if the value is longer than
   *    maxlength. Setting the attribute to a negative or non-numeric value, allows view values of
   *    any length.
   * @param {string=} pattern Similar to `ngPattern` except that the attribute value is the actual string
   *    that contains the regular expression body that will be converted to a regular expression
   *    as in the ngPattern directive.
   * @param {string=} ngPattern Sets `pattern` validation error key if the ngModel value does not match
   *    a RegExp found by evaluating the Angular expression given in the attribute value.
   *    If the expression evaluates to a RegExp object, then this is used directly.
   *    If the expression evaluates to a string, then it will be converted to a RegExp
   *    after wrapping it in `^` and `$` characters. For instance, `"abc"` will be converted to
   *    `new RegExp('^abc$')`.<br />
   *    **Note:** Avoid using the `g` flag on the RegExp, as it will cause each successive search to
   *    start at the index of the last search's match, thus not taking the whole input value into
   *    account.
   * @param {string=} ngChange Angular expression to be executed when input changes due to user
   *    interaction with the input element.
   *
   * @example
      <example name="number-input-directive" module="numberExample">
        <file name="index.html">
         <script>
           angular.module('numberExample', [])
             .controller('ExampleController', ['$scope', function($scope) {
               $scope.example = {
                 value: 12
               };
             }]);
         </script>
         <form name="myForm" ng-controller="ExampleController">
           <label>Number:
             <input type="number" name="input" ng-model="example.value"
                    min="0" max="99" required>
          </label>
           <div role="alert">
             <span class="error" ng-show="myForm.input.$error.required">
               Required!</span>
             <span class="error" ng-show="myForm.input.$error.number">
               Not valid number!</span>
           </div>
           <tt>value = {{example.value}}</tt><br/>
           <tt>myForm.input.$valid = {{myForm.input.$valid}}</tt><br/>
           <tt>myForm.input.$error = {{myForm.input.$error}}</tt><br/>
           <tt>myForm.$valid = {{myForm.$valid}}</tt><br/>
           <tt>myForm.$error.required = {{!!myForm.$error.required}}</tt><br/>
          </form>
        </file>
        <file name="protractor.js" type="protractor">
          var value = element(by.binding('example.value'));
          var valid = element(by.binding('myForm.input.$valid'));
          var input = element(by.model('example.value'));

          it('should initialize to model', function() {
            expect(value.getText()).toContain('12');
            expect(valid.getText()).toContain('true');
          });

          it('should be invalid if empty', function() {
            input.clear();
            input.sendKeys('');
            expect(value.getText()).toEqual('value =');
            expect(valid.getText()).toContain('false');
          });

          it('should be invalid if over max', function() {
            input.clear();
            input.sendKeys('123');
            expect(value.getText()).toEqual('value =');
            expect(valid.getText()).toContain('false');
          });
        </file>
      </example>
   */
  'number': numberInputType,


  /**
   * @ngdoc input
   * @name input[url]
   *
   * @description
   * Text input with URL validation. Sets the `url` validation error key if the content is not a
   * valid URL.
   *
   * <div class="alert alert-warning">
   * **Note:** `input[url]` uses a regex to validate urls that is derived from the regex
   * used in Chromium. If you need stricter validation, you can use `ng-pattern` or modify
   * the built-in validators (see the {@link guide/forms Forms guide})
   * </div>
   *
   * @param {string} ngModel Assignable angular expression to data-bind to.
   * @param {string=} name Property name of the form under which the control is published.
   * @param {string=} required Sets `required` validation error key if the value is not entered.
   * @param {string=} ngRequired Adds `required` attribute and `required` validation constraint to
   *    the element when the ngRequired expression evaluates to true. Use `ngRequired` instead of
   *    `required` when you want to data-bind to the `required` attribute.
   * @param {number=} ngMinlength Sets `minlength` validation error key if the value is shorter than
   *    minlength.
   * @param {number=} ngMaxlength Sets `maxlength` validation error key if the value is longer than
   *    maxlength. Setting the attribute to a negative or non-numeric value, allows view values of
   *    any length.
   * @param {string=} pattern Similar to `ngPattern` except that the attribute value is the actual string
   *    that contains the regular expression body that will be converted to a regular expression
   *    as in the ngPattern directive.
   * @param {string=} ngPattern Sets `pattern` validation error key if the ngModel value does not match
   *    a RegExp found by evaluating the Angular expression given in the attribute value.
   *    If the expression evaluates to a RegExp object, then this is used directly.
   *    If the expression evaluates to a string, then it will be converted to a RegExp
   *    after wrapping it in `^` and `$` characters. For instance, `"abc"` will be converted to
   *    `new RegExp('^abc$')`.<br />
   *    **Note:** Avoid using the `g` flag on the RegExp, as it will cause each successive search to
   *    start at the index of the last search's match, thus not taking the whole input value into
   *    account.
   * @param {string=} ngChange Angular expression to be executed when input changes due to user
   *    interaction with the input element.
   *
   * @example
      <example name="url-input-directive" module="urlExample">
        <file name="index.html">
         <script>
           angular.module('urlExample', [])
             .controller('ExampleController', ['$scope', function($scope) {
               $scope.url = {
                 text: 'http://google.com'
               };
             }]);
         </script>
         <form name="myForm" ng-controller="ExampleController">
           <label>URL:
             <input type="url" name="input" ng-model="url.text" required>
           <label>
           <div role="alert">
             <span class="error" ng-show="myForm.input.$error.required">
               Required!</span>
             <span class="error" ng-show="myForm.input.$error.url">
               Not valid url!</span>
           </div>
           <tt>text = {{url.text}}</tt><br/>
           <tt>myForm.input.$valid = {{myForm.input.$valid}}</tt><br/>
           <tt>myForm.input.$error = {{myForm.input.$error}}</tt><br/>
           <tt>myForm.$valid = {{myForm.$valid}}</tt><br/>
           <tt>myForm.$error.required = {{!!myForm.$error.required}}</tt><br/>
           <tt>myForm.$error.url = {{!!myForm.$error.url}}</tt><br/>
          </form>
        </file>
        <file name="protractor.js" type="protractor">
          var text = element(by.binding('url.text'));
          var valid = element(by.binding('myForm.input.$valid'));
          var input = element(by.model('url.text'));

          it('should initialize to model', function() {
            expect(text.getText()).toContain('http://google.com');
            expect(valid.getText()).toContain('true');
          });

          it('should be invalid if empty', function() {
            input.clear();
            input.sendKeys('');

            expect(text.getText()).toEqual('text =');
            expect(valid.getText()).toContain('false');
          });

          it('should be invalid if not url', function() {
            input.clear();
            input.sendKeys('box');

            expect(valid.getText()).toContain('false');
          });
        </file>
      </example>
   */
  'url': urlInputType,


  /**
   * @ngdoc input
   * @name input[email]
   *
   * @description
   * Text input with email validation. Sets the `email` validation error key if not a valid email
   * address.
   *
   * <div class="alert alert-warning">
   * **Note:** `input[email]` uses a regex to validate email addresses that is derived from the regex
   * used in Chromium. If you need stricter validation (e.g. requiring a top-level domain), you can
   * use `ng-pattern` or modify the built-in validators (see the {@link guide/forms Forms guide})
   * </div>
   *
   * @param {string} ngModel Assignable angular expression to data-bind to.
   * @param {string=} name Property name of the form under which the control is published.
   * @param {string=} required Sets `required` validation error key if the value is not entered.
   * @param {string=} ngRequired Adds `required` attribute and `required` validation constraint to
   *    the element when the ngRequired expression evaluates to true. Use `ngRequired` instead of
   *    `required` when you want to data-bind to the `required` attribute.
   * @param {number=} ngMinlength Sets `minlength` validation error key if the value is shorter than
   *    minlength.
   * @param {number=} ngMaxlength Sets `maxlength` validation error key if the value is longer than
   *    maxlength. Setting the attribute to a negative or non-numeric value, allows view values of
   *    any length.
   * @param {string=} pattern Similar to `ngPattern` except that the attribute value is the actual string
   *    that contains the regular expression body that will be converted to a regular expression
   *    as in the ngPattern directive.
   * @param {string=} ngPattern Sets `pattern` validation error key if the ngModel value does not match
   *    a RegExp found by evaluating the Angular expression given in the attribute value.
   *    If the expression evaluates to a RegExp object, then this is used directly.
   *    If the expression evaluates to a string, then it will be converted to a RegExp
   *    after wrapping it in `^` and `$` characters. For instance, `"abc"` will be converted to
   *    `new RegExp('^abc$')`.<br />
   *    **Note:** Avoid using the `g` flag on the RegExp, as it will cause each successive search to
   *    start at the index of the last search's match, thus not taking the whole input value into
   *    account.
   * @param {string=} ngChange Angular expression to be executed when input changes due to user
   *    interaction with the input element.
   *
   * @example
      <example name="email-input-directive" module="emailExample">
        <file name="index.html">
         <script>
           angular.module('emailExample', [])
             .controller('ExampleController', ['$scope', function($scope) {
               $scope.email = {
                 text: 'me@example.com'
               };
             }]);
         </script>
           <form name="myForm" ng-controller="ExampleController">
             <label>Email:
               <input type="email" name="input" ng-model="email.text" required>
             </label>
             <div role="alert">
               <span class="error" ng-show="myForm.input.$error.required">
                 Required!</span>
               <span class="error" ng-show="myForm.input.$error.email">
                 Not valid email!</span>
             </div>
             <tt>text = {{email.text}}</tt><br/>
             <tt>myForm.input.$valid = {{myForm.input.$valid}}</tt><br/>
             <tt>myForm.input.$error = {{myForm.input.$error}}</tt><br/>
             <tt>myForm.$valid = {{myForm.$valid}}</tt><br/>
             <tt>myForm.$error.required = {{!!myForm.$error.required}}</tt><br/>
             <tt>myForm.$error.email = {{!!myForm.$error.email}}</tt><br/>
           </form>
         </file>
        <file name="protractor.js" type="protractor">
          var text = element(by.binding('email.text'));
          var valid = element(by.binding('myForm.input.$valid'));
          var input = element(by.model('email.text'));

          it('should initialize to model', function() {
            expect(text.getText()).toContain('me@example.com');
            expect(valid.getText()).toContain('true');
          });

          it('should be invalid if empty', function() {
            input.clear();
            input.sendKeys('');
            expect(text.getText()).toEqual('text =');
            expect(valid.getText()).toContain('false');
          });

          it('should be invalid if not email', function() {
            input.clear();
            input.sendKeys('xxx');

            expect(valid.getText()).toContain('false');
          });
        </file>
      </example>
   */
  'email': emailInputType,


  /**
   * @ngdoc input
   * @name input[radio]
   *
   * @description
   * HTML radio button.
   *
   * @param {string} ngModel Assignable angular expression to data-bind to.
   * @param {string} value The value to which the `ngModel` expression should be set when selected.
   *    Note that `value` only supports `string` values, i.e. the scope model needs to be a string,
   *    too. Use `ngValue` if you need complex models (`number`, `object`, ...).
   * @param {string=} name Property name of the form under which the control is published.
   * @param {string=} ngChange Angular expression to be executed when input changes due to user
   *    interaction with the input element.
   * @param {string} ngValue Angular expression to which `ngModel` will be be set when the radio
   *    is selected. Should be used instead of the `value` attribute if you need
   *    a non-string `ngModel` (`boolean`, `array`, ...).
   *
   * @example
      <example name="radio-input-directive" module="radioExample">
        <file name="index.html">
         <script>
           angular.module('radioExample', [])
             .controller('ExampleController', ['$scope', function($scope) {
               $scope.color = {
                 name: 'blue'
               };
               $scope.specialValue = {
                 "id": "12345",
                 "value": "green"
               };
             }]);
         </script>
         <form name="myForm" ng-controller="ExampleController">
           <label>
             <input type="radio" ng-model="color.name" value="red">
             Red
           </label><br/>
           <label>
             <input type="radio" ng-model="color.name" ng-value="specialValue">
             Green
           </label><br/>
           <label>
             <input type="radio" ng-model="color.name" value="blue">
             Blue
           </label><br/>
           <tt>color = {{color.name | json}}</tt><br/>
          </form>
          Note that `ng-value="specialValue"` sets radio item's value to be the value of `$scope.specialValue`.
        </file>
        <file name="protractor.js" type="protractor">
          it('should change state', function() {
            var color = element(by.binding('color.name'));

            expect(color.getText()).toContain('blue');

            element.all(by.model('color.name')).get(0).click();

            expect(color.getText()).toContain('red');
          });
        </file>
      </example>
   */
  'radio': radioInputType,


  /**
   * @ngdoc input
   * @name input[checkbox]
   *
   * @description
   * HTML checkbox.
   *
   * @param {string} ngModel Assignable angular expression to data-bind to.
   * @param {string=} name Property name of the form under which the control is published.
   * @param {expression=} ngTrueValue The value to which the expression should be set when selected.
   * @param {expression=} ngFalseValue The value to which the expression should be set when not selected.
   * @param {string=} ngChange Angular expression to be executed when input changes due to user
   *    interaction with the input element.
   *
   * @example
      <example name="checkbox-input-directive" module="checkboxExample">
        <file name="index.html">
         <script>
           angular.module('checkboxExample', [])
             .controller('ExampleController', ['$scope', function($scope) {
               $scope.checkboxModel = {
                value1 : true,
                value2 : 'YES'
              };
             }]);
         </script>
         <form name="myForm" ng-controller="ExampleController">
           <label>Value1:
             <input type="checkbox" ng-model="checkboxModel.value1">
           </label><br/>
           <label>Value2:
             <input type="checkbox" ng-model="checkboxModel.value2"
                    ng-true-value="'YES'" ng-false-value="'NO'">
            </label><br/>
           <tt>value1 = {{checkboxModel.value1}}</tt><br/>
           <tt>value2 = {{checkboxModel.value2}}</tt><br/>
          </form>
        </file>
        <file name="protractor.js" type="protractor">
          it('should change state', function() {
            var value1 = element(by.binding('checkboxModel.value1'));
            var value2 = element(by.binding('checkboxModel.value2'));

            expect(value1.getText()).toContain('true');
            expect(value2.getText()).toContain('YES');

            element(by.model('checkboxModel.value1')).click();
            element(by.model('checkboxModel.value2')).click();

            expect(value1.getText()).toContain('false');
            expect(value2.getText()).toContain('NO');
          });
        </file>
      </example>
   */
  'checkbox': checkboxInputType,

  'hidden': noop,
  'button': noop,
  'submit': noop,
  'reset': noop,
  'file': noop
};

function stringBasedInputType(ctrl) {
  ctrl.$formatters.push(function(value) {
    return ctrl.$isEmpty(value) ? value : value.toString();
  });
}

function textInputType(scope, element, attr, ctrl, $sniffer, $browser) {
  baseInputType(scope, element, attr, ctrl, $sniffer, $browser);
  stringBasedInputType(ctrl);
}

function baseInputType(scope, element, attr, ctrl, $sniffer, $browser) {
  var type = lowercase(element[0].type);

  // In composition mode, users are still inputing intermediate text buffer,
  // hold the listener until composition is done.
  // More about composition events: https://developer.mozilla.org/en-US/docs/Web/API/CompositionEvent
  if (!$sniffer.android) {
    var composing = false;

    element.on('compositionstart', function(data) {
      composing = true;
    });

    element.on('compositionend', function() {
      composing = false;
      listener();
    });
  }

  var listener = function(ev) {
    if (timeout) {
      $browser.defer.cancel(timeout);
      timeout = null;
    }
    if (composing) return;
    var value = element.val(),
        event = ev && ev.type;

    // By default we will trim the value
    // If the attribute ng-trim exists we will avoid trimming
    // If input type is 'password', the value is never trimmed
    if (type !== 'password' && (!attr.ngTrim || attr.ngTrim !== 'false')) {
      value = trim(value);
    }

    // If a control is suffering from bad input (due to native validators), browsers discard its
    // value, so it may be necessary to revalidate (by calling $setViewValue again) even if the
    // control's value is the same empty value twice in a row.
    if (ctrl.$viewValue !== value || (value === '' && ctrl.$$hasNativeValidators)) {
      ctrl.$setViewValue(value, event);
    }
  };

  // if the browser does support "input" event, we are fine - except on IE9 which doesn't fire the
  // input event on backspace, delete or cut
  if ($sniffer.hasEvent('input')) {
    element.on('input', listener);
  } else {
    var timeout;

    var deferListener = function(ev, input, origValue) {
      if (!timeout) {
        timeout = $browser.defer(function() {
          timeout = null;
          if (!input || input.value !== origValue) {
            listener(ev);
          }
        });
      }
    };

    element.on('keydown', function(event) {
      var key = event.keyCode;

      // ignore
      //    command            modifiers                   arrows
      if (key === 91 || (15 < key && key < 19) || (37 <= key && key <= 40)) return;

      deferListener(event, this, this.value);
    });

    // if user modifies input value using context menu in IE, we need "paste" and "cut" events to catch it
    if ($sniffer.hasEvent('paste')) {
      element.on('paste cut', deferListener);
    }
  }

  // if user paste into input using mouse on older browser
  // or form autocomplete on newer browser, we need "change" event to catch it
  element.on('change', listener);

  ctrl.$render = function() {
    // Workaround for Firefox validation #12102.
    var value = ctrl.$isEmpty(ctrl.$viewValue) ? '' : ctrl.$viewValue;
    if (element.val() !== value) {
      element.val(value);
    }
  };
}

function weekParser(isoWeek, existingDate) {
  if (isDate(isoWeek)) {
    return isoWeek;
  }

  if (isString(isoWeek)) {
    WEEK_REGEXP.lastIndex = 0;
    var parts = WEEK_REGEXP.exec(isoWeek);
    if (parts) {
      var year = +parts[1],
          week = +parts[2],
          hours = 0,
          minutes = 0,
          seconds = 0,
          milliseconds = 0,
          firstThurs = getFirstThursdayOfYear(year),
          addDays = (week - 1) * 7;

      if (existingDate) {
        hours = existingDate.getHours();
        minutes = existingDate.getMinutes();
        seconds = existingDate.getSeconds();
        milliseconds = existingDate.getMilliseconds();
      }

      return new Date(year, 0, firstThurs.getDate() + addDays, hours, minutes, seconds, milliseconds);
    }
  }

  return NaN;
}

function createDateParser(regexp, mapping) {
  return function(iso, date) {
    var parts, map;

    if (isDate(iso)) {
      return iso;
    }

    if (isString(iso)) {
      // When a date is JSON'ified to wraps itself inside of an extra
      // set of double quotes. This makes the date parsing code unable
      // to match the date string and parse it as a date.
      if (iso.charAt(0) == '"' && iso.charAt(iso.length - 1) == '"') {
        iso = iso.substring(1, iso.length - 1);
      }
      if (ISO_DATE_REGEXP.test(iso)) {
        return new Date(iso);
      }
      regexp.lastIndex = 0;
      parts = regexp.exec(iso);

      if (parts) {
        parts.shift();
        if (date) {
          map = {
            yyyy: date.getFullYear(),
            MM: date.getMonth() + 1,
            dd: date.getDate(),
            HH: date.getHours(),
            mm: date.getMinutes(),
            ss: date.getSeconds(),
            sss: date.getMilliseconds() / 1000
          };
        } else {
          map = { yyyy: 1970, MM: 1, dd: 1, HH: 0, mm: 0, ss: 0, sss: 0 };
        }

        forEach(parts, function(part, index) {
          if (index < mapping.length) {
            map[mapping[index]] = +part;
          }
        });
        return new Date(map.yyyy, map.MM - 1, map.dd, map.HH, map.mm, map.ss || 0, map.sss * 1000 || 0);
      }
    }

    return NaN;
  };
}

function createDateInputType(type, regexp, parseDate, format) {
  return function dynamicDateInputType(scope, element, attr, ctrl, $sniffer, $browser, $filter) {
    badInputChecker(scope, element, attr, ctrl);
    baseInputType(scope, element, attr, ctrl, $sniffer, $browser);
    var timezone = ctrl && ctrl.$options && ctrl.$options.timezone;
    var previousDate;

    ctrl.$$parserName = type;
    ctrl.$parsers.push(function(value) {
      if (ctrl.$isEmpty(value)) return null;
      if (regexp.test(value)) {
        // Note: We cannot read ctrl.$modelValue, as there might be a different
        // parser/formatter in the processing chain so that the model
        // contains some different data format!
        var parsedDate = parseDate(value, previousDate);
        if (timezone) {
          parsedDate = convertTimezoneToLocal(parsedDate, timezone);
        }
        return parsedDate;
      }
      return undefined;
    });

    ctrl.$formatters.push(function(value) {
      if (value && !isDate(value)) {
        throw ngModelMinErr('datefmt', 'Expected `{0}` to be a date', value);
      }
      if (isValidDate(value)) {
        previousDate = value;
        if (previousDate && timezone) {
          previousDate = convertTimezoneToLocal(previousDate, timezone, true);
        }
        return $filter('date')(value, format, timezone);
      } else {
        previousDate = null;
        return '';
      }
    });

    if (isDefined(attr.min) || attr.ngMin) {
      var minVal;
      ctrl.$validators.min = function(value) {
        return !isValidDate(value) || isUndefined(minVal) || parseDate(value) >= minVal;
      };
      attr.$observe('min', function(val) {
        minVal = parseObservedDateValue(val);
        ctrl.$validate();
      });
    }

    if (isDefined(attr.max) || attr.ngMax) {
      var maxVal;
      ctrl.$validators.max = function(value) {
        return !isValidDate(value) || isUndefined(maxVal) || parseDate(value) <= maxVal;
      };
      attr.$observe('max', function(val) {
        maxVal = parseObservedDateValue(val);
        ctrl.$validate();
      });
    }

    function isValidDate(value) {
      // Invalid Date: getTime() returns NaN
      return value && !(value.getTime && value.getTime() !== value.getTime());
    }

    function parseObservedDateValue(val) {
      return isDefined(val) && !isDate(val) ? parseDate(val) || undefined : val;
    }
  };
}

function badInputChecker(scope, element, attr, ctrl) {
  var node = element[0];
  var nativeValidation = ctrl.$$hasNativeValidators = isObject(node.validity);
  if (nativeValidation) {
    ctrl.$parsers.push(function(value) {
      var validity = element.prop(VALIDITY_STATE_PROPERTY) || {};
      // Detect bug in FF35 for input[email] (https://bugzilla.mozilla.org/show_bug.cgi?id=1064430):
      // - also sets validity.badInput (should only be validity.typeMismatch).
      // - see http://www.whatwg.org/specs/web-apps/current-work/multipage/forms.html#e-mail-state-(type=email)
      // - can ignore this case as we can still read out the erroneous email...
      return validity.badInput && !validity.typeMismatch ? undefined : value;
    });
  }
}

function numberInputType(scope, element, attr, ctrl, $sniffer, $browser) {
  badInputChecker(scope, element, attr, ctrl);
  baseInputType(scope, element, attr, ctrl, $sniffer, $browser);

  ctrl.$$parserName = 'number';
  ctrl.$parsers.push(function(value) {
    if (ctrl.$isEmpty(value))      return null;
    if (NUMBER_REGEXP.test(value)) return parseFloat(value);
    return undefined;
  });

  ctrl.$formatters.push(function(value) {
    if (!ctrl.$isEmpty(value)) {
      if (!isNumber(value)) {
        throw ngModelMinErr('numfmt', 'Expected `{0}` to be a number', value);
      }
      value = value.toString();
    }
    return value;
  });

  if (isDefined(attr.min) || attr.ngMin) {
    var minVal;
    ctrl.$validators.min = function(value) {
      return ctrl.$isEmpty(value) || isUndefined(minVal) || value >= minVal;
    };

    attr.$observe('min', function(val) {
      if (isDefined(val) && !isNumber(val)) {
        val = parseFloat(val, 10);
      }
      minVal = isNumber(val) && !isNaN(val) ? val : undefined;
      // TODO(matsko): implement validateLater to reduce number of validations
      ctrl.$validate();
    });
  }

  if (isDefined(attr.max) || attr.ngMax) {
    var maxVal;
    ctrl.$validators.max = function(value) {
      return ctrl.$isEmpty(value) || isUndefined(maxVal) || value <= maxVal;
    };

    attr.$observe('max', function(val) {
      if (isDefined(val) && !isNumber(val)) {
        val = parseFloat(val, 10);
      }
      maxVal = isNumber(val) && !isNaN(val) ? val : undefined;
      // TODO(matsko): implement validateLater to reduce number of validations
      ctrl.$validate();
    });
  }
}

function urlInputType(scope, element, attr, ctrl, $sniffer, $browser) {
  // Note: no badInputChecker here by purpose as `url` is only a validation
  // in browsers, i.e. we can always read out input.value even if it is not valid!
  baseInputType(scope, element, attr, ctrl, $sniffer, $browser);
  stringBasedInputType(ctrl);

  ctrl.$$parserName = 'url';
  ctrl.$validators.url = function(modelValue, viewValue) {
    var value = modelValue || viewValue;
    return ctrl.$isEmpty(value) || URL_REGEXP.test(value);
  };
}

function emailInputType(scope, element, attr, ctrl, $sniffer, $browser) {
  // Note: no badInputChecker here by purpose as `url` is only a validation
  // in browsers, i.e. we can always read out input.value even if it is not valid!
  baseInputType(scope, element, attr, ctrl, $sniffer, $browser);
  stringBasedInputType(ctrl);

  ctrl.$$parserName = 'email';
  ctrl.$validators.email = function(modelValue, viewValue) {
    var value = modelValue || viewValue;
    return ctrl.$isEmpty(value) || EMAIL_REGEXP.test(value);
  };
}

function radioInputType(scope, element, attr, ctrl) {
  // make the name unique, if not defined
  if (isUndefined(attr.name)) {
    element.attr('name', nextUid());
  }

  var listener = function(ev) {
    if (element[0].checked) {
      ctrl.$setViewValue(attr.value, ev && ev.type);
    }
  };

  element.on('click', listener);

  ctrl.$render = function() {
    var value = attr.value;
    element[0].checked = (value == ctrl.$viewValue);
  };

  attr.$observe('value', ctrl.$render);
}

function parseConstantExpr($parse, context, name, expression, fallback) {
  var parseFn;
  if (isDefined(expression)) {
    parseFn = $parse(expression);
    if (!parseFn.constant) {
      throw ngModelMinErr('constexpr', 'Expected constant expression for `{0}`, but saw ' +
                                   '`{1}`.', name, expression);
    }
    return parseFn(context);
  }
  return fallback;
}

function checkboxInputType(scope, element, attr, ctrl, $sniffer, $browser, $filter, $parse) {
  var trueValue = parseConstantExpr($parse, scope, 'ngTrueValue', attr.ngTrueValue, true);
  var falseValue = parseConstantExpr($parse, scope, 'ngFalseValue', attr.ngFalseValue, false);

  var listener = function(ev) {
    ctrl.$setViewValue(element[0].checked, ev && ev.type);
  };

  element.on('click', listener);

  ctrl.$render = function() {
    element[0].checked = ctrl.$viewValue;
  };

  // Override the standard `$isEmpty` because the $viewValue of an empty checkbox is always set to `false`
  // This is because of the parser below, which compares the `$modelValue` with `trueValue` to convert
  // it to a boolean.
  ctrl.$isEmpty = function(value) {
    return value === false;
  };

  ctrl.$formatters.push(function(value) {
    return equals(value, trueValue);
  });

  ctrl.$parsers.push(function(value) {
    return value ? trueValue : falseValue;
  });
}


/**
 * @ngdoc directive
 * @name textarea
 * @restrict E
 *
 * @description
 * HTML textarea element control with angular data-binding. The data-binding and validation
 * properties of this element are exactly the same as those of the
 * {@link ng.directive:input input element}.
 *
 * @param {string} ngModel Assignable angular expression to data-bind to.
 * @param {string=} name Property name of the form under which the control is published.
 * @param {string=} required Sets `required` validation error key if the value is not entered.
 * @param {string=} ngRequired Adds `required` attribute and `required` validation constraint to
 *    the element when the ngRequired expression evaluates to true. Use `ngRequired` instead of
 *    `required` when you want to data-bind to the `required` attribute.
 * @param {number=} ngMinlength Sets `minlength` validation error key if the value is shorter than
 *    minlength.
 * @param {number=} ngMaxlength Sets `maxlength` validation error key if the value is longer than
 *    maxlength. Setting the attribute to a negative or non-numeric value, allows view values of any
 *    length.
 * @param {string=} ngPattern Sets `pattern` validation error key if the ngModel value does not match
 *    a RegExp found by evaluating the Angular expression given in the attribute value.
 *    If the expression evaluates to a RegExp object, then this is used directly.
 *    If the expression evaluates to a string, then it will be converted to a RegExp
 *    after wrapping it in `^` and `$` characters. For instance, `"abc"` will be converted to
 *    `new RegExp('^abc$')`.<br />
 *    **Note:** Avoid using the `g` flag on the RegExp, as it will cause each successive search to
 *    start at the index of the last search's match, thus not taking the whole input value into
 *    account.
 * @param {string=} ngChange Angular expression to be executed when input changes due to user
 *    interaction with the input element.
 * @param {boolean=} [ngTrim=true] If set to false Angular will not automatically trim the input.
 */


/**
 * @ngdoc directive
 * @name input
 * @restrict E
 *
 * @description
 * HTML input element control. When used together with {@link ngModel `ngModel`}, it provides data-binding,
 * input state control, and validation.
 * Input control follows HTML5 input types and polyfills the HTML5 validation behavior for older browsers.
 *
 * <div class="alert alert-warning">
 * **Note:** Not every feature offered is available for all input types.
 * Specifically, data binding and event handling via `ng-model` is unsupported for `input[file]`.
 * </div>
 *
 * @param {string} ngModel Assignable angular expression to data-bind to.
 * @param {string=} name Property name of the form under which the control is published.
 * @param {string=} required Sets `required` validation error key if the value is not entered.
 * @param {boolean=} ngRequired Sets `required` attribute if set to true
 * @param {number=} ngMinlength Sets `minlength` validation error key if the value is shorter than
 *    minlength.
 * @param {number=} ngMaxlength Sets `maxlength` validation error key if the value is longer than
 *    maxlength. Setting the attribute to a negative or non-numeric value, allows view values of any
 *    length.
 * @param {string=} ngPattern Sets `pattern` validation error key if the ngModel value does not match
 *    a RegExp found by evaluating the Angular expression given in the attribute value.
 *    If the expression evaluates to a RegExp object, then this is used directly.
 *    If the expression evaluates to a string, then it will be converted to a RegExp
 *    after wrapping it in `^` and `$` characters. For instance, `"abc"` will be converted to
 *    `new RegExp('^abc$')`.<br />
 *    **Note:** Avoid using the `g` flag on the RegExp, as it will cause each successive search to
 *    start at the index of the last search's match, thus not taking the whole input value into
 *    account.
 * @param {string=} ngChange Angular expression to be executed when input changes due to user
 *    interaction with the input element.
 * @param {boolean=} [ngTrim=true] If set to false Angular will not automatically trim the input.
 *    This parameter is ignored for input[type=password] controls, which will never trim the
 *    input.
 *
 * @example
    <example name="input-directive" module="inputExample">
      <file name="index.html">
       <script>
          angular.module('inputExample', [])
            .controller('ExampleController', ['$scope', function($scope) {
              $scope.user = {name: 'guest', last: 'visitor'};
            }]);
       </script>
       <div ng-controller="ExampleController">
         <form name="myForm">
           <label>
              User name:
              <input type="text" name="userName" ng-model="user.name" required>
           </label>
           <div role="alert">
             <span class="error" ng-show="myForm.userName.$error.required">
              Required!</span>
           </div>
           <label>
              Last name:
              <input type="text" name="lastName" ng-model="user.last"
              ng-minlength="3" ng-maxlength="10">
           </label>
           <div role="alert">
             <span class="error" ng-show="myForm.lastName.$error.minlength">
               Too short!</span>
             <span class="error" ng-show="myForm.lastName.$error.maxlength">
               Too long!</span>
           </div>
         </form>
         <hr>
         <tt>user = {{user}}</tt><br/>
         <tt>myForm.userName.$valid = {{myForm.userName.$valid}}</tt><br/>
         <tt>myForm.userName.$error = {{myForm.userName.$error}}</tt><br/>
         <tt>myForm.lastName.$valid = {{myForm.lastName.$valid}}</tt><br/>
         <tt>myForm.lastName.$error = {{myForm.lastName.$error}}</tt><br/>
         <tt>myForm.$valid = {{myForm.$valid}}</tt><br/>
         <tt>myForm.$error.required = {{!!myForm.$error.required}}</tt><br/>
         <tt>myForm.$error.minlength = {{!!myForm.$error.minlength}}</tt><br/>
         <tt>myForm.$error.maxlength = {{!!myForm.$error.maxlength}}</tt><br/>
       </div>
      </file>
      <file name="protractor.js" type="protractor">
        var user = element(by.exactBinding('user'));
        var userNameValid = element(by.binding('myForm.userName.$valid'));
        var lastNameValid = element(by.binding('myForm.lastName.$valid'));
        var lastNameError = element(by.binding('myForm.lastName.$error'));
        var formValid = element(by.binding('myForm.$valid'));
        var userNameInput = element(by.model('user.name'));
        var userLastInput = element(by.model('user.last'));

        it('should initialize to model', function() {
          expect(user.getText()).toContain('{"name":"guest","last":"visitor"}');
          expect(userNameValid.getText()).toContain('true');
          expect(formValid.getText()).toContain('true');
        });

        it('should be invalid if empty when required', function() {
          userNameInput.clear();
          userNameInput.sendKeys('');

          expect(user.getText()).toContain('{"last":"visitor"}');
          expect(userNameValid.getText()).toContain('false');
          expect(formValid.getText()).toContain('false');
        });

        it('should be valid if empty when min length is set', function() {
          userLastInput.clear();
          userLastInput.sendKeys('');

          expect(user.getText()).toContain('{"name":"guest","last":""}');
          expect(lastNameValid.getText()).toContain('true');
          expect(formValid.getText()).toContain('true');
        });

        it('should be invalid if less than required min length', function() {
          userLastInput.clear();
          userLastInput.sendKeys('xx');

          expect(user.getText()).toContain('{"name":"guest"}');
          expect(lastNameValid.getText()).toContain('false');
          expect(lastNameError.getText()).toContain('minlength');
          expect(formValid.getText()).toContain('false');
        });

        it('should be invalid if longer than max length', function() {
          userLastInput.clear();
          userLastInput.sendKeys('some ridiculously long name');

          expect(user.getText()).toContain('{"name":"guest"}');
          expect(lastNameValid.getText()).toContain('false');
          expect(lastNameError.getText()).toContain('maxlength');
          expect(formValid.getText()).toContain('false');
        });
      </file>
    </example>
 */
var inputDirective = ['$browser', '$sniffer', '$filter', '$parse',
    function($browser, $sniffer, $filter, $parse) {
  return {
    restrict: 'E',
    require: ['?ngModel'],
    link: {
      pre: function(scope, element, attr, ctrls) {
        if (ctrls[0]) {
          (inputType[lowercase(attr.type)] || inputType.text)(scope, element, attr, ctrls[0], $sniffer,
                                                              $browser, $filter, $parse);
        }
      }
    }
  };
}];



var CONSTANT_VALUE_REGEXP = /^(true|false|\d+)$/;
/**
 * @ngdoc directive
 * @name ngValue
 *
 * @description
 * Binds the given expression to the value of `<option>` or {@link input[radio] `input[radio]`},
 * so that when the element is selected, the {@link ngModel `ngModel`} of that element is set to
 * the bound value.
 *
 * `ngValue` is useful when dynamically generating lists of radio buttons using
 * {@link ngRepeat `ngRepeat`}, as shown below.
 *
 * Likewise, `ngValue` can be used to generate `<option>` elements for
 * the {@link select `select`} element. In that case however, only strings are supported
 * for the `value `attribute, so the resulting `ngModel` will always be a string.
 * Support for `select` models with non-string values is available via `ngOptions`.
 *
 * @element input
 * @param {string=} ngValue angular expression, whose value will be bound to the `value` attribute
 *   of the `input` element
 *
 * @example
    <example name="ngValue-directive" module="valueExample">
      <file name="index.html">
       <script>
          angular.module('valueExample', [])
            .controller('ExampleController', ['$scope', function($scope) {
              $scope.names = ['pizza', 'unicorns', 'robots'];
              $scope.my = { favorite: 'unicorns' };
            }]);
       </script>
        <form ng-controller="ExampleController">
          <h2>Which is your favorite?</h2>
            <label ng-repeat="name in names" for="{{name}}">
              {{name}}
              <input type="radio"
                     ng-model="my.favorite"
                     ng-value="name"
                     id="{{name}}"
                     name="favorite">
            </label>
          <div>You chose {{my.favorite}}</div>
        </form>
      </file>
      <file name="protractor.js" type="protractor">
        var favorite = element(by.binding('my.favorite'));

        it('should initialize to model', function() {
          expect(favorite.getText()).toContain('unicorns');
        });
        it('should bind the values to the inputs', function() {
          element.all(by.model('my.favorite')).get(0).click();
          expect(favorite.getText()).toContain('pizza');
        });
      </file>
    </example>
 */
var ngValueDirective = function() {
  return {
    restrict: 'A',
    priority: 100,
    compile: function(tpl, tplAttr) {
      if (CONSTANT_VALUE_REGEXP.test(tplAttr.ngValue)) {
        return function ngValueConstantLink(scope, elm, attr) {
          attr.$set('value', scope.$eval(attr.ngValue));
        };
      } else {
        return function ngValueLink(scope, elm, attr) {
          scope.$watch(attr.ngValue, function valueWatchAction(value) {
            attr.$set('value', value);
          });
        };
      }
    }
  };
};

/**
 * @ngdoc directive
 * @name ngBind
 * @restrict AC
 *
 * @description
 * The `ngBind` attribute tells Angular to replace the text content of the specified HTML element
 * with the value of a given expression, and to update the text content when the value of that
 * expression changes.
 *
 * Typically, you don't use `ngBind` directly, but instead you use the double curly markup like
 * `{{ expression }}` which is similar but less verbose.
 *
 * It is preferable to use `ngBind` instead of `{{ expression }}` if a template is momentarily
 * displayed by the browser in its raw state before Angular compiles it. Since `ngBind` is an
 * element attribute, it makes the bindings invisible to the user while the page is loading.
 *
 * An alternative solution to this problem would be using the
 * {@link ng.directive:ngCloak ngCloak} directive.
 *
 *
 * @element ANY
 * @param {expression} ngBind {@link guide/expression Expression} to evaluate.
 *
 * @example
 * Enter a name in the Live Preview text box; the greeting below the text box changes instantly.
   <example module="bindExample">
     <file name="index.html">
       <script>
         angular.module('bindExample', [])
           .controller('ExampleController', ['$scope', function($scope) {
             $scope.name = 'Whirled';
           }]);
       </script>
       <div ng-controller="ExampleController">
         <label>Enter name: <input type="text" ng-model="name"></label><br>
         Hello <span ng-bind="name"></span>!
       </div>
     </file>
     <file name="protractor.js" type="protractor">
       it('should check ng-bind', function() {
         var nameInput = element(by.model('name'));

         expect(element(by.binding('name')).getText()).toBe('Whirled');
         nameInput.clear();
         nameInput.sendKeys('world');
         expect(element(by.binding('name')).getText()).toBe('world');
       });
     </file>
   </example>
 */
var ngBindDirective = ['$compile', function($compile) {
  return {
    restrict: 'AC',
    compile: function ngBindCompile(templateElement) {
      $compile.$$addBindingClass(templateElement);
      return function ngBindLink(scope, element, attr) {
        $compile.$$addBindingInfo(element, attr.ngBind);
        element = element[0];
        scope.$watch(attr.ngBind, function ngBindWatchAction(value) {
          element.textContent = isUndefined(value) ? '' : value;
        });
      };
    }
  };
}];


/**
 * @ngdoc directive
 * @name ngBindTemplate
 *
 * @description
 * The `ngBindTemplate` directive specifies that the element
 * text content should be replaced with the interpolation of the template
 * in the `ngBindTemplate` attribute.
 * Unlike `ngBind`, the `ngBindTemplate` can contain multiple `{{` `}}`
 * expressions. This directive is needed since some HTML elements
 * (such as TITLE and OPTION) cannot contain SPAN elements.
 *
 * @element ANY
 * @param {string} ngBindTemplate template of form
 *   <tt>{{</tt> <tt>expression</tt> <tt>}}</tt> to eval.
 *
 * @example
 * Try it here: enter text in text box and watch the greeting change.
   <example module="bindExample">
     <file name="index.html">
       <script>
         angular.module('bindExample', [])
           .controller('ExampleController', ['$scope', function($scope) {
             $scope.salutation = 'Hello';
             $scope.name = 'World';
           }]);
       </script>
       <div ng-controller="ExampleController">
        <label>Salutation: <input type="text" ng-model="salutation"></label><br>
        <label>Name: <input type="text" ng-model="name"></label><br>
        <pre ng-bind-template="{{salutation}} {{name}}!"></pre>
       </div>
     </file>
     <file name="protractor.js" type="protractor">
       it('should check ng-bind', function() {
         var salutationElem = element(by.binding('salutation'));
         var salutationInput = element(by.model('salutation'));
         var nameInput = element(by.model('name'));

         expect(salutationElem.getText()).toBe('Hello World!');

         salutationInput.clear();
         salutationInput.sendKeys('Greetings');
         nameInput.clear();
         nameInput.sendKeys('user');

         expect(salutationElem.getText()).toBe('Greetings user!');
       });
     </file>
   </example>
 */
var ngBindTemplateDirective = ['$interpolate', '$compile', function($interpolate, $compile) {
  return {
    compile: function ngBindTemplateCompile(templateElement) {
      $compile.$$addBindingClass(templateElement);
      return function ngBindTemplateLink(scope, element, attr) {
        var interpolateFn = $interpolate(element.attr(attr.$attr.ngBindTemplate));
        $compile.$$addBindingInfo(element, interpolateFn.expressions);
        element = element[0];
        attr.$observe('ngBindTemplate', function(value) {
          element.textContent = isUndefined(value) ? '' : value;
        });
      };
    }
  };
}];


/**
 * @ngdoc directive
 * @name ngBindHtml
 *
 * @description
 * Evaluates the expression and inserts the resulting HTML into the element in a secure way. By default,
 * the resulting HTML content will be sanitized using the {@link ngSanitize.$sanitize $sanitize} service.
 * To utilize this functionality, ensure that `$sanitize` is available, for example, by including {@link
 * ngSanitize} in your module's dependencies (not in core Angular). In order to use {@link ngSanitize}
 * in your module's dependencies, you need to include "angular-sanitize.js" in your application.
 *
 * You may also bypass sanitization for values you know are safe. To do so, bind to
 * an explicitly trusted value via {@link ng.$sce#trustAsHtml $sce.trustAsHtml}.  See the example
 * under {@link ng.$sce#show-me-an-example-using-sce- Strict Contextual Escaping (SCE)}.
 *
 * Note: If a `$sanitize` service is unavailable and the bound value isn't explicitly trusted, you
 * will have an exception (instead of an exploit.)
 *
 * @element ANY
 * @param {expression} ngBindHtml {@link guide/expression Expression} to evaluate.
 *
 * @example

   <example module="bindHtmlExample" deps="angular-sanitize.js">
     <file name="index.html">
       <div ng-controller="ExampleController">
        <p ng-bind-html="myHTML"></p>
       </div>
     </file>

     <file name="script.js">
       angular.module('bindHtmlExample', ['ngSanitize'])
         .controller('ExampleController', ['$scope', function($scope) {
           $scope.myHTML =
              'I am an <code>HTML</code>string with ' +
              '<a href="#">links!</a> and other <em>stuff</em>';
         }]);
     </file>

     <file name="protractor.js" type="protractor">
       it('should check ng-bind-html', function() {
         expect(element(by.binding('myHTML')).getText()).toBe(
             'I am an HTMLstring with links! and other stuff');
       });
     </file>
   </example>
 */
var ngBindHtmlDirective = ['$sce', '$parse', '$compile', function($sce, $parse, $compile) {
  return {
    restrict: 'A',
    compile: function ngBindHtmlCompile(tElement, tAttrs) {
      var ngBindHtmlGetter = $parse(tAttrs.ngBindHtml);
      var ngBindHtmlWatch = $parse(tAttrs.ngBindHtml, function getStringValue(value) {
        return (value || '').toString();
      });
      $compile.$$addBindingClass(tElement);

      return function ngBindHtmlLink(scope, element, attr) {
        $compile.$$addBindingInfo(element, attr.ngBindHtml);

        scope.$watch(ngBindHtmlWatch, function ngBindHtmlWatchAction() {
          // we re-evaluate the expr because we want a TrustedValueHolderType
          // for $sce, not a string
          element.html($sce.getTrustedHtml(ngBindHtmlGetter(scope)) || '');
        });
      };
    }
  };
}];

/**
 * @ngdoc directive
 * @name ngChange
 *
 * @description
 * Evaluate the given expression when the user changes the input.
 * The expression is evaluated immediately, unlike the JavaScript onchange event
 * which only triggers at the end of a change (usually, when the user leaves the
 * form element or presses the return key).
 *
 * The `ngChange` expression is only evaluated when a change in the input value causes
 * a new value to be committed to the model.
 *
 * It will not be evaluated:
 * * if the value returned from the `$parsers` transformation pipeline has not changed
 * * if the input has continued to be invalid since the model will stay `null`
 * * if the model is changed programmatically and not by a change to the input value
 *
 *
 * Note, this directive requires `ngModel` to be present.
 *
 * @element input
 * @param {expression} ngChange {@link guide/expression Expression} to evaluate upon change
 * in input value.
 *
 * @example
 * <example name="ngChange-directive" module="changeExample">
 *   <file name="index.html">
 *     <script>
 *       angular.module('changeExample', [])
 *         .controller('ExampleController', ['$scope', function($scope) {
 *           $scope.counter = 0;
 *           $scope.change = function() {
 *             $scope.counter++;
 *           };
 *         }]);
 *     </script>
 *     <div ng-controller="ExampleController">
 *       <input type="checkbox" ng-model="confirmed" ng-change="change()" id="ng-change-example1" />
 *       <input type="checkbox" ng-model="confirmed" id="ng-change-example2" />
 *       <label for="ng-change-example2">Confirmed</label><br />
 *       <tt>debug = {{confirmed}}</tt><br/>
 *       <tt>counter = {{counter}}</tt><br/>
 *     </div>
 *   </file>
 *   <file name="protractor.js" type="protractor">
 *     var counter = element(by.binding('counter'));
 *     var debug = element(by.binding('confirmed'));
 *
 *     it('should evaluate the expression if changing from view', function() {
 *       expect(counter.getText()).toContain('0');
 *
 *       element(by.id('ng-change-example1')).click();
 *
 *       expect(counter.getText()).toContain('1');
 *       expect(debug.getText()).toContain('true');
 *     });
 *
 *     it('should not evaluate the expression if changing from model', function() {
 *       element(by.id('ng-change-example2')).click();

 *       expect(counter.getText()).toContain('0');
 *       expect(debug.getText()).toContain('true');
 *     });
 *   </file>
 * </example>
 */
var ngChangeDirective = valueFn({
  restrict: 'A',
  require: 'ngModel',
  link: function(scope, element, attr, ctrl) {
    ctrl.$viewChangeListeners.push(function() {
      scope.$eval(attr.ngChange);
    });
  }
});

function classDirective(name, selector) {
  name = 'ngClass' + name;
  return ['$animate', function($animate) {
    return {
      restrict: 'AC',
      link: function(scope, element, attr) {
        var oldVal;

        scope.$watch(attr[name], ngClassWatchAction, true);

        attr.$observe('class', function(value) {
          ngClassWatchAction(scope.$eval(attr[name]));
        });


        if (name !== 'ngClass') {
          scope.$watch('$index', function($index, old$index) {
            // jshint bitwise: false
            var mod = $index & 1;
            if (mod !== (old$index & 1)) {
              var classes = arrayClasses(scope.$eval(attr[name]));
              mod === selector ?
                addClasses(classes) :
                removeClasses(classes);
            }
          });
        }

        function addClasses(classes) {
          var newClasses = digestClassCounts(classes, 1);
          attr.$addClass(newClasses);
        }

        function removeClasses(classes) {
          var newClasses = digestClassCounts(classes, -1);
          attr.$removeClass(newClasses);
        }

        function digestClassCounts(classes, count) {
          // Use createMap() to prevent class assumptions involving property
          // names in Object.prototype
          var classCounts = element.data('$classCounts') || createMap();
          var classesToUpdate = [];
          forEach(classes, function(className) {
            if (count > 0 || classCounts[className]) {
              classCounts[className] = (classCounts[className] || 0) + count;
              if (classCounts[className] === +(count > 0)) {
                classesToUpdate.push(className);
              }
            }
          });
          element.data('$classCounts', classCounts);
          return classesToUpdate.join(' ');
        }

        function updateClasses(oldClasses, newClasses) {
          var toAdd = arrayDifference(newClasses, oldClasses);
          var toRemove = arrayDifference(oldClasses, newClasses);
          toAdd = digestClassCounts(toAdd, 1);
          toRemove = digestClassCounts(toRemove, -1);
          if (toAdd && toAdd.length) {
            $animate.addClass(element, toAdd);
          }
          if (toRemove && toRemove.length) {
            $animate.removeClass(element, toRemove);
          }
        }

        function ngClassWatchAction(newVal) {
          if (selector === true || scope.$index % 2 === selector) {
            var newClasses = arrayClasses(newVal || []);
            if (!oldVal) {
              addClasses(newClasses);
            } else if (!equals(newVal,oldVal)) {
              var oldClasses = arrayClasses(oldVal);
              updateClasses(oldClasses, newClasses);
            }
          }
          oldVal = shallowCopy(newVal);
        }
      }
    };

    function arrayDifference(tokens1, tokens2) {
      var values = [];

      outer:
      for (var i = 0; i < tokens1.length; i++) {
        var token = tokens1[i];
        for (var j = 0; j < tokens2.length; j++) {
          if (token == tokens2[j]) continue outer;
        }
        values.push(token);
      }
      return values;
    }

    function arrayClasses(classVal) {
      var classes = [];
      if (isArray(classVal)) {
        forEach(classVal, function(v) {
          classes = classes.concat(arrayClasses(v));
        });
        return classes;
      } else if (isString(classVal)) {
        return classVal.split(' ');
      } else if (isObject(classVal)) {
        forEach(classVal, function(v, k) {
          if (v) {
            classes = classes.concat(k.split(' '));
          }
        });
        return classes;
      }
      return classVal;
    }
  }];
}

/**
 * @ngdoc directive
 * @name ngClass
 * @restrict AC
 *
 * @description
 * The `ngClass` directive allows you to dynamically set CSS classes on an HTML element by databinding
 * an expression that represents all classes to be added.
 *
 * The directive operates in three different ways, depending on which of three types the expression
 * evaluates to:
 *
 * 1. If the expression evaluates to a string, the string should be one or more space-delimited class
 * names.
 *
 * 2. If the expression evaluates to an object, then for each key-value pair of the
 * object with a truthy value the corresponding key is used as a class name.
 *
 * 3. If the expression evaluates to an array, each element of the array should either be a string as in
 * type 1 or an object as in type 2. This means that you can mix strings and objects together in an array
 * to give you more control over what CSS classes appear. See the code below for an example of this.
 *
 *
 * The directive won't add duplicate classes if a particular class was already set.
 *
 * When the expression changes, the previously added classes are removed and only then are the
 * new classes added.
 *
 * @animations
 * **add** - happens just before the class is applied to the elements
 *
 * **remove** - happens just before the class is removed from the element
 *
 * @element ANY
 * @param {expression} ngClass {@link guide/expression Expression} to eval. The result
 *   of the evaluation can be a string representing space delimited class
 *   names, an array, or a map of class names to boolean values. In the case of a map, the
 *   names of the properties whose values are truthy will be added as css classes to the
 *   element.
 *
 * @example Example that demonstrates basic bindings via ngClass directive.
   <example>
     <file name="index.html">
       <p ng-class="{strike: deleted, bold: important, 'has-error': error}">Map Syntax Example</p>
       <label>
          <input type="checkbox" ng-model="deleted">
          deleted (apply "strike" class)
       </label><br>
       <label>
          <input type="checkbox" ng-model="important">
          important (apply "bold" class)
       </label><br>
       <label>
          <input type="checkbox" ng-model="error">
          error (apply "has-error" class)
       </label>
       <hr>
       <p ng-class="style">Using String Syntax</p>
       <input type="text" ng-model="style"
              placeholder="Type: bold strike red" aria-label="Type: bold strike red">
       <hr>
       <p ng-class="[style1, style2, style3]">Using Array Syntax</p>
       <input ng-model="style1"
              placeholder="Type: bold, strike or red" aria-label="Type: bold, strike or red"><br>
       <input ng-model="style2"
              placeholder="Type: bold, strike or red" aria-label="Type: bold, strike or red 2"><br>
       <input ng-model="style3"
              placeholder="Type: bold, strike or red" aria-label="Type: bold, strike or red 3"><br>
       <hr>
       <p ng-class="[style4, {orange: warning}]">Using Array and Map Syntax</p>
       <input ng-model="style4" placeholder="Type: bold, strike" aria-label="Type: bold, strike"><br>
       <label><input type="checkbox" ng-model="warning"> warning (apply "orange" class)</label>
     </file>
     <file name="style.css">
       .strike {
           text-decoration: line-through;
       }
       .bold {
           font-weight: bold;
       }
       .red {
           color: red;
       }
       .has-error {
           color: red;
           background-color: yellow;
       }
       .orange {
           color: orange;
       }
     </file>
     <file name="protractor.js" type="protractor">
       var ps = element.all(by.css('p'));

       it('should let you toggle the class', function() {

         expect(ps.first().getAttribute('class')).not.toMatch(/bold/);
         expect(ps.first().getAttribute('class')).not.toMatch(/has-error/);

         element(by.model('important')).click();
         expect(ps.first().getAttribute('class')).toMatch(/bold/);

         element(by.model('error')).click();
         expect(ps.first().getAttribute('class')).toMatch(/has-error/);
       });

       it('should let you toggle string example', function() {
         expect(ps.get(1).getAttribute('class')).toBe('');
         element(by.model('style')).clear();
         element(by.model('style')).sendKeys('red');
         expect(ps.get(1).getAttribute('class')).toBe('red');
       });

       it('array example should have 3 classes', function() {
         expect(ps.get(2).getAttribute('class')).toBe('');
         element(by.model('style1')).sendKeys('bold');
         element(by.model('style2')).sendKeys('strike');
         element(by.model('style3')).sendKeys('red');
         expect(ps.get(2).getAttribute('class')).toBe('bold strike red');
       });

       it('array with map example should have 2 classes', function() {
         expect(ps.last().getAttribute('class')).toBe('');
         element(by.model('style4')).sendKeys('bold');
         element(by.model('warning')).click();
         expect(ps.last().getAttribute('class')).toBe('bold orange');
       });
     </file>
   </example>

   ## Animations

   The example below demonstrates how to perform animations using ngClass.

   <example module="ngAnimate" deps="angular-animate.js" animations="true">
     <file name="index.html">
      <input id="setbtn" type="button" value="set" ng-click="myVar='my-class'">
      <input id="clearbtn" type="button" value="clear" ng-click="myVar=''">
      <br>
      <span class="base-class" ng-class="myVar">Sample Text</span>
     </file>
     <file name="style.css">
       .base-class {
         transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 0.5s;
       }

       .base-class.my-class {
         color: red;
         font-size:3em;
       }
     </file>
     <file name="protractor.js" type="protractor">
       it('should check ng-class', function() {
         expect(element(by.css('.base-class')).getAttribute('class')).not.
           toMatch(/my-class/);

         element(by.id('setbtn')).click();

         expect(element(by.css('.base-class')).getAttribute('class')).
           toMatch(/my-class/);

         element(by.id('clearbtn')).click();

         expect(element(by.css('.base-class')).getAttribute('class')).not.
           toMatch(/my-class/);
       });
     </file>
   </example>


   ## ngClass and pre-existing CSS3 Transitions/Animations
   The ngClass directive still supports CSS3 Transitions/Animations even if they do not follow the ngAnimate CSS naming structure.
   Upon animation ngAnimate will apply supplementary CSS classes to track the start and end of an animation, but this will not hinder
   any pre-existing CSS transitions already on the element. To get an idea of what happens during a class-based animation, be sure
   to view the step by step details of {@link $animate#addClass $animate.addClass} and
   {@link $animate#removeClass $animate.removeClass}.
 */
var ngClassDirective = classDirective('', true);

/**
 * @ngdoc directive
 * @name ngClassOdd
 * @restrict AC
 *
 * @description
 * The `ngClassOdd` and `ngClassEven` directives work exactly as
 * {@link ng.directive:ngClass ngClass}, except they work in
 * conjunction with `ngRepeat` and take effect only on odd (even) rows.
 *
 * This directive can be applied only within the scope of an
 * {@link ng.directive:ngRepeat ngRepeat}.
 *
 * @element ANY
 * @param {expression} ngClassOdd {@link guide/expression Expression} to eval. The result
 *   of the evaluation can be a string representing space delimited class names or an array.
 *
 * @example
   <example>
     <file name="index.html">
        <ol ng-init="names=['John', 'Mary', 'Cate', 'Suz']">
          <li ng-repeat="name in names">
           <span ng-class-odd="'odd'" ng-class-even="'even'">
             {{name}}
           </span>
          </li>
        </ol>
     </file>
     <file name="style.css">
       .odd {
         color: red;
       }
       .even {
         color: blue;
       }
     </file>
     <file name="protractor.js" type="protractor">
       it('should check ng-class-odd and ng-class-even', function() {
         expect(element(by.repeater('name in names').row(0).column('name')).getAttribute('class')).
           toMatch(/odd/);
         expect(element(by.repeater('name in names').row(1).column('name')).getAttribute('class')).
           toMatch(/even/);
       });
     </file>
   </example>
 */
var ngClassOddDirective = classDirective('Odd', 0);

/**
 * @ngdoc directive
 * @name ngClassEven
 * @restrict AC
 *
 * @description
 * The `ngClassOdd` and `ngClassEven` directives work exactly as
 * {@link ng.directive:ngClass ngClass}, except they work in
 * conjunction with `ngRepeat` and take effect only on odd (even) rows.
 *
 * This directive can be applied only within the scope of an
 * {@link ng.directive:ngRepeat ngRepeat}.
 *
 * @element ANY
 * @param {expression} ngClassEven {@link guide/expression Expression} to eval. The
 *   result of the evaluation can be a string representing space delimited class names or an array.
 *
 * @example
   <example>
     <file name="index.html">
        <ol ng-init="names=['John', 'Mary', 'Cate', 'Suz']">
          <li ng-repeat="name in names">
           <span ng-class-odd="'odd'" ng-class-even="'even'">
             {{name}} &nbsp; &nbsp; &nbsp;
           </span>
          </li>
        </ol>
     </file>
     <file name="style.css">
       .odd {
         color: red;
       }
       .even {
         color: blue;
       }
     </file>
     <file name="protractor.js" type="protractor">
       it('should check ng-class-odd and ng-class-even', function() {
         expect(element(by.repeater('name in names').row(0).column('name')).getAttribute('class')).
           toMatch(/odd/);
         expect(element(by.repeater('name in names').row(1).column('name')).getAttribute('class')).
           toMatch(/even/);
       });
     </file>
   </example>
 */
var ngClassEvenDirective = classDirective('Even', 1);

/**
 * @ngdoc directive
 * @name ngCloak
 * @restrict AC
 *
 * @description
 * The `ngCloak` directive is used to prevent the Angular html template from being briefly
 * displayed by the browser in its raw (uncompiled) form while your application is loading. Use this
 * directive to avoid the undesirable flicker effect caused by the html template display.
 *
 * The directive can be applied to the `<body>` element, but the preferred usage is to apply
 * multiple `ngCloak` directives to small portions of the page to permit progressive rendering
 * of the browser view.
 *
 * `ngCloak` works in cooperation with the following css rule embedded within `angular.js` and
 * `angular.min.js`.
 * For CSP mode please add `angular-csp.css` to your html file (see {@link ng.directive:ngCsp ngCsp}).
 *
 * ```css
 * [ng\:cloak], [ng-cloak], [data-ng-cloak], [x-ng-cloak], .ng-cloak, .x-ng-cloak {
 *   display: none !important;
 * }
 * ```
 *
 * When this css rule is loaded by the browser, all html elements (including their children) that
 * are tagged with the `ngCloak` directive are hidden. When Angular encounters this directive
 * during the compilation of the template it deletes the `ngCloak` element attribute, making
 * the compiled element visible.
 *
 * For the best result, the `angular.js` script must be loaded in the head section of the html
 * document; alternatively, the css rule above must be included in the external stylesheet of the
 * application.
 *
 * @element ANY
 *
 * @example
   <example>
     <file name="index.html">
        <div id="template1" ng-cloak>{{ 'hello' }}</div>
        <div id="template2" class="ng-cloak">{{ 'world' }}</div>
     </file>
     <file name="protractor.js" type="protractor">
       it('should remove the template directive and css class', function() {
         expect($('#template1').getAttribute('ng-cloak')).
           toBeNull();
         expect($('#template2').getAttribute('ng-cloak')).
           toBeNull();
       });
     </file>
   </example>
 *
 */
var ngCloakDirective = ngDirective({
  compile: function(element, attr) {
    attr.$set('ngCloak', undefined);
    element.removeClass('ng-cloak');
  }
});

/**
 * @ngdoc directive
 * @name ngController
 *
 * @description
 * The `ngController` directive attaches a controller class to the view. This is a key aspect of how angular
 * supports the principles behind the Model-View-Controller design pattern.
 *
 * MVC components in angular:
 *
 * * Model â€” Models are the properties of a scope; scopes are attached to the DOM where scope properties
 *   are accessed through bindings.
 * * View â€” The template (HTML with data bindings) that is rendered into the View.
 * * Controller â€” The `ngController` directive specifies a Controller class; the class contains business
 *   logic behind the application to decorate the scope with functions and values
 *
 * Note that you can also attach controllers to the DOM by declaring it in a route definition
 * via the {@link ngRoute.$route $route} service. A common mistake is to declare the controller
 * again using `ng-controller` in the template itself.  This will cause the controller to be attached
 * and executed twice.
 *
 * @element ANY
 * @scope
 * @priority 500
 * @param {expression} ngController Name of a constructor function registered with the current
 * {@link ng.$controllerProvider $controllerProvider} or an {@link guide/expression expression}
 * that on the current scope evaluates to a constructor function.
 *
 * The controller instance can be published into a scope property by specifying
 * `ng-controller="as propertyName"`.
 *
 * If the current `$controllerProvider` is configured to use globals (via
 * {@link ng.$controllerProvider#allowGlobals `$controllerProvider.allowGlobals()` }), this may
 * also be the name of a globally accessible constructor function (not recommended).
 *
 * @example
 * Here is a simple form for editing user contact information. Adding, removing, clearing, and
 * greeting are methods declared on the controller (see source tab). These methods can
 * easily be called from the angular markup. Any changes to the data are automatically reflected
 * in the View without the need for a manual update.
 *
 * Two different declaration styles are included below:
 *
 * * one binds methods and properties directly onto the controller using `this`:
 * `ng-controller="SettingsController1 as settings"`
 * * one injects `$scope` into the controller:
 * `ng-controller="SettingsController2"`
 *
 * The second option is more common in the Angular community, and is generally used in boilerplates
 * and in this guide. However, there are advantages to binding properties directly to the controller
 * and avoiding scope.
 *
 * * Using `controller as` makes it obvious which controller you are accessing in the template when
 * multiple controllers apply to an element.
 * * If you are writing your controllers as classes you have easier access to the properties and
 * methods, which will appear on the scope, from inside the controller code.
 * * Since there is always a `.` in the bindings, you don't have to worry about prototypal
 * inheritance masking primitives.
 *
 * This example demonstrates the `controller as` syntax.
 *
 * <example name="ngControllerAs" module="controllerAsExample">
 *   <file name="index.html">
 *    <div id="ctrl-as-exmpl" ng-controller="SettingsController1 as settings">
 *      <label>Name: <input type="text" ng-model="settings.name"/></label>
 *      <button ng-click="settings.greet()">greet</button><br/>
 *      Contact:
 *      <ul>
 *        <li ng-repeat="contact in settings.contacts">
 *          <select ng-model="contact.type" aria-label="Contact method" id="select_{{$index}}">
 *             <option>phone</option>
 *             <option>email</option>
 *          </select>
 *          <input type="text" ng-model="contact.value" aria-labelledby="select_{{$index}}" />
 *          <button ng-click="settings.clearContact(contact)">clear</button>
 *          <button ng-click="settings.removeContact(contact)" aria-label="Remove">X</button>
 *        </li>
 *        <li><button ng-click="settings.addContact()">add</button></li>
 *     </ul>
 *    </div>
 *   </file>
 *   <file name="app.js">
 *    angular.module('controllerAsExample', [])
 *      .controller('SettingsController1', SettingsController1);
 *
 *    function SettingsController1() {
 *      this.name = "John Smith";
 *      this.contacts = [
 *        {type: 'phone', value: '408 555 1212'},
 *        {type: 'email', value: 'john.smith@example.org'} ];
 *    }
 *
 *    SettingsController1.prototype.greet = function() {
 *      alert(this.name);
 *    };
 *
 *    SettingsController1.prototype.addContact = function() {
 *      this.contacts.push({type: 'email', value: 'yourname@example.org'});
 *    };
 *
 *    SettingsController1.prototype.removeContact = function(contactToRemove) {
 *     var index = this.contacts.indexOf(contactToRemove);
 *      this.contacts.splice(index, 1);
 *    };
 *
 *    SettingsController1.prototype.clearContact = function(contact) {
 *      contact.type = 'phone';
 *      contact.value = '';
 *    };
 *   </file>
 *   <file name="protractor.js" type="protractor">
 *     it('should check controller as', function() {
 *       var container = element(by.id('ctrl-as-exmpl'));
 *         expect(container.element(by.model('settings.name'))
 *           .getAttribute('value')).toBe('John Smith');
 *
 *       var firstRepeat =
 *           container.element(by.repeater('contact in settings.contacts').row(0));
 *       var secondRepeat =
 *           container.element(by.repeater('contact in settings.contacts').row(1));
 *
 *       expect(firstRepeat.element(by.model('contact.value')).getAttribute('value'))
 *           .toBe('408 555 1212');
 *
 *       expect(secondRepeat.element(by.model('contact.value')).getAttribute('value'))
 *           .toBe('john.smith@example.org');
 *
 *       firstRepeat.element(by.buttonText('clear')).click();
 *
 *       expect(firstRepeat.element(by.model('contact.value')).getAttribute('value'))
 *           .toBe('');
 *
 *       container.element(by.buttonText('add')).click();
 *
 *       expect(container.element(by.repeater('contact in settings.contacts').row(2))
 *           .element(by.model('contact.value'))
 *           .getAttribute('value'))
 *           .toBe('yourname@example.org');
 *     });
 *   </file>
 * </example>
 *
 * This example demonstrates the "attach to `$scope`" style of controller.
 *
 * <example name="ngController" module="controllerExample">
 *  <file name="index.html">
 *   <div id="ctrl-exmpl" ng-controller="SettingsController2">
 *     <label>Name: <input type="text" ng-model="name"/></label>
 *     <button ng-click="greet()">greet</button><br/>
 *     Contact:
 *     <ul>
 *       <li ng-repeat="contact in contacts">
 *         <select ng-model="contact.type" id="select_{{$index}}">
 *            <option>phone</option>
 *            <option>email</option>
 *         </select>
 *         <input type="text" ng-model="contact.value" aria-labelledby="select_{{$index}}" />
 *         <button ng-click="clearContact(contact)">clear</button>
 *         <button ng-click="removeContact(contact)">X</button>
 *       </li>
 *       <li>[ <button ng-click="addContact()">add</button> ]</li>
 *    </ul>
 *   </div>
 *  </file>
 *  <file name="app.js">
 *   angular.module('controllerExample', [])
 *     .controller('SettingsController2', ['$scope', SettingsController2]);
 *
 *   function SettingsController2($scope) {
 *     $scope.name = "John Smith";
 *     $scope.contacts = [
 *       {type:'phone', value:'408 555 1212'},
 *       {type:'email', value:'john.smith@example.org'} ];
 *
 *     $scope.greet = function() {
 *       alert($scope.name);
 *     };
 *
 *     $scope.addContact = function() {
 *       $scope.contacts.push({type:'email', value:'yourname@example.org'});
 *     };
 *
 *     $scope.removeContact = function(contactToRemove) {
 *       var index = $scope.contacts.indexOf(contactToRemove);
 *       $scope.contacts.splice(index, 1);
 *     };
 *
 *     $scope.clearContact = function(contact) {
 *       contact.type = 'phone';
 *       contact.value = '';
 *     };
 *   }
 *  </file>
 *  <file name="protractor.js" type="protractor">
 *    it('should check controller', function() {
 *      var container = element(by.id('ctrl-exmpl'));
 *
 *      expect(container.element(by.model('name'))
 *          .getAttribute('value')).toBe('John Smith');
 *
 *      var firstRepeat =
 *          container.element(by.repeater('contact in contacts').row(0));
 *      var secondRepeat =
 *          container.element(by.repeater('contact in contacts').row(1));
 *
 *      expect(firstRepeat.element(by.model('contact.value')).getAttribute('value'))
 *          .toBe('408 555 1212');
 *      expect(secondRepeat.element(by.model('contact.value')).getAttribute('value'))
 *          .toBe('john.smith@example.org');
 *
 *      firstRepeat.element(by.buttonText('clear')).click();
 *
 *      expect(firstRepeat.element(by.model('contact.value')).getAttribute('value'))
 *          .toBe('');
 *
 *      container.element(by.buttonText('add')).click();
 *
 *      expect(container.element(by.repeater('contact in contacts').row(2))
 *          .element(by.model('contact.value'))
 *          .getAttribute('value'))
 *          .toBe('yourname@example.org');
 *    });
 *  </file>
 *</example>

 */
var ngControllerDirective = [function() {
  return {
    restrict: 'A',
    scope: true,
    controller: '@',
    priority: 500
  };
}];

/**
 * @ngdoc directive
 * @name ngCsp
 *
 * @element html
 * @description
 *
 * Angular has some features that can break certain
 * [CSP (Content Security Policy)](https://developer.mozilla.org/en/Security/CSP) rules.
 *
 * If you intend to implement these rules then you must tell Angular not to use these features.
 *
 * This is necessary when developing things like Google Chrome Extensions or Universal Windows Apps.
 *
 *
 * The following rules affect Angular:
 *
 * * `unsafe-eval`: this rule forbids apps to use `eval` or `Function(string)` generated functions
 * (among other things). Angular makes use of this in the {@link $parse} service to provide a 30%
 * increase in the speed of evaluating Angular expressions.
 *
 * * `unsafe-inline`: this rule forbids apps from inject custom styles into the document. Angular
 * makes use of this to include some CSS rules (e.g. {@link ngCloak} and {@link ngHide}).
 * To make these directives work when a CSP rule is blocking inline styles, you must link to the
 * `angular-csp.css` in your HTML manually.
 *
 * If you do not provide `ngCsp` then Angular tries to autodetect if CSP is blocking unsafe-eval
 * and automatically deactivates this feature in the {@link $parse} service. This autodetection,
 * however, triggers a CSP error to be logged in the console:
 *
 * ```
 * Refused to evaluate a string as JavaScript because 'unsafe-eval' is not an allowed source of
 * script in the following Content Security Policy directive: "default-src 'self'". Note that
 * 'script-src' was not explicitly set, so 'default-src' is used as a fallback.
 * ```
 *
 * This error is harmless but annoying. To prevent the error from showing up, put the `ngCsp`
 * directive on an element of the HTML document that appears before the `<script>` tag that loads
 * the `angular.js` file.
 *
 * *Note: This directive is only available in the `ng-csp` and `data-ng-csp` attribute form.*
 *
 * You can specify which of the CSP related Angular features should be deactivated by providing
 * a value for the `ng-csp` attribute. The options are as follows:
 *
 * * no-inline-style: this stops Angular from injecting CSS styles into the DOM
 *
 * * no-unsafe-eval: this stops Angular from optimising $parse with unsafe eval of strings
 *
 * You can use these values in the following combinations:
 *
 *
 * * No declaration means that Angular will assume that you can do inline styles, but it will do
 * a runtime check for unsafe-eval. E.g. `<body>`. This is backwardly compatible with previous versions
 * of Angular.
 *
 * * A simple `ng-csp` (or `data-ng-csp`) attribute will tell Angular to deactivate both inline
 * styles and unsafe eval. E.g. `<body ng-csp>`. This is backwardly compatible with previous versions
 * of Angular.
 *
 * * Specifying only `no-unsafe-eval` tells Angular that we must not use eval, but that we can inject
 * inline styles. E.g. `<body ng-csp="no-unsafe-eval">`.
 *
 * * Specifying only `no-inline-style` tells Angular that we must not inject styles, but that we can
 * run eval - no automcatic check for unsafe eval will occur. E.g. `<body ng-csp="no-inline-style">`
 *
 * * Specifying both `no-unsafe-eval` and `no-inline-style` tells Angular that we must not inject
 * styles nor use eval, which is the same as an empty: ng-csp.
 * E.g.`<body ng-csp="no-inline-style;no-unsafe-eval">`
 *
 * @example
 * This example shows how to apply the `ngCsp` directive to the `html` tag.
   ```html
     <!doctype html>
     <html ng-app ng-csp>
     ...
     ...
     </html>
   ```
  * @example
      // Note: the suffix `.csp` in the example name triggers
      // csp mode in our http server!
      <example name="example.csp" module="cspExample" ng-csp="true">
        <file name="index.html">
          <div ng-controller="MainController as ctrl">
            <div>
              <button ng-click="ctrl.inc()" id="inc">Increment</button>
              <span id="counter">
                {{ctrl.counter}}
              </span>
            </div>

            <div>
              <button ng-click="ctrl.evil()" id="evil">Evil</button>
              <span id="evilError">
                {{ctrl.evilError}}
              </span>
            </div>
          </div>
        </file>
        <file name="script.js">
           angular.module('cspExample', [])
             .controller('MainController', function() {
                this.counter = 0;
                this.inc = function() {
                  this.counter++;
                };
                this.evil = function() {
                  // jshint evil:true
                  try {
                    eval('1+2');
                  } catch (e) {
                    this.evilError = e.message;
                  }
                };
              });
        </file>
        <file name="protractor.js" type="protractor">
          var util, webdriver;

          var incBtn = element(by.id('inc'));
          var counter = element(by.id('counter'));
          var evilBtn = element(by.id('evil'));
          var evilError = element(by.id('evilError'));

          function getAndClearSevereErrors() {
            return browser.manage().logs().get('browser').then(function(browserLog) {
              return browserLog.filter(function(logEntry) {
                return logEntry.level.value > webdriver.logging.Level.WARNING.value;
              });
            });
          }

          function clearErrors() {
            getAndClearSevereErrors();
          }

          function expectNoErrors() {
            getAndClearSevereErrors().then(function(filteredLog) {
              expect(filteredLog.length).toEqual(0);
              if (filteredLog.length) {
                console.log('browser console errors: ' + util.inspect(filteredLog));
              }
            });
          }

          function expectError(regex) {
            getAndClearSevereErrors().then(function(filteredLog) {
              var found = false;
              filteredLog.forEach(function(log) {
                if (log.message.match(regex)) {
                  found = true;
                }
              });
              if (!found) {
                throw new Error('expected an error that matches ' + regex);
              }
            });
          }

          beforeEach(function() {
            util = require('util');
            webdriver = require('protractor/node_modules/selenium-webdriver');
          });

          // For now, we only test on Chrome,
          // as Safari does not load the page with Protractor's injected scripts,
          // and Firefox webdriver always disables content security policy (#6358)
          if (browser.params.browser !== 'chrome') {
            return;
          }

          it('should not report errors when the page is loaded', function() {
            // clear errors so we are not dependent on previous tests
            clearErrors();
            // Need to reload the page as the page is already loaded when
            // we come here
            browser.driver.getCurrentUrl().then(function(url) {
              browser.get(url);
            });
            expectNoErrors();
          });

          it('should evaluate expressions', function() {
            expect(counter.getText()).toEqual('0');
            incBtn.click();
            expect(counter.getText()).toEqual('1');
            expectNoErrors();
          });

          it('should throw and report an error when using "eval"', function() {
            evilBtn.click();
            expect(evilError.getText()).toMatch(/Content Security Policy/);
            expectError(/Content Security Policy/);
          });
        </file>
      </example>
  */

// ngCsp is not implemented as a proper directive any more, because we need it be processed while we
// bootstrap the system (before $parse is instantiated), for this reason we just have
// the csp() fn that looks for the `ng-csp` attribute anywhere in the current doc

/**
 * @ngdoc directive
 * @name ngClick
 *
 * @description
 * The ngClick directive allows you to specify custom behavior when
 * an element is clicked.
 *
 * @element ANY
 * @priority 0
 * @param {expression} ngClick {@link guide/expression Expression} to evaluate upon
 * click. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
   <example>
     <file name="index.html">
      <button ng-click="count = count + 1" ng-init="count=0">
        Increment
      </button>
      <span>
        count: {{count}}
      </span>
     </file>
     <file name="protractor.js" type="protractor">
       it('should check ng-click', function() {
         expect(element(by.binding('count')).getText()).toMatch('0');
         element(by.css('button')).click();
         expect(element(by.binding('count')).getText()).toMatch('1');
       });
     </file>
   </example>
 */
/*
 * A collection of directives that allows creation of custom event handlers that are defined as
 * angular expressions and are compiled and executed within the current scope.
 */
var ngEventDirectives = {};

// For events that might fire synchronously during DOM manipulation
// we need to execute their event handlers asynchronously using $evalAsync,
// so that they are not executed in an inconsistent state.
var forceAsyncEvents = {
  'blur': true,
  'focus': true
};
forEach(
  'click dblclick mousedown mouseup mouseover mouseout mousemove mouseenter mouseleave keydown keyup keypress submit focus blur copy cut paste'.split(' '),
  function(eventName) {
    var directiveName = directiveNormalize('ng-' + eventName);
    ngEventDirectives[directiveName] = ['$parse', '$rootScope', function($parse, $rootScope) {
      return {
        restrict: 'A',
        compile: function($element, attr) {
          // We expose the powerful $event object on the scope that provides access to the Window,
          // etc. that isn't protected by the fast paths in $parse.  We explicitly request better
          // checks at the cost of speed since event handler expressions are not executed as
          // frequently as regular change detection.
          var fn = $parse(attr[directiveName], /* interceptorFn */ null, /* expensiveChecks */ true);
          return function ngEventHandler(scope, element) {
            element.on(eventName, function(event) {
              var callback = function() {
                fn(scope, {$event:event});
              };
              if (forceAsyncEvents[eventName] && $rootScope.$$phase) {
                scope.$evalAsync(callback);
              } else {
                scope.$apply(callback);
              }
            });
          };
        }
      };
    }];
  }
);

/**
 * @ngdoc directive
 * @name ngDblclick
 *
 * @description
 * The `ngDblclick` directive allows you to specify custom behavior on a dblclick event.
 *
 * @element ANY
 * @priority 0
 * @param {expression} ngDblclick {@link guide/expression Expression} to evaluate upon
 * a dblclick. (The Event object is available as `$event`)
 *
 * @example
   <example>
     <file name="index.html">
      <button ng-dblclick="count = count + 1" ng-init="count=0">
        Increment (on double click)
      </button>
      count: {{count}}
     </file>
   </example>
 */


/**
 * @ngdoc directive
 * @name ngMousedown
 *
 * @description
 * The ngMousedown directive allows you to specify custom behavior on mousedown event.
 *
 * @element ANY
 * @priority 0
 * @param {expression} ngMousedown {@link guide/expression Expression} to evaluate upon
 * mousedown. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
   <example>
     <file name="index.html">
      <button ng-mousedown="count = count + 1" ng-init="count=0">
        Increment (on mouse down)
      </button>
      count: {{count}}
     </file>
   </example>
 */


/**
 * @ngdoc directive
 * @name ngMouseup
 *
 * @description
 * Specify custom behavior on mouseup event.
 *
 * @element ANY
 * @priority 0
 * @param {expression} ngMouseup {@link guide/expression Expression} to evaluate upon
 * mouseup. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
   <example>
     <file name="index.html">
      <button ng-mouseup="count = count + 1" ng-init="count=0">
        Increment (on mouse up)
      </button>
      count: {{count}}
     </file>
   </example>
 */

/**
 * @ngdoc directive
 * @name ngMouseover
 *
 * @description
 * Specify custom behavior on mouseover event.
 *
 * @element ANY
 * @priority 0
 * @param {expression} ngMouseover {@link guide/expression Expression} to evaluate upon
 * mouseover. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
   <example>
     <file name="index.html">
      <button ng-mouseover="count = count + 1" ng-init="count=0">
        Increment (when mouse is over)
      </button>
      count: {{count}}
     </file>
   </example>
 */


/**
 * @ngdoc directive
 * @name ngMouseenter
 *
 * @description
 * Specify custom behavior on mouseenter event.
 *
 * @element ANY
 * @priority 0
 * @param {expression} ngMouseenter {@link guide/expression Expression} to evaluate upon
 * mouseenter. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
   <example>
     <file name="index.html">
      <button ng-mouseenter="count = count + 1" ng-init="count=0">
        Increment (when mouse enters)
      </button>
      count: {{count}}
     </file>
   </example>
 */


/**
 * @ngdoc directive
 * @name ngMouseleave
 *
 * @description
 * Specify custom behavior on mouseleave event.
 *
 * @element ANY
 * @priority 0
 * @param {expression} ngMouseleave {@link guide/expression Expression} to evaluate upon
 * mouseleave. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
   <example>
     <file name="index.html">
      <button ng-mouseleave="count = count + 1" ng-init="count=0">
        Increment (when mouse leaves)
      </button>
      count: {{count}}
     </file>
   </example>
 */


/**
 * @ngdoc directive
 * @name ngMousemove
 *
 * @description
 * Specify custom behavior on mousemove event.
 *
 * @element ANY
 * @priority 0
 * @param {expression} ngMousemove {@link guide/expression Expression} to evaluate upon
 * mousemove. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
   <example>
     <file name="index.html">
      <button ng-mousemove="count = count + 1" ng-init="count=0">
        Increment (when mouse moves)
      </button>
      count: {{count}}
     </file>
   </example>
 */


/**
 * @ngdoc directive
 * @name ngKeydown
 *
 * @description
 * Specify custom behavior on keydown event.
 *
 * @element ANY
 * @priority 0
 * @param {expression} ngKeydown {@link guide/expression Expression} to evaluate upon
 * keydown. (Event object is available as `$event` and can be interrogated for keyCode, altKey, etc.)
 *
 * @example
   <example>
     <file name="index.html">
      <input ng-keydown="count = count + 1" ng-init="count=0">
      key down count: {{count}}
     </file>
   </example>
 */


/**
 * @ngdoc directive
 * @name ngKeyup
 *
 * @description
 * Specify custom behavior on keyup event.
 *
 * @element ANY
 * @priority 0
 * @param {expression} ngKeyup {@link guide/expression Expression} to evaluate upon
 * keyup. (Event object is available as `$event` and can be interrogated for keyCode, altKey, etc.)
 *
 * @example
   <example>
     <file name="index.html">
       <p>Typing in the input box below updates the key count</p>
       <input ng-keyup="count = count + 1" ng-init="count=0"> key up count: {{count}}

       <p>Typing in the input box below updates the keycode</p>
       <input ng-keyup="event=$event">
       <p>event keyCode: {{ event.keyCode }}</p>
       <p>event altKey: {{ event.altKey }}</p>
     </file>
   </example>
 */


/**
 * @ngdoc directive
 * @name ngKeypress
 *
 * @description
 * Specify custom behavior on keypress event.
 *
 * @element ANY
 * @param {expression} ngKeypress {@link guide/expression Expression} to evaluate upon
 * keypress. ({@link guide/expression#-event- Event object is available as `$event`}
 * and can be interrogated for keyCode, altKey, etc.)
 *
 * @example
   <example>
     <file name="index.html">
      <input ng-keypress="count = count + 1" ng-init="count=0">
      key press count: {{count}}
     </file>
   </example>
 */


/**
 * @ngdoc directive
 * @name ngSubmit
 *
 * @description
 * Enables binding angular expressions to onsubmit events.
 *
 * Additionally it prevents the default action (which for form means sending the request to the
 * server and reloading the current page), but only if the form does not contain `action`,
 * `data-action`, or `x-action` attributes.
 *
 * <div class="alert alert-warning">
 * **Warning:** Be careful not to cause "double-submission" by using both the `ngClick` and
 * `ngSubmit` handlers together. See the
 * {@link form#submitting-a-form-and-preventing-the-default-action `form` directive documentation}
 * for a detailed discussion of when `ngSubmit` may be triggered.
 * </div>
 *
 * @element form
 * @priority 0
 * @param {expression} ngSubmit {@link guide/expression Expression} to eval.
 * ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
   <example module="submitExample">
     <file name="index.html">
      <script>
        angular.module('submitExample', [])
          .controller('ExampleController', ['$scope', function($scope) {
            $scope.list = [];
            $scope.text = 'hello';
            $scope.submit = function() {
              if ($scope.text) {
                $scope.list.push(this.text);
                $scope.text = '';
              }
            };
          }]);
      </script>
      <form ng-submit="submit()" ng-controller="ExampleController">
        Enter text and hit enter:
        <input type="text" ng-model="text" name="text" />
        <input type="submit" id="submit" value="Submit" />
        <pre>list={{list}}</pre>
      </form>
     </file>
     <file name="protractor.js" type="protractor">
       it('should check ng-submit', function() {
         expect(element(by.binding('list')).getText()).toBe('list=[]');
         element(by.css('#submit')).click();
         expect(element(by.binding('list')).getText()).toContain('hello');
         expect(element(by.model('text')).getAttribute('value')).toBe('');
       });
       it('should ignore empty strings', function() {
         expect(element(by.binding('list')).getText()).toBe('list=[]');
         element(by.css('#submit')).click();
         element(by.css('#submit')).click();
         expect(element(by.binding('list')).getText()).toContain('hello');
        });
     </file>
   </example>
 */

/**
 * @ngdoc directive
 * @name ngFocus
 *
 * @description
 * Specify custom behavior on focus event.
 *
 * Note: As the `focus` event is executed synchronously when calling `input.focus()`
 * AngularJS executes the expression using `scope.$evalAsync` if the event is fired
 * during an `$apply` to ensure a consistent state.
 *
 * @element window, input, select, textarea, a
 * @priority 0
 * @param {expression} ngFocus {@link guide/expression Expression} to evaluate upon
 * focus. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
 * See {@link ng.directive:ngClick ngClick}
 */

/**
 * @ngdoc directive
 * @name ngBlur
 *
 * @description
 * Specify custom behavior on blur event.
 *
 * A [blur event](https://developer.mozilla.org/en-US/docs/Web/Events/blur) fires when
 * an element has lost focus.
 *
 * Note: As the `blur` event is executed synchronously also during DOM manipulations
 * (e.g. removing a focussed input),
 * AngularJS executes the expression using `scope.$evalAsync` if the event is fired
 * during an `$apply` to ensure a consistent state.
 *
 * @element window, input, select, textarea, a
 * @priority 0
 * @param {expression} ngBlur {@link guide/expression Expression} to evaluate upon
 * blur. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
 * See {@link ng.directive:ngClick ngClick}
 */

/**
 * @ngdoc directive
 * @name ngCopy
 *
 * @description
 * Specify custom behavior on copy event.
 *
 * @element window, input, select, textarea, a
 * @priority 0
 * @param {expression} ngCopy {@link guide/expression Expression} to evaluate upon
 * copy. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
   <example>
     <file name="index.html">
      <input ng-copy="copied=true" ng-init="copied=false; value='copy me'" ng-model="value">
      copied: {{copied}}
     </file>
   </example>
 */

/**
 * @ngdoc directive
 * @name ngCut
 *
 * @description
 * Specify custom behavior on cut event.
 *
 * @element window, input, select, textarea, a
 * @priority 0
 * @param {expression} ngCut {@link guide/expression Expression} to evaluate upon
 * cut. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
   <example>
     <file name="index.html">
      <input ng-cut="cut=true" ng-init="cut=false; value='cut me'" ng-model="value">
      cut: {{cut}}
     </file>
   </example>
 */

/**
 * @ngdoc directive
 * @name ngPaste
 *
 * @description
 * Specify custom behavior on paste event.
 *
 * @element window, input, select, textarea, a
 * @priority 0
 * @param {expression} ngPaste {@link guide/expression Expression} to evaluate upon
 * paste. ({@link guide/expression#-event- Event object is available as `$event`})
 *
 * @example
   <example>
     <file name="index.html">
      <input ng-paste="paste=true" ng-init="paste=false" placeholder='paste here'>
      pasted: {{paste}}
     </file>
   </example>
 */

/**
 * @ngdoc directive
 * @name ngIf
 * @restrict A
 * @multiElement
 *
 * @description
 * The `ngIf` directive removes or recreates a portion of the DOM tree based on an
 * {expression}. If the expression assigned to `ngIf` evaluates to a false
 * value then the element is removed from the DOM, otherwise a clone of the
 * element is reinserted into the DOM.
 *
 * `ngIf` differs from `ngShow` and `ngHide` in that `ngIf` completely removes and recreates the
 * element in the DOM rather than changing its visibility via the `display` css property.  A common
 * case when this difference is significant is when using css selectors that rely on an element's
 * position within the DOM, such as the `:first-child` or `:last-child` pseudo-classes.
 *
 * Note that when an element is removed using `ngIf` its scope is destroyed and a new scope
 * is created when the element is restored.  The scope created within `ngIf` inherits from
 * its parent scope using
 * [prototypal inheritance](https://github.com/angular/angular.js/wiki/Understanding-Scopes#javascript-prototypal-inheritance).
 * An important implication of this is if `ngModel` is used within `ngIf` to bind to
 * a javascript primitive defined in the parent scope. In this case any modifications made to the
 * variable within the child scope will override (hide) the value in the parent scope.
 *
 * Also, `ngIf` recreates elements using their compiled state. An example of this behavior
 * is if an element's class attribute is directly modified after it's compiled, using something like
 * jQuery's `.addClass()` method, and the element is later removed. When `ngIf` recreates the element
 * the added class will be lost because the original compiled state is used to regenerate the element.
 *
 * Additionally, you can provide animations via the `ngAnimate` module to animate the `enter`
 * and `leave` effects.
 *
 * @animations
 * enter - happens just after the `ngIf` contents change and a new DOM element is created and injected into the `ngIf` container
 * leave - happens just before the `ngIf` contents are removed from the DOM
 *
 * @element ANY
 * @scope
 * @priority 600
 * @param {expression} ngIf If the {@link guide/expression expression} is falsy then
 *     the element is removed from the DOM tree. If it is truthy a copy of the compiled
 *     element is added to the DOM tree.
 *
 * @example
  <example module="ngAnimate" deps="angular-animate.js" animations="true">
    <file name="index.html">
      <label>Click me: <input type="checkbox" ng-model="checked" ng-init="checked=true" /></label><br/>
      Show when checked:
      <span ng-if="checked" class="animate-if">
        This is removed when the checkbox is unchecked.
      </span>
    </file>
    <file name="animations.css">
      .animate-if {
        background:white;
        border:1px solid black;
        padding:10px;
      }

      .animate-if.ng-enter, .animate-if.ng-leave {
        transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 0.5s;
      }

      .animate-if.ng-enter,
      .animate-if.ng-leave.ng-leave-active {
        opacity:0;
      }

      .animate-if.ng-leave,
      .animate-if.ng-enter.ng-enter-active {
        opacity:1;
      }
    </file>
  </example>
 */
var ngIfDirective = ['$animate', function($animate) {
  return {
    multiElement: true,
    transclude: 'element',
    priority: 600,
    terminal: true,
    restrict: 'A',
    $$tlb: true,
    link: function($scope, $element, $attr, ctrl, $transclude) {
        var block, childScope, previousElements;
        $scope.$watch($attr.ngIf, function ngIfWatchAction(value) {

          if (value) {
            if (!childScope) {
              $transclude(function(clone, newScope) {
                childScope = newScope;
                clone[clone.length++] = document.createComment(' end ngIf: ' + $attr.ngIf + ' ');
                // Note: We only need the first/last node of the cloned nodes.
                // However, we need to keep the reference to the jqlite wrapper as it might be changed later
                // by a directive with templateUrl when its template arrives.
                block = {
                  clone: clone
                };
                $animate.enter(clone, $element.parent(), $element);
              });
            }
          } else {
            if (previousElements) {
              previousElements.remove();
              previousElements = null;
            }
            if (childScope) {
              childScope.$destroy();
              childScope = null;
            }
            if (block) {
              previousElements = getBlockNodes(block.clone);
              $animate.leave(previousElements).then(function() {
                previousElements = null;
              });
              block = null;
            }
          }
        });
    }
  };
}];

/**
 * @ngdoc directive
 * @name ngInclude
 * @restrict ECA
 *
 * @description
 * Fetches, compiles and includes an external HTML fragment.
 *
 * By default, the template URL is restricted to the same domain and protocol as the
 * application document. This is done by calling {@link $sce#getTrustedResourceUrl
 * $sce.getTrustedResourceUrl} on it. To load templates from other domains or protocols
 * you may either {@link ng.$sceDelegateProvider#resourceUrlWhitelist whitelist them} or
 * {@link $sce#trustAsResourceUrl wrap them} as trusted values. Refer to Angular's {@link
 * ng.$sce Strict Contextual Escaping}.
 *
 * In addition, the browser's
 * [Same Origin Policy](https://code.google.com/p/browsersec/wiki/Part2#Same-origin_policy_for_XMLHttpRequest)
 * and [Cross-Origin Resource Sharing (CORS)](http://www.w3.org/TR/cors/)
 * policy may further restrict whether the template is successfully loaded.
 * For example, `ngInclude` won't work for cross-domain requests on all browsers and for `file://`
 * access on some browsers.
 *
 * @animations
 * enter - animation is used to bring new content into the browser.
 * leave - animation is used to animate existing content away.
 *
 * The enter and leave animation occur concurrently.
 *
 * @scope
 * @priority 400
 *
 * @param {string} ngInclude|src angular expression evaluating to URL. If the source is a string constant,
 *                 make sure you wrap it in **single** quotes, e.g. `src="'myPartialTemplate.html'"`.
 * @param {string=} onload Expression to evaluate when a new partial is loaded.
 *                  <div class="alert alert-warning">
 *                  **Note:** When using onload on SVG elements in IE11, the browser will try to call
 *                  a function with the name on the window element, which will usually throw a
 *                  "function is undefined" error. To fix this, you can instead use `data-onload` or a
 *                  different form that {@link guide/directive#normalization matches} `onload`.
 *                  </div>
   *
 * @param {string=} autoscroll Whether `ngInclude` should call {@link ng.$anchorScroll
 *                  $anchorScroll} to scroll the viewport after the content is loaded.
 *
 *                  - If the attribute is not set, disable scrolling.
 *                  - If the attribute is set without value, enable scrolling.
 *                  - Otherwise enable scrolling only if the expression evaluates to truthy value.
 *
 * @example
  <example module="includeExample" deps="angular-animate.js" animations="true">
    <file name="index.html">
     <div ng-controller="ExampleController">
       <select ng-model="template" ng-options="t.name for t in templates">
        <option value="">(blank)</option>
       </select>
       url of the template: <code>{{template.url}}</code>
       <hr/>
       <div class="slide-animate-container">
         <div class="slide-animate" ng-include="template.url"></div>
       </div>
     </div>
    </file>
    <file name="script.js">
      angular.module('includeExample', ['ngAnimate'])
        .controller('ExampleController', ['$scope', function($scope) {
          $scope.templates =
            [ { name: 'template1.html', url: 'template1.html'},
              { name: 'template2.html', url: 'template2.html'} ];
          $scope.template = $scope.templates[0];
        }]);
     </file>
    <file name="template1.html">
      Content of template1.html
    </file>
    <file name="template2.html">
      Content of template2.html
    </file>
    <file name="animations.css">
      .slide-animate-container {
        position:relative;
        background:white;
        border:1px solid black;
        height:40px;
        overflow:hidden;
      }

      .slide-animate {
        padding:10px;
      }

      .slide-animate.ng-enter, .slide-animate.ng-leave {
        transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 0.5s;

        position:absolute;
        top:0;
        left:0;
        right:0;
        bottom:0;
        display:block;
        padding:10px;
      }

      .slide-animate.ng-enter {
        top:-50px;
      }
      .slide-animate.ng-enter.ng-enter-active {
        top:0;
      }

      .slide-animate.ng-leave {
        top:0;
      }
      .slide-animate.ng-leave.ng-leave-active {
        top:50px;
      }
    </file>
    <file name="protractor.js" type="protractor">
      var templateSelect = element(by.model('template'));
      var includeElem = element(by.css('[ng-include]'));

      it('should load template1.html', function() {
        expect(includeElem.getText()).toMatch(/Content of template1.html/);
      });

      it('should load template2.html', function() {
        if (browser.params.browser == 'firefox') {
          // Firefox can't handle using selects
          // See https://github.com/angular/protractor/issues/480
          return;
        }
        templateSelect.click();
        templateSelect.all(by.css('option')).get(2).click();
        expect(includeElem.getText()).toMatch(/Content of template2.html/);
      });

      it('should change to blank', function() {
        if (browser.params.browser == 'firefox') {
          // Firefox can't handle using selects
          return;
        }
        templateSelect.click();
        templateSelect.all(by.css('option')).get(0).click();
        expect(includeElem.isPresent()).toBe(false);
      });
    </file>
  </example>
 */


/**
 * @ngdoc event
 * @name ngInclude#$includeContentRequested
 * @eventType emit on the scope ngInclude was declared in
 * @description
 * Emitted every time the ngInclude content is requested.
 *
 * @param {Object} angularEvent Synthetic event object.
 * @param {String} src URL of content to load.
 */


/**
 * @ngdoc event
 * @name ngInclude#$includeContentLoaded
 * @eventType emit on the current ngInclude scope
 * @description
 * Emitted every time the ngInclude content is reloaded.
 *
 * @param {Object} angularEvent Synthetic event object.
 * @param {String} src URL of content to load.
 */


/**
 * @ngdoc event
 * @name ngInclude#$includeContentError
 * @eventType emit on the scope ngInclude was declared in
 * @description
 * Emitted when a template HTTP request yields an erroneous response (status < 200 || status > 299)
 *
 * @param {Object} angularEvent Synthetic event object.
 * @param {String} src URL of content to load.
 */
var ngIncludeDirective = ['$templateRequest', '$anchorScroll', '$animate',
                  function($templateRequest,   $anchorScroll,   $animate) {
  return {
    restrict: 'ECA',
    priority: 400,
    terminal: true,
    transclude: 'element',
    controller: angular.noop,
    compile: function(element, attr) {
      var srcExp = attr.ngInclude || attr.src,
          onloadExp = attr.onload || '',
          autoScrollExp = attr.autoscroll;

      return function(scope, $element, $attr, ctrl, $transclude) {
        var changeCounter = 0,
            currentScope,
            previousElement,
            currentElement;

        var cleanupLastIncludeContent = function() {
          if (previousElement) {
            previousElement.remove();
            previousElement = null;
          }
          if (currentScope) {
            currentScope.$destroy();
            currentScope = null;
          }
          if (currentElement) {
            $animate.leave(currentElement).then(function() {
              previousElement = null;
            });
            previousElement = currentElement;
            currentElement = null;
          }
        };

        scope.$watch(srcExp, function ngIncludeWatchAction(src) {
          var afterAnimation = function() {
            if (isDefined(autoScrollExp) && (!autoScrollExp || scope.$eval(autoScrollExp))) {
              $anchorScroll();
            }
          };
          var thisChangeId = ++changeCounter;

          if (src) {
            //set the 2nd param to true to ignore the template request error so that the inner
            //contents and scope can be cleaned up.
            $templateRequest(src, true).then(function(response) {
              if (thisChangeId !== changeCounter) return;
              var newScope = scope.$new();
              ctrl.template = response;

              // Note: This will also link all children of ng-include that were contained in the original
              // html. If that content contains controllers, ... they could pollute/change the scope.
              // However, using ng-include on an element with additional content does not make sense...
              // Note: We can't remove them in the cloneAttchFn of $transclude as that
              // function is called before linking the content, which would apply child
              // directives to non existing elements.
              var clone = $transclude(newScope, function(clone) {
                cleanupLastIncludeContent();
                $animate.enter(clone, null, $element).then(afterAnimation);
              });

              currentScope = newScope;
              currentElement = clone;

              currentScope.$emit('$includeContentLoaded', src);
              scope.$eval(onloadExp);
            }, function() {
              if (thisChangeId === changeCounter) {
                cleanupLastIncludeContent();
                scope.$emit('$includeContentError', src);
              }
            });
            scope.$emit('$includeContentRequested', src);
          } else {
            cleanupLastIncludeContent();
            ctrl.template = null;
          }
        });
      };
    }
  };
}];

// This directive is called during the $transclude call of the first `ngInclude` directive.
// It will replace and compile the content of the element with the loaded template.
// We need this directive so that the element content is already filled when
// the link function of another directive on the same element as ngInclude
// is called.
var ngIncludeFillContentDirective = ['$compile',
  function($compile) {
    return {
      restrict: 'ECA',
      priority: -400,
      require: 'ngInclude',
      link: function(scope, $element, $attr, ctrl) {
        if (/SVG/.test($element[0].toString())) {
          // WebKit: https://bugs.webkit.org/show_bug.cgi?id=135698 --- SVG elements do not
          // support innerHTML, so detect this here and try to generate the contents
          // specially.
          $element.empty();
          $compile(jqLiteBuildFragment(ctrl.template, document).childNodes)(scope,
              function namespaceAdaptedClone(clone) {
            $element.append(clone);
          }, {futureParentElement: $element});
          return;
        }

        $element.html(ctrl.template);
        $compile($element.contents())(scope);
      }
    };
  }];

/**
 * @ngdoc directive
 * @name ngInit
 * @restrict AC
 *
 * @description
 * The `ngInit` directive allows you to evaluate an expression in the
 * current scope.
 *
 * <div class="alert alert-danger">
 * This directive can be abused to add unnecessary amounts of logic into your templates.
 * There are only a few appropriate uses of `ngInit`, such as for aliasing special properties of
 * {@link ng.directive:ngRepeat `ngRepeat`}, as seen in the demo below; and for injecting data via
 * server side scripting. Besides these few cases, you should use {@link guide/controller controllers}
 * rather than `ngInit` to initialize values on a scope.
 * </div>
 *
 * <div class="alert alert-warning">
 * **Note**: If you have assignment in `ngInit` along with a {@link ng.$filter `filter`}, make
 * sure you have parentheses to ensure correct operator precedence:
 * <pre class="prettyprint">
 * `<div ng-init="test1 = ($index | toString)"></div>`
 * </pre>
 * </div>
 *
 * @priority 450
 *
 * @element ANY
 * @param {expression} ngInit {@link guide/expression Expression} to eval.
 *
 * @example
   <example module="initExample">
     <file name="index.html">
   <script>
     angular.module('initExample', [])
       .controller('ExampleController', ['$scope', function($scope) {
         $scope.list = [['a', 'b'], ['c', 'd']];
       }]);
   </script>
   <div ng-controller="ExampleController">
     <div ng-repeat="innerList in list" ng-init="outerIndex = $index">
       <div ng-repeat="value in innerList" ng-init="innerIndex = $index">
          <span class="example-init">list[ {{outerIndex}} ][ {{innerIndex}} ] = {{value}};</span>
       </div>
     </div>
   </div>
     </file>
     <file name="protractor.js" type="protractor">
       it('should alias index positions', function() {
         var elements = element.all(by.css('.example-init'));
         expect(elements.get(0).getText()).toBe('list[ 0 ][ 0 ] = a;');
         expect(elements.get(1).getText()).toBe('list[ 0 ][ 1 ] = b;');
         expect(elements.get(2).getText()).toBe('list[ 1 ][ 0 ] = c;');
         expect(elements.get(3).getText()).toBe('list[ 1 ][ 1 ] = d;');
       });
     </file>
   </example>
 */
var ngInitDirective = ngDirective({
  priority: 450,
  compile: function() {
    return {
      pre: function(scope, element, attrs) {
        scope.$eval(attrs.ngInit);
      }
    };
  }
});

/**
 * @ngdoc directive
 * @name ngList
 *
 * @description
 * Text input that converts between a delimited string and an array of strings. The default
 * delimiter is a comma followed by a space - equivalent to `ng-list=", "`. You can specify a custom
 * delimiter as the value of the `ngList` attribute - for example, `ng-list=" | "`.
 *
 * The behaviour of the directive is affected by the use of the `ngTrim` attribute.
 * * If `ngTrim` is set to `"false"` then whitespace around both the separator and each
 *   list item is respected. This implies that the user of the directive is responsible for
 *   dealing with whitespace but also allows you to use whitespace as a delimiter, such as a
 *   tab or newline character.
 * * Otherwise whitespace around the delimiter is ignored when splitting (although it is respected
 *   when joining the list items back together) and whitespace around each list item is stripped
 *   before it is added to the model.
 *
 * ### Example with Validation
 *
 * <example name="ngList-directive" module="listExample">
 *   <file name="app.js">
 *      angular.module('listExample', [])
 *        .controller('ExampleController', ['$scope', function($scope) {
 *          $scope.names = ['morpheus', 'neo', 'trinity'];
 *        }]);
 *   </file>
 *   <file name="index.html">
 *    <form name="myForm" ng-controller="ExampleController">
 *      <label>List: <input name="namesInput" ng-model="names" ng-list required></label>
 *      <span role="alert">
 *        <span class="error" ng-show="myForm.namesInput.$error.required">
 *        Required!</span>
 *      </span>
 *      <br>
 *      <tt>names = {{names}}</tt><br/>
 *      <tt>myForm.namesInput.$valid = {{myForm.namesInput.$valid}}</tt><br/>
 *      <tt>myForm.namesInput.$error = {{myForm.namesInput.$error}}</tt><br/>
 *      <tt>myForm.$valid = {{myForm.$valid}}</tt><br/>
 *      <tt>myForm.$error.required = {{!!myForm.$error.required}}</tt><br/>
 *     </form>
 *   </file>
 *   <file name="protractor.js" type="protractor">
 *     var listInput = element(by.model('names'));
 *     var names = element(by.exactBinding('names'));
 *     var valid = element(by.binding('myForm.namesInput.$valid'));
 *     var error = element(by.css('span.error'));
 *
 *     it('should initialize to model', function() {
 *       expect(names.getText()).toContain('["morpheus","neo","trinity"]');
 *       expect(valid.getText()).toContain('true');
 *       expect(error.getCssValue('display')).toBe('none');
 *     });
 *
 *     it('should be invalid if empty', function() {
 *       listInput.clear();
 *       listInput.sendKeys('');
 *
 *       expect(names.getText()).toContain('');
 *       expect(valid.getText()).toContain('false');
 *       expect(error.getCssValue('display')).not.toBe('none');
 *     });
 *   </file>
 * </example>
 *
 * ### Example - splitting on newline
 * <example name="ngList-directive-newlines">
 *   <file name="index.html">
 *    <textarea ng-model="list" ng-list="&#10;" ng-trim="false"></textarea>
 *    <pre>{{ list | json }}</pre>
 *   </file>
 *   <file name="protractor.js" type="protractor">
 *     it("should split the text by newlines", function() {
 *       var listInput = element(by.model('list'));
 *       var output = element(by.binding('list | json'));
 *       listInput.sendKeys('abc\ndef\nghi');
 *       expect(output.getText()).toContain('[\n  "abc",\n  "def",\n  "ghi"\n]');
 *     });
 *   </file>
 * </example>
 *
 * @element input
 * @param {string=} ngList optional delimiter that should be used to split the value.
 */
var ngListDirective = function() {
  return {
    restrict: 'A',
    priority: 100,
    require: 'ngModel',
    link: function(scope, element, attr, ctrl) {
      // We want to control whitespace trimming so we use this convoluted approach
      // to access the ngList attribute, which doesn't pre-trim the attribute
      var ngList = element.attr(attr.$attr.ngList) || ', ';
      var trimValues = attr.ngTrim !== 'false';
      var separator = trimValues ? trim(ngList) : ngList;

      var parse = function(viewValue) {
        // If the viewValue is invalid (say required but empty) it will be `undefined`
        if (isUndefined(viewValue)) return;

        var list = [];

        if (viewValue) {
          forEach(viewValue.split(separator), function(value) {
            if (value) list.push(trimValues ? trim(value) : value);
          });
        }

        return list;
      };

      ctrl.$parsers.push(parse);
      ctrl.$formatters.push(function(value) {
        if (isArray(value)) {
          return value.join(ngList);
        }

        return undefined;
      });

      // Override the standard $isEmpty because an empty array means the input is empty.
      ctrl.$isEmpty = function(value) {
        return !value || !value.length;
      };
    }
  };
};

/* global VALID_CLASS: true,
  INVALID_CLASS: true,
  PRISTINE_CLASS: true,
  DIRTY_CLASS: true,
  UNTOUCHED_CLASS: true,
  TOUCHED_CLASS: true,
*/

var VALID_CLASS = 'ng-valid',
    INVALID_CLASS = 'ng-invalid',
    PRISTINE_CLASS = 'ng-pristine',
    DIRTY_CLASS = 'ng-dirty',
    UNTOUCHED_CLASS = 'ng-untouched',
    TOUCHED_CLASS = 'ng-touched',
    PENDING_CLASS = 'ng-pending';

var ngModelMinErr = minErr('ngModel');

/**
 * @ngdoc type
 * @name ngModel.NgModelController
 *
 * @property {*} $viewValue The actual value from the control's view. For `input` elements, this is a
 * String. See {@link ngModel.NgModelController#$setViewValue} for information about when the $viewValue
 * is set.
 * @property {*} $modelValue The value in the model that the control is bound to.
 * @property {Array.<Function>} $parsers Array of functions to execute, as a pipeline, whenever
       the control reads value from the DOM. The functions are called in array order, each passing
       its return value through to the next. The last return value is forwarded to the
       {@link ngModel.NgModelController#$validators `$validators`} collection.

Parsers are used to sanitize / convert the {@link ngModel.NgModelController#$viewValue
`$viewValue`}.

Returning `undefined` from a parser means a parse error occurred. In that case,
no {@link ngModel.NgModelController#$validators `$validators`} will run and the `ngModel`
will be set to `undefined` unless {@link ngModelOptions `ngModelOptions.allowInvalid`}
is set to `true`. The parse error is stored in `ngModel.$error.parse`.

 *
 * @property {Array.<Function>} $formatters Array of functions to execute, as a pipeline, whenever
       the model value changes. The functions are called in reverse array order, each passing the value through to the
       next. The last return value is used as the actual DOM value.
       Used to format / convert values for display in the control.
 * ```js
 * function formatter(value) {
 *   if (value) {
 *     return value.toUpperCase();
 *   }
 * }
 * ngModel.$formatters.push(formatter);
 * ```
 *
 * @property {Object.<string, function>} $validators A collection of validators that are applied
 *      whenever the model value changes. The key value within the object refers to the name of the
 *      validator while the function refers to the validation operation. The validation operation is
 *      provided with the model value as an argument and must return a true or false value depending
 *      on the response of that validation.
 *
 * ```js
 * ngModel.$validators.validCharacters = function(modelValue, viewValue) {
 *   var value = modelValue || viewValue;
 *   return /[0-9]+/.test(value) &&
 *          /[a-z]+/.test(value) &&
 *          /[A-Z]+/.test(value) &&
 *          /\W+/.test(value);
 * };
 * ```
 *
 * @property {Object.<string, function>} $asyncValidators A collection of validations that are expected to
 *      perform an asynchronous validation (e.g. a HTTP request). The validation function that is provided
 *      is expected to return a promise when it is run during the model validation process. Once the promise
 *      is delivered then the validation status will be set to true when fulfilled and false when rejected.
 *      When the asynchronous validators are triggered, each of the validators will run in parallel and the model
 *      value will only be updated once all validators have been fulfilled. As long as an asynchronous validator
 *      is unfulfilled, its key will be added to the controllers `$pending` property. Also, all asynchronous validators
 *      will only run once all synchronous validators have passed.
 *
 * Please note that if $http is used then it is important that the server returns a success HTTP response code
 * in order to fulfill the validation and a status level of `4xx` in order to reject the validation.
 *
 * ```js
 * ngModel.$asyncValidators.uniqueUsername = function(modelValue, viewValue) {
 *   var value = modelValue || viewValue;
 *
 *   // Lookup user by username
 *   return $http.get('/api/users/' + value).
 *      then(function resolved() {
 *        //username exists, this means validation fails
 *        return $q.reject('exists');
 *      }, function rejected() {
 *        //username does not exist, therefore this validation passes
 *        return true;
 *      });
 * };
 * ```
 *
 * @property {Array.<Function>} $viewChangeListeners Array of functions to execute whenever the
 *     view value has changed. It is called with no arguments, and its return value is ignored.
 *     This can be used in place of additional $watches against the model value.
 *
 * @property {Object} $error An object hash with all failing validator ids as keys.
 * @property {Object} $pending An object hash with all pending validator ids as keys.
 *
 * @property {boolean} $untouched True if control has not lost focus yet.
 * @property {boolean} $touched True if control has lost focus.
 * @property {boolean} $pristine True if user has not interacted with the control yet.
 * @property {boolean} $dirty True if user has already interacted with the control.
 * @property {boolean} $valid True if there is no error.
 * @property {boolean} $invalid True if at least one error on the control.
 * @property {string} $name The name attribute of the control.
 *
 * @description
 *
 * `NgModelController` provides API for the {@link ngModel `ngModel`} directive.
 * The controller contains services for data-binding, validation, CSS updates, and value formatting
 * and parsing. It purposefully does not contain any logic which deals with DOM rendering or
 * listening to DOM events.
 * Such DOM related logic should be provided by other directives which make use of
 * `NgModelController` for data-binding to control elements.
 * Angular provides this DOM logic for most {@link input `input`} elements.
 * At the end of this page you can find a {@link ngModel.NgModelController#custom-control-example
 * custom control example} that uses `ngModelController` to bind to `contenteditable` elements.
 *
 * @example
 * ### Custom Control Example
 * This example shows how to use `NgModelController` with a custom control to achieve
 * data-binding. Notice how different directives (`contenteditable`, `ng-model`, and `required`)
 * collaborate together to achieve the desired result.
 *
 * `contenteditable` is an HTML5 attribute, which tells the browser to let the element
 * contents be edited in place by the user.
 *
 * We are using the {@link ng.service:$sce $sce} service here and include the {@link ngSanitize $sanitize}
 * module to automatically remove "bad" content like inline event listener (e.g. `<span onclick="...">`).
 * However, as we are using `$sce` the model can still decide to provide unsafe content if it marks
 * that content using the `$sce` service.
 *
 * <example name="NgModelController" module="customControl" deps="angular-sanitize.js">
    <file name="style.css">
      [contenteditable] {
        border: 1px solid black;
        background-color: white;
        min-height: 20px;
      }

      .ng-invalid {
        border: 1px solid red;
      }

    </file>
    <file name="script.js">
      angular.module('customControl', ['ngSanitize']).
        directive('contenteditable', ['$sce', function($sce) {
          return {
            restrict: 'A', // only activate on element attribute
            require: '?ngModel', // get a hold of NgModelController
            link: function(scope, element, attrs, ngModel) {
              if (!ngModel) return; // do nothing if no ng-model

              // Specify how UI should be updated
              ngModel.$render = function() {
                element.html($sce.getTrustedHtml(ngModel.$viewValue || ''));
              };

              // Listen for change events to enable binding
              element.on('blur keyup change', function() {
                scope.$evalAsync(read);
              });
              read(); // initialize

              // Write data to the model
              function read() {
                var html = element.html();
                // When we clear the content editable the browser leaves a <br> behind
                // If strip-br attribute is provided then we strip this out
                if ( attrs.stripBr && html == '<br>' ) {
                  html = '';
                }
                ngModel.$setViewValue(html);
              }
            }
          };
        }]);
    </file>
    <file name="index.html">
      <form name="myForm">
       <div contenteditable
            name="myWidget" ng-model="userContent"
            strip-br="true"
            required>Change me!</div>
        <span ng-show="myForm.myWidget.$error.required">Required!</span>
       <hr>
       <textarea ng-model="userContent" aria-label="Dynamic textarea"></textarea>
      </form>
    </file>
    <file name="protractor.js" type="protractor">
    it('should data-bind and become invalid', function() {
      if (browser.params.browser == 'safari' || browser.params.browser == 'firefox') {
        // SafariDriver can't handle contenteditable
        // and Firefox driver can't clear contenteditables very well
        return;
      }
      var contentEditable = element(by.css('[contenteditable]'));
      var content = 'Change me!';

      expect(contentEditable.getText()).toEqual(content);

      contentEditable.clear();
      contentEditable.sendKeys(protractor.Key.BACK_SPACE);
      expect(contentEditable.getText()).toEqual('');
      expect(contentEditable.getAttribute('class')).toMatch(/ng-invalid-required/);
    });
    </file>
 * </example>
 *
 *
 */
var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$parse', '$animate', '$timeout', '$rootScope', '$q', '$interpolate',
    function($scope, $exceptionHandler, $attr, $element, $parse, $animate, $timeout, $rootScope, $q, $interpolate) {
  this.$viewValue = Number.NaN;
  this.$modelValue = Number.NaN;
  this.$$rawModelValue = undefined; // stores the parsed modelValue / model set from scope regardless of validity.
  this.$validators = {};
  this.$asyncValidators = {};
  this.$parsers = [];
  this.$formatters = [];
  this.$viewChangeListeners = [];
  this.$untouched = true;
  this.$touched = false;
  this.$pristine = true;
  this.$dirty = false;
  this.$valid = true;
  this.$invalid = false;
  this.$error = {}; // keep invalid keys here
  this.$$success = {}; // keep valid keys here
  this.$pending = undefined; // keep pending keys here
  this.$name = $interpolate($attr.name || '', false)($scope);
  this.$$parentForm = nullFormCtrl;

  var parsedNgModel = $parse($attr.ngModel),
      parsedNgModelAssign = parsedNgModel.assign,
      ngModelGet = parsedNgModel,
      ngModelSet = parsedNgModelAssign,
      pendingDebounce = null,
      parserValid,
      ctrl = this;

  this.$$setOptions = function(options) {
    ctrl.$options = options;
    if (options && options.getterSetter) {
      var invokeModelGetter = $parse($attr.ngModel + '()'),
          invokeModelSetter = $parse($attr.ngModel + '($$$p)');

      ngModelGet = function($scope) {
        var modelValue = parsedNgModel($scope);
        if (isFunction(modelValue)) {
          modelValue = invokeModelGetter($scope);
        }
        return modelValue;
      };
      ngModelSet = function($scope, newValue) {
        if (isFunction(parsedNgModel($scope))) {
          invokeModelSetter($scope, {$$$p: ctrl.$modelValue});
        } else {
          parsedNgModelAssign($scope, ctrl.$modelValue);
        }
      };
    } else if (!parsedNgModel.assign) {
      throw ngModelMinErr('nonassign', "Expression '{0}' is non-assignable. Element: {1}",
          $attr.ngModel, startingTag($element));
    }
  };

  /**
   * @ngdoc method
   * @name ngModel.NgModelController#$render
   *
   * @description
   * Called when the view needs to be updated. It is expected that the user of the ng-model
   * directive will implement this method.
   *
   * The `$render()` method is invoked in the following situations:
   *
   * * `$rollbackViewValue()` is called.  If we are rolling back the view value to the last
   *   committed value then `$render()` is called to update the input control.
   * * The value referenced by `ng-model` is changed programmatically and both the `$modelValue` and
   *   the `$viewValue` are different from last time.
   *
   * Since `ng-model` does not do a deep watch, `$render()` is only invoked if the values of
   * `$modelValue` and `$viewValue` are actually different from their previous value. If `$modelValue`
   * or `$viewValue` are objects (rather than a string or number) then `$render()` will not be
   * invoked if you only change a property on the objects.
   */
  this.$render = noop;

  /**
   * @ngdoc method
   * @name ngModel.NgModelController#$isEmpty
   *
   * @description
   * This is called when we need to determine if the value of an input is empty.
   *
   * For instance, the required directive does this to work out if the input has data or not.
   *
   * The default `$isEmpty` function checks whether the value is `undefined`, `''`, `null` or `NaN`.
   *
   * You can override this for input directives whose concept of being empty is different from the
   * default. The `checkboxInputType` directive does this because in its case a value of `false`
   * implies empty.
   *
   * @param {*} value The value of the input to check for emptiness.
   * @returns {boolean} True if `value` is "empty".
   */
  this.$isEmpty = function(value) {
    return isUndefined(value) || value === '' || value === null || value !== value;
  };

  var currentValidationRunId = 0;

  /**
   * @ngdoc method
   * @name ngModel.NgModelController#$setValidity
   *
   * @description
   * Change the validity state, and notify the form.
   *
   * This method can be called within $parsers/$formatters or a custom validation implementation.
   * However, in most cases it should be sufficient to use the `ngModel.$validators` and
   * `ngModel.$asyncValidators` collections which will call `$setValidity` automatically.
   *
   * @param {string} validationErrorKey Name of the validator. The `validationErrorKey` will be assigned
   *        to either `$error[validationErrorKey]` or `$pending[validationErrorKey]`
   *        (for unfulfilled `$asyncValidators`), so that it is available for data-binding.
   *        The `validationErrorKey` should be in camelCase and will get converted into dash-case
   *        for class name. Example: `myError` will result in `ng-valid-my-error` and `ng-invalid-my-error`
   *        class and can be bound to as  `{{someForm.someControl.$error.myError}}` .
   * @param {boolean} isValid Whether the current state is valid (true), invalid (false), pending (undefined),
   *                          or skipped (null). Pending is used for unfulfilled `$asyncValidators`.
   *                          Skipped is used by Angular when validators do not run because of parse errors and
   *                          when `$asyncValidators` do not run because any of the `$validators` failed.
   */
  addSetValidityMethod({
    ctrl: this,
    $element: $element,
    set: function(object, property) {
      object[property] = true;
    },
    unset: function(object, property) {
      delete object[property];
    },
    $animate: $animate
  });

  /**
   * @ngdoc method
   * @name ngModel.NgModelController#$setPristine
   *
   * @description
   * Sets the control to its pristine state.
   *
   * This method can be called to remove the `ng-dirty` class and set the control to its pristine
   * state (`ng-pristine` class). A model is considered to be pristine when the control
   * has not been changed from when first compiled.
   */
  this.$setPristine = function() {
    ctrl.$dirty = false;
    ctrl.$pristine = true;
    $animate.removeClass($element, DIRTY_CLASS);
    $animate.addClass($element, PRISTINE_CLASS);
  };

  /**
   * @ngdoc method
   * @name ngModel.NgModelController#$setDirty
   *
   * @description
   * Sets the control to its dirty state.
   *
   * This method can be called to remove the `ng-pristine` class and set the control to its dirty
   * state (`ng-dirty` class). A model is considered to be dirty when the control has been changed
   * from when first compiled.
   */
  this.$setDirty = function() {
    ctrl.$dirty = true;
    ctrl.$pristine = false;
    $animate.removeClass($element, PRISTINE_CLASS);
    $animate.addClass($element, DIRTY_CLASS);
    ctrl.$$parentForm.$setDirty();
  };

  /**
   * @ngdoc method
   * @name ngModel.NgModelController#$setUntouched
   *
   * @description
   * Sets the control to its untouched state.
   *
   * This method can be called to remove the `ng-touched` class and set the control to its
   * untouched state (`ng-untouched` class). Upon compilation, a model is set as untouched
   * by default, however this function can be used to restore that state if the model has
   * already been touched by the user.
   */
  this.$setUntouched = function() {
    ctrl.$touched = false;
    ctrl.$untouched = true;
    $animate.setClass($element, UNTOUCHED_CLASS, TOUCHED_CLASS);
  };

  /**
   * @ngdoc method
   * @name ngModel.NgModelController#$setTouched
   *
   * @description
   * Sets the control to its touched state.
   *
   * This method can be called to remove the `ng-untouched` class and set the control to its
   * touched state (`ng-touched` class). A model is considered to be touched when the user has
   * first focused the control element and then shifted focus away from the control (blur event).
   */
  this.$setTouched = function() {
    ctrl.$touched = true;
    ctrl.$untouched = false;
    $animate.setClass($element, TOUCHED_CLASS, UNTOUCHED_CLASS);
  };

  /**
   * @ngdoc method
   * @name ngModel.NgModelController#$rollbackViewValue
   *
   * @description
   * Cancel an update and reset the input element's value to prevent an update to the `$modelValue`,
   * which may be caused by a pending debounced event or because the input is waiting for a some
   * future event.
   *
   * If you have an input that uses `ng-model-options` to set up debounced events or events such
   * as blur you can have a situation where there is a period when the `$viewValue`
   * is out of synch with the ngModel's `$modelValue`.
   *
   * In this case, you can run into difficulties if you try to update the ngModel's `$modelValue`
   * programmatically before these debounced/future events have resolved/occurred, because Angular's
   * dirty checking mechanism is not able to tell whether the model has actually changed or not.
   *
   * The `$rollbackViewValue()` method should be called before programmatically changing the model of an
   * input which may have such events pending. This is important in order to make sure that the
   * input field will be updated with the new model value and any pending operations are cancelled.
   *
   * <example name="ng-model-cancel-update" module="cancel-update-example">
   *   <file name="app.js">
   *     angular.module('cancel-update-example', [])
   *
   *     .controller('CancelUpdateController', ['$scope', function($scope) {
   *       $scope.resetWithCancel = function(e) {
   *         if (e.keyCode == 27) {
   *           $scope.myForm.myInput1.$rollbackViewValue();
   *           $scope.myValue = '';
   *         }
   *       };
   *       $scope.resetWithoutCancel = function(e) {
   *         if (e.keyCode == 27) {
   *           $scope.myValue = '';
   *         }
   *       };
   *     }]);
   *   </file>
   *   <file name="index.html">
   *     <div ng-controller="CancelUpdateController">
   *       <p>Try typing something in each input.  See that the model only updates when you
   *          blur off the input.
   *        </p>
   *        <p>Now see what happens if you start typing then press the Escape key</p>
   *
   *       <form name="myForm" ng-model-options="{ updateOn: 'blur' }">
   *         <p id="inputDescription1">With $rollbackViewValue()</p>
   *         <input name="myInput1" aria-describedby="inputDescription1" ng-model="myValue"
   *                ng-keydown="resetWithCancel($event)"><br/>
   *         myValue: "{{ myValue }}"
   *
   *         <p id="inputDescription2">Without $rollbackViewValue()</p>
   *         <input name="myInput2" aria-describedby="inputDescription2" ng-model="myValue"
   *                ng-keydown="resetWithoutCancel($event)"><br/>
   *         myValue: "{{ myValue }}"
   *       </form>
   *     </div>
   *   </file>
   * </example>
   */
  this.$rollbackViewValue = function() {
    $timeout.cancel(pendingDebounce);
    ctrl.$viewValue = ctrl.$$lastCommittedViewValue;
    ctrl.$render();
  };

  /**
   * @ngdoc method
   * @name ngModel.NgModelController#$validate
   *
   * @description
   * Runs each of the registered validators (first synchronous validators and then
   * asynchronous validators).
   * If the validity changes to invalid, the model will be set to `undefined`,
   * unless {@link ngModelOptions `ngModelOptions.allowInvalid`} is `true`.
   * If the validity changes to valid, it will set the model to the last available valid
   * `$modelValue`, i.e. either the last parsed value or the last value set from the scope.
   */
  this.$validate = function() {
    // ignore $validate before model is initialized
    if (isNumber(ctrl.$modelValue) && isNaN(ctrl.$modelValue)) {
      return;
    }

    var viewValue = ctrl.$$lastCommittedViewValue;
    // Note: we use the $$rawModelValue as $modelValue might have been
    // set to undefined during a view -> model update that found validation
    // errors. We can't parse the view here, since that could change
    // the model although neither viewValue nor the model on the scope changed
    var modelValue = ctrl.$$rawModelValue;

    var prevValid = ctrl.$valid;
    var prevModelValue = ctrl.$modelValue;

    var allowInvalid = ctrl.$options && ctrl.$options.allowInvalid;

    ctrl.$$runValidators(modelValue, viewValue, function(allValid) {
      // If there was no change in validity, don't update the model
      // This prevents changing an invalid modelValue to undefined
      if (!allowInvalid && prevValid !== allValid) {
        // Note: Don't check ctrl.$valid here, as we could have
        // external validators (e.g. calculated on the server),
        // that just call $setValidity and need the model value
        // to calculate their validity.
        ctrl.$modelValue = allValid ? modelValue : undefined;

        if (ctrl.$modelValue !== prevModelValue) {
          ctrl.$$writeModelToScope();
        }
      }
    });

  };

  this.$$runValidators = function(modelValue, viewValue, doneCallback) {
    currentValidationRunId++;
    var localValidationRunId = currentValidationRunId;

    // check parser error
    if (!processParseErrors()) {
      validationDone(false);
      return;
    }
    if (!processSyncValidators()) {
      validationDone(false);
      return;
    }
    processAsyncValidators();

    function processParseErrors() {
      var errorKey = ctrl.$$parserName || 'parse';
      if (isUndefined(parserValid)) {
        setValidity(errorKey, null);
      } else {
        if (!parserValid) {
          forEach(ctrl.$validators, function(v, name) {
            setValidity(name, null);
          });
          forEach(ctrl.$asyncValidators, function(v, name) {
            setValidity(name, null);
          });
        }
        // Set the parse error last, to prevent unsetting it, should a $validators key == parserName
        setValidity(errorKey, parserValid);
        return parserValid;
      }
      return true;
    }

    function processSyncValidators() {
      var syncValidatorsValid = true;
      forEach(ctrl.$validators, function(validator, name) {
        var result = validator(modelValue, viewValue);
        syncValidatorsValid = syncValidatorsValid && result;
        setValidity(name, result);
      });
      if (!syncValidatorsValid) {
        forEach(ctrl.$asyncValidators, function(v, name) {
          setValidity(name, null);
        });
        return false;
      }
      return true;
    }

    function processAsyncValidators() {
      var validatorPromises = [];
      var allValid = true;
      forEach(ctrl.$asyncValidators, function(validator, name) {
        var promise = validator(modelValue, viewValue);
        if (!isPromiseLike(promise)) {
          throw ngModelMinErr("$asyncValidators",
            "Expected asynchronous validator to return a promise but got '{0}' instead.", promise);
        }
        setValidity(name, undefined);
        validatorPromises.push(promise.then(function() {
          setValidity(name, true);
        }, function(error) {
          allValid = false;
          setValidity(name, false);
        }));
      });
      if (!validatorPromises.length) {
        validationDone(true);
      } else {
        $q.all(validatorPromises).then(function() {
          validationDone(allValid);
        }, noop);
      }
    }

    function setValidity(name, isValid) {
      if (localValidationRunId === currentValidationRunId) {
        ctrl.$setValidity(name, isValid);
      }
    }

    function validationDone(allValid) {
      if (localValidationRunId === currentValidationRunId) {

        doneCallback(allValid);
      }
    }
  };

  /**
   * @ngdoc method
   * @name ngModel.NgModelController#$commitViewValue
   *
   * @description
   * Commit a pending update to the `$modelValue`.
   *
   * Updates may be pending by a debounced event or because the input is waiting for a some future
   * event defined in `ng-model-options`. this method is rarely needed as `NgModelController`
   * usually handles calling this in response to input events.
   */
  this.$commitViewValue = function() {
    var viewValue = ctrl.$viewValue;

    $timeout.cancel(pendingDebounce);

    // If the view value has not changed then we should just exit, except in the case where there is
    // a native validator on the element. In this case the validation state may have changed even though
    // the viewValue has stayed empty.
    if (ctrl.$$lastCommittedViewValue === viewValue && (viewValue !== '' || !ctrl.$$hasNativeValidators)) {
      return;
    }
    ctrl.$$lastCommittedViewValue = viewValue;

    // change to dirty
    if (ctrl.$pristine) {
      this.$setDirty();
    }
    this.$$parseAndValidate();
  };

  this.$$parseAndValidate = function() {
    var viewValue = ctrl.$$lastCommittedViewValue;
    var modelValue = viewValue;
    parserValid = isUndefined(modelValue) ? undefined : true;

    if (parserValid) {
      for (var i = 0; i < ctrl.$parsers.length; i++) {
        modelValue = ctrl.$parsers[i](modelValue);
        if (isUndefined(modelValue)) {
          parserValid = false;
          break;
        }
      }
    }
    if (isNumber(ctrl.$modelValue) && isNaN(ctrl.$modelValue)) {
      // ctrl.$modelValue has not been touched yet...
      ctrl.$modelValue = ngModelGet($scope);
    }
    var prevModelValue = ctrl.$modelValue;
    var allowInvalid = ctrl.$options && ctrl.$options.allowInvalid;
    ctrl.$$rawModelValue = modelValue;

    if (allowInvalid) {
      ctrl.$modelValue = modelValue;
      writeToModelIfNeeded();
    }

    // Pass the $$lastCommittedViewValue here, because the cached viewValue might be out of date.
    // This can happen if e.g. $setViewValue is called from inside a parser
    ctrl.$$runValidators(modelValue, ctrl.$$lastCommittedViewValue, function(allValid) {
      if (!allowInvalid) {
        // Note: Don't check ctrl.$valid here, as we could have
        // external validators (e.g. calculated on the server),
        // that just call $setValidity and need the model value
        // to calculate their validity.
        ctrl.$modelValue = allValid ? modelValue : undefined;
        writeToModelIfNeeded();
      }
    });

    function writeToModelIfNeeded() {
      if (ctrl.$modelValue !== prevModelValue) {
        ctrl.$$writeModelToScope();
      }
    }
  };

  this.$$writeModelToScope = function() {
    ngModelSet($scope, ctrl.$modelValue);
    forEach(ctrl.$viewChangeListeners, function(listener) {
      try {
        listener();
      } catch (e) {
        $exceptionHandler(e);
      }
    });
  };

  /**
   * @ngdoc method
   * @name ngModel.NgModelController#$setViewValue
   *
   * @description
   * Update the view value.
   *
   * This method should be called when a control wants to change the view value; typically,
   * this is done from within a DOM event handler. For example, the {@link ng.directive:input input}
   * directive calls it when the value of the input changes and {@link ng.directive:select select}
   * calls it when an option is selected.
   *
   * When `$setViewValue` is called, the new `value` will be staged for committing through the `$parsers`
   * and `$validators` pipelines. If there are no special {@link ngModelOptions} specified then the staged
   * value sent directly for processing, finally to be applied to `$modelValue` and then the
   * **expression** specified in the `ng-model` attribute. Lastly, all the registered change listeners,
   * in the `$viewChangeListeners` list, are called.
   *
   * In case the {@link ng.directive:ngModelOptions ngModelOptions} directive is used with `updateOn`
   * and the `default` trigger is not listed, all those actions will remain pending until one of the
   * `updateOn` events is triggered on the DOM element.
   * All these actions will be debounced if the {@link ng.directive:ngModelOptions ngModelOptions}
   * directive is used with a custom debounce for this particular event.
   * Note that a `$digest` is only triggered once the `updateOn` events are fired, or if `debounce`
   * is specified, once the timer runs out.
   *
   * When used with standard inputs, the view value will always be a string (which is in some cases
   * parsed into another type, such as a `Date` object for `input[date]`.)
   * However, custom controls might also pass objects to this method. In this case, we should make
   * a copy of the object before passing it to `$setViewValue`. This is because `ngModel` does not
   * perform a deep watch of objects, it only looks for a change of identity. If you only change
   * the property of the object then ngModel will not realise that the object has changed and
   * will not invoke the `$parsers` and `$validators` pipelines. For this reason, you should
   * not change properties of the copy once it has been passed to `$setViewValue`.
   * Otherwise you may cause the model value on the scope to change incorrectly.
   *
   * <div class="alert alert-info">
   * In any case, the value passed to the method should always reflect the current value
   * of the control. For example, if you are calling `$setViewValue` for an input element,
   * you should pass the input DOM value. Otherwise, the control and the scope model become
   * out of sync. It's also important to note that `$setViewValue` does not call `$render` or change
   * the control's DOM value in any way. If we want to change the control's DOM value
   * programmatically, we should update the `ngModel` scope expression. Its new value will be
   * picked up by the model controller, which will run it through the `$formatters`, `$render` it
   * to update the DOM, and finally call `$validate` on it.
   * </div>
   *
   * @param {*} value value from the view.
   * @param {string} trigger Event that triggered the update.
   */
  this.$setViewValue = function(value, trigger) {
    ctrl.$viewValue = value;
    if (!ctrl.$options || ctrl.$options.updateOnDefault) {
      ctrl.$$debounceViewValueCommit(trigger);
    }
  };

  this.$$debounceViewValueCommit = function(trigger) {
    var debounceDelay = 0,
        options = ctrl.$options,
        debounce;

    if (options && isDefined(options.debounce)) {
      debounce = options.debounce;
      if (isNumber(debounce)) {
        debounceDelay = debounce;
      } else if (isNumber(debounce[trigger])) {
        debounceDelay = debounce[trigger];
      } else if (isNumber(debounce['default'])) {
        debounceDelay = debounce['default'];
      }
    }

    $timeout.cancel(pendingDebounce);
    if (debounceDelay) {
      pendingDebounce = $timeout(function() {
        ctrl.$commitViewValue();
      }, debounceDelay);
    } else if ($rootScope.$$phase) {
      ctrl.$commitViewValue();
    } else {
      $scope.$apply(function() {
        ctrl.$commitViewValue();
      });
    }
  };

  // model -> value
  // Note: we cannot use a normal scope.$watch as we want to detect the following:
  // 1. scope value is 'a'
  // 2. user enters 'b'
  // 3. ng-change kicks in and reverts scope value to 'a'
  //    -> scope value did not change since the last digest as
  //       ng-change executes in apply phase
  // 4. view should be changed back to 'a'
  $scope.$watch(function ngModelWatch() {
    var modelValue = ngModelGet($scope);

    // if scope model value and ngModel value are out of sync
    // TODO(perf): why not move this to the action fn?
    if (modelValue !== ctrl.$modelValue &&
       // checks for NaN is needed to allow setting the model to NaN when there's an asyncValidator
       (ctrl.$modelValue === ctrl.$modelValue || modelValue === modelValue)
    ) {
      ctrl.$modelValue = ctrl.$$rawModelValue = modelValue;
      parserValid = undefined;

      var formatters = ctrl.$formatters,
          idx = formatters.length;

      var viewValue = modelValue;
      while (idx--) {
        viewValue = formatters[idx](viewValue);
      }
      if (ctrl.$viewValue !== viewValue) {
        ctrl.$viewValue = ctrl.$$lastCommittedViewValue = viewValue;
        ctrl.$render();

        ctrl.$$runValidators(modelValue, viewValue, noop);
      }
    }

    return modelValue;
  });
}];


/**
 * @ngdoc directive
 * @name ngModel
 *
 * @element input
 * @priority 1
 *
 * @description
 * The `ngModel` directive binds an `input`,`select`, `textarea` (or custom form control) to a
 * property on the scope using {@link ngModel.NgModelController NgModelController},
 * which is created and exposed by this directive.
 *
 * `ngModel` is responsible for:
 *
 * - Binding the view into the model, which other directives such as `input`, `textarea` or `select`
 *   require.
 * - Providing validation behavior (i.e. required, number, email, url).
 * - Keeping the state of the control (valid/invalid, dirty/pristine, touched/untouched, validation errors).
 * - Setting related css classes on the element (`ng-valid`, `ng-invalid`, `ng-dirty`, `ng-pristine`, `ng-touched`, `ng-untouched`) including animations.
 * - Registering the control with its parent {@link ng.directive:form form}.
 *
 * Note: `ngModel` will try to bind to the property given by evaluating the expression on the
 * current scope. If the property doesn't already exist on this scope, it will be created
 * implicitly and added to the scope.
 *
 * For best practices on using `ngModel`, see:
 *
 *  - [Understanding Scopes](https://github.com/angular/angular.js/wiki/Understanding-Scopes)
 *
 * For basic examples, how to use `ngModel`, see:
 *
 *  - {@link ng.directive:input input}
 *    - {@link input[text] text}
 *    - {@link input[checkbox] checkbox}
 *    - {@link input[radio] radio}
 *    - {@link input[number] number}
 *    - {@link input[email] email}
 *    - {@link input[url] url}
 *    - {@link input[date] date}
 *    - {@link input[datetime-local] datetime-local}
 *    - {@link input[time] time}
 *    - {@link input[month] month}
 *    - {@link input[week] week}
 *  - {@link ng.directive:select select}
 *  - {@link ng.directive:textarea textarea}
 *
 * # CSS classes
 * The following CSS classes are added and removed on the associated input/select/textarea element
 * depending on the validity of the model.
 *
 *  - `ng-valid`: the model is valid
 *  - `ng-invalid`: the model is invalid
 *  - `ng-valid-[key]`: for each valid key added by `$setValidity`
 *  - `ng-invalid-[key]`: for each invalid key added by `$setValidity`
 *  - `ng-pristine`: the control hasn't been interacted with yet
 *  - `ng-dirty`: the control has been interacted with
 *  - `ng-touched`: the control has been blurred
 *  - `ng-untouched`: the control hasn't been blurred
 *  - `ng-pending`: any `$asyncValidators` are unfulfilled
 *
 * Keep in mind that ngAnimate can detect each of these classes when added and removed.
 *
 * ## Animation Hooks
 *
 * Animations within models are triggered when any of the associated CSS classes are added and removed
 * on the input element which is attached to the model. These classes are: `.ng-pristine`, `.ng-dirty`,
 * `.ng-invalid` and `.ng-valid` as well as any other validations that are performed on the model itself.
 * The animations that are triggered within ngModel are similar to how they work in ngClass and
 * animations can be hooked into using CSS transitions, keyframes as well as JS animations.
 *
 * The following example shows a simple way to utilize CSS transitions to style an input element
 * that has been rendered as invalid after it has been validated:
 *
 * <pre>
 * //be sure to include ngAnimate as a module to hook into more
 * //advanced animations
 * .my-input {
 *   transition:0.5s linear all;
 *   background: white;
 * }
 * .my-input.ng-invalid {
 *   background: red;
 *   color:white;
 * }
 * </pre>
 *
 * @example
 * <example deps="angular-animate.js" animations="true" fixBase="true" module="inputExample">
     <file name="index.html">
       <script>
        angular.module('inputExample', [])
          .controller('ExampleController', ['$scope', function($scope) {
            $scope.val = '1';
          }]);
       </script>
       <style>
         .my-input {
           transition:all linear 0.5s;
           background: transparent;
         }
         .my-input.ng-invalid {
           color:white;
           background: red;
         }
       </style>
       <p id="inputDescription">
        Update input to see transitions when valid/invalid.
        Integer is a valid value.
       </p>
       <form name="testForm" ng-controller="ExampleController">
         <input ng-model="val" ng-pattern="/^\d+$/" name="anim" class="my-input"
                aria-describedby="inputDescription" />
       </form>
     </file>
 * </example>
 *
 * ## Binding to a getter/setter
 *
 * Sometimes it's helpful to bind `ngModel` to a getter/setter function.  A getter/setter is a
 * function that returns a representation of the model when called with zero arguments, and sets
 * the internal state of a model when called with an argument. It's sometimes useful to use this
 * for models that have an internal representation that's different from what the model exposes
 * to the view.
 *
 * <div class="alert alert-success">
 * **Best Practice:** It's best to keep getters fast because Angular is likely to call them more
 * frequently than other parts of your code.
 * </div>
 *
 * You use this behavior by adding `ng-model-options="{ getterSetter: true }"` to an element that
 * has `ng-model` attached to it. You can also add `ng-model-options="{ getterSetter: true }"` to
 * a `<form>`, which will enable this behavior for all `<input>`s within it. See
 * {@link ng.directive:ngModelOptions `ngModelOptions`} for more.
 *
 * The following example shows how to use `ngModel` with a getter/setter:
 *
 * @example
 * <example name="ngModel-getter-setter" module="getterSetterExample">
     <file name="index.html">
       <div ng-controller="ExampleController">
         <form name="userForm">
           <label>Name:
             <input type="text" name="userName"
                    ng-model="user.name"
                    ng-model-options="{ getterSetter: true }" />
           </label>
         </form>
         <pre>user.name = <span ng-bind="user.name()"></span></pre>
       </div>
     </file>
     <file name="app.js">
       angular.module('getterSetterExample', [])
         .controller('ExampleController', ['$scope', function($scope) {
           var _name = 'Brian';
           $scope.user = {
             name: function(newName) {
              // Note that newName can be undefined for two reasons:
              // 1. Because it is called as a getter and thus called with no arguments
              // 2. Because the property should actually be set to undefined. This happens e.g. if the
              //    input is invalid
              return arguments.length ? (_name = newName) : _name;
             }
           };
         }]);
     </file>
 * </example>
 */
var ngModelDirective = ['$rootScope', function($rootScope) {
  return {
    restrict: 'A',
    require: ['ngModel', '^?form', '^?ngModelOptions'],
    controller: NgModelController,
    // Prelink needs to run before any input directive
    // so that we can set the NgModelOptions in NgModelController
    // before anyone else uses it.
    priority: 1,
    compile: function ngModelCompile(element) {
      // Setup initial state of the control
      element.addClass(PRISTINE_CLASS).addClass(UNTOUCHED_CLASS).addClass(VALID_CLASS);

      return {
        pre: function ngModelPreLink(scope, element, attr, ctrls) {
          var modelCtrl = ctrls[0],
              formCtrl = ctrls[1] || modelCtrl.$$parentForm;

          modelCtrl.$$setOptions(ctrls[2] && ctrls[2].$options);

          // notify others, especially parent forms
          formCtrl.$addControl(modelCtrl);

          attr.$observe('name', function(newValue) {
            if (modelCtrl.$name !== newValue) {
              modelCtrl.$$parentForm.$$renameControl(modelCtrl, newValue);
            }
          });

          scope.$on('$destroy', function() {
            modelCtrl.$$parentForm.$removeControl(modelCtrl);
          });
        },
        post: function ngModelPostLink(scope, element, attr, ctrls) {
          var modelCtrl = ctrls[0];
          if (modelCtrl.$options && modelCtrl.$options.updateOn) {
            element.on(modelCtrl.$options.updateOn, function(ev) {
              modelCtrl.$$debounceViewValueCommit(ev && ev.type);
            });
          }

          element.on('blur', function(ev) {
            if (modelCtrl.$touched) return;

            if ($rootScope.$$phase) {
              scope.$evalAsync(modelCtrl.$setTouched);
            } else {
              scope.$apply(modelCtrl.$setTouched);
            }
          });
        }
      };
    }
  };
}];

var DEFAULT_REGEXP = /(\s+|^)default(\s+|$)/;

/**
 * @ngdoc directive
 * @name ngModelOptions
 *
 * @description
 * Allows tuning how model updates are done. Using `ngModelOptions` you can specify a custom list of
 * events that will trigger a model update and/or a debouncing delay so that the actual update only
 * takes place when a timer expires; this timer will be reset after another change takes place.
 *
 * Given the nature of `ngModelOptions`, the value displayed inside input fields in the view might
 * be different from the value in the actual model. This means that if you update the model you
 * should also invoke {@link ngModel.NgModelController `$rollbackViewValue`} on the relevant input field in
 * order to make sure it is synchronized with the model and that any debounced action is canceled.
 *
 * The easiest way to reference the control's {@link ngModel.NgModelController `$rollbackViewValue`}
 * method is by making sure the input is placed inside a form that has a `name` attribute. This is
 * important because `form` controllers are published to the related scope under the name in their
 * `name` attribute.
 *
 * Any pending changes will take place immediately when an enclosing form is submitted via the
 * `submit` event. Note that `ngClick` events will occur before the model is updated. Use `ngSubmit`
 * to have access to the updated model.
 *
 * `ngModelOptions` has an effect on the element it's declared on and its descendants.
 *
 * @param {Object} ngModelOptions options to apply to the current model. Valid keys are:
 *   - `updateOn`: string specifying which event should the input be bound to. You can set several
 *     events using an space delimited list. There is a special event called `default` that
 *     matches the default events belonging of the control.
 *   - `debounce`: integer value which contains the debounce model update value in milliseconds. A
 *     value of 0 triggers an immediate update. If an object is supplied instead, you can specify a
 *     custom value for each event. For example:
 *     `ng-model-options="{ updateOn: 'default blur', debounce: { 'default': 500, 'blur': 0 } }"`
 *   - `allowInvalid`: boolean value which indicates that the model can be set with values that did
 *     not validate correctly instead of the default behavior of setting the model to undefined.
 *   - `getterSetter`: boolean value which determines whether or not to treat functions bound to
       `ngModel` as getters/setters.
 *   - `timezone`: Defines the timezone to be used to read/write the `Date` instance in the model for
 *     `<input type="date">`, `<input type="time">`, ... . It understands UTC/GMT and the
 *     continental US time zone abbreviations, but for general use, use a time zone offset, for
 *     example, `'+0430'` (4 hours, 30 minutes east of the Greenwich meridian)
 *     If not specified, the timezone of the browser will be used.
 *
 * @example

  The following example shows how to override immediate updates. Changes on the inputs within the
  form will update the model only when the control loses focus (blur event). If `escape` key is
  pressed while the input field is focused, the value is reset to the value in the current model.

  <example name="ngModelOptions-directive-blur" module="optionsExample">
    <file name="index.html">
      <div ng-controller="ExampleController">
        <form name="userForm">
          <label>Name:
            <input type="text" name="userName"
                   ng-model="user.name"
                   ng-model-options="{ updateOn: 'blur' }"
                   ng-keyup="cancel($event)" />
          </label><br />
          <label>Other data:
            <input type="text" ng-model="user.data" />
          </label><br />
        </form>
        <pre>user.name = <span ng-bind="user.name"></span></pre>
        <pre>user.data = <span ng-bind="user.data"></span></pre>
      </div>
    </file>
    <file name="app.js">
      angular.module('optionsExample', [])
        .controller('ExampleController', ['$scope', function($scope) {
          $scope.user = { name: 'John', data: '' };

          $scope.cancel = function(e) {
            if (e.keyCode == 27) {
              $scope.userForm.userName.$rollbackViewValue();
            }
          };
        }]);
    </file>
    <file name="protractor.js" type="protractor">
      var model = element(by.binding('user.name'));
      var input = element(by.model('user.name'));
      var other = element(by.model('user.data'));

      it('should allow custom events', function() {
        input.sendKeys(' Doe');
        input.click();
        expect(model.getText()).toEqual('John');
        other.click();
        expect(model.getText()).toEqual('John Doe');
      });

      it('should $rollbackViewValue when model changes', function() {
        input.sendKeys(' Doe');
        expect(input.getAttribute('value')).toEqual('John Doe');
        input.sendKeys(protractor.Key.ESCAPE);
        expect(input.getAttribute('value')).toEqual('John');
        other.click();
        expect(model.getText()).toEqual('John');
      });
    </file>
  </example>

  This one shows how to debounce model changes. Model will be updated only 1 sec after last change.
  If the `Clear` button is pressed, any debounced action is canceled and the value becomes empty.

  <example name="ngModelOptions-directive-debounce" module="optionsExample">
    <file name="index.html">
      <div ng-controller="ExampleController">
        <form name="userForm">
          <label>Name:
            <input type="text" name="userName"
                   ng-model="user.name"
                   ng-model-options="{ debounce: 1000 }" />
          </label>
          <button ng-click="userForm.userName.$rollbackViewValue(); user.name=''">Clear</button>
          <br />
        </form>
        <pre>user.name = <span ng-bind="user.name"></span></pre>
      </div>
    </file>
    <file name="app.js">
      angular.module('optionsExample', [])
        .controller('ExampleController', ['$scope', function($scope) {
          $scope.user = { name: 'Igor' };
        }]);
    </file>
  </example>

  This one shows how to bind to getter/setters:

  <example name="ngModelOptions-directive-getter-setter" module="getterSetterExample">
    <file name="index.html">
      <div ng-controller="ExampleController">
        <form name="userForm">
          <label>Name:
            <input type="text" name="userName"
                   ng-model="user.name"
                   ng-model-options="{ getterSetter: true }" />
          </label>
        </form>
        <pre>user.name = <span ng-bind="user.name()"></span></pre>
      </div>
    </file>
    <file name="app.js">
      angular.module('getterSetterExample', [])
        .controller('ExampleController', ['$scope', function($scope) {
          var _name = 'Brian';
          $scope.user = {
            name: function(newName) {
              // Note that newName can be undefined for two reasons:
              // 1. Because it is called as a getter and thus called with no arguments
              // 2. Because the property should actually be set to undefined. This happens e.g. if the
              //    input is invalid
              return arguments.length ? (_name = newName) : _name;
            }
          };
        }]);
    </file>
  </example>
 */
var ngModelOptionsDirective = function() {
  return {
    restrict: 'A',
    controller: ['$scope', '$attrs', function($scope, $attrs) {
      var that = this;
      this.$options = copy($scope.$eval($attrs.ngModelOptions));
      // Allow adding/overriding bound events
      if (isDefined(this.$options.updateOn)) {
        this.$options.updateOnDefault = false;
        // extract "default" pseudo-event from list of events that can trigger a model update
        this.$options.updateOn = trim(this.$options.updateOn.replace(DEFAULT_REGEXP, function() {
          that.$options.updateOnDefault = true;
          return ' ';
        }));
      } else {
        this.$options.updateOnDefault = true;
      }
    }]
  };
};



// helper methods
function addSetValidityMethod(context) {
  var ctrl = context.ctrl,
      $element = context.$element,
      classCache = {},
      set = context.set,
      unset = context.unset,
      $animate = context.$animate;

  classCache[INVALID_CLASS] = !(classCache[VALID_CLASS] = $element.hasClass(VALID_CLASS));

  ctrl.$setValidity = setValidity;

  function setValidity(validationErrorKey, state, controller) {
    if (isUndefined(state)) {
      createAndSet('$pending', validationErrorKey, controller);
    } else {
      unsetAndCleanup('$pending', validationErrorKey, controller);
    }
    if (!isBoolean(state)) {
      unset(ctrl.$error, validationErrorKey, controller);
      unset(ctrl.$$success, validationErrorKey, controller);
    } else {
      if (state) {
        unset(ctrl.$error, validationErrorKey, controller);
        set(ctrl.$$success, validationErrorKey, controller);
      } else {
        set(ctrl.$error, validationErrorKey, controller);
        unset(ctrl.$$success, validationErrorKey, controller);
      }
    }
    if (ctrl.$pending) {
      cachedToggleClass(PENDING_CLASS, true);
      ctrl.$valid = ctrl.$invalid = undefined;
      toggleValidationCss('', null);
    } else {
      cachedToggleClass(PENDING_CLASS, false);
      ctrl.$valid = isObjectEmpty(ctrl.$error);
      ctrl.$invalid = !ctrl.$valid;
      toggleValidationCss('', ctrl.$valid);
    }

    // re-read the state as the set/unset methods could have
    // combined state in ctrl.$error[validationError] (used for forms),
    // where setting/unsetting only increments/decrements the value,
    // and does not replace it.
    var combinedState;
    if (ctrl.$pending && ctrl.$pending[validationErrorKey]) {
      combinedState = undefined;
    } else if (ctrl.$error[validationErrorKey]) {
      combinedState = false;
    } else if (ctrl.$$success[validationErrorKey]) {
      combinedState = true;
    } else {
      combinedState = null;
    }

    toggleValidationCss(validationErrorKey, combinedState);
    ctrl.$$parentForm.$setValidity(validationErrorKey, combinedState, ctrl);
  }

  function createAndSet(name, value, controller) {
    if (!ctrl[name]) {
      ctrl[name] = {};
    }
    set(ctrl[name], value, controller);
  }

  function unsetAndCleanup(name, value, controller) {
    if (ctrl[name]) {
      unset(ctrl[name], value, controller);
    }
    if (isObjectEmpty(ctrl[name])) {
      ctrl[name] = undefined;
    }
  }

  function cachedToggleClass(className, switchValue) {
    if (switchValue && !classCache[className]) {
      $animate.addClass($element, className);
      classCache[className] = true;
    } else if (!switchValue && classCache[className]) {
      $animate.removeClass($element, className);
      classCache[className] = false;
    }
  }

  function toggleValidationCss(validationErrorKey, isValid) {
    validationErrorKey = validationErrorKey ? '-' + snake_case(validationErrorKey, '-') : '';

    cachedToggleClass(VALID_CLASS + validationErrorKey, isValid === true);
    cachedToggleClass(INVALID_CLASS + validationErrorKey, isValid === false);
  }
}

function isObjectEmpty(obj) {
  if (obj) {
    for (var prop in obj) {
      if (obj.hasOwnProperty(prop)) {
        return false;
      }
    }
  }
  return true;
}

/**
 * @ngdoc directive
 * @name ngNonBindable
 * @restrict AC
 * @priority 1000
 *
 * @description
 * The `ngNonBindable` directive tells Angular not to compile or bind the contents of the current
 * DOM element. This is useful if the element contains what appears to be Angular directives and
 * bindings but which should be ignored by Angular. This could be the case if you have a site that
 * displays snippets of code, for instance.
 *
 * @element ANY
 *
 * @example
 * In this example there are two locations where a simple interpolation binding (`{{}}`) is present,
 * but the one wrapped in `ngNonBindable` is left alone.
 *
 * @example
    <example>
      <file name="index.html">
        <div>Normal: {{1 + 2}}</div>
        <div ng-non-bindable>Ignored: {{1 + 2}}</div>
      </file>
      <file name="protractor.js" type="protractor">
       it('should check ng-non-bindable', function() {
         expect(element(by.binding('1 + 2')).getText()).toContain('3');
         expect(element.all(by.css('div')).last().getText()).toMatch(/1 \+ 2/);
       });
      </file>
    </example>
 */
var ngNonBindableDirective = ngDirective({ terminal: true, priority: 1000 });

/* global jqLiteRemove */

var ngOptionsMinErr = minErr('ngOptions');

/**
 * @ngdoc directive
 * @name ngOptions
 * @restrict A
 *
 * @description
 *
 * The `ngOptions` attribute can be used to dynamically generate a list of `<option>`
 * elements for the `<select>` element using the array or object obtained by evaluating the
 * `ngOptions` comprehension expression.
 *
 * In many cases, `ngRepeat` can be used on `<option>` elements instead of `ngOptions` to achieve a
 * similar result. However, `ngOptions` provides some benefits such as reducing memory and
 * increasing speed by not creating a new scope for each repeated instance, as well as providing
 * more flexibility in how the `<select>`'s model is assigned via the `select` **`as`** part of the
 * comprehension expression. `ngOptions` should be used when the `<select>` model needs to be bound
 *  to a non-string value. This is because an option element can only be bound to string values at
 * present.
 *
 * When an item in the `<select>` menu is selected, the array element or object property
 * represented by the selected option will be bound to the model identified by the `ngModel`
 * directive.
 *
 * Optionally, a single hard-coded `<option>` element, with the value set to an empty string, can
 * be nested into the `<select>` element. This element will then represent the `null` or "not selected"
 * option. See example below for demonstration.
 *
 * ## Complex Models (objects or collections)
 *
 * By default, `ngModel` watches the model by reference, not value. This is important to know when
 * binding the select to a model that is an object or a collection.
 *
 * One issue occurs if you want to preselect an option. For example, if you set
 * the model to an object that is equal to an object in your collection, `ngOptions` won't be able to set the selection,
 * because the objects are not identical. So by default, you should always reference the item in your collection
 * for preselections, e.g.: `$scope.selected = $scope.collection[3]`.
 *
 * Another solution is to use a `track by` clause, because then `ngOptions` will track the identity
 * of the item not by reference, but by the result of the `track by` expression. For example, if your
 * collection items have an id property, you would `track by item.id`.
 *
 * A different issue with objects or collections is that ngModel won't detect if an object property or
 * a collection item changes. For that reason, `ngOptions` additionally watches the model using
 * `$watchCollection`, when the expression contains a `track by` clause or the the select has the `multiple` attribute.
 * This allows ngOptions to trigger a re-rendering of the options even if the actual object/collection
 * has not changed identity, but only a property on the object or an item in the collection changes.
 *
 * Note that `$watchCollection` does a shallow comparison of the properties of the object (or the items in the collection
 * if the model is an array). This means that changing a property deeper than the first level inside the
 * object/collection will not trigger a re-rendering.
 *
 * ## `select` **`as`**
 *
 * Using `select` **`as`** will bind the result of the `select` expression to the model, but
 * the value of the `<select>` and `<option>` html elements will be either the index (for array data sources)
 * or property name (for object data sources) of the value within the collection. If a **`track by`** expression
 * is used, the result of that expression will be set as the value of the `option` and `select` elements.
 *
 *
 * ### `select` **`as`** and **`track by`**
 *
 * <div class="alert alert-warning">
 * Be careful when using `select` **`as`** and **`track by`** in the same expression.
 * </div>
 *
 * Given this array of items on the $scope:
 *
 * ```js
 * $scope.items = [{
 *   id: 1,
 *   label: 'aLabel',
 *   subItem: { name: 'aSubItem' }
 * }, {
 *   id: 2,
 *   label: 'bLabel',
 *   subItem: { name: 'bSubItem' }
 * }];
 * ```
 *
 * This will work:
 *
 * ```html
 * <select ng-options="item as item.label for item in items track by item.id" ng-model="selected"></select>
 * ```
 * ```js
 * $scope.selected = $scope.items[0];
 * ```
 *
 * but this will not work:
 *
 * ```html
 * <select ng-options="item.subItem as item.label for item in items track by item.id" ng-model="selected"></select>
 * ```
 * ```js
 * $scope.selected = $scope.items[0].subItem;
 * ```
 *
 * In both examples, the **`track by`** expression is applied successfully to each `item` in the
 * `items` array. Because the selected option has been set programmatically in the controller, the
 * **`track by`** expression is also applied to the `ngModel` value. In the first example, the
 * `ngModel` value is `items[0]` and the **`track by`** expression evaluates to `items[0].id` with
 * no issue. In the second example, the `ngModel` value is `items[0].subItem` and the **`track by`**
 * expression evaluates to `items[0].subItem.id` (which is undefined). As a result, the model value
 * is not matched against any `<option>` and the `<select>` appears as having no selected value.
 *
 *
 * @param {string} ngModel Assignable angular expression to data-bind to.
 * @param {string=} name Property name of the form under which the control is published.
 * @param {string=} required The control is considered valid only if value is entered.
 * @param {string=} ngRequired Adds `required` attribute and `required` validation constraint to
 *    the element when the ngRequired expression evaluates to true. Use `ngRequired` instead of
 *    `required` when you want to data-bind to the `required` attribute.
 * @param {comprehension_expression=} ngOptions in one of the following forms:
 *
 *   * for array data sources:
 *     * `label` **`for`** `value` **`in`** `array`
 *     * `select` **`as`** `label` **`for`** `value` **`in`** `array`
 *     * `label` **`group by`** `group` **`for`** `value` **`in`** `array`
 *     * `label` **`disable when`** `disable` **`for`** `value` **`in`** `array`
 *     * `label` **`group by`** `group` **`for`** `value` **`in`** `array` **`track by`** `trackexpr`
 *     * `label` **`disable when`** `disable` **`for`** `value` **`in`** `array` **`track by`** `trackexpr`
 *     * `label` **`for`** `value` **`in`** `array` | orderBy:`orderexpr` **`track by`** `trackexpr`
 *        (for including a filter with `track by`)
 *   * for object data sources:
 *     * `label` **`for (`**`key` **`,`** `value`**`) in`** `object`
 *     * `select` **`as`** `label` **`for (`**`key` **`,`** `value`**`) in`** `object`
 *     * `label` **`group by`** `group` **`for (`**`key`**`,`** `value`**`) in`** `object`
 *     * `label` **`disable when`** `disable` **`for (`**`key`**`,`** `value`**`) in`** `object`
 *     * `select` **`as`** `label` **`group by`** `group`
 *         **`for` `(`**`key`**`,`** `value`**`) in`** `object`
 *     * `select` **`as`** `label` **`disable when`** `disable`
 *         **`for` `(`**`key`**`,`** `value`**`) in`** `object`
 *
 * Where:
 *
 *   * `array` / `object`: an expression which evaluates to an array / object to iterate over.
 *   * `value`: local variable which will refer to each item in the `array` or each property value
 *      of `object` during iteration.
 *   * `key`: local variable which will refer to a property name in `object` during iteration.
 *   * `label`: The result of this expression will be the label for `<option>` element. The
 *     `expression` will most likely refer to the `value` variable (e.g. `value.propertyName`).
 *   * `select`: The result of this expression will be bound to the model of the parent `<select>`
 *      element. If not specified, `select` expression will default to `value`.
 *   * `group`: The result of this expression will be used to group options using the `<optgroup>`
 *      DOM element.
 *   * `disable`: The result of this expression will be used to disable the rendered `<option>`
 *      element. Return `true` to disable.
 *   * `trackexpr`: Used when working with an array of objects. The result of this expression will be
 *      used to identify the objects in the array. The `trackexpr` will most likely refer to the
 *     `value` variable (e.g. `value.propertyName`). With this the selection is preserved
 *      even when the options are recreated (e.g. reloaded from the server).
 *
 * @example
    <example module="selectExample">
      <file name="index.html">
        <script>
        angular.module('selectExample', [])
          .controller('ExampleController', ['$scope', function($scope) {
            $scope.colors = [
              {name:'black', shade:'dark'},
              {name:'white', shade:'light', notAnOption: true},
              {name:'red', shade:'dark'},
              {name:'blue', shade:'dark', notAnOption: true},
              {name:'yellow', shade:'light', notAnOption: false}
            ];
            $scope.myColor = $scope.colors[2]; // red
          }]);
        </script>
        <div ng-controller="ExampleController">
          <ul>
            <li ng-repeat="color in colors">
              <label>Name: <input ng-model="color.name"></label>
              <label><input type="checkbox" ng-model="color.notAnOption"> Disabled?</label>
              <button ng-click="colors.splice($index, 1)" aria-label="Remove">X</button>
            </li>
            <li>
              <button ng-click="colors.push({})">add</button>
            </li>
          </ul>
          <hr/>
          <label>Color (null not allowed):
            <select ng-model="myColor" ng-options="color.name for color in colors"></select>
          </label><br/>
          <label>Color (null allowed):
          <span  class="nullable">
            <select ng-model="myColor" ng-options="color.name for color in colors">
              <option value="">-- choose color --</option>
            </select>
          </span></label><br/>

          <label>Color grouped by shade:
            <select ng-model="myColor" ng-options="color.name group by color.shade for color in colors">
            </select>
          </label><br/>

          <label>Color grouped by shade, with some disabled:
            <select ng-model="myColor"
                  ng-options="color.name group by color.shade disable when color.notAnOption for color in colors">
            </select>
          </label><br/>



          Select <button ng-click="myColor = { name:'not in list', shade: 'other' }">bogus</button>.
          <br/>
          <hr/>
          Currently selected: {{ {selected_color:myColor} }}
          <div style="border:solid 1px black; height:20px"
               ng-style="{'background-color':myColor.name}">
          </div>
        </div>
      </file>
      <file name="protractor.js" type="protractor">
         it('should check ng-options', function() {
           expect(element(by.binding('{selected_color:myColor}')).getText()).toMatch('red');
           element.all(by.model('myColor')).first().click();
           element.all(by.css('select[ng-model="myColor"] option')).first().click();
           expect(element(by.binding('{selected_color:myColor}')).getText()).toMatch('black');
           element(by.css('.nullable select[ng-model="myColor"]')).click();
           element.all(by.css('.nullable select[ng-model="myColor"] option')).first().click();
           expect(element(by.binding('{selected_color:myColor}')).getText()).toMatch('null');
         });
      </file>
    </example>
 */

// jshint maxlen: false
//                     //00001111111111000000000002222222222000000000000000000000333333333300000000000000000000000004444444444400000000000005555555555555550000000006666666666666660000000777777777777777000000000000000888888888800000000000000000009999999999
var NG_OPTIONS_REGEXP = /^\s*([\s\S]+?)(?:\s+as\s+([\s\S]+?))?(?:\s+group\s+by\s+([\s\S]+?))?(?:\s+disable\s+when\s+([\s\S]+?))?\s+for\s+(?:([\$\w][\$\w]*)|(?:\(\s*([\$\w][\$\w]*)\s*,\s*([\$\w][\$\w]*)\s*\)))\s+in\s+([\s\S]+?)(?:\s+track\s+by\s+([\s\S]+?))?$/;
                        // 1: value expression (valueFn)
                        // 2: label expression (displayFn)
                        // 3: group by expression (groupByFn)
                        // 4: disable when expression (disableWhenFn)
                        // 5: array item variable name
                        // 6: object item key variable name
                        // 7: object item value variable name
                        // 8: collection expression
                        // 9: track by expression
// jshint maxlen: 100


var ngOptionsDirective = ['$compile', '$parse', function($compile, $parse) {

  function parseOptionsExpression(optionsExp, selectElement, scope) {

    var match = optionsExp.match(NG_OPTIONS_REGEXP);
    if (!(match)) {
      throw ngOptionsMinErr('iexp',
        "Expected expression in form of " +
        "'_select_ (as _label_)? for (_key_,)?_value_ in _collection_'" +
        " but got '{0}'. Element: {1}",
        optionsExp, startingTag(selectElement));
    }

    // Extract the parts from the ngOptions expression

    // The variable name for the value of the item in the collection
    var valueName = match[5] || match[7];
    // The variable name for the key of the item in the collection
    var keyName = match[6];

    // An expression that generates the viewValue for an option if there is a label expression
    var selectAs = / as /.test(match[0]) && match[1];
    // An expression that is used to track the id of each object in the options collection
    var trackBy = match[9];
    // An expression that generates the viewValue for an option if there is no label expression
    var valueFn = $parse(match[2] ? match[1] : valueName);
    var selectAsFn = selectAs && $parse(selectAs);
    var viewValueFn = selectAsFn || valueFn;
    var trackByFn = trackBy && $parse(trackBy);

    // Get the value by which we are going to track the option
    // if we have a trackFn then use that (passing scope and locals)
    // otherwise just hash the given viewValue
    var getTrackByValueFn = trackBy ?
                              function(value, locals) { return trackByFn(scope, locals); } :
                              function getHashOfValue(value) { return hashKey(value); };
    var getTrackByValue = function(value, key) {
      return getTrackByValueFn(value, getLocals(value, key));
    };

    var displayFn = $parse(match[2] || match[1]);
    var groupByFn = $parse(match[3] || '');
    var disableWhenFn = $parse(match[4] || '');
    var valuesFn = $parse(match[8]);

    var locals = {};
    var getLocals = keyName ? function(value, key) {
      locals[keyName] = key;
      locals[valueName] = value;
      return locals;
    } : function(value) {
      locals[valueName] = value;
      return locals;
    };


    function Option(selectValue, viewValue, label, group, disabled) {
      this.selectValue = selectValue;
      this.viewValue = viewValue;
      this.label = label;
      this.group = group;
      this.disabled = disabled;
    }

    function getOptionValuesKeys(optionValues) {
      var optionValuesKeys;

      if (!keyName && isArrayLike(optionValues)) {
        optionValuesKeys = optionValues;
      } else {
        // if object, extract keys, in enumeration order, unsorted
        optionValuesKeys = [];
        for (var itemKey in optionValues) {
          if (optionValues.hasOwnProperty(itemKey) && itemKey.charAt(0) !== '$') {
            optionValuesKeys.push(itemKey);
          }
        }
      }
      return optionValuesKeys;
    }

    return {
      trackBy: trackBy,
      getTrackByValue: getTrackByValue,
      getWatchables: $parse(valuesFn, function(optionValues) {
        // Create a collection of things that we would like to watch (watchedArray)
        // so that they can all be watched using a single $watchCollection
        // that only runs the handler once if anything changes
        var watchedArray = [];
        optionValues = optionValues || [];

        var optionValuesKeys = getOptionValuesKeys(optionValues);
        var optionValuesLength = optionValuesKeys.length;
        for (var index = 0; index < optionValuesLength; index++) {
          var key = (optionValues === optionValuesKeys) ? index : optionValuesKeys[index];
          var value = optionValues[key];

          var locals = getLocals(optionValues[key], key);
          var selectValue = getTrackByValueFn(optionValues[key], locals);
          watchedArray.push(selectValue);

          // Only need to watch the displayFn if there is a specific label expression
          if (match[2] || match[1]) {
            var label = displayFn(scope, locals);
            watchedArray.push(label);
          }

          // Only need to watch the disableWhenFn if there is a specific disable expression
          if (match[4]) {
            var disableWhen = disableWhenFn(scope, locals);
            watchedArray.push(disableWhen);
          }
        }
        return watchedArray;
      }),

      getOptions: function() {

        var optionItems = [];
        var selectValueMap = {};

        // The option values were already computed in the `getWatchables` fn,
        // which must have been called to trigger `getOptions`
        var optionValues = valuesFn(scope) || [];
        var optionValuesKeys = getOptionValuesKeys(optionValues);
        var optionValuesLength = optionValuesKeys.length;

        for (var index = 0; index < optionValuesLength; index++) {
          var key = (optionValues === optionValuesKeys) ? index : optionValuesKeys[index];
          var value = optionValues[key];
          var locals = getLocals(value, key);
          var viewValue = viewValueFn(scope, locals);
          var selectValue = getTrackByValueFn(viewValue, locals);
          var label = displayFn(scope, locals);
          var group = groupByFn(scope, locals);
          var disabled = disableWhenFn(scope, locals);
          var optionItem = new Option(selectValue, viewValue, label, group, disabled);

          optionItems.push(optionItem);
          selectValueMap[selectValue] = optionItem;
        }

        return {
          items: optionItems,
          selectValueMap: selectValueMap,
          getOptionFromViewValue: function(value) {
            return selectValueMap[getTrackByValue(value)];
          },
          getViewValueFromOption: function(option) {
            // If the viewValue could be an object that may be mutated by the application,
            // we need to make a copy and not return the reference to the value on the option.
            return trackBy ? angular.copy(option.viewValue) : option.viewValue;
          }
        };
      }
    };
  }


  // we can't just jqLite('<option>') since jqLite is not smart enough
  // to create it in <select> and IE barfs otherwise.
  var optionTemplate = document.createElement('option'),
      optGroupTemplate = document.createElement('optgroup');


    function ngOptionsPostLink(scope, selectElement, attr, ctrls) {

      // if ngModel is not defined, we don't need to do anything
      var ngModelCtrl = ctrls[1];
      if (!ngModelCtrl) return;

      var selectCtrl = ctrls[0];
      var multiple = attr.multiple;

      // The emptyOption allows the application developer to provide their own custom "empty"
      // option when the viewValue does not match any of the option values.
      var emptyOption;
      for (var i = 0, children = selectElement.children(), ii = children.length; i < ii; i++) {
        if (children[i].value === '') {
          emptyOption = children.eq(i);
          break;
        }
      }

      var providedEmptyOption = !!emptyOption;

      var unknownOption = jqLite(optionTemplate.cloneNode(false));
      unknownOption.val('?');

      var options;
      var ngOptions = parseOptionsExpression(attr.ngOptions, selectElement, scope);


      var renderEmptyOption = function() {
        if (!providedEmptyOption) {
          selectElement.prepend(emptyOption);
        }
        selectElement.val('');
        emptyOption.prop('selected', true); // needed for IE
        emptyOption.attr('selected', true);
      };

      var removeEmptyOption = function() {
        if (!providedEmptyOption) {
          emptyOption.remove();
        }
      };


      var renderUnknownOption = function() {
        selectElement.prepend(unknownOption);
        selectElement.val('?');
        unknownOption.prop('selected', true); // needed for IE
        unknownOption.attr('selected', true);
      };

      var removeUnknownOption = function() {
        unknownOption.remove();
      };

      // Update the controller methods for multiple selectable options
      if (!multiple) {

        selectCtrl.writeValue = function writeNgOptionsValue(value) {
          var option = options.getOptionFromViewValue(value);

          if (option && !option.disabled) {
            if (selectElement[0].value !== option.selectValue) {
              removeUnknownOption();
              removeEmptyOption();

              selectElement[0].value = option.selectValue;
              option.element.selected = true;
              option.element.setAttribute('selected', 'selected');
            }
          } else {
            if (value === null || providedEmptyOption) {
              removeUnknownOption();
              renderEmptyOption();
            } else {
              removeEmptyOption();
              renderUnknownOption();
            }
          }
        };

        selectCtrl.readValue = function readNgOptionsValue() {

          var selectedOption = options.selectValueMap[selectElement.val()];

          if (selectedOption && !selectedOption.disabled) {
            removeEmptyOption();
            removeUnknownOption();
            return options.getViewValueFromOption(selectedOption);
          }
          return null;
        };

        // If we are using `track by` then we must watch the tracked value on the model
        // since ngModel only watches for object identity change
        if (ngOptions.trackBy) {
          scope.$watch(
            function() { return ngOptions.getTrackByValue(ngModelCtrl.$viewValue); },
            function() { ngModelCtrl.$render(); }
          );
        }

      } else {

        ngModelCtrl.$isEmpty = function(value) {
          return !value || value.length === 0;
        };


        selectCtrl.writeValue = function writeNgOptionsMultiple(value) {
          options.items.forEach(function(option) {
            option.element.selected = false;
          });

          if (value) {
            value.forEach(function(item) {
              var option = options.getOptionFromViewValue(item);
              if (option && !option.disabled) option.element.selected = true;
            });
          }
        };


        selectCtrl.readValue = function readNgOptionsMultiple() {
          var selectedValues = selectElement.val() || [],
              selections = [];

          forEach(selectedValues, function(value) {
            var option = options.selectValueMap[value];
            if (option && !option.disabled) selections.push(options.getViewValueFromOption(option));
          });

          return selections;
        };

        // If we are using `track by` then we must watch these tracked values on the model
        // since ngModel only watches for object identity change
        if (ngOptions.trackBy) {

          scope.$watchCollection(function() {
            if (isArray(ngModelCtrl.$viewValue)) {
              return ngModelCtrl.$viewValue.map(function(value) {
                return ngOptions.getTrackByValue(value);
              });
            }
          }, function() {
            ngModelCtrl.$render();
          });

        }
      }


      if (providedEmptyOption) {

        // we need to remove it before calling selectElement.empty() because otherwise IE will
        // remove the label from the element. wtf?
        emptyOption.remove();

        // compile the element since there might be bindings in it
        $compile(emptyOption)(scope);

        // remove the class, which is added automatically because we recompile the element and it
        // becomes the compilation root
        emptyOption.removeClass('ng-scope');
      } else {
        emptyOption = jqLite(optionTemplate.cloneNode(false));
      }

      // We need to do this here to ensure that the options object is defined
      // when we first hit it in writeNgOptionsValue
      updateOptions();

      // We will re-render the option elements if the option values or labels change
      scope.$watchCollection(ngOptions.getWatchables, updateOptions);

      // ------------------------------------------------------------------ //


      function updateOptionElement(option, element) {
        option.element = element;
        element.disabled = option.disabled;
        // NOTE: The label must be set before the value, otherwise IE10/11/EDGE create unresponsive
        // selects in certain circumstances when multiple selects are next to each other and display
        // the option list in listbox style, i.e. the select is [multiple], or specifies a [size].
        // See https://github.com/angular/angular.js/issues/11314 for more info.
        // This is unfortunately untestable with unit / e2e tests
        if (option.label !== element.label) {
          element.label = option.label;
          element.textContent = option.label;
        }
        if (option.value !== element.value) element.value = option.selectValue;
      }

      function addOrReuseElement(parent, current, type, templateElement) {
        var element;
        // Check whether we can reuse the next element
        if (current && lowercase(current.nodeName) === type) {
          // The next element is the right type so reuse it
          element = current;
        } else {
          // The next element is not the right type so create a new one
          element = templateElement.cloneNode(false);
          if (!current) {
            // There are no more elements so just append it to the select
            parent.appendChild(element);
          } else {
            // The next element is not a group so insert the new one
            parent.insertBefore(element, current);
          }
        }
        return element;
      }


      function removeExcessElements(current) {
        var next;
        while (current) {
          next = current.nextSibling;
          jqLiteRemove(current);
          current = next;
        }
      }


      function skipEmptyAndUnknownOptions(current) {
        var emptyOption_ = emptyOption && emptyOption[0];
        var unknownOption_ = unknownOption && unknownOption[0];

        // We cannot rely on the extracted empty option being the same as the compiled empty option,
        // because the compiled empty option might have been replaced by a comment because
        // it had an "element" transclusion directive on it (such as ngIf)
        if (emptyOption_ || unknownOption_) {
          while (current &&
                (current === emptyOption_ ||
                current === unknownOption_ ||
                current.nodeType === NODE_TYPE_COMMENT ||
                current.value === '')) {
            current = current.nextSibling;
          }
        }
        return current;
      }


      function updateOptions() {

        var previousValue = options && selectCtrl.readValue();

        options = ngOptions.getOptions();

        var groupMap = {};
        var currentElement = selectElement[0].firstChild;

        // Ensure that the empty option is always there if it was explicitly provided
        if (providedEmptyOption) {
          selectElement.prepend(emptyOption);
        }

        currentElement = skipEmptyAndUnknownOptions(currentElement);

        options.items.forEach(function updateOption(option) {
          var group;
          var groupElement;
          var optionElement;

          if (option.group) {

            // This option is to live in a group
            // See if we have already created this group
            group = groupMap[option.group];

            if (!group) {

              // We have not already created this group
              groupElement = addOrReuseElement(selectElement[0],
                                               currentElement,
                                               'optgroup',
                                               optGroupTemplate);
              // Move to the next element
              currentElement = groupElement.nextSibling;

              // Update the label on the group element
              groupElement.label = option.group;

              // Store it for use later
              group = groupMap[option.group] = {
                groupElement: groupElement,
                currentOptionElement: groupElement.firstChild
              };

            }

            // So now we have a group for this option we add the option to the group
            optionElement = addOrReuseElement(group.groupElement,
                                              group.currentOptionElement,
                                              'option',
                                              optionTemplate);
            updateOptionElement(option, optionElement);
            // Move to the next element
            group.currentOptionElement = optionElement.nextSibling;

          } else {

            // This option is not in a group
            optionElement = addOrReuseElement(selectElement[0],
                                              currentElement,
                                              'option',
                                              optionTemplate);
            updateOptionElement(option, optionElement);
            // Move to the next element
            currentElement = optionElement.nextSibling;
          }
        });


        // Now remove all excess options and group
        Object.keys(groupMap).forEach(function(key) {
          removeExcessElements(groupMap[key].currentOptionElement);
        });
        removeExcessElements(currentElement);

        ngModelCtrl.$render();

        // Check to see if the value has changed due to the update to the options
        if (!ngModelCtrl.$isEmpty(previousValue)) {
          var nextValue = selectCtrl.readValue();
          if (ngOptions.trackBy ? !equals(previousValue, nextValue) : previousValue !== nextValue) {
            ngModelCtrl.$setViewValue(nextValue);
            ngModelCtrl.$render();
          }
        }

      }
  }

  return {
    restrict: 'A',
    terminal: true,
    require: ['select', '?ngModel'],
    link: {
      pre: function ngOptionsPreLink(scope, selectElement, attr, ctrls) {
        // Deactivate the SelectController.register method to prevent
        // option directives from accidentally registering themselves
        // (and unwanted $destroy handlers etc.)
        ctrls[0].registerOption = noop;
      },
      post: ngOptionsPostLink
    }
  };
}];

/**
 * @ngdoc directive
 * @name ngPluralize
 * @restrict EA
 *
 * @description
 * `ngPluralize` is a directive that displays messages according to en-US localization rules.
 * These rules are bundled with angular.js, but can be overridden
 * (see {@link guide/i18n Angular i18n} dev guide). You configure ngPluralize directive
 * by specifying the mappings between
 * [plural categories](http://unicode.org/repos/cldr-tmp/trunk/diff/supplemental/language_plural_rules.html)
 * and the strings to be displayed.
 *
 * # Plural categories and explicit number rules
 * There are two
 * [plural categories](http://unicode.org/repos/cldr-tmp/trunk/diff/supplemental/language_plural_rules.html)
 * in Angular's default en-US locale: "one" and "other".
 *
 * While a plural category may match many numbers (for example, in en-US locale, "other" can match
 * any number that is not 1), an explicit number rule can only match one number. For example, the
 * explicit number rule for "3" matches the number 3. There are examples of plural categories
 * and explicit number rules throughout the rest of this documentation.
 *
 * # Configuring ngPluralize
 * You configure ngPluralize by providing 2 attributes: `count` and `when`.
 * You can also provide an optional attribute, `offset`.
 *
 * The value of the `count` attribute can be either a string or an {@link guide/expression
 * Angular expression}; these are evaluated on the current scope for its bound value.
 *
 * The `when` attribute specifies the mappings between plural categories and the actual
 * string to be displayed. The value of the attribute should be a JSON object.
 *
 * The following example shows how to configure ngPluralize:
 *
 * ```html
 * <ng-pluralize count="personCount"
                 when="{'0': 'Nobody is viewing.',
 *                      'one': '1 person is viewing.',
 *                      'other': '{} people are viewing.'}">
 * </ng-pluralize>
 *```
 *
 * In the example, `"0: Nobody is viewing."` is an explicit number rule. If you did not
 * specify this rule, 0 would be matched to the "other" category and "0 people are viewing"
 * would be shown instead of "Nobody is viewing". You can specify an explicit number rule for
 * other numbers, for example 12, so that instead of showing "12 people are viewing", you can
 * show "a dozen people are viewing".
 *
 * You can use a set of closed braces (`{}`) as a placeholder for the number that you want substituted
 * into pluralized strings. In the previous example, Angular will replace `{}` with
 * <span ng-non-bindable>`{{personCount}}`</span>. The closed braces `{}` is a placeholder
 * for <span ng-non-bindable>{{numberExpression}}</span>.
 *
 * If no rule is defined for a category, then an empty string is displayed and a warning is generated.
 * Note that some locales define more categories than `one` and `other`. For example, fr-fr defines `few` and `many`.
 *
 * # Configuring ngPluralize with offset
 * The `offset` attribute allows further customization of pluralized text, which can result in
 * a better user experience. For example, instead of the message "4 people are viewing this document",
 * you might display "John, Kate and 2 others are viewing this document".
 * The offset attribute allows you to offset a number by any desired value.
 * Let's take a look at an example:
 *
 * ```html
 * <ng-pluralize count="personCount" offset=2
 *               when="{'0': 'Nobody is viewing.',
 *                      '1': '{{person1}} is viewing.',
 *                      '2': '{{person1}} and {{person2}} are viewing.',
 *                      'one': '{{person1}}, {{person2}} and one other person are viewing.',
 *                      'other': '{{person1}}, {{person2}} and {} other people are viewing.'}">
 * </ng-pluralize>
 * ```
 *
 * Notice that we are still using two plural categories(one, other), but we added
 * three explicit number rules 0, 1 and 2.
 * When one person, perhaps John, views the document, "John is viewing" will be shown.
 * When three people view the document, no explicit number rule is found, so
 * an offset of 2 is taken off 3, and Angular uses 1 to decide the plural category.
 * In this case, plural category 'one' is matched and "John, Mary and one other person are viewing"
 * is shown.
 *
 * Note that when you specify offsets, you must provide explicit number rules for
 * numbers from 0 up to and including the offset. If you use an offset of 3, for example,
 * you must provide explicit number rules for 0, 1, 2 and 3. You must also provide plural strings for
 * plural categories "one" and "other".
 *
 * @param {string|expression} count The variable to be bound to.
 * @param {string} when The mapping between plural category to its corresponding strings.
 * @param {number=} offset Offset to deduct from the total number.
 *
 * @example
    <example module="pluralizeExample">
      <file name="index.html">
        <script>
          angular.module('pluralizeExample', [])
            .controller('ExampleController', ['$scope', function($scope) {
              $scope.person1 = 'Igor';
              $scope.person2 = 'Misko';
              $scope.personCount = 1;
            }]);
        </script>
        <div ng-controller="ExampleController">
          <label>Person 1:<input type="text" ng-model="person1" value="Igor" /></label><br/>
          <label>Person 2:<input type="text" ng-model="person2" value="Misko" /></label><br/>
          <label>Number of People:<input type="text" ng-model="personCount" value="1" /></label><br/>

          <!--- Example with simple pluralization rules for en locale --->
          Without Offset:
          <ng-pluralize count="personCount"
                        when="{'0': 'Nobody is viewing.',
                               'one': '1 person is viewing.',
                               'other': '{} people are viewing.'}">
          </ng-pluralize><br>

          <!--- Example with offset --->
          With Offset(2):
          <ng-pluralize count="personCount" offset=2
                        when="{'0': 'Nobody is viewing.',
                               '1': '{{person1}} is viewing.',
                               '2': '{{person1}} and {{person2}} are viewing.',
                               'one': '{{person1}}, {{person2}} and one other person are viewing.',
                               'other': '{{person1}}, {{person2}} and {} other people are viewing.'}">
          </ng-pluralize>
        </div>
      </file>
      <file name="protractor.js" type="protractor">
        it('should show correct pluralized string', function() {
          var withoutOffset = element.all(by.css('ng-pluralize')).get(0);
          var withOffset = element.all(by.css('ng-pluralize')).get(1);
          var countInput = element(by.model('personCount'));

          expect(withoutOffset.getText()).toEqual('1 person is viewing.');
          expect(withOffset.getText()).toEqual('Igor is viewing.');

          countInput.clear();
          countInput.sendKeys('0');

          expect(withoutOffset.getText()).toEqual('Nobody is viewing.');
          expect(withOffset.getText()).toEqual('Nobody is viewing.');

          countInput.clear();
          countInput.sendKeys('2');

          expect(withoutOffset.getText()).toEqual('2 people are viewing.');
          expect(withOffset.getText()).toEqual('Igor and Misko are viewing.');

          countInput.clear();
          countInput.sendKeys('3');

          expect(withoutOffset.getText()).toEqual('3 people are viewing.');
          expect(withOffset.getText()).toEqual('Igor, Misko and one other person are viewing.');

          countInput.clear();
          countInput.sendKeys('4');

          expect(withoutOffset.getText()).toEqual('4 people are viewing.');
          expect(withOffset.getText()).toEqual('Igor, Misko and 2 other people are viewing.');
        });
        it('should show data-bound names', function() {
          var withOffset = element.all(by.css('ng-pluralize')).get(1);
          var personCount = element(by.model('personCount'));
          var person1 = element(by.model('person1'));
          var person2 = element(by.model('person2'));
          personCount.clear();
          personCount.sendKeys('4');
          person1.clear();
          person1.sendKeys('Di');
          person2.clear();
          person2.sendKeys('Vojta');
          expect(withOffset.getText()).toEqual('Di, Vojta and 2 other people are viewing.');
        });
      </file>
    </example>
 */
var ngPluralizeDirective = ['$locale', '$interpolate', '$log', function($locale, $interpolate, $log) {
  var BRACE = /{}/g,
      IS_WHEN = /^when(Minus)?(.+)$/;

  return {
    link: function(scope, element, attr) {
      var numberExp = attr.count,
          whenExp = attr.$attr.when && element.attr(attr.$attr.when), // we have {{}} in attrs
          offset = attr.offset || 0,
          whens = scope.$eval(whenExp) || {},
          whensExpFns = {},
          startSymbol = $interpolate.startSymbol(),
          endSymbol = $interpolate.endSymbol(),
          braceReplacement = startSymbol + numberExp + '-' + offset + endSymbol,
          watchRemover = angular.noop,
          lastCount;

      forEach(attr, function(expression, attributeName) {
        var tmpMatch = IS_WHEN.exec(attributeName);
        if (tmpMatch) {
          var whenKey = (tmpMatch[1] ? '-' : '') + lowercase(tmpMatch[2]);
          whens[whenKey] = element.attr(attr.$attr[attributeName]);
        }
      });
      forEach(whens, function(expression, key) {
        whensExpFns[key] = $interpolate(expression.replace(BRACE, braceReplacement));

      });

      scope.$watch(numberExp, function ngPluralizeWatchAction(newVal) {
        var count = parseFloat(newVal);
        var countIsNaN = isNaN(count);

        if (!countIsNaN && !(count in whens)) {
          // If an explicit number rule such as 1, 2, 3... is defined, just use it.
          // Otherwise, check it against pluralization rules in $locale service.
          count = $locale.pluralCat(count - offset);
        }

        // If both `count` and `lastCount` are NaN, we don't need to re-register a watch.
        // In JS `NaN !== NaN`, so we have to exlicitly check.
        if ((count !== lastCount) && !(countIsNaN && isNumber(lastCount) && isNaN(lastCount))) {
          watchRemover();
          var whenExpFn = whensExpFns[count];
          if (isUndefined(whenExpFn)) {
            if (newVal != null) {
              $log.debug("ngPluralize: no rule defined for '" + count + "' in " + whenExp);
            }
            watchRemover = noop;
            updateElementText();
          } else {
            watchRemover = scope.$watch(whenExpFn, updateElementText);
          }
          lastCount = count;
        }
      });

      function updateElementText(newText) {
        element.text(newText || '');
      }
    }
  };
}];

/**
 * @ngdoc directive
 * @name ngRepeat
 * @multiElement
 *
 * @description
 * The `ngRepeat` directive instantiates a template once per item from a collection. Each template
 * instance gets its own scope, where the given loop variable is set to the current collection item,
 * and `$index` is set to the item index or key.
 *
 * Special properties are exposed on the local scope of each template instance, including:
 *
 * | Variable  | Type            | Details                                                                     |
 * |-----------|-----------------|-----------------------------------------------------------------------------|
 * | `$index`  | {@type number}  | iterator offset of the repeated element (0..length-1)                       |
 * | `$first`  | {@type boolean} | true if the repeated element is first in the iterator.                      |
 * | `$middle` | {@type boolean} | true if the repeated element is between the first and last in the iterator. |
 * | `$last`   | {@type boolean} | true if the repeated element is last in the iterator.                       |
 * | `$even`   | {@type boolean} | true if the iterator position `$index` is even (otherwise false).           |
 * | `$odd`    | {@type boolean} | true if the iterator position `$index` is odd (otherwise false).            |
 *
 * <div class="alert alert-info">
 *   Creating aliases for these properties is possible with {@link ng.directive:ngInit `ngInit`}.
 *   This may be useful when, for instance, nesting ngRepeats.
 * </div>
 *
 *
 * # Iterating over object properties
 *
 * It is possible to get `ngRepeat` to iterate over the properties of an object using the following
 * syntax:
 *
 * ```js
 * <div ng-repeat="(key, value) in myObj"> ... </div>
 * ```
 *
 * You need to be aware that the JavaScript specification does not define the order of keys
 * returned for an object. (To mitigate this in Angular 1.3 the `ngRepeat` directive
 * used to sort the keys alphabetically.)
 *
 * Version 1.4 removed the alphabetic sorting. We now rely on the order returned by the browser
 * when running `for key in myObj`. It seems that browsers generally follow the strategy of providing
 * keys in the order in which they were defined, although there are exceptions when keys are deleted
 * and reinstated. See the [MDN page on `delete` for more info](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/delete#Cross-browser_notes).
 *
 * If this is not desired, the recommended workaround is to convert your object into an array
 * that is sorted into the order that you prefer before providing it to `ngRepeat`.  You could
 * do this with a filter such as [toArrayFilter](http://ngmodules.org/modules/angular-toArrayFilter)
 * or implement a `$watch` on the object yourself.
 *
 *
 * # Tracking and Duplicates
 *
 * `ngRepeat` uses {@link $rootScope.Scope#$watchCollection $watchCollection} to detect changes in
 * the collection. When a change happens, ngRepeat then makes the corresponding changes to the DOM:
 *
 * * When an item is added, a new instance of the template is added to the DOM.
 * * When an item is removed, its template instance is removed from the DOM.
 * * When items are reordered, their respective templates are reordered in the DOM.
 *
 * To minimize creation of DOM elements, `ngRepeat` uses a function
 * to "keep track" of all items in the collection and their corresponding DOM elements.
 * For example, if an item is added to the collection, ngRepeat will know that all other items
 * already have DOM elements, and will not re-render them.
 *
 * The default tracking function (which tracks items by their identity) does not allow
 * duplicate items in arrays. This is because when there are duplicates, it is not possible
 * to maintain a one-to-one mapping between collection items and DOM elements.
 *
 * If you do need to repeat duplicate items, you can substitute the default tracking behavior
 * with your own using the `track by` expression.
 *
 * For example, you may track items by the index of each item in the collection, using the
 * special scope property `$index`:
 * ```html
 *    <div ng-repeat="n in [42, 42, 43, 43] track by $index">
 *      {{n}}
 *    </div>
 * ```
 *
 * You may also use arbitrary expressions in `track by`, including references to custom functions
 * on the scope:
 * ```html
 *    <div ng-repeat="n in [42, 42, 43, 43] track by myTrackingFunction(n)">
 *      {{n}}
 *    </div>
 * ```
 *
 * <div class="alert alert-success">
 * If you are working with objects that have an identifier property, you should track
 * by the identifier instead of the whole object. Should you reload your data later, `ngRepeat`
 * will not have to rebuild the DOM elements for items it has already rendered, even if the
 * JavaScript objects in the collection have been substituted for new ones. For large collections,
 * this signifincantly improves rendering performance. If you don't have a unique identifier,
 * `track by $index` can also provide a performance boost.
 * </div>
 * ```html
 *    <div ng-repeat="model in collection track by model.id">
 *      {{model.name}}
 *    </div>
 * ```
 *
 * When no `track by` expression is provided, it is equivalent to tracking by the built-in
 * `$id` function, which tracks items by their identity:
 * ```html
 *    <div ng-repeat="obj in collection track by $id(obj)">
 *      {{obj.prop}}
 *    </div>
 * ```
 *
 * <div class="alert alert-warning">
 * **Note:** `track by` must always be the last expression:
 * </div>
 * ```
 * <div ng-repeat="model in collection | orderBy: 'id' as filtered_result track by model.id">
 *     {{model.name}}
 * </div>
 * ```
 *
 * # Special repeat start and end points
 * To repeat a series of elements instead of just one parent element, ngRepeat (as well as other ng directives) supports extending
 * the range of the repeater by defining explicit start and end points by using **ng-repeat-start** and **ng-repeat-end** respectively.
 * The **ng-repeat-start** directive works the same as **ng-repeat**, but will repeat all the HTML code (including the tag it's defined on)
 * up to and including the ending HTML tag where **ng-repeat-end** is placed.
 *
 * The example below makes use of this feature:
 * ```html
 *   <header ng-repeat-start="item in items">
 *     Header {{ item }}
 *   </header>
 *   <div class="body">
 *     Body {{ item }}
 *   </div>
 *   <footer ng-repeat-end>
 *     Footer {{ item }}
 *   </footer>
 * ```
 *
 * And with an input of {@type ['A','B']} for the items variable in the example above, the output will evaluate to:
 * ```html
 *   <header>
 *     Header A
 *   </header>
 *   <div class="body">
 *     Body A
 *   </div>
 *   <footer>
 *     Footer A
 *   </footer>
 *   <header>
 *     Header B
 *   </header>
 *   <div class="body">
 *     Body B
 *   </div>
 *   <footer>
 *     Footer B
 *   </footer>
 * ```
 *
 * The custom start and end points for ngRepeat also support all other HTML directive syntax flavors provided in AngularJS (such
 * as **data-ng-repeat-start**, **x-ng-repeat-start** and **ng:repeat-start**).
 *
 * @animations
 * **.enter** - when a new item is added to the list or when an item is revealed after a filter
 *
 * **.leave** - when an item is removed from the list or when an item is filtered out
 *
 * **.move** - when an adjacent item is filtered out causing a reorder or when the item contents are reordered
 *
 * @element ANY
 * @scope
 * @priority 1000
 * @param {repeat_expression} ngRepeat The expression indicating how to enumerate a collection. These
 *   formats are currently supported:
 *
 *   * `variable in expression` â€“ where variable is the user defined loop variable and `expression`
 *     is a scope expression giving the collection to enumerate.
 *
 *     For example: `album in artist.albums`.
 *
 *   * `(key, value) in expression` â€“ where `key` and `value` can be any user defined identifiers,
 *     and `expression` is the scope expression giving the collection to enumerate.
 *
 *     For example: `(name, age) in {'adam':10, 'amalie':12}`.
 *
 *   * `variable in expression track by tracking_expression` â€“ You can also provide an optional tracking expression
 *     which can be used to associate the objects in the collection with the DOM elements. If no tracking expression
 *     is specified, ng-repeat associates elements by identity. It is an error to have
 *     more than one tracking expression value resolve to the same key. (This would mean that two distinct objects are
 *     mapped to the same DOM element, which is not possible.)
 *
 *     Note that the tracking expression must come last, after any filters, and the alias expression.
 *
 *     For example: `item in items` is equivalent to `item in items track by $id(item)`. This implies that the DOM elements
 *     will be associated by item identity in the array.
 *
 *     For example: `item in items track by $id(item)`. A built in `$id()` function can be used to assign a unique
 *     `$$hashKey` property to each item in the array. This property is then used as a key to associated DOM elements
 *     with the corresponding item in the array by identity. Moving the same object in array would move the DOM
 *     element in the same way in the DOM.
 *
 *     For example: `item in items track by item.id` is a typical pattern when the items come from the database. In this
 *     case the object identity does not matter. Two objects are considered equivalent as long as their `id`
 *     property is same.
 *
 *     For example: `item in items | filter:searchText track by item.id` is a pattern that might be used to apply a filter
 *     to items in conjunction with a tracking expression.
 *
 *   * `variable in expression as alias_expression` â€“ You can also provide an optional alias expression which will then store the
 *     intermediate results of the repeater after the filters have been applied. Typically this is used to render a special message
 *     when a filter is active on the repeater, but the filtered result set is empty.
 *
 *     For example: `item in items | filter:x as results` will store the fragment of the repeated items as `results`, but only after
 *     the items have been processed through the filter.
 *
 *     Please note that `as [variable name] is not an operator but rather a part of ngRepeat micro-syntax so it can be used only at the end
 *     (and not as operator, inside an expression).
 *
 *     For example: `item in items | filter : x | orderBy : order | limitTo : limit as results` .
 *
 * @example
 * This example initializes the scope to a list of names and
 * then uses `ngRepeat` to display every person:
  <example module="ngAnimate" deps="angular-animate.js" animations="true">
    <file name="index.html">
      <div ng-init="friends = [
        {name:'John', age:25, gender:'boy'},
        {name:'Jessie', age:30, gender:'girl'},
        {name:'Johanna', age:28, gender:'girl'},
        {name:'Joy', age:15, gender:'girl'},
        {name:'Mary', age:28, gender:'girl'},
        {name:'Peter', age:95, gender:'boy'},
        {name:'Sebastian', age:50, gender:'boy'},
        {name:'Erika', age:27, gender:'girl'},
        {name:'Patrick', age:40, gender:'boy'},
        {name:'Samantha', age:60, gender:'girl'}
      ]">
        I have {{friends.length}} friends. They are:
        <input type="search" ng-model="q" placeholder="filter friends..." aria-label="filter friends" />
        <ul class="example-animate-container">
          <li class="animate-repeat" ng-repeat="friend in friends | filter:q as results">
            [{{$index + 1}}] {{friend.name}} who is {{friend.age}} years old.
          </li>
          <li class="animate-repeat" ng-if="results.length == 0">
            <strong>No results found...</strong>
          </li>
        </ul>
      </div>
    </file>
    <file name="animations.css">
      .example-animate-container {
        background:white;
        border:1px solid black;
        list-style:none;
        margin:0;
        padding:0 10px;
      }

      .animate-repeat {
        line-height:40px;
        list-style:none;
        box-sizing:border-box;
      }

      .animate-repeat.ng-move,
      .animate-repeat.ng-enter,
      .animate-repeat.ng-leave {
        transition:all linear 0.5s;
      }

      .animate-repeat.ng-leave.ng-leave-active,
      .animate-repeat.ng-move,
      .animate-repeat.ng-enter {
        opacity:0;
        max-height:0;
      }

      .animate-repeat.ng-leave,
      .animate-repeat.ng-move.ng-move-active,
      .animate-repeat.ng-enter.ng-enter-active {
        opacity:1;
        max-height:40px;
      }
    </file>
    <file name="protractor.js" type="protractor">
      var friends = element.all(by.repeater('friend in friends'));

      it('should render initial data set', function() {
        expect(friends.count()).toBe(10);
        expect(friends.get(0).getText()).toEqual('[1] John who is 25 years old.');
        expect(friends.get(1).getText()).toEqual('[2] Jessie who is 30 years old.');
        expect(friends.last().getText()).toEqual('[10] Samantha who is 60 years old.');
        expect(element(by.binding('friends.length')).getText())
            .toMatch("I have 10 friends. They are:");
      });

       it('should update repeater when filter predicate changes', function() {
         expect(friends.count()).toBe(10);

         element(by.model('q')).sendKeys('ma');

         expect(friends.count()).toBe(2);
         expect(friends.get(0).getText()).toEqual('[1] Mary who is 28 years old.');
         expect(friends.last().getText()).toEqual('[2] Samantha who is 60 years old.');
       });
      </file>
    </example>
 */
var ngRepeatDirective = ['$parse', '$animate', function($parse, $animate) {
  var NG_REMOVED = '$$NG_REMOVED';
  var ngRepeatMinErr = minErr('ngRepeat');

  var updateScope = function(scope, index, valueIdentifier, value, keyIdentifier, key, arrayLength) {
    // TODO(perf): generate setters to shave off ~40ms or 1-1.5%
    scope[valueIdentifier] = value;
    if (keyIdentifier) scope[keyIdentifier] = key;
    scope.$index = index;
    scope.$first = (index === 0);
    scope.$last = (index === (arrayLength - 1));
    scope.$middle = !(scope.$first || scope.$last);
    // jshint bitwise: false
    scope.$odd = !(scope.$even = (index&1) === 0);
    // jshint bitwise: true
  };

  var getBlockStart = function(block) {
    return block.clone[0];
  };

  var getBlockEnd = function(block) {
    return block.clone[block.clone.length - 1];
  };


  return {
    restrict: 'A',
    multiElement: true,
    transclude: 'element',
    priority: 1000,
    terminal: true,
    $$tlb: true,
    compile: function ngRepeatCompile($element, $attr) {
      var expression = $attr.ngRepeat;
      var ngRepeatEndComment = document.createComment(' end ngRepeat: ' + expression + ' ');

      var match = expression.match(/^\s*([\s\S]+?)\s+in\s+([\s\S]+?)(?:\s+as\s+([\s\S]+?))?(?:\s+track\s+by\s+([\s\S]+?))?\s*$/);

      if (!match) {
        throw ngRepeatMinErr('iexp', "Expected expression in form of '_item_ in _collection_[ track by _id_]' but got '{0}'.",
            expression);
      }

      var lhs = match[1];
      var rhs = match[2];
      var aliasAs = match[3];
      var trackByExp = match[4];

      match = lhs.match(/^(?:(\s*[\$\w]+)|\(\s*([\$\w]+)\s*,\s*([\$\w]+)\s*\))$/);

      if (!match) {
        throw ngRepeatMinErr('iidexp', "'_item_' in '_item_ in _collection_' should be an identifier or '(_key_, _value_)' expression, but got '{0}'.",
            lhs);
      }
      var valueIdentifier = match[3] || match[1];
      var keyIdentifier = match[2];

      if (aliasAs && (!/^[$a-zA-Z_][$a-zA-Z0-9_]*$/.test(aliasAs) ||
          /^(null|undefined|this|\$index|\$first|\$middle|\$last|\$even|\$odd|\$parent|\$root|\$id)$/.test(aliasAs))) {
        throw ngRepeatMinErr('badident', "alias '{0}' is invalid --- must be a valid JS identifier which is not a reserved name.",
          aliasAs);
      }

      var trackByExpGetter, trackByIdExpFn, trackByIdArrayFn, trackByIdObjFn;
      var hashFnLocals = {$id: hashKey};

      if (trackByExp) {
        trackByExpGetter = $parse(trackByExp);
      } else {
        trackByIdArrayFn = function(key, value) {
          return hashKey(value);
        };
        trackByIdObjFn = function(key) {
          return key;
        };
      }

      return function ngRepeatLink($scope, $element, $attr, ctrl, $transclude) {

        if (trackByExpGetter) {
          trackByIdExpFn = function(key, value, index) {
            // assign key, value, and $index to the locals so that they can be used in hash functions
            if (keyIdentifier) hashFnLocals[keyIdentifier] = key;
            hashFnLocals[valueIdentifier] = value;
            hashFnLocals.$index = index;
            return trackByExpGetter($scope, hashFnLocals);
          };
        }

        // Store a list of elements from previous run. This is a hash where key is the item from the
        // iterator, and the value is objects with following properties.
        //   - scope: bound scope
        //   - element: previous element.
        //   - index: position
        //
        // We are using no-proto object so that we don't need to guard against inherited props via
        // hasOwnProperty.
        var lastBlockMap = createMap();

        //watch props
        $scope.$watchCollection(rhs, function ngRepeatAction(collection) {
          var index, length,
              previousNode = $element[0],     // node that cloned nodes should be inserted after
                                              // initialized to the comment node anchor
              nextNode,
              // Same as lastBlockMap but it has the current state. It will become the
              // lastBlockMap on the next iteration.
              nextBlockMap = createMap(),
              collectionLength,
              key, value, // key/value of iteration
              trackById,
              trackByIdFn,
              collectionKeys,
              block,       // last object information {scope, element, id}
              nextBlockOrder,
              elementsToRemove;

          if (aliasAs) {
            $scope[aliasAs] = collection;
          }

          if (isArrayLike(collection)) {
            collectionKeys = collection;
            trackByIdFn = trackByIdExpFn || trackByIdArrayFn;
          } else {
            trackByIdFn = trackByIdExpFn || trackByIdObjFn;
            // if object, extract keys, in enumeration order, unsorted
            collectionKeys = [];
            for (var itemKey in collection) {
              if (hasOwnProperty.call(collection, itemKey) && itemKey.charAt(0) !== '$') {
                collectionKeys.push(itemKey);
              }
            }
          }

          collectionLength = collectionKeys.length;
          nextBlockOrder = new Array(collectionLength);

          // locate existing items
          for (index = 0; index < collectionLength; index++) {
            key = (collection === collectionKeys) ? index : collectionKeys[index];
            value = collection[key];
            trackById = trackByIdFn(key, value, index);
            if (lastBlockMap[trackById]) {
              // found previously seen block
              block = lastBlockMap[trackById];
              delete lastBlockMap[trackById];
              nextBlockMap[trackById] = block;
              nextBlockOrder[index] = block;
            } else if (nextBlockMap[trackById]) {
              // if collision detected. restore lastBlockMap and throw an error
              forEach(nextBlockOrder, function(block) {
                if (block && block.scope) lastBlockMap[block.id] = block;
              });
              throw ngRepeatMinErr('dupes',
                  "Duplicates in a repeater are not allowed. Use 'track by' expression to specify unique keys. Repeater: {0}, Duplicate key: {1}, Duplicate value: {2}",
                  expression, trackById, value);
            } else {
              // new never before seen block
              nextBlockOrder[index] = {id: trackById, scope: undefined, clone: undefined};
              nextBlockMap[trackById] = true;
            }
          }

          // remove leftover items
          for (var blockKey in lastBlockMap) {
            block = lastBlockMap[blockKey];
            elementsToRemove = getBlockNodes(block.clone);
            $animate.leave(elementsToRemove);
            if (elementsToRemove[0].parentNode) {
              // if the element was not removed yet because of pending animation, mark it as deleted
              // so that we can ignore it later
              for (index = 0, length = elementsToRemove.length; index < length; index++) {
                elementsToRemove[index][NG_REMOVED] = true;
              }
            }
            block.scope.$destroy();
          }

          // we are not using forEach for perf reasons (trying to avoid #call)
          for (index = 0; index < collectionLength; index++) {
            key = (collection === collectionKeys) ? index : collectionKeys[index];
            value = collection[key];
            block = nextBlockOrder[index];

            if (block.scope) {
              // if we have already seen this object, then we need to reuse the
              // associated scope/element

              nextNode = previousNode;

              // skip nodes that are already pending removal via leave animation
              do {
                nextNode = nextNode.nextSibling;
              } while (nextNode && nextNode[NG_REMOVED]);

              if (getBlockStart(block) != nextNode) {
                // existing item which got moved
                $animate.move(getBlockNodes(block.clone), null, jqLite(previousNode));
              }
              previousNode = getBlockEnd(block);
              updateScope(block.scope, index, valueIdentifier, value, keyIdentifier, key, collectionLength);
            } else {
              // new item which we don't know about
              $transclude(function ngRepeatTransclude(clone, scope) {
                block.scope = scope;
                // http://jsperf.com/clone-vs-createcomment
                var endNode = ngRepeatEndComment.cloneNode(false);
                clone[clone.length++] = endNode;

                // TODO(perf): support naked previousNode in `enter` to avoid creation of jqLite wrapper?
                $animate.enter(clone, null, jqLite(previousNode));
                previousNode = endNode;
                // Note: We only need the first/last node of the cloned nodes.
                // However, we need to keep the reference to the jqlite wrapper as it might be changed later
                // by a directive with templateUrl when its template arrives.
                block.clone = clone;
                nextBlockMap[block.id] = block;
                updateScope(block.scope, index, valueIdentifier, value, keyIdentifier, key, collectionLength);
              });
            }
          }
          lastBlockMap = nextBlockMap;
        });
      };
    }
  };
}];

var NG_HIDE_CLASS = 'ng-hide';
var NG_HIDE_IN_PROGRESS_CLASS = 'ng-hide-animate';
/**
 * @ngdoc directive
 * @name ngShow
 * @multiElement
 *
 * @description
 * The `ngShow` directive shows or hides the given HTML element based on the expression
 * provided to the `ngShow` attribute. The element is shown or hidden by removing or adding
 * the `.ng-hide` CSS class onto the element. The `.ng-hide` CSS class is predefined
 * in AngularJS and sets the display style to none (using an !important flag).
 * For CSP mode please add `angular-csp.css` to your html file (see {@link ng.directive:ngCsp ngCsp}).
 *
 * ```html
 * <!-- when $scope.myValue is truthy (element is visible) -->
 * <div ng-show="myValue"></div>
 *
 * <!-- when $scope.myValue is falsy (element is hidden) -->
 * <div ng-show="myValue" class="ng-hide"></div>
 * ```
 *
 * When the `ngShow` expression evaluates to a falsy value then the `.ng-hide` CSS class is added to the class
 * attribute on the element causing it to become hidden. When truthy, the `.ng-hide` CSS class is removed
 * from the element causing the element not to appear hidden.
 *
 * ## Why is !important used?
 *
 * You may be wondering why !important is used for the `.ng-hide` CSS class. This is because the `.ng-hide` selector
 * can be easily overridden by heavier selectors. For example, something as simple
 * as changing the display style on a HTML list item would make hidden elements appear visible.
 * This also becomes a bigger issue when dealing with CSS frameworks.
 *
 * By using !important, the show and hide behavior will work as expected despite any clash between CSS selector
 * specificity (when !important isn't used with any conflicting styles). If a developer chooses to override the
 * styling to change how to hide an element then it is just a matter of using !important in their own CSS code.
 *
 * ### Overriding `.ng-hide`
 *
 * By default, the `.ng-hide` class will style the element with `display: none!important`. If you wish to change
 * the hide behavior with ngShow/ngHide then this can be achieved by restating the styles for the `.ng-hide`
 * class CSS. Note that the selector that needs to be used is actually `.ng-hide:not(.ng-hide-animate)` to cope
 * with extra animation classes that can be added.
 *
 * ```css
 * .ng-hide:not(.ng-hide-animate) {
 *   /&#42; this is just another form of hiding an element &#42;/
 *   display: block!important;
 *   position: absolute;
 *   top: -9999px;
 *   left: -9999px;
 * }
 * ```
 *
 * By default you don't need to override in CSS anything and the animations will work around the display style.
 *
 * ## A note about animations with `ngShow`
 *
 * Animations in ngShow/ngHide work with the show and hide events that are triggered when the directive expression
 * is true and false. This system works like the animation system present with ngClass except that
 * you must also include the !important flag to override the display property
 * so that you can perform an animation when the element is hidden during the time of the animation.
 *
 * ```css
 * //
 * //a working example can be found at the bottom of this page
 * //
 * .my-element.ng-hide-add, .my-element.ng-hide-remove {
 *   /&#42; this is required as of 1.3x to properly
 *      apply all styling in a show/hide animation &#42;/
 *   transition: 0s linear all;
 * }
 *
 * .my-element.ng-hide-add-active,
 * .my-element.ng-hide-remove-active {
 *   /&#42; the transition is defined in the active class &#42;/
 *   transition: 1s linear all;
 * }
 *
 * .my-element.ng-hide-add { ... }
 * .my-element.ng-hide-add.ng-hide-add-active { ... }
 * .my-element.ng-hide-remove { ... }
 * .my-element.ng-hide-remove.ng-hide-remove-active { ... }
 * ```
 *
 * Keep in mind that, as of AngularJS version 1.3.0-beta.11, there is no need to change the display
 * property to block during animation states--ngAnimate will handle the style toggling automatically for you.
 *
 * @animations
 * addClass: `.ng-hide` - happens after the `ngShow` expression evaluates to a truthy value and the just before contents are set to visible
 * removeClass: `.ng-hide` - happens after the `ngShow` expression evaluates to a non truthy value and just before the contents are set to hidden
 *
 * @element ANY
 * @param {expression} ngShow If the {@link guide/expression expression} is truthy
 *     then the element is shown or hidden respectively.
 *
 * @example
  <example module="ngAnimate" deps="angular-animate.js" animations="true">
    <file name="index.html">
      Click me: <input type="checkbox" ng-model="checked" aria-label="Toggle ngHide"><br/>
      <div>
        Show:
        <div class="check-element animate-show" ng-show="checked">
          <span class="glyphicon glyphicon-thumbs-up"></span> I show up when your checkbox is checked.
        </div>
      </div>
      <div>
        Hide:
        <div class="check-element animate-show" ng-hide="checked">
          <span class="glyphicon glyphicon-thumbs-down"></span> I hide when your checkbox is checked.
        </div>
      </div>
    </file>
    <file name="glyphicons.css">
      @import url(../../components/bootstrap-3.1.1/css/bootstrap.css);
    </file>
    <file name="animations.css">
      .animate-show {
        line-height: 20px;
        opacity: 1;
        padding: 10px;
        border: 1px solid black;
        background: white;
      }

      .animate-show.ng-hide-add, .animate-show.ng-hide-remove {
        transition: all linear 0.5s;
      }

      .animate-show.ng-hide {
        line-height: 0;
        opacity: 0;
        padding: 0 10px;
      }

      .check-element {
        padding: 10px;
        border: 1px solid black;
        background: white;
      }
    </file>
    <file name="protractor.js" type="protractor">
      var thumbsUp = element(by.css('span.glyphicon-thumbs-up'));
      var thumbsDown = element(by.css('span.glyphicon-thumbs-down'));

      it('should check ng-show / ng-hide', function() {
        expect(thumbsUp.isDisplayed()).toBeFalsy();
        expect(thumbsDown.isDisplayed()).toBeTruthy();

        element(by.model('checked')).click();

        expect(thumbsUp.isDisplayed()).toBeTruthy();
        expect(thumbsDown.isDisplayed()).toBeFalsy();
      });
    </file>
  </example>
 */
var ngShowDirective = ['$animate', function($animate) {
  return {
    restrict: 'A',
    multiElement: true,
    link: function(scope, element, attr) {
      scope.$watch(attr.ngShow, function ngShowWatchAction(value) {
        // we're adding a temporary, animation-specific class for ng-hide since this way
        // we can control when the element is actually displayed on screen without having
        // to have a global/greedy CSS selector that breaks when other animations are run.
        // Read: https://github.com/angular/angular.js/issues/9103#issuecomment-58335845
        $animate[value ? 'removeClass' : 'addClass'](element, NG_HIDE_CLASS, {
          tempClasses: NG_HIDE_IN_PROGRESS_CLASS
        });
      });
    }
  };
}];


/**
 * @ngdoc directive
 * @name ngHide
 * @multiElement
 *
 * @description
 * The `ngHide` directive shows or hides the given HTML element based on the expression
 * provided to the `ngHide` attribute. The element is shown or hidden by removing or adding
 * the `ng-hide` CSS class onto the element. The `.ng-hide` CSS class is predefined
 * in AngularJS and sets the display style to none (using an !important flag).
 * For CSP mode please add `angular-csp.css` to your html file (see {@link ng.directive:ngCsp ngCsp}).
 *
 * ```html
 * <!-- when $scope.myValue is truthy (element is hidden) -->
 * <div ng-hide="myValue" class="ng-hide"></div>
 *
 * <!-- when $scope.myValue is falsy (element is visible) -->
 * <div ng-hide="myValue"></div>
 * ```
 *
 * When the `ngHide` expression evaluates to a truthy value then the `.ng-hide` CSS class is added to the class
 * attribute on the element causing it to become hidden. When falsy, the `.ng-hide` CSS class is removed
 * from the element causing the element not to appear hidden.
 *
 * ## Why is !important used?
 *
 * You may be wondering why !important is used for the `.ng-hide` CSS class. This is because the `.ng-hide` selector
 * can be easily overridden by heavier selectors. For example, something as simple
 * as changing the display style on a HTML list item would make hidden elements appear visible.
 * This also becomes a bigger issue when dealing with CSS frameworks.
 *
 * By using !important, the show and hide behavior will work as expected despite any clash between CSS selector
 * specificity (when !important isn't used with any conflicting styles). If a developer chooses to override the
 * styling to change how to hide an element then it is just a matter of using !important in their own CSS code.
 *
 * ### Overriding `.ng-hide`
 *
 * By default, the `.ng-hide` class will style the element with `display: none!important`. If you wish to change
 * the hide behavior with ngShow/ngHide then this can be achieved by restating the styles for the `.ng-hide`
 * class in CSS:
 *
 * ```css
 * .ng-hide {
 *   /&#42; this is just another form of hiding an element &#42;/
 *   display: block!important;
 *   position: absolute;
 *   top: -9999px;
 *   left: -9999px;
 * }
 * ```
 *
 * By default you don't need to override in CSS anything and the animations will work around the display style.
 *
 * ## A note about animations with `ngHide`
 *
 * Animations in ngShow/ngHide work with the show and hide events that are triggered when the directive expression
 * is true and false. This system works like the animation system present with ngClass, except that the `.ng-hide`
 * CSS class is added and removed for you instead of your own CSS class.
 *
 * ```css
 * //
 * //a working example can be found at the bottom of this page
 * //
 * .my-element.ng-hide-add, .my-element.ng-hide-remove {
 *   transition: 0.5s linear all;
 * }
 *
 * .my-element.ng-hide-add { ... }
 * .my-element.ng-hide-add.ng-hide-add-active { ... }
 * .my-element.ng-hide-remove { ... }
 * .my-element.ng-hide-remove.ng-hide-remove-active { ... }
 * ```
 *
 * Keep in mind that, as of AngularJS version 1.3.0-beta.11, there is no need to change the display
 * property to block during animation states--ngAnimate will handle the style toggling automatically for you.
 *
 * @animations
 * removeClass: `.ng-hide` - happens after the `ngHide` expression evaluates to a truthy value and just before the contents are set to hidden
 * addClass: `.ng-hide` - happens after the `ngHide` expression evaluates to a non truthy value and just before the contents are set to visible
 *
 * @element ANY
 * @param {expression} ngHide If the {@link guide/expression expression} is truthy then
 *     the element is shown or hidden respectively.
 *
 * @example
  <example module="ngAnimate" deps="angular-animate.js" animations="true">
    <file name="index.html">
      Click me: <input type="checkbox" ng-model="checked" aria-label="Toggle ngShow"><br/>
      <div>
        Show:
        <div class="check-element animate-hide" ng-show="checked">
          <span class="glyphicon glyphicon-thumbs-up"></span> I show up when your checkbox is checked.
        </div>
      </div>
      <div>
        Hide:
        <div class="check-element animate-hide" ng-hide="checked">
          <span class="glyphicon glyphicon-thumbs-down"></span> I hide when your checkbox is checked.
        </div>
      </div>
    </file>
    <file name="glyphicons.css">
      @import url(../../components/bootstrap-3.1.1/css/bootstrap.css);
    </file>
    <file name="animations.css">
      .animate-hide {
        transition: all linear 0.5s;
        line-height: 20px;
        opacity: 1;
        padding: 10px;
        border: 1px solid black;
        background: white;
      }

      .animate-hide.ng-hide {
        line-height: 0;
        opacity: 0;
        padding: 0 10px;
      }

      .check-element {
        padding: 10px;
        border: 1px solid black;
        background: white;
      }
    </file>
    <file name="protractor.js" type="protractor">
      var thumbsUp = element(by.css('span.glyphicon-thumbs-up'));
      var thumbsDown = element(by.css('span.glyphicon-thumbs-down'));

      it('should check ng-show / ng-hide', function() {
        expect(thumbsUp.isDisplayed()).toBeFalsy();
        expect(thumbsDown.isDisplayed()).toBeTruthy();

        element(by.model('checked')).click();

        expect(thumbsUp.isDisplayed()).toBeTruthy();
        expect(thumbsDown.isDisplayed()).toBeFalsy();
      });
    </file>
  </example>
 */
var ngHideDirective = ['$animate', function($animate) {
  return {
    restrict: 'A',
    multiElement: true,
    link: function(scope, element, attr) {
      scope.$watch(attr.ngHide, function ngHideWatchAction(value) {
        // The comment inside of the ngShowDirective explains why we add and
        // remove a temporary class for the show/hide animation
        $animate[value ? 'addClass' : 'removeClass'](element,NG_HIDE_CLASS, {
          tempClasses: NG_HIDE_IN_PROGRESS_CLASS
        });
      });
    }
  };
}];

/**
 * @ngdoc directive
 * @name ngStyle
 * @restrict AC
 *
 * @description
 * The `ngStyle` directive allows you to set CSS style on an HTML element conditionally.
 *
 * @element ANY
 * @param {expression} ngStyle
 *
 * {@link guide/expression Expression} which evals to an
 * object whose keys are CSS style names and values are corresponding values for those CSS
 * keys.
 *
 * Since some CSS style names are not valid keys for an object, they must be quoted.
 * See the 'background-color' style in the example below.
 *
 * @example
   <example>
     <file name="index.html">
        <input type="button" value="set color" ng-click="myStyle={color:'red'}">
        <input type="button" value="set background" ng-click="myStyle={'background-color':'blue'}">
        <input type="button" value="clear" ng-click="myStyle={}">
        <br/>
        <span ng-style="myStyle">Sample Text</span>
        <pre>myStyle={{myStyle}}</pre>
     </file>
     <file name="style.css">
       span {
         color: black;
       }
     </file>
     <file name="protractor.js" type="protractor">
       var colorSpan = element(by.css('span'));

       it('should check ng-style', function() {
         expect(colorSpan.getCssValue('color')).toBe('rgba(0, 0, 0, 1)');
         element(by.css('input[value=\'set color\']')).click();
         expect(colorSpan.getCssValue('color')).toBe('rgba(255, 0, 0, 1)');
         element(by.css('input[value=clear]')).click();
         expect(colorSpan.getCssValue('color')).toBe('rgba(0, 0, 0, 1)');
       });
     </file>
   </example>
 */
var ngStyleDirective = ngDirective(function(scope, element, attr) {
  scope.$watch(attr.ngStyle, function ngStyleWatchAction(newStyles, oldStyles) {
    if (oldStyles && (newStyles !== oldStyles)) {
      forEach(oldStyles, function(val, style) { element.css(style, '');});
    }
    if (newStyles) element.css(newStyles);
  }, true);
});

/**
 * @ngdoc directive
 * @name ngSwitch
 * @restrict EA
 *
 * @description
 * The `ngSwitch` directive is used to conditionally swap DOM structure on your template based on a scope expression.
 * Elements within `ngSwitch` but without `ngSwitchWhen` or `ngSwitchDefault` directives will be preserved at the location
 * as specified in the template.
 *
 * The directive itself works similar to ngInclude, however, instead of downloading template code (or loading it
 * from the template cache), `ngSwitch` simply chooses one of the nested elements and makes it visible based on which element
 * matches the value obtained from the evaluated expression. In other words, you define a container element
 * (where you place the directive), place an expression on the **`on="..."` attribute**
 * (or the **`ng-switch="..."` attribute**), define any inner elements inside of the directive and place
 * a when attribute per element. The when attribute is used to inform ngSwitch which element to display when the on
 * expression is evaluated. If a matching expression is not found via a when attribute then an element with the default
 * attribute is displayed.
 *
 * <div class="alert alert-info">
 * Be aware that the attribute values to match against cannot be expressions. They are interpreted
 * as literal string values to match against.
 * For example, **`ng-switch-when="someVal"`** will match against the string `"someVal"` not against the
 * value of the expression `$scope.someVal`.
 * </div>

 * @animations
 * enter - happens after the ngSwitch contents change and the matched child element is placed inside the container
 * leave - happens just after the ngSwitch contents change and just before the former contents are removed from the DOM
 *
 * @usage
 *
 * ```
 * <ANY ng-switch="expression">
 *   <ANY ng-switch-when="matchValue1">...</ANY>
 *   <ANY ng-switch-when="matchValue2">...</ANY>
 *   <ANY ng-switch-default>...</ANY>
 * </ANY>
 * ```
 *
 *
 * @scope
 * @priority 1200
 * @param {*} ngSwitch|on expression to match against <code>ng-switch-when</code>.
 * On child elements add:
 *
 * * `ngSwitchWhen`: the case statement to match against. If match then this
 *   case will be displayed. If the same match appears multiple times, all the
 *   elements will be displayed.
 * * `ngSwitchDefault`: the default case when no other case match. If there
 *   are multiple default cases, all of them will be displayed when no other
 *   case match.
 *
 *
 * @example
  <example module="switchExample" deps="angular-animate.js" animations="true">
    <file name="index.html">
      <div ng-controller="ExampleController">
        <select ng-model="selection" ng-options="item for item in items">
        </select>
        <code>selection={{selection}}</code>
        <hr/>
        <div class="animate-switch-container"
          ng-switch on="selection">
            <div class="animate-switch" ng-switch-when="settings">Settings Div</div>
            <div class="animate-switch" ng-switch-when="home">Home Span</div>
            <div class="animate-switch" ng-switch-default>default</div>
        </div>
      </div>
    </file>
    <file name="script.js">
      angular.module('switchExample', ['ngAnimate'])
        .controller('ExampleController', ['$scope', function($scope) {
          $scope.items = ['settings', 'home', 'other'];
          $scope.selection = $scope.items[0];
        }]);
    </file>
    <file name="animations.css">
      .animate-switch-container {
        position:relative;
        background:white;
        border:1px solid black;
        height:40px;
        overflow:hidden;
      }

      .animate-switch {
        padding:10px;
      }

      .animate-switch.ng-animate {
        transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 0.5s;

        position:absolute;
        top:0;
        left:0;
        right:0;
        bottom:0;
      }

      .animate-switch.ng-leave.ng-leave-active,
      .animate-switch.ng-enter {
        top:-50px;
      }
      .animate-switch.ng-leave,
      .animate-switch.ng-enter.ng-enter-active {
        top:0;
      }
    </file>
    <file name="protractor.js" type="protractor">
      var switchElem = element(by.css('[ng-switch]'));
      var select = element(by.model('selection'));

      it('should start in settings', function() {
        expect(switchElem.getText()).toMatch(/Settings Div/);
      });
      it('should change to home', function() {
        select.all(by.css('option')).get(1).click();
        expect(switchElem.getText()).toMatch(/Home Span/);
      });
      it('should select default', function() {
        select.all(by.css('option')).get(2).click();
        expect(switchElem.getText()).toMatch(/default/);
      });
    </file>
  </example>
 */
var ngSwitchDirective = ['$animate', function($animate) {
  return {
    require: 'ngSwitch',

    // asks for $scope to fool the BC controller module
    controller: ['$scope', function ngSwitchController() {
     this.cases = {};
    }],
    link: function(scope, element, attr, ngSwitchController) {
      var watchExpr = attr.ngSwitch || attr.on,
          selectedTranscludes = [],
          selectedElements = [],
          previousLeaveAnimations = [],
          selectedScopes = [];

      var spliceFactory = function(array, index) {
          return function() { array.splice(index, 1); };
      };

      scope.$watch(watchExpr, function ngSwitchWatchAction(value) {
        var i, ii;
        for (i = 0, ii = previousLeaveAnimations.length; i < ii; ++i) {
          $animate.cancel(previousLeaveAnimations[i]);
        }
        previousLeaveAnimations.length = 0;

        for (i = 0, ii = selectedScopes.length; i < ii; ++i) {
          var selected = getBlockNodes(selectedElements[i].clone);
          selectedScopes[i].$destroy();
          var promise = previousLeaveAnimations[i] = $animate.leave(selected);
          promise.then(spliceFactory(previousLeaveAnimations, i));
        }

        selectedElements.length = 0;
        selectedScopes.length = 0;

        if ((selectedTranscludes = ngSwitchController.cases['!' + value] || ngSwitchController.cases['?'])) {
          forEach(selectedTranscludes, function(selectedTransclude) {
            selectedTransclude.transclude(function(caseElement, selectedScope) {
              selectedScopes.push(selectedScope);
              var anchor = selectedTransclude.element;
              caseElement[caseElement.length++] = document.createComment(' end ngSwitchWhen: ');
              var block = { clone: caseElement };

              selectedElements.push(block);
              $animate.enter(caseElement, anchor.parent(), anchor);
            });
          });
        }
      });
    }
  };
}];

var ngSwitchWhenDirective = ngDirective({
  transclude: 'element',
  priority: 1200,
  require: '^ngSwitch',
  multiElement: true,
  link: function(scope, element, attrs, ctrl, $transclude) {
    ctrl.cases['!' + attrs.ngSwitchWhen] = (ctrl.cases['!' + attrs.ngSwitchWhen] || []);
    ctrl.cases['!' + attrs.ngSwitchWhen].push({ transclude: $transclude, element: element });
  }
});

var ngSwitchDefaultDirective = ngDirective({
  transclude: 'element',
  priority: 1200,
  require: '^ngSwitch',
  multiElement: true,
  link: function(scope, element, attr, ctrl, $transclude) {
    ctrl.cases['?'] = (ctrl.cases['?'] || []);
    ctrl.cases['?'].push({ transclude: $transclude, element: element });
   }
});

/**
 * @ngdoc directive
 * @name ngTransclude
 * @restrict EAC
 *
 * @description
 * Directive that marks the insertion point for the transcluded DOM of the nearest parent directive that uses transclusion.
 *
 * Any existing content of the element that this directive is placed on will be removed before the transcluded content is inserted.
 *
 * @element ANY
 *
 * @example
   <example module="transcludeExample">
     <file name="index.html">
       <script>
         angular.module('transcludeExample', [])
          .directive('pane', function(){
             return {
               restrict: 'E',
               transclude: true,
               scope: { title:'@' },
               template: '<div style="border: 1px solid black;">' +
                           '<div style="background-color: gray">{{title}}</div>' +
                           '<ng-transclude></ng-transclude>' +
                         '</div>'
             };
         })
         .controller('ExampleController', ['$scope', function($scope) {
           $scope.title = 'Lorem Ipsum';
           $scope.text = 'Neque porro quisquam est qui dolorem ipsum quia dolor...';
         }]);
       </script>
       <div ng-controller="ExampleController">
         <input ng-model="title" aria-label="title"> <br/>
         <textarea ng-model="text" aria-label="text"></textarea> <br/>
         <pane title="{{title}}">{{text}}</pane>
       </div>
     </file>
     <file name="protractor.js" type="protractor">
        it('should have transcluded', function() {
          var titleElement = element(by.model('title'));
          titleElement.clear();
          titleElement.sendKeys('TITLE');
          var textElement = element(by.model('text'));
          textElement.clear();
          textElement.sendKeys('TEXT');
          expect(element(by.binding('title')).getText()).toEqual('TITLE');
          expect(element(by.binding('text')).getText()).toEqual('TEXT');
        });
     </file>
   </example>
 *
 */
var ngTranscludeDirective = ngDirective({
  restrict: 'EAC',
  link: function($scope, $element, $attrs, controller, $transclude) {
    if (!$transclude) {
      throw minErr('ngTransclude')('orphan',
       'Illegal use of ngTransclude directive in the template! ' +
       'No parent directive that requires a transclusion found. ' +
       'Element: {0}',
       startingTag($element));
    }

    $transclude(function(clone) {
      $element.empty();
      $element.append(clone);
    });
  }
});

/**
 * @ngdoc directive
 * @name script
 * @restrict E
 *
 * @description
 * Load the content of a `<script>` element into {@link ng.$templateCache `$templateCache`}, so that the
 * template can be used by {@link ng.directive:ngInclude `ngInclude`},
 * {@link ngRoute.directive:ngView `ngView`}, or {@link guide/directive directives}. The type of the
 * `<script>` element must be specified as `text/ng-template`, and a cache name for the template must be
 * assigned through the element's `id`, which can then be used as a directive's `templateUrl`.
 *
 * @param {string} type Must be set to `'text/ng-template'`.
 * @param {string} id Cache name of the template.
 *
 * @example
  <example>
    <file name="index.html">
      <script type="text/ng-template" id="/tpl.html">
        Content of the template.
      </script>

      <a ng-click="currentTpl='/tpl.html'" id="tpl-link">Load inlined template</a>
      <div id="tpl-content" ng-include src="currentTpl"></div>
    </file>
    <file name="protractor.js" type="protractor">
      it('should load template defined inside script tag', function() {
        element(by.css('#tpl-link')).click();
        expect(element(by.css('#tpl-content')).getText()).toMatch(/Content of the template/);
      });
    </file>
  </example>
 */
var scriptDirective = ['$templateCache', function($templateCache) {
  return {
    restrict: 'E',
    terminal: true,
    compile: function(element, attr) {
      if (attr.type == 'text/ng-template') {
        var templateUrl = attr.id,
            text = element[0].text;

        $templateCache.put(templateUrl, text);
      }
    }
  };
}];

var noopNgModelController = { $setViewValue: noop, $render: noop };

function chromeHack(optionElement) {
  // Workaround for https://code.google.com/p/chromium/issues/detail?id=381459
  // Adding an <option selected="selected"> element to a <select required="required"> should
  // automatically select the new element
  if (optionElement[0].hasAttribute('selected')) {
    optionElement[0].selected = true;
  }
}

/**
 * @ngdoc type
 * @name  select.SelectController
 * @description
 * The controller for the `<select>` directive. This provides support for reading
 * and writing the selected value(s) of the control and also coordinates dynamically
 * added `<option>` elements, perhaps by an `ngRepeat` directive.
 */
var SelectController =
        ['$element', '$scope', '$attrs', function($element, $scope, $attrs) {

  var self = this,
      optionsMap = new HashMap();

  // If the ngModel doesn't get provided then provide a dummy noop version to prevent errors
  self.ngModelCtrl = noopNgModelController;

  // The "unknown" option is one that is prepended to the list if the viewValue
  // does not match any of the options. When it is rendered the value of the unknown
  // option is '? XXX ?' where XXX is the hashKey of the value that is not known.
  //
  // We can't just jqLite('<option>') since jqLite is not smart enough
  // to create it in <select> and IE barfs otherwise.
  self.unknownOption = jqLite(document.createElement('option'));
  self.renderUnknownOption = function(val) {
    var unknownVal = '? ' + hashKey(val) + ' ?';
    self.unknownOption.val(unknownVal);
    $element.prepend(self.unknownOption);
    $element.val(unknownVal);
  };

  $scope.$on('$destroy', function() {
    // disable unknown option so that we don't do work when the whole select is being destroyed
    self.renderUnknownOption = noop;
  });

  self.removeUnknownOption = function() {
    if (self.unknownOption.parent()) self.unknownOption.remove();
  };


  // Read the value of the select control, the implementation of this changes depending
  // upon whether the select can have multiple values and whether ngOptions is at work.
  self.readValue = function readSingleValue() {
    self.removeUnknownOption();
    return $element.val();
  };


  // Write the value to the select control, the implementation of this changes depending
  // upon whether the select can have multiple values and whether ngOptions is at work.
  self.writeValue = function writeSingleValue(value) {
    if (self.hasOption(value)) {
      self.removeUnknownOption();
      $element.val(value);
      if (value === '') self.emptyOption.prop('selected', true); // to make IE9 happy
    } else {
      if (value == null && self.emptyOption) {
        self.removeUnknownOption();
        $element.val('');
      } else {
        self.renderUnknownOption(value);
      }
    }
  };


  // Tell the select control that an option, with the given value, has been added
  self.addOption = function(value, element) {
    assertNotHasOwnProperty(value, '"option value"');
    if (value === '') {
      self.emptyOption = element;
    }
    var count = optionsMap.get(value) || 0;
    optionsMap.put(value, count + 1);
    self.ngModelCtrl.$render();
    chromeHack(element);
  };

  // Tell the select control that an option, with the given value, has been removed
  self.removeOption = function(value) {
    var count = optionsMap.get(value);
    if (count) {
      if (count === 1) {
        optionsMap.remove(value);
        if (value === '') {
          self.emptyOption = undefined;
        }
      } else {
        optionsMap.put(value, count - 1);
      }
    }
  };

  // Check whether the select control has an option matching the given value
  self.hasOption = function(value) {
    return !!optionsMap.get(value);
  };


  self.registerOption = function(optionScope, optionElement, optionAttrs, interpolateValueFn, interpolateTextFn) {

    if (interpolateValueFn) {
      // The value attribute is interpolated
      var oldVal;
      optionAttrs.$observe('value', function valueAttributeObserveAction(newVal) {
        if (isDefined(oldVal)) {
          self.removeOption(oldVal);
        }
        oldVal = newVal;
        self.addOption(newVal, optionElement);
      });
    } else if (interpolateTextFn) {
      // The text content is interpolated
      optionScope.$watch(interpolateTextFn, function interpolateWatchAction(newVal, oldVal) {
        optionAttrs.$set('value', newVal);
        if (oldVal !== newVal) {
          self.removeOption(oldVal);
        }
        self.addOption(newVal, optionElement);
      });
    } else {
      // The value attribute is static
      self.addOption(optionAttrs.value, optionElement);
    }

    optionElement.on('$destroy', function() {
      self.removeOption(optionAttrs.value);
      self.ngModelCtrl.$render();
    });
  };
}];

/**
 * @ngdoc directive
 * @name select
 * @restrict E
 *
 * @description
 * HTML `SELECT` element with angular data-binding.
 *
 * The `select` directive is used together with {@link ngModel `ngModel`} to provide data-binding
 * between the scope and the `<select>` control (including setting default values).
 * ÃŒt also handles dynamic `<option>` elements, which can be added using the {@link ngRepeat `ngRepeat}` or
 * {@link ngOptions `ngOptions`} directives.
 *
 * When an item in the `<select>` menu is selected, the value of the selected option will be bound
 * to the model identified by the `ngModel` directive. With static or repeated options, this is
 * the content of the `value` attribute or the textContent of the `<option>`, if the value attribute is missing.
 * If you want dynamic value attributes, you can use interpolation inside the value attribute.
 *
 * <div class="alert alert-warning">
 * Note that the value of a `select` directive used without `ngOptions` is always a string.
 * When the model needs to be bound to a non-string value, you must either explictly convert it
 * using a directive (see example below) or use `ngOptions` to specify the set of options.
 * This is because an option element can only be bound to string values at present.
 * </div>
 *
 * If the viewValue of `ngModel` does not match any of the options, then the control
 * will automatically add an "unknown" option, which it then removes when the mismatch is resolved.
 *
 * Optionally, a single hard-coded `<option>` element, with the value set to an empty string, can
 * be nested into the `<select>` element. This element will then represent the `null` or "not selected"
 * option. See example below for demonstration.
 *
 * <div class="alert alert-info">
 * In many cases, `ngRepeat` can be used on `<option>` elements instead of {@link ng.directive:ngOptions
 * ngOptions} to achieve a similar result. However, `ngOptions` provides some benefits, such as
 * more flexibility in how the `<select>`'s model is assigned via the `select` **`as`** part of the
 * comprehension expression, and additionally in reducing memory and increasing speed by not creating
 * a new scope for each repeated instance.
 * </div>
 *
 *
 * @param {string} ngModel Assignable angular expression to data-bind to.
 * @param {string=} name Property name of the form under which the control is published.
 * @param {string=} multiple Allows multiple options to be selected. The selected values will be
 *     bound to the model as an array.
 * @param {string=} required Sets `required` validation error key if the value is not entered.
 * @param {string=} ngRequired Adds required attribute and required validation constraint to
 * the element when the ngRequired expression evaluates to true. Use ngRequired instead of required
 * when you want to data-bind to the required attribute.
 * @param {string=} ngChange Angular expression to be executed when selected option(s) changes due to user
 *    interaction with the select element.
 * @param {string=} ngOptions sets the options that the select is populated with and defines what is
 * set on the model on selection. See {@link ngOptions `ngOptions`}.
 *
 * @example
 * ### Simple `select` elements with static options
 *
 * <example name="static-select" module="staticSelect">
 * <file name="index.html">
 * <div ng-controller="ExampleController">
 *   <form name="myForm">
 *     <label for="singleSelect"> Single select: </label><br>
 *     <select name="singleSelect" ng-model="data.singleSelect">
 *       <option value="option-1">Option 1</option>
 *       <option value="option-2">Option 2</option>
 *     </select><br>
 *
 *     <label for="singleSelect"> Single select with "not selected" option and dynamic option values: </label><br>
 *     <select name="singleSelect" id="singleSelect" ng-model="data.singleSelect">
 *       <option value="">---Please select---</option> <!-- not selected / blank option -->
 *       <option value="{{data.option1}}">Option 1</option> <!-- interpolation -->
 *       <option value="option-2">Option 2</option>
 *     </select><br>
 *     <button ng-click="forceUnknownOption()">Force unknown option</button><br>
 *     <tt>singleSelect = {{data.singleSelect}}</tt>
 *
 *     <hr>
 *     <label for="multipleSelect"> Multiple select: </label><br>
 *     <select name="multipleSelect" id="multipleSelect" ng-model="data.multipleSelect" multiple>
 *       <option value="option-1">Option 1</option>
 *       <option value="option-2">Option 2</option>
 *       <option value="option-3">Option 3</option>
 *     </select><br>
 *     <tt>multipleSelect = {{data.multipleSelect}}</tt><br/>
 *   </form>
 * </div>
 * </file>
 * <file name="app.js">
 *  angular.module('staticSelect', [])
 *    .controller('ExampleController', ['$scope', function($scope) {
 *      $scope.data = {
 *       singleSelect: null,
 *       multipleSelect: [],
 *       option1: 'option-1',
 *      };
 *
 *      $scope.forceUnknownOption = function() {
 *        $scope.data.singleSelect = 'nonsense';
 *      };
 *   }]);
 * </file>
 *</example>
 *
 * ### Using `ngRepeat` to generate `select` options
 * <example name="ngrepeat-select" module="ngrepeatSelect">
 * <file name="index.html">
 * <div ng-controller="ExampleController">
 *   <form name="myForm">
 *     <label for="repeatSelect"> Repeat select: </label>
 *     <select name="repeatSelect" id="repeatSelect" ng-model="data.repeatSelect">
 *       <option ng-repeat="option in data.availableOptions" value="{{option.id}}">{{option.name}}</option>
 *     </select>
 *   </form>
 *   <hr>
 *   <tt>repeatSelect = {{data.repeatSelect}}</tt><br/>
 * </div>
 * </file>
 * <file name="app.js">
 *  angular.module('ngrepeatSelect', [])
 *    .controller('ExampleController', ['$scope', function($scope) {
 *      $scope.data = {
 *       repeatSelect: null,
 *       availableOptions: [
 *         {id: '1', name: 'Option A'},
 *         {id: '2', name: 'Option B'},
 *         {id: '3', name: 'Option C'}
 *       ],
 *      };
 *   }]);
 * </file>
 *</example>
 *
 *
 * ### Using `select` with `ngOptions` and setting a default value
 * See the {@link ngOptions ngOptions documentation} for more `ngOptions` usage examples.
 *
 * <example name="select-with-default-values" module="defaultValueSelect">
 * <file name="index.html">
 * <div ng-controller="ExampleController">
 *   <form name="myForm">
 *     <label for="mySelect">Make a choice:</label>
 *     <select name="mySelect" id="mySelect"
 *       ng-options="option.name for option in data.availableOptions track by option.id"
 *       ng-model="data.selectedOption"></select>
 *   </form>
 *   <hr>
 *   <tt>option = {{data.selectedOption}}</tt><br/>
 * </div>
 * </file>
 * <file name="app.js">
 *  angular.module('defaultValueSelect', [])
 *    .controller('ExampleController', ['$scope', function($scope) {
 *      $scope.data = {
 *       availableOptions: [
 *         {id: '1', name: 'Option A'},
 *         {id: '2', name: 'Option B'},
 *         {id: '3', name: 'Option C'}
 *       ],
 *       selectedOption: {id: '3', name: 'Option C'} //This sets the default value of the select in the ui
 *       };
 *   }]);
 * </file>
 *</example>
 *
 *
 * ### Binding `select` to a non-string value via `ngModel` parsing / formatting
 *
 * <example name="select-with-non-string-options" module="nonStringSelect">
 *   <file name="index.html">
 *     <select ng-model="model.id" convert-to-number>
 *       <option value="0">Zero</option>
 *       <option value="1">One</option>
 *       <option value="2">Two</option>
 *     </select>
 *     {{ model }}
 *   </file>
 *   <file name="app.js">
 *     angular.module('nonStringSelect', [])
 *       .run(function($rootScope) {
 *         $rootScope.model = { id: 2 };
 *       })
 *       .directive('convertToNumber', function() {
 *         return {
 *           require: 'ngModel',
 *           link: function(scope, element, attrs, ngModel) {
 *             ngModel.$parsers.push(function(val) {
 *               return parseInt(val, 10);
 *             });
 *             ngModel.$formatters.push(function(val) {
 *               return '' + val;
 *             });
 *           }
 *         };
 *       });
 *   </file>
 *   <file name="protractor.js" type="protractor">
 *     it('should initialize to model', function() {
 *       var select = element(by.css('select'));
 *       expect(element(by.model('model.id')).$('option:checked').getText()).toEqual('Two');
 *     });
 *   </file>
 * </example>
 *
 */
var selectDirective = function() {

  return {
    restrict: 'E',
    require: ['select', '?ngModel'],
    controller: SelectController,
    priority: 1,
    link: {
      pre: selectPreLink
    }
  };

  function selectPreLink(scope, element, attr, ctrls) {

      // if ngModel is not defined, we don't need to do anything
      var ngModelCtrl = ctrls[1];
      if (!ngModelCtrl) return;

      var selectCtrl = ctrls[0];

      selectCtrl.ngModelCtrl = ngModelCtrl;

      // We delegate rendering to the `writeValue` method, which can be changed
      // if the select can have multiple selected values or if the options are being
      // generated by `ngOptions`
      ngModelCtrl.$render = function() {
        selectCtrl.writeValue(ngModelCtrl.$viewValue);
      };

      // When the selected item(s) changes we delegate getting the value of the select control
      // to the `readValue` method, which can be changed if the select can have multiple
      // selected values or if the options are being generated by `ngOptions`
      element.on('change', function() {
        scope.$apply(function() {
          ngModelCtrl.$setViewValue(selectCtrl.readValue());
        });
      });

      // If the select allows multiple values then we need to modify how we read and write
      // values from and to the control; also what it means for the value to be empty and
      // we have to add an extra watch since ngModel doesn't work well with arrays - it
      // doesn't trigger rendering if only an item in the array changes.
      if (attr.multiple) {

        // Read value now needs to check each option to see if it is selected
        selectCtrl.readValue = function readMultipleValue() {
          var array = [];
          forEach(element.find('option'), function(option) {
            if (option.selected) {
              array.push(option.value);
            }
          });
          return array;
        };

        // Write value now needs to set the selected property of each matching option
        selectCtrl.writeValue = function writeMultipleValue(value) {
          var items = new HashMap(value);
          forEach(element.find('option'), function(option) {
            option.selected = isDefined(items.get(option.value));
          });
        };

        // we have to do it on each watch since ngModel watches reference, but
        // we need to work of an array, so we need to see if anything was inserted/removed
        var lastView, lastViewRef = NaN;
        scope.$watch(function selectMultipleWatch() {
          if (lastViewRef === ngModelCtrl.$viewValue && !equals(lastView, ngModelCtrl.$viewValue)) {
            lastView = shallowCopy(ngModelCtrl.$viewValue);
            ngModelCtrl.$render();
          }
          lastViewRef = ngModelCtrl.$viewValue;
        });

        // If we are a multiple select then value is now a collection
        // so the meaning of $isEmpty changes
        ngModelCtrl.$isEmpty = function(value) {
          return !value || value.length === 0;
        };

      }
    }
};


// The option directive is purely designed to communicate the existence (or lack of)
// of dynamically created (and destroyed) option elements to their containing select
// directive via its controller.
var optionDirective = ['$interpolate', function($interpolate) {
  return {
    restrict: 'E',
    priority: 100,
    compile: function(element, attr) {

      if (isDefined(attr.value)) {
        // If the value attribute is defined, check if it contains an interpolation
        var interpolateValueFn = $interpolate(attr.value, true);
      } else {
        // If the value attribute is not defined then we fall back to the
        // text content of the option element, which may be interpolated
        var interpolateTextFn = $interpolate(element.text(), true);
        if (!interpolateTextFn) {
          attr.$set('value', element.text());
        }
      }

      return function(scope, element, attr) {

        // This is an optimization over using ^^ since we don't want to have to search
        // all the way to the root of the DOM for every single option element
        var selectCtrlName = '$selectController',
            parent = element.parent(),
            selectCtrl = parent.data(selectCtrlName) ||
              parent.parent().data(selectCtrlName); // in case we are in optgroup

        if (selectCtrl) {
          selectCtrl.registerOption(scope, element, attr, interpolateValueFn, interpolateTextFn);
        }
      };
    }
  };
}];

var styleDirective = valueFn({
  restrict: 'E',
  terminal: false
});

var requiredDirective = function() {
  return {
    restrict: 'A',
    require: '?ngModel',
    link: function(scope, elm, attr, ctrl) {
      if (!ctrl) return;
      attr.required = true; // force truthy in case we are on non input element

      ctrl.$validators.required = function(modelValue, viewValue) {
        return !attr.required || !ctrl.$isEmpty(viewValue);
      };

      attr.$observe('required', function() {
        ctrl.$validate();
      });
    }
  };
};


var patternDirective = function() {
  return {
    restrict: 'A',
    require: '?ngModel',
    link: function(scope, elm, attr, ctrl) {
      if (!ctrl) return;

      var regexp, patternExp = attr.ngPattern || attr.pattern;
      attr.$observe('pattern', function(regex) {
        if (isString(regex) && regex.length > 0) {
          regex = new RegExp('^' + regex + '$');
        }

        if (regex && !regex.test) {
          throw minErr('ngPattern')('noregexp',
            'Expected {0} to be a RegExp but was {1}. Element: {2}', patternExp,
            regex, startingTag(elm));
        }

        regexp = regex || undefined;
        ctrl.$validate();
      });

      ctrl.$validators.pattern = function(modelValue, viewValue) {
        // HTML5 pattern constraint validates the input value, so we validate the viewValue
        return ctrl.$isEmpty(viewValue) || isUndefined(regexp) || regexp.test(viewValue);
      };
    }
  };
};


var maxlengthDirective = function() {
  return {
    restrict: 'A',
    require: '?ngModel',
    link: function(scope, elm, attr, ctrl) {
      if (!ctrl) return;

      var maxlength = -1;
      attr.$observe('maxlength', function(value) {
        var intVal = toInt(value);
        maxlength = isNaN(intVal) ? -1 : intVal;
        ctrl.$validate();
      });
      ctrl.$validators.maxlength = function(modelValue, viewValue) {
        return (maxlength < 0) || ctrl.$isEmpty(viewValue) || (viewValue.length <= maxlength);
      };
    }
  };
};

var minlengthDirective = function() {
  return {
    restrict: 'A',
    require: '?ngModel',
    link: function(scope, elm, attr, ctrl) {
      if (!ctrl) return;

      var minlength = 0;
      attr.$observe('minlength', function(value) {
        minlength = toInt(value) || 0;
        ctrl.$validate();
      });
      ctrl.$validators.minlength = function(modelValue, viewValue) {
        return ctrl.$isEmpty(viewValue) || viewValue.length >= minlength;
      };
    }
  };
};

if (window.angular.bootstrap) {
  //AngularJS is already loaded, so we can return here...
  console.log('WARNING: Tried to load angular more than once.');
  return;
}

//try to bind to jquery now so that one can write jqLite(document).ready()
//but we will rebind on bootstrap again.
bindJQuery();

publishExternalAPI(angular);

angular.module("ngLocale", [], ["$provide", function($provide) {
var PLURAL_CATEGORY = {ZERO: "zero", ONE: "one", TWO: "two", FEW: "few", MANY: "many", OTHER: "other"};
function getDecimals(n) {
  n = n + '';
  var i = n.indexOf('.');
  return (i == -1) ? 0 : n.length - i - 1;
}

function getVF(n, opt_precision) {
  var v = opt_precision;

  if (undefined === v) {
    v = Math.min(getDecimals(n), 3);
  }

  var base = Math.pow(10, v);
  var f = ((n * base) | 0) % base;
  return {v: v, f: f};
}

$provide.value("$locale", {
  "DATETIME_FORMATS": {
    "AMPMS": [
      "AM",
      "PM"
    ],
    "DAY": [
      "Sunday",
      "Monday",
      "Tuesday",
      "Wednesday",
      "Thursday",
      "Friday",
      "Saturday"
    ],
    "ERANAMES": [
      "Before Christ",
      "Anno Domini"
    ],
    "ERAS": [
      "BC",
      "AD"
    ],
    "FIRSTDAYOFWEEK": 0,
    "MONTH": [
      "January",
      "February",
      "March",
      "April",
      "May",
      "June",
      "July",
      "August",
      "September",
      "October",
      "November",
      "December"
    ],
    "SHORTDAY": [
      "Sun",
      "Mon",
      "Tue",
      "Wed",
      "Thu",
      "Fri",
      "Sat"
    ],
    "SHORTMONTH": [
      "Jan",
      "Feb",
      "Mar",
      "Apr",
      "May",
      "Jun",
      "Jul",
      "Aug",
      "Sep",
      "Oct",
      "Nov",
      "Dec"
    ],
    "WEEKENDRANGE": [
      5,
      6
    ],
    "fullDate": "EEEE, MMMM d, y",
    "longDate": "MMMM d, y",
    "medium": "MMM d, y h:mm:ss a",
    "mediumDate": "MMM d, y",
    "mediumTime": "h:mm:ss a",
    "short": "M/d/yy h:mm a",
    "shortDate": "M/d/yy",
    "shortTime": "h:mm a"
  },
  "NUMBER_FORMATS": {
    "CURRENCY_SYM": "$",
    "DECIMAL_SEP": ".",
    "GROUP_SEP": ",",
    "PATTERNS": [
      {
        "gSize": 3,
        "lgSize": 3,
        "maxFrac": 3,
        "minFrac": 0,
        "minInt": 1,
        "negPre": "-",
        "negSuf": "",
        "posPre": "",
        "posSuf": ""
      },
      {
        "gSize": 3,
        "lgSize": 3,
        "maxFrac": 2,
        "minFrac": 2,
        "minInt": 1,
        "negPre": "-\u00a4",
        "negSuf": "",
        "posPre": "\u00a4",
        "posSuf": ""
      }
    ]
  },
  "id": "en-us",
  "pluralCat": function(n, opt_precision) {  var i = n | 0;  var vf = getVF(n, opt_precision);  if (i == 1 && vf.v == 0) {    return PLURAL_CATEGORY.ONE;  }  return PLURAL_CATEGORY.OTHER;}
});
}]);

  jqLite(document).ready(function() {
    angularInit(document, bootstrap);
  });

})(window, document);

!window.angular.$$csp().noInlineStyle && window.angular.element(document.head).prepend('<style type="text/css">@charset "UTF-8";[ng\\:cloak],[ng-cloak],[data-ng-cloak],[x-ng-cloak],.ng-cloak,.x-ng-cloak,.ng-hide:not(.ng-hide-animate){display:none !important;}ng\\:form{display:block;}.ng-animate-shim{visibility:hidden;}.ng-anchor{position:absolute;}</style>');;
/**
 * @license AngularJS v1.4.8
 * (c) 2010-2015 Google, Inc. http://angularjs.org
 * License: MIT
 */
(function (window, angular, undefined) {
    'use strict';

    /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
     *     Any commits to this file should be reviewed with security in mind.  *
     *   Changes to this file can potentially create security vulnerabilities. *
     *          An approval from 2 Core members with history of modifying      *
     *                         this file is required.                          *
     *                                                                         *
     *  Does the change somehow allow for arbitrary javascript to be executed? *
     *    Or allows for someone to change the prototype of built-in objects?   *
     *     Or gives undesired access to variables likes document or window?    *
     * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */

    var $sanitizeMinErr = angular.$$minErr('$sanitize');

    /**
     * @ngdoc module
     * @name ngSanitize
     * @description
     *
     * # ngSanitize
     *
     * The `ngSanitize` module provides functionality to sanitize HTML.
     *
     *
     * <div doc-module-components="ngSanitize"></div>
     *
     * See {@link ngSanitize.$sanitize `$sanitize`} for usage.
     */

    /*
     * HTML Parser By Misko Hevery (misko@hevery.com)
     * based on:  HTML Parser By John Resig (ejohn.org)
     * Original code by Erik Arvidsson, Mozilla Public License
     * http://erik.eae.net/simplehtmlparser/simplehtmlparser.js
     *
     * // Use like so:
     * htmlParser(htmlString, {
     *     start: function(tag, attrs, unary) {},
     *     end: function(tag) {},
     *     chars: function(text) {},
     *     comment: function(text) {}
     * });
     *
     */


    /**
     * @ngdoc service
     * @name $sanitize
     * @kind function
     *
     * @description
     *   The input is sanitized by parsing the HTML into tokens. All safe tokens (from a whitelist) are
     *   then serialized back to properly escaped html string. This means that no unsafe input can make
     *   it into the returned string, however, since our parser is more strict than a typical browser
     *   parser, it's possible that some obscure input, which would be recognized as valid HTML by a
     *   browser, won't make it through the sanitizer. The input may also contain SVG markup.
     *   The whitelist is configured using the functions `aHrefSanitizationWhitelist` and
     *   `imgSrcSanitizationWhitelist` of {@link ng.$compileProvider `$compileProvider`}.
     *
     * @param {string} html HTML input.
     * @returns {string} Sanitized HTML.
     *
     * @example
       <example module="sanitizeExample" deps="angular-sanitize.js">
       <file name="index.html">
         <script>
             angular.module('sanitizeExample', ['ngSanitize'])
               .controller('ExampleController', ['$scope', '$sce', function($scope, $sce) {
                 $scope.snippet =
                   '<p style="color:blue">an html\n' +
                   '<em onmouseover="this.textContent=\'PWN3D!\'">click here</em>\n' +
                   'snippet</p>';
                 $scope.deliberatelyTrustDangerousSnippet = function() {
                   return $sce.trustAsHtml($scope.snippet);
                 };
               }]);
         </script>
         <div ng-controller="ExampleController">
            Snippet: <textarea ng-model="snippet" cols="60" rows="3"></textarea>
           <table>
             <tr>
               <td>Directive</td>
               <td>How</td>
               <td>Source</td>
               <td>Rendered</td>
             </tr>
             <tr id="bind-html-with-sanitize">
               <td>ng-bind-html</td>
               <td>Automatically uses $sanitize</td>
               <td><pre>&lt;div ng-bind-html="snippet"&gt;<br/>&lt;/div&gt;</pre></td>
               <td><div ng-bind-html="snippet"></div></td>
             </tr>
             <tr id="bind-html-with-trust">
               <td>ng-bind-html</td>
               <td>Bypass $sanitize by explicitly trusting the dangerous value</td>
               <td>
               <pre>&lt;div ng-bind-html="deliberatelyTrustDangerousSnippet()"&gt;
    &lt;/div&gt;</pre>
               </td>
               <td><div ng-bind-html="deliberatelyTrustDangerousSnippet()"></div></td>
             </tr>
             <tr id="bind-default">
               <td>ng-bind</td>
               <td>Automatically escapes</td>
               <td><pre>&lt;div ng-bind="snippet"&gt;<br/>&lt;/div&gt;</pre></td>
               <td><div ng-bind="snippet"></div></td>
             </tr>
           </table>
           </div>
       </file>
       <file name="protractor.js" type="protractor">
         it('should sanitize the html snippet by default', function() {
           expect(element(by.css('#bind-html-with-sanitize div')).getInnerHtml()).
             toBe('<p>an html\n<em>click here</em>\nsnippet</p>');
         });
    
         it('should inline raw snippet if bound to a trusted value', function() {
           expect(element(by.css('#bind-html-with-trust div')).getInnerHtml()).
             toBe("<p style=\"color:blue\">an html\n" +
                  "<em onmouseover=\"this.textContent='PWN3D!'\">click here</em>\n" +
                  "snippet</p>");
         });
    
         it('should escape snippet without any filter', function() {
           expect(element(by.css('#bind-default div')).getInnerHtml()).
             toBe("&lt;p style=\"color:blue\"&gt;an html\n" +
                  "&lt;em onmouseover=\"this.textContent='PWN3D!'\"&gt;click here&lt;/em&gt;\n" +
                  "snippet&lt;/p&gt;");
         });
    
         it('should update', function() {
           element(by.model('snippet')).clear();
           element(by.model('snippet')).sendKeys('new <b onclick="alert(1)">text</b>');
           expect(element(by.css('#bind-html-with-sanitize div')).getInnerHtml()).
             toBe('new <b>text</b>');
           expect(element(by.css('#bind-html-with-trust div')).getInnerHtml()).toBe(
             'new <b onclick="alert(1)">text</b>');
           expect(element(by.css('#bind-default div')).getInnerHtml()).toBe(
             "new &lt;b onclick=\"alert(1)\"&gt;text&lt;/b&gt;");
         });
       </file>
       </example>
     */
    function $SanitizeProvider() {
        this.$get = ['$$sanitizeUri', function ($$sanitizeUri) {
            return function (html) {
                var buf = [];
                htmlParser(html, htmlSanitizeWriter(buf, function (uri, isImage) {
                    return !/^unsafe/.test($$sanitizeUri(uri, isImage));
                }));
                return buf.join('');
            };
        }];
    }

    function sanitizeText(chars) {
        var buf = [];
        var writer = htmlSanitizeWriter(buf, angular.noop);
        writer.chars(chars);
        return buf.join('');
    }


    // Regular Expressions for parsing tags and attributes
    var START_TAG_REGEXP =
           /^<((?:[a-zA-Z])[\w:-]*)((?:\s+[\w:-]+(?:\s*=\s*(?:(?:"[^"]*")|(?:'[^']*')|[^>\s]+))?)*)\s*(\/?)\s*(>?)/,
      END_TAG_REGEXP = /^<\/\s*([\w:-]+)[^>]*>/,
      ATTR_REGEXP = /([\w:-]+)(?:\s*=\s*(?:(?:"((?:[^"])*)")|(?:'((?:[^'])*)')|([^>\s]+)))?/g,
      BEGIN_TAG_REGEXP = /^</,
      BEGING_END_TAGE_REGEXP = /^<\//,
      COMMENT_REGEXP = /<!--(.*?)-->/g,
      DOCTYPE_REGEXP = /<!DOCTYPE([^>]*?)>/i,
      CDATA_REGEXP = /<!\[CDATA\[(.*?)]]>/g,
      SURROGATE_PAIR_REGEXP = /[\uD800-\uDBFF][\uDC00-\uDFFF]/g,
      // Match everything outside of normal chars and " (quote character)
      NON_ALPHANUMERIC_REGEXP = /([^\#-~| |!])/g;


    // Good source of info about elements and attributes
    // http://dev.w3.org/html5/spec/Overview.html#semantics
    // http://simon.html5.org/html-elements

    // Safe Void Elements - HTML5
    // http://dev.w3.org/html5/spec/Overview.html#void-elements
    var voidElements = makeMap("area,br,col,hr,img,wbr");

    // Elements that you can, intentionally, leave open (and which close themselves)
    // http://dev.w3.org/html5/spec/Overview.html#optional-tags
    var optionalEndTagBlockElements = makeMap("colgroup,dd,dt,li,p,tbody,td,tfoot,th,thead,tr"),
        optionalEndTagInlineElements = makeMap("rp,rt"),
        optionalEndTagElements = angular.extend({},
                                                optionalEndTagInlineElements,
                                                optionalEndTagBlockElements);

    // Safe Block Elements - HTML5
    var blockElements = angular.extend({}, optionalEndTagBlockElements, makeMap("address,article," +
            "aside,blockquote,caption,center,del,dir,div,dl,figure,figcaption,footer,h1,h2,h3,h4,h5," +
            "h6,header,hgroup,hr,ins,map,menu,nav,ol,pre,script,section,table,ul"));

    // Inline Elements - HTML5
    var inlineElements = angular.extend({}, optionalEndTagInlineElements, makeMap("a,abbr,acronym,b," +
            "bdi,bdo,big,br,cite,code,del,dfn,em,font,i,img,ins,kbd,label,map,mark,q,ruby,rp,rt,s," +
            "samp,small,span,strike,strong,sub,sup,time,tt,u,var"));

    // SVG Elements
    // https://wiki.whatwg.org/wiki/Sanitization_rules#svg_Elements
    // Note: the elements animate,animateColor,animateMotion,animateTransform,set are intentionally omitted.
    // They can potentially allow for arbitrary javascript to be executed. See #11290
    var svgElements = makeMap("circle,defs,desc,ellipse,font-face,font-face-name,font-face-src,g,glyph," +
            "hkern,image,linearGradient,line,marker,metadata,missing-glyph,mpath,path,polygon,polyline," +
            "radialGradient,rect,stop,svg,switch,text,title,tspan,use");

    // Special Elements (can contain anything)
    var specialElements = makeMap("script,style");

    var validElements = angular.extend({},
                                       voidElements,
                                       blockElements,
                                       inlineElements,
                                       optionalEndTagElements,
                                       svgElements);

    //Attributes that have href and hence need to be sanitized
    var uriAttrs = makeMap("background,cite,href,longdesc,src,usemap,xlink:href");

    var htmlAttrs = makeMap('abbr,align,alt,axis,bgcolor,border,cellpadding,cellspacing,class,clear,' +
        'color,cols,colspan,compact,coords,dir,face,headers,height,hreflang,hspace,' +
        'ismap,lang,language,nohref,nowrap,rel,rev,rows,rowspan,rules,' +
        'scope,scrolling,shape,size,span,start,summary,tabindex,target,title,type,' +
        'valign,value,vspace,width');

    // SVG attributes (without "id" and "name" attributes)
    // https://wiki.whatwg.org/wiki/Sanitization_rules#svg_Attributes
    var svgAttrs = makeMap('accent-height,accumulate,additive,alphabetic,arabic-form,ascent,' +
        'baseProfile,bbox,begin,by,calcMode,cap-height,class,color,color-rendering,content,' +
        'cx,cy,d,dx,dy,descent,display,dur,end,fill,fill-rule,font-family,font-size,font-stretch,' +
        'font-style,font-variant,font-weight,from,fx,fy,g1,g2,glyph-name,gradientUnits,hanging,' +
        'height,horiz-adv-x,horiz-origin-x,ideographic,k,keyPoints,keySplines,keyTimes,lang,' +
        'marker-end,marker-mid,marker-start,markerHeight,markerUnits,markerWidth,mathematical,' +
        'max,min,offset,opacity,orient,origin,overline-position,overline-thickness,panose-1,' +
        'path,pathLength,points,preserveAspectRatio,r,refX,refY,repeatCount,repeatDur,' +
        'requiredExtensions,requiredFeatures,restart,rotate,rx,ry,slope,stemh,stemv,stop-color,' +
        'stop-opacity,strikethrough-position,strikethrough-thickness,stroke,stroke-dasharray,' +
        'stroke-dashoffset,stroke-linecap,stroke-linejoin,stroke-miterlimit,stroke-opacity,' +
        'stroke-width,systemLanguage,target,text-anchor,to,transform,type,u1,u2,underline-position,' +
        'underline-thickness,unicode,unicode-range,units-per-em,values,version,viewBox,visibility,' +
        'width,widths,x,x-height,x1,x2,xlink:actuate,xlink:arcrole,xlink:role,xlink:show,xlink:title,' +
        'xlink:type,xml:base,xml:lang,xml:space,xmlns,xmlns:xlink,y,y1,y2,zoomAndPan', true);

    var validAttrs = angular.extend({},
                                    uriAttrs,
                                    svgAttrs,
                                    htmlAttrs);

    function makeMap(str, lowercaseKeys) {
        var obj = {}, items = str.split(','), i;
        for (i = 0; i < items.length; i++) {
            obj[lowercaseKeys ? angular.lowercase(items[i]) : items[i]] = true;
        }
        return obj;
    }


    /**
     * @example
     * htmlParser(htmlString, {
     *     start: function(tag, attrs, unary) {},
     *     end: function(tag) {},
     *     chars: function(text) {},
     *     comment: function(text) {}
     * });
     *
     * @param {string} html string
     * @param {object} handler
     */
    function htmlParser(html, handler) {
        if (typeof html !== 'string') {
            if (html === null || typeof html === 'undefined') {
                html = '';
            } else {
                html = '' + html;
            }
        }
        var index, chars, match, stack = [], last = html, text;
        stack.last = function () { return stack[stack.length - 1]; };

        while (html) {
            text = '';
            chars = true;

            // Make sure we're not in a script or style element
            if (!stack.last() || !specialElements[stack.last()]) {

                // Comment
                if (html.indexOf("<!--") === 0) {
                    // comments containing -- are not allowed unless they terminate the comment
                    index = html.indexOf("--", 4);

                    if (index >= 0 && html.lastIndexOf("-->", index) === index) {
                        if (handler.comment) handler.comment(html.substring(4, index));
                        html = html.substring(index + 3);
                        chars = false;
                    }
                    // DOCTYPE
                } else if (DOCTYPE_REGEXP.test(html)) {
                    match = html.match(DOCTYPE_REGEXP);

                    if (match) {
                        html = html.replace(match[0], '');
                        chars = false;
                    }
                    // end tag
                } else if (BEGING_END_TAGE_REGEXP.test(html)) {
                    match = html.match(END_TAG_REGEXP);

                    if (match) {
                        html = html.substring(match[0].length);
                        match[0].replace(END_TAG_REGEXP, parseEndTag);
                        chars = false;
                    }

                    // start tag
                } else if (BEGIN_TAG_REGEXP.test(html)) {
                    match = html.match(START_TAG_REGEXP);

                    if (match) {
                        // We only have a valid start-tag if there is a '>'.
                        if (match[4]) {
                            html = html.substring(match[0].length);
                            match[0].replace(START_TAG_REGEXP, parseStartTag);
                        }
                        chars = false;
                    } else {
                        // no ending tag found --- this piece should be encoded as an entity.
                        text += '<';
                        html = html.substring(1);
                    }
                }

                if (chars) {
                    index = html.indexOf("<");

                    text += index < 0 ? html : html.substring(0, index);
                    html = index < 0 ? "" : html.substring(index);

                    if (handler.chars) handler.chars(decodeEntities(text));
                }

            } else {
                // IE versions 9 and 10 do not understand the regex '[^]', so using a workaround with [\W\w].
                html = html.replace(new RegExp("([\\W\\w]*)<\\s*\\/\\s*" + stack.last() + "[^>]*>", 'i'),
                  function (all, text) {
                      text = text.replace(COMMENT_REGEXP, "$1").replace(CDATA_REGEXP, "$1");

                      if (handler.chars) handler.chars(decodeEntities(text));

                      return "";
                  });

                parseEndTag("", stack.last());
            }

            if (html == last) {
                throw $sanitizeMinErr('badparse', "The sanitizer was unable to parse the following block " +
                                                  "of html: {0}", html);
            }
            last = html;
        }

        // Clean up any remaining tags
        parseEndTag();

        function parseStartTag(tag, tagName, rest, unary) {
            tagName = angular.lowercase(tagName);
            if (blockElements[tagName]) {
                while (stack.last() && inlineElements[stack.last()]) {
                    parseEndTag("", stack.last());
                }
            }

            if (optionalEndTagElements[tagName] && stack.last() == tagName) {
                parseEndTag("", tagName);
            }

            unary = voidElements[tagName] || !!unary;

            if (!unary) {
                stack.push(tagName);
            }

            var attrs = {};

            rest.replace(ATTR_REGEXP,
              function (match, name, doubleQuotedValue, singleQuotedValue, unquotedValue) {
                  var value = doubleQuotedValue
                    || singleQuotedValue
                    || unquotedValue
                    || '';

                  attrs[name] = decodeEntities(value);
              });
            if (handler.start) handler.start(tagName, attrs, unary);
        }

        function parseEndTag(tag, tagName) {
            var pos = 0, i;
            tagName = angular.lowercase(tagName);
            if (tagName) {
                // Find the closest opened tag of the same type
                for (pos = stack.length - 1; pos >= 0; pos--) {
                    if (stack[pos] == tagName) break;
                }
            }

            if (pos >= 0) {
                // Close all the open elements, up the stack
                for (i = stack.length - 1; i >= pos; i--)
                    if (handler.end) handler.end(stack[i]);

                // Remove the open elements from the stack
                stack.length = pos;
            }
        }
    }

    var hiddenPre = document.createElement("pre");
    /**
     * decodes all entities into regular string
     * @param value
     * @returns {string} A string with decoded entities.
     */
    function decodeEntities(value) {
        if (!value) { return ''; }

        hiddenPre.innerHTML = value.replace(/</g, "&lt;");
        // innerText depends on styling as it doesn't display hidden elements.
        // Therefore, it's better to use textContent not to cause unnecessary reflows.
        return hiddenPre.textContent;
    }

    /**
     * Escapes all potentially dangerous characters, so that the
     * resulting string can be safely inserted into attribute or
     * element text.
     * @param value
     * @returns {string} escaped text
     */
    function encodeEntities(value) {
        return value.
          replace(/&/g, '&amp;').
          replace(SURROGATE_PAIR_REGEXP, function (value) {
              var hi = value.charCodeAt(0);
              var low = value.charCodeAt(1);
              return '&#' + (((hi - 0xD800) * 0x400) + (low - 0xDC00) + 0x10000) + ';';
          }).
          replace(NON_ALPHANUMERIC_REGEXP, function (value) {
              return '&#' + value.charCodeAt(0) + ';';
          }).
          replace(/</g, '&lt;').
          replace(/>/g, '&gt;');
    }

    /**
     * create an HTML/XML writer which writes to buffer
     * @param {Array} buf use buf.jain('') to get out sanitized html string
     * @returns {object} in the form of {
     *     start: function(tag, attrs, unary) {},
     *     end: function(tag) {},
     *     chars: function(text) {},
     *     comment: function(text) {}
     * }
     */
    function htmlSanitizeWriter(buf, uriValidator) {
        var ignore = false;
        var out = angular.bind(buf, buf.push);
        return {
            start: function (tag, attrs, unary) {
                tag = angular.lowercase(tag);
                if (!ignore && specialElements[tag]) {
                    ignore = tag;
                }
                if (!ignore && validElements[tag] === true) {
                    out('<');
                    out(tag);
                    angular.forEach(attrs, function (value, key) {
                        var lkey = angular.lowercase(key);
                        var isImage = (tag === 'img' && lkey === 'src') || (lkey === 'background');
                        if (validAttrs[lkey] === true &&
                          (uriAttrs[lkey] !== true || uriValidator(value, isImage))) {
                            out(' ');
                            out(key);
                            out('="');
                            out(encodeEntities(value));
                            out('"');
                        }
                    });
                    out(unary ? '/>' : '>');
                }
            },
            end: function (tag) {
                tag = angular.lowercase(tag);
                if (!ignore && validElements[tag] === true) {
                    out('</');
                    out(tag);
                    out('>');
                }
                if (tag == ignore) {
                    ignore = false;
                }
            },
            chars: function (chars) {
                if (!ignore) {
                    out(encodeEntities(chars));
                }
            }
        };
    }


    // define ngSanitize module and register $sanitize service
    angular.module('ngSanitize', []).provider('$sanitize', $SanitizeProvider);

    /* global sanitizeText: false */

    /**
     * @ngdoc filter
     * @name linky
     * @kind function
     *
     * @description
     * Finds links in text input and turns them into html links. Supports http/https/ftp/mailto and
     * plain email address links.
     *
     * Requires the {@link ngSanitize `ngSanitize`} module to be installed.
     *
     * @param {string} text Input text.
     * @param {string} target Window (_blank|_self|_parent|_top) or named frame to open links in.
     * @returns {string} Html-linkified text.
     *
     * @usage
       <span ng-bind-html="linky_expression | linky"></span>
     *
     * @example
       <example module="linkyExample" deps="angular-sanitize.js">
         <file name="index.html">
           <script>
             angular.module('linkyExample', ['ngSanitize'])
               .controller('ExampleController', ['$scope', function($scope) {
                 $scope.snippet =
                   'Pretty text with some links:\n'+
                   'http://angularjs.org/,\n'+
                   'mailto:us@somewhere.org,\n'+
                   'another@somewhere.org,\n'+
                   'and one more: ftp://127.0.0.1/.';
                 $scope.snippetWithTarget = 'http://angularjs.org/';
               }]);
           </script>
           <div ng-controller="ExampleController">
           Snippet: <textarea ng-model="snippet" cols="60" rows="3"></textarea>
           <table>
             <tr>
               <td>Filter</td>
               <td>Source</td>
               <td>Rendered</td>
             </tr>
             <tr id="linky-filter">
               <td>linky filter</td>
               <td>
                 <pre>&lt;div ng-bind-html="snippet | linky"&gt;<br>&lt;/div&gt;</pre>
               </td>
               <td>
                 <div ng-bind-html="snippet | linky"></div>
               </td>
             </tr>
             <tr id="linky-target">
              <td>linky target</td>
              <td>
                <pre>&lt;div ng-bind-html="snippetWithTarget | linky:'_blank'"&gt;<br>&lt;/div&gt;</pre>
              </td>
              <td>
                <div ng-bind-html="snippetWithTarget | linky:'_blank'"></div>
              </td>
             </tr>
             <tr id="escaped-html">
               <td>no filter</td>
               <td><pre>&lt;div ng-bind="snippet"&gt;<br>&lt;/div&gt;</pre></td>
               <td><div ng-bind="snippet"></div></td>
             </tr>
           </table>
         </file>
         <file name="protractor.js" type="protractor">
           it('should linkify the snippet with urls', function() {
             expect(element(by.id('linky-filter')).element(by.binding('snippet | linky')).getText()).
                 toBe('Pretty text with some links: http://angularjs.org/, us@somewhere.org, ' +
                      'another@somewhere.org, and one more: ftp://127.0.0.1/.');
             expect(element.all(by.css('#linky-filter a')).count()).toEqual(4);
           });
    
           it('should not linkify snippet without the linky filter', function() {
             expect(element(by.id('escaped-html')).element(by.binding('snippet')).getText()).
                 toBe('Pretty text with some links: http://angularjs.org/, mailto:us@somewhere.org, ' +
                      'another@somewhere.org, and one more: ftp://127.0.0.1/.');
             expect(element.all(by.css('#escaped-html a')).count()).toEqual(0);
           });
    
           it('should update', function() {
             element(by.model('snippet')).clear();
             element(by.model('snippet')).sendKeys('new http://link.');
             expect(element(by.id('linky-filter')).element(by.binding('snippet | linky')).getText()).
                 toBe('new http://link.');
             expect(element.all(by.css('#linky-filter a')).count()).toEqual(1);
             expect(element(by.id('escaped-html')).element(by.binding('snippet')).getText())
                 .toBe('new http://link.');
           });
    
           it('should work with the target property', function() {
            expect(element(by.id('linky-target')).
                element(by.binding("snippetWithTarget | linky:'_blank'")).getText()).
                toBe('http://angularjs.org/');
            expect(element(by.css('#linky-target a')).getAttribute('target')).toEqual('_blank');
           });
         </file>
       </example>
     */
    angular.module('ngSanitize').filter('linky', ['$sanitize', function ($sanitize) {
        var LINKY_URL_REGEXP =
              /((ftp|https?):\/\/|(www\.)|(mailto:)?[A-Za-z0-9._%+-]+@)\S*[^\s.;,(){}<>"\u201d\u2019]/i,
            MAILTO_REGEXP = /^mailto:/i;

        return function (text, target) {
            if (!text) return text;
            var match;
            var raw = text;
            var html = [];
            var url;
            var i;
            while ((match = raw.match(LINKY_URL_REGEXP))) {
                // We can not end in these as they are sometimes found at the end of the sentence
                url = match[0];
                // if we did not match ftp/http/www/mailto then assume mailto
                if (!match[2] && !match[4]) {
                    url = (match[3] ? 'http://' : 'mailto:') + url;
                }
                i = match.index;
                addText(raw.substr(0, i));
                addLink(url, match[0].replace(MAILTO_REGEXP, ''));
                raw = raw.substring(i + match[0].length);
            }
            addText(raw);
            return $sanitize(html.join(''));

            function addText(text) {
                if (!text) {
                    return;
                }
                html.push(sanitizeText(text));
            }

            function addLink(url, text) {
                html.push('<a ');
                if (angular.isDefined(target)) {
                    html.push('target="',
                              target,
                              '" ');
                }
                html.push('href="',
                          url.replace(/"/g, '&quot;'),
                          '">');
                addText(text);
                html.push('</a>');
            }
        };
    }]);


})(window, window.angular);;
'use strict';
angular.module("ngLocale", [], ["$provide", function ($provide) {
    var PLURAL_CATEGORY = { ZERO: "zero", ONE: "one", TWO: "two", FEW: "few", MANY: "many", OTHER: "other" };
    function getDecimals(n) {
        n = n + '';
        var i = n.indexOf('.');
        return (i == -1) ? 0 : n.length - i - 1;
    }

    function getVF(n, opt_precision) {
        var v = opt_precision;

        if (undefined === v) {
            v = Math.min(getDecimals(n), 3);
        }

        var base = Math.pow(10, v);
        var f = ((n * base) | 0) % base;
        return { v: v, f: f };
    }

    $provide.value("$locale", {
        "DATETIME_FORMATS": {
            "AMPMS": [
              "vorm.",
              "nachm."
            ],
            "DAY": [
              "Sonntag",
              "Montag",
              "Dienstag",
              "Mittwoch",
              "Donnerstag",
              "Freitag",
              "Samstag"
            ],
            "ERANAMES": [
              "v. Chr.",
              "n. Chr."
            ],
            "ERAS": [
              "v. Chr.",
              "n. Chr."
            ],
            "FIRSTDAYOFWEEK": 1,
            "MONTH": [
              "Januar",
              "Februar",
              "M\u00e4rz",
              "April",
              "Mai",
              "Juni",
              "Juli",
              "August",
              "September",
              "Oktober",
              "November",
              "Dezember"
            ],
            "SHORTDAY": [
              "So.",
              "Mo.",
              "Di.",
              "Mi.",
              "Do.",
              "Fr.",
              "Sa."
            ],
            "SHORTMONTH": [
              "Jan.",
              "Feb.",
              "M\u00e4rz",
              "Apr.",
              "Mai",
              "Juni",
              "Juli",
              "Aug.",
              "Sep.",
              "Okt.",
              "Nov.",
              "Dez."
            ],
            "WEEKENDRANGE": [
              5,
              6
            ],
            "fullDate": "EEEE, d. MMMM y",
            "longDate": "d. MMMM y",
            "medium": "dd.MM.y HH:mm:ss",
            "mediumDate": "dd.MM.y",
            "mediumTime": "HH:mm:ss",
            "short": "dd.MM.yy HH:mm",
            "shortDate": "dd.MM.yy",
            "shortTime": "HH:mm"
        },
        "NUMBER_FORMATS": {
            "CURRENCY_SYM": "CHF",
            "DECIMAL_SEP": ".",
            "GROUP_SEP": "'",
            "PATTERNS": [
              {
                  "gSize": 3,
                  "lgSize": 3,
                  "maxFrac": 3,
                  "minFrac": 0,
                  "minInt": 1,
                  "negPre": "-",
                  "negSuf": "",
                  "posPre": "",
                  "posSuf": ""
              },
              {
                  "gSize": 3,
                  "lgSize": 3,
                  "maxFrac": 2,
                  "minFrac": 2,
                  "minInt": 1,
                  "negPre": "\u00a4-",
                  "negSuf": "",
                  "posPre": "\u00a4\u00a0",
                  "posSuf": ""
              }
            ]
        },
        "id": "de-ch",
        "pluralCat": function (n, opt_precision) { var i = n | 0; var vf = getVF(n, opt_precision); if (i == 1 && vf.v == 0) { return PLURAL_CATEGORY.ONE; } return PLURAL_CATEGORY.OTHER; }
    });
}]);;
/**
 * @license AngularJS v1.4.8
 * (c) 2010-2015 Google, Inc. http://angularjs.org
 * License: MIT
 */
(function (window, angular, undefined) {
    'use strict';

    var $resourceMinErr = angular.$$minErr('$resource');

    // Helper functions and regex to lookup a dotted path on an object
    // stopping at undefined/null.  The path must be composed of ASCII
    // identifiers (just like $parse)
    var MEMBER_NAME_REGEX = /^(\.[a-zA-Z_$@][0-9a-zA-Z_$@]*)+$/;

    function isValidDottedPath(path) {
        return (path != null && path !== '' && path !== 'hasOwnProperty' &&
            MEMBER_NAME_REGEX.test('.' + path));
    }

    function lookupDottedPath(obj, path) {
        if (!isValidDottedPath(path)) {
            throw $resourceMinErr('badmember', 'Dotted member path "@{0}" is invalid.', path);
        }
        var keys = path.split('.');
        for (var i = 0, ii = keys.length; i < ii && angular.isDefined(obj) ; i++) {
            var key = keys[i];
            obj = (obj !== null) ? obj[key] : undefined;
        }
        return obj;
    }

    /**
     * Create a shallow copy of an object and clear other fields from the destination
     */
    function shallowClearAndCopy(src, dst) {
        dst = dst || {};

        angular.forEach(dst, function (value, key) {
            delete dst[key];
        });

        for (var key in src) {
            if (src.hasOwnProperty(key) && !(key.charAt(0) === '$' && key.charAt(1) === '$')) {
                dst[key] = src[key];
            }
        }

        return dst;
    }

    /**
     * @ngdoc module
     * @name ngResource
     * @description
     *
     * # ngResource
     *
     * The `ngResource` module provides interaction support with RESTful services
     * via the $resource service.
     *
     *
     * <div doc-module-components="ngResource"></div>
     *
     * See {@link ngResource.$resource `$resource`} for usage.
     */

    /**
     * @ngdoc service
     * @name $resource
     * @requires $http
     *
     * @description
     * A factory which creates a resource object that lets you interact with
     * [RESTful](http://en.wikipedia.org/wiki/Representational_State_Transfer) server-side data sources.
     *
     * The returned resource object has action methods which provide high-level behaviors without
     * the need to interact with the low level {@link ng.$http $http} service.
     *
     * Requires the {@link ngResource `ngResource`} module to be installed.
     *
     * By default, trailing slashes will be stripped from the calculated URLs,
     * which can pose problems with server backends that do not expect that
     * behavior.  This can be disabled by configuring the `$resourceProvider` like
     * this:
     *
     * ```js
         app.config(['$resourceProvider', function($resourceProvider) {
           // Don't strip trailing slashes from calculated URLs
           $resourceProvider.defaults.stripTrailingSlashes = false;
         }]);
     * ```
     *
     * @param {string} url A parameterized URL template with parameters prefixed by `:` as in
     *   `/user/:username`. If you are using a URL with a port number (e.g.
     *   `http://example.com:8080/api`), it will be respected.
     *
     *   If you are using a url with a suffix, just add the suffix, like this:
     *   `$resource('http://example.com/resource.json')` or `$resource('http://example.com/:id.json')`
     *   or even `$resource('http://example.com/resource/:resource_id.:format')`
     *   If the parameter before the suffix is empty, :resource_id in this case, then the `/.` will be
     *   collapsed down to a single `.`.  If you need this sequence to appear and not collapse then you
     *   can escape it with `/\.`.
     *
     * @param {Object=} paramDefaults Default values for `url` parameters. These can be overridden in
     *   `actions` methods. If any of the parameter value is a function, it will be executed every time
     *   when a param value needs to be obtained for a request (unless the param was overridden).
     *
     *   Each key value in the parameter object is first bound to url template if present and then any
     *   excess keys are appended to the url search query after the `?`.
     *
     *   Given a template `/path/:verb` and parameter `{verb:'greet', salutation:'Hello'}` results in
     *   URL `/path/greet?salutation=Hello`.
     *
     *   If the parameter value is prefixed with `@` then the value for that parameter will be extracted
     *   from the corresponding property on the `data` object (provided when calling an action method).  For
     *   example, if the `defaultParam` object is `{someParam: '@someProp'}` then the value of `someParam`
     *   will be `data.someProp`.
     *
     * @param {Object.<Object>=} actions Hash with declaration of custom actions that should extend
     *   the default set of resource actions. The declaration should be created in the format of {@link
     *   ng.$http#usage $http.config}:
     *
     *       {action1: {method:?, params:?, isArray:?, headers:?, ...},
     *        action2: {method:?, params:?, isArray:?, headers:?, ...},
     *        ...}
     *
     *   Where:
     *
     *   - **`action`** â€“ {string} â€“ The name of action. This name becomes the name of the method on
     *     your resource object.
     *   - **`method`** â€“ {string} â€“ Case insensitive HTTP method (e.g. `GET`, `POST`, `PUT`,
     *     `DELETE`, `JSONP`, etc).
     *   - **`params`** â€“ {Object=} â€“ Optional set of pre-bound parameters for this action. If any of
     *     the parameter value is a function, it will be executed every time when a param value needs to
     *     be obtained for a request (unless the param was overridden).
     *   - **`url`** â€“ {string} â€“ action specific `url` override. The url templating is supported just
     *     like for the resource-level urls.
     *   - **`isArray`** â€“ {boolean=} â€“ If true then the returned object for this action is an array,
     *     see `returns` section.
     *   - **`transformRequest`** â€“
     *     `{function(data, headersGetter)|Array.<function(data, headersGetter)>}` â€“
     *     transform function or an array of such functions. The transform function takes the http
     *     request body and headers and returns its transformed (typically serialized) version.
     *     By default, transformRequest will contain one function that checks if the request data is
     *     an object and serializes to using `angular.toJson`. To prevent this behavior, set
     *     `transformRequest` to an empty array: `transformRequest: []`
     *   - **`transformResponse`** â€“
     *     `{function(data, headersGetter)|Array.<function(data, headersGetter)>}` â€“
     *     transform function or an array of such functions. The transform function takes the http
     *     response body and headers and returns its transformed (typically deserialized) version.
     *     By default, transformResponse will contain one function that checks if the response looks like
     *     a JSON string and deserializes it using `angular.fromJson`. To prevent this behavior, set
     *     `transformResponse` to an empty array: `transformResponse: []`
     *   - **`cache`** â€“ `{boolean|Cache}` â€“ If true, a default $http cache will be used to cache the
     *     GET request, otherwise if a cache instance built with
     *     {@link ng.$cacheFactory $cacheFactory}, this cache will be used for
     *     caching.
     *   - **`timeout`** â€“ `{number|Promise}` â€“ timeout in milliseconds, or {@link ng.$q promise} that
     *     should abort the request when resolved.
     *   - **`withCredentials`** - `{boolean}` - whether to set the `withCredentials` flag on the
     *     XHR object. See
     *     [requests with credentials](https://developer.mozilla.org/en/http_access_control#section_5)
     *     for more information.
     *   - **`responseType`** - `{string}` - see
     *     [requestType](https://developer.mozilla.org/en-US/docs/DOM/XMLHttpRequest#responseType).
     *   - **`interceptor`** - `{Object=}` - The interceptor object has two optional methods -
     *     `response` and `responseError`. Both `response` and `responseError` interceptors get called
     *     with `http response` object. See {@link ng.$http $http interceptors}.
     *
     * @param {Object} options Hash with custom settings that should extend the
     *   default `$resourceProvider` behavior.  The only supported option is
     *
     *   Where:
     *
     *   - **`stripTrailingSlashes`** â€“ {boolean} â€“ If true then the trailing
     *   slashes from any calculated URL will be stripped. (Defaults to true.)
     *
     * @returns {Object} A resource "class" object with methods for the default set of resource actions
     *   optionally extended with custom `actions`. The default set contains these actions:
     *   ```js
     *   { 'get':    {method:'GET'},
     *     'save':   {method:'POST'},
     *     'query':  {method:'GET', isArray:true},
     *     'remove': {method:'DELETE'},
     *     'delete': {method:'DELETE'} };
     *   ```
     *
     *   Calling these methods invoke an {@link ng.$http} with the specified http method,
     *   destination and parameters. When the data is returned from the server then the object is an
     *   instance of the resource class. The actions `save`, `remove` and `delete` are available on it
     *   as  methods with the `$` prefix. This allows you to easily perform CRUD operations (create,
     *   read, update, delete) on server-side data like this:
     *   ```js
     *   var User = $resource('/user/:userId', {userId:'@id'});
     *   var user = User.get({userId:123}, function() {
     *     user.abc = true;
     *     user.$save();
     *   });
     *   ```
     *
     *   It is important to realize that invoking a $resource object method immediately returns an
     *   empty reference (object or array depending on `isArray`). Once the data is returned from the
     *   server the existing reference is populated with the actual data. This is a useful trick since
     *   usually the resource is assigned to a model which is then rendered by the view. Having an empty
     *   object results in no rendering, once the data arrives from the server then the object is
     *   populated with the data and the view automatically re-renders itself showing the new data. This
     *   means that in most cases one never has to write a callback function for the action methods.
     *
     *   The action methods on the class object or instance object can be invoked with the following
     *   parameters:
     *
     *   - HTTP GET "class" actions: `Resource.action([parameters], [success], [error])`
     *   - non-GET "class" actions: `Resource.action([parameters], postData, [success], [error])`
     *   - non-GET instance actions:  `instance.$action([parameters], [success], [error])`
     *
     *
     *   Success callback is called with (value, responseHeaders) arguments, where the value is
     *   the populated resource instance or collection object. The error callback is called
     *   with (httpResponse) argument.
     *
     *   Class actions return empty instance (with additional properties below).
     *   Instance actions return promise of the action.
     *
     *   The Resource instances and collection have these additional properties:
     *
     *   - `$promise`: the {@link ng.$q promise} of the original server interaction that created this
     *     instance or collection.
     *
     *     On success, the promise is resolved with the same resource instance or collection object,
     *     updated with data from server. This makes it easy to use in
     *     {@link ngRoute.$routeProvider resolve section of $routeProvider.when()} to defer view
     *     rendering until the resource(s) are loaded.
     *
     *     On failure, the promise is resolved with the {@link ng.$http http response} object, without
     *     the `resource` property.
     *
     *     If an interceptor object was provided, the promise will instead be resolved with the value
     *     returned by the interceptor.
     *
     *   - `$resolved`: `true` after first server interaction is completed (either with success or
     *      rejection), `false` before that. Knowing if the Resource has been resolved is useful in
     *      data-binding.
     *
     * @example
     *
     * # Credit card resource
     *
     * ```js
         // Define CreditCard class
         var CreditCard = $resource('/user/:userId/card/:cardId',
          {userId:123, cardId:'@id'}, {
           charge: {method:'POST', params:{charge:true}}
          });
    
         // We can retrieve a collection from the server
         var cards = CreditCard.query(function() {
           // GET: /user/123/card
           // server returns: [ {id:456, number:'1234', name:'Smith'} ];
    
           var card = cards[0];
           // each item is an instance of CreditCard
           expect(card instanceof CreditCard).toEqual(true);
           card.name = "J. Smith";
           // non GET methods are mapped onto the instances
           card.$save();
           // POST: /user/123/card/456 {id:456, number:'1234', name:'J. Smith'}
           // server returns: {id:456, number:'1234', name: 'J. Smith'};
    
           // our custom method is mapped as well.
           card.$charge({amount:9.99});
           // POST: /user/123/card/456?amount=9.99&charge=true {id:456, number:'1234', name:'J. Smith'}
         });
    
         // we can create an instance as well
         var newCard = new CreditCard({number:'0123'});
         newCard.name = "Mike Smith";
         newCard.$save();
         // POST: /user/123/card {number:'0123', name:'Mike Smith'}
         // server returns: {id:789, number:'0123', name: 'Mike Smith'};
         expect(newCard.id).toEqual(789);
     * ```
     *
     * The object returned from this function execution is a resource "class" which has "static" method
     * for each action in the definition.
     *
     * Calling these methods invoke `$http` on the `url` template with the given `method`, `params` and
     * `headers`.
     * When the data is returned from the server then the object is an instance of the resource type and
     * all of the non-GET methods are available with `$` prefix. This allows you to easily support CRUD
     * operations (create, read, update, delete) on server-side data.
    
       ```js
         var User = $resource('/user/:userId', {userId:'@id'});
         User.get({userId:123}, function(user) {
           user.abc = true;
           user.$save();
         });
       ```
     *
     * It's worth noting that the success callback for `get`, `query` and other methods gets passed
     * in the response that came from the server as well as $http header getter function, so one
     * could rewrite the above example and get access to http headers as:
     *
       ```js
         var User = $resource('/user/:userId', {userId:'@id'});
         User.get({userId:123}, function(u, getResponseHeaders){
           u.abc = true;
           u.$save(function(u, putResponseHeaders) {
             //u => saved user object
             //putResponseHeaders => $http header getter
           });
         });
       ```
     *
     * You can also access the raw `$http` promise via the `$promise` property on the object returned
     *
       ```
         var User = $resource('/user/:userId', {userId:'@id'});
         User.get({userId:123})
             .$promise.then(function(user) {
               $scope.user = user;
             });
       ```
    
     * # Creating a custom 'PUT' request
     * In this example we create a custom method on our resource to make a PUT request
     * ```js
     *    var app = angular.module('app', ['ngResource', 'ngRoute']);
     *
     *    // Some APIs expect a PUT request in the format URL/object/ID
     *    // Here we are creating an 'update' method
     *    app.factory('Notes', ['$resource', function($resource) {
     *    return $resource('/notes/:id', null,
     *        {
     *            'update': { method:'PUT' }
     *        });
     *    }]);
     *
     *    // In our controller we get the ID from the URL using ngRoute and $routeParams
     *    // We pass in $routeParams and our Notes factory along with $scope
     *    app.controller('NotesCtrl', ['$scope', '$routeParams', 'Notes',
                                          function($scope, $routeParams, Notes) {
     *    // First get a note object from the factory
     *    var note = Notes.get({ id:$routeParams.id });
     *    $id = note.id;
     *
     *    // Now call update passing in the ID first then the object you are updating
     *    Notes.update({ id:$id }, note);
     *
     *    // This will PUT /notes/ID with the note object in the request payload
     *    }]);
     * ```
     */
    angular.module('ngResource', ['ng']).
      provider('$resource', function () {
          var PROTOCOL_AND_DOMAIN_REGEX = /^https?:\/\/[^\/]*/;
          var provider = this;

          this.defaults = {
              // Strip slashes by default
              stripTrailingSlashes: true,

              // Default actions configuration
              actions: {
                  'get': { method: 'GET' },
                  'save': { method: 'POST' },
                  'query': { method: 'GET', isArray: true },
                  'remove': { method: 'DELETE' },
                  'delete': { method: 'DELETE' }
              }
          };

          this.$get = ['$http', '$q', function ($http, $q) {

              var noop = angular.noop,
                forEach = angular.forEach,
                extend = angular.extend,
                copy = angular.copy,
                isFunction = angular.isFunction;

              /**
               * We need our custom method because encodeURIComponent is too aggressive and doesn't follow
               * http://www.ietf.org/rfc/rfc3986.txt with regards to the character set
               * (pchar) allowed in path segments:
               *    segment       = *pchar
               *    pchar         = unreserved / pct-encoded / sub-delims / ":" / "@"
               *    pct-encoded   = "%" HEXDIG HEXDIG
               *    unreserved    = ALPHA / DIGIT / "-" / "." / "_" / "~"
               *    sub-delims    = "!" / "$" / "&" / "'" / "(" / ")"
               *                     / "*" / "+" / "," / ";" / "="
               */
              function encodeUriSegment(val) {
                  return encodeUriQuery(val, true).
                    replace(/%26/gi, '&').
                    replace(/%3D/gi, '=').
                    replace(/%2B/gi, '+');
              }


              /**
               * This method is intended for encoding *key* or *value* parts of query component. We need a
               * custom method because encodeURIComponent is too aggressive and encodes stuff that doesn't
               * have to be encoded per http://tools.ietf.org/html/rfc3986:
               *    query       = *( pchar / "/" / "?" )
               *    pchar         = unreserved / pct-encoded / sub-delims / ":" / "@"
               *    unreserved    = ALPHA / DIGIT / "-" / "." / "_" / "~"
               *    pct-encoded   = "%" HEXDIG HEXDIG
               *    sub-delims    = "!" / "$" / "&" / "'" / "(" / ")"
               *                     / "*" / "+" / "," / ";" / "="
               */
              function encodeUriQuery(val, pctEncodeSpaces) {
                  return encodeURIComponent(val).
                    replace(/%40/gi, '@').
                    replace(/%3A/gi, ':').
                    replace(/%24/g, '$').
                    replace(/%2C/gi, ',').
                    replace(/%20/g, (pctEncodeSpaces ? '%20' : '+'));
              }

              function Route(template, defaults) {
                  this.template = template;
                  this.defaults = extend({}, provider.defaults, defaults);
                  this.urlParams = {};
              }

              Route.prototype = {
                  setUrlParams: function (config, params, actionUrl) {
                      var self = this,
                        url = actionUrl || self.template,
                        val,
                        encodedVal,
                        protocolAndDomain = '';

                      var urlParams = self.urlParams = {};
                      forEach(url.split(/\W/), function (param) {
                          if (param === 'hasOwnProperty') {
                              throw $resourceMinErr('badname', "hasOwnProperty is not a valid parameter name.");
                          }
                          if (!(new RegExp("^\\d+$").test(param)) && param &&
                            (new RegExp("(^|[^\\\\]):" + param + "(\\W|$)").test(url))) {
                              urlParams[param] = true;
                          }
                      });
                      url = url.replace(/\\:/g, ':');
                      url = url.replace(PROTOCOL_AND_DOMAIN_REGEX, function (match) {
                          protocolAndDomain = match;
                          return '';
                      });

                      params = params || {};
                      forEach(self.urlParams, function (_, urlParam) {
                          val = params.hasOwnProperty(urlParam) ? params[urlParam] : self.defaults[urlParam];
                          if (angular.isDefined(val) && val !== null) {
                              encodedVal = encodeUriSegment(val);
                              url = url.replace(new RegExp(":" + urlParam + "(\\W|$)", "g"), function (match, p1) {
                                  return encodedVal + p1;
                              });
                          } else {
                              url = url.replace(new RegExp("(\/?):" + urlParam + "(\\W|$)", "g"), function (match,
                                  leadingSlashes, tail) {
                                  if (tail.charAt(0) == '/') {
                                      return tail;
                                  } else {
                                      return leadingSlashes + tail;
                                  }
                              });
                          }
                      });

                      // strip trailing slashes and set the url (unless this behavior is specifically disabled)
                      if (self.defaults.stripTrailingSlashes) {
                          url = url.replace(/\/+$/, '') || '/';
                      }

                      // then replace collapse `/.` if found in the last URL path segment before the query
                      // E.g. `http://url.com/id./format?q=x` becomes `http://url.com/id.format?q=x`
                      url = url.replace(/\/\.(?=\w+($|\?))/, '.');
                      // replace escaped `/\.` with `/.`
                      config.url = protocolAndDomain + url.replace(/\/\\\./, '/.');


                      // set params - delegate param encoding to $http
                      forEach(params, function (value, key) {
                          if (!self.urlParams[key]) {
                              config.params = config.params || {};
                              config.params[key] = value;
                          }
                      });
                  }
              };


              function resourceFactory(url, paramDefaults, actions, options) {
                  var route = new Route(url, options);

                  actions = extend({}, provider.defaults.actions, actions);

                  function extractParams(data, actionParams) {
                      var ids = {};
                      actionParams = extend({}, paramDefaults, actionParams);
                      forEach(actionParams, function (value, key) {
                          if (isFunction(value)) { value = value(); }
                          ids[key] = value && value.charAt && value.charAt(0) == '@' ?
                            lookupDottedPath(data, value.substr(1)) : value;
                      });
                      return ids;
                  }

                  function defaultResponseInterceptor(response) {
                      return response.resource;
                  }

                  function Resource(value) {
                      shallowClearAndCopy(value || {}, this);
                  }

                  Resource.prototype.toJSON = function () {
                      var data = extend({}, this);
                      delete data.$promise;
                      delete data.$resolved;
                      return data;
                  };

                  forEach(actions, function (action, name) {
                      var hasBody = /^(POST|PUT|PATCH)$/i.test(action.method);

                      Resource[name] = function (a1, a2, a3, a4) {
                          var params = {}, data, success, error;

                          /* jshint -W086 */ /* (purposefully fall through case statements) */
                          switch (arguments.length) {
                              case 4:
                                  error = a4;
                                  success = a3;
                                  //fallthrough
                              case 3:
                              case 2:
                                  if (isFunction(a2)) {
                                      if (isFunction(a1)) {
                                          success = a1;
                                          error = a2;
                                          break;
                                      }

                                      success = a2;
                                      error = a3;
                                      //fallthrough
                                  } else {
                                      params = a1;
                                      data = a2;
                                      success = a3;
                                      break;
                                  }
                              case 1:
                                  if (isFunction(a1)) success = a1;
                                  else if (hasBody) data = a1;
                                  else params = a1;
                                  break;
                              case 0: break;
                              default:
                                  throw $resourceMinErr('badargs',
                                    "Expected up to 4 arguments [params, data, success, error], got {0} arguments",
                                    arguments.length);
                          }
                          /* jshint +W086 */ /* (purposefully fall through case statements) */

                          var isInstanceCall = this instanceof Resource;
                          var value = isInstanceCall ? data : (action.isArray ? [] : new Resource(data));
                          var httpConfig = {};
                          var responseInterceptor = action.interceptor && action.interceptor.response ||
                            defaultResponseInterceptor;
                          var responseErrorInterceptor = action.interceptor && action.interceptor.responseError ||
                            undefined;

                          forEach(action, function (value, key) {
                              switch (key) {
                                  default:
                                      httpConfig[key] = copy(value);
                                      break;
                                  case 'params':
                                  case 'isArray':
                                  case 'interceptor':
                                      break;
                                  case 'timeout':
                                      httpConfig[key] = value;
                                      break;
                              }
                          });

                          if (hasBody) httpConfig.data = data;
                          route.setUrlParams(httpConfig,
                            extend({}, extractParams(data, action.params || {}), params),
                            action.url);

                          var promise = $http(httpConfig).then(function (response) {
                              var data = response.data,
                                promise = value.$promise;

                              if (data) {
                                  // Need to convert action.isArray to boolean in case it is undefined
                                  // jshint -W018
                                  if (angular.isArray(data) !== (!!action.isArray)) {
                                      throw $resourceMinErr('badcfg',
                                          'Error in resource configuration for action `{0}`. Expected response to ' +
                                          'contain an {1} but got an {2} (Request: {3} {4})', name, action.isArray ? 'array' : 'object',
                                        angular.isArray(data) ? 'array' : 'object', httpConfig.method, httpConfig.url);
                                  }
                                  // jshint +W018
                                  if (action.isArray) {
                                      value.length = 0;
                                      forEach(data, function (item) {
                                          if (typeof item === "object") {
                                              value.push(new Resource(item));
                                          } else {
                                              // Valid JSON values may be string literals, and these should not be converted
                                              // into objects. These items will not have access to the Resource prototype
                                              // methods, but unfortunately there
                                              value.push(item);
                                          }
                                      });
                                  } else {
                                      shallowClearAndCopy(data, value);
                                      value.$promise = promise;
                                  }
                              }

                              value.$resolved = true;

                              response.resource = value;

                              return response;
                          }, function (response) {
                              value.$resolved = true;

                              (error || noop)(response);

                              return $q.reject(response);
                          });

                          promise = promise.then(
                            function (response) {
                                var value = responseInterceptor(response);
                                (success || noop)(value, response.headers);
                                return value;
                            },
                            responseErrorInterceptor);

                          if (!isInstanceCall) {
                              // we are creating instance / collection
                              // - set the initial promise
                              // - return the instance / collection
                              value.$promise = promise;
                              value.$resolved = false;

                              return value;
                          }

                          // instance call
                          return promise;
                      };


                      Resource.prototype['$' + name] = function (params, success, error) {
                          if (isFunction(params)) {
                              error = success; success = params; params = {};
                          }
                          var result = Resource[name].call(this, params, this, success, error);
                          return result.$promise || result;
                      };
                  });

                  Resource.bind = function (additionalParamDefaults) {
                      return resourceFactory(url, extend({}, paramDefaults, additionalParamDefaults), actions);
                  };

                  return Resource;
              }

              return resourceFactory;
          }];
      });


})(window, window.angular);;
/**
 * @license AngularJS v1.4.8
 * (c) 2010-2015 Google, Inc. http://angularjs.org
 * License: MIT
 */
(function (window, angular, undefined) {
    'use strict';

    /**
     * @ngdoc module
     * @name ngCookies
     * @description
     *
     * # ngCookies
     *
     * The `ngCookies` module provides a convenient wrapper for reading and writing browser cookies.
     *
     *
     * <div doc-module-components="ngCookies"></div>
     *
     * See {@link ngCookies.$cookies `$cookies`} for usage.
     */


    angular.module('ngCookies', ['ng']).
      /**
       * @ngdoc provider
       * @name $cookiesProvider
       * @description
       * Use `$cookiesProvider` to change the default behavior of the {@link ngCookies.$cookies $cookies} service.
       * */
       provider('$cookies', [function $CookiesProvider() {
           /**
            * @ngdoc property
            * @name $cookiesProvider#defaults
            * @description
            *
            * Object containing default options to pass when setting cookies.
            *
            * The object may have following properties:
            *
            * - **path** - `{string}` - The cookie will be available only for this path and its
            *   sub-paths. By default, this would be the URL that appears in your base tag.
            * - **domain** - `{string}` - The cookie will be available only for this domain and
            *   its sub-domains. For obvious security reasons the user agent will not accept the
            *   cookie if the current domain is not a sub domain or equals to the requested domain.
            * - **expires** - `{string|Date}` - String of the form "Wdy, DD Mon YYYY HH:MM:SS GMT"
            *   or a Date object indicating the exact date/time this cookie will expire.
            * - **secure** - `{boolean}` - The cookie will be available only in secured connection.
            *
            * Note: by default the address that appears in your `<base>` tag will be used as path.
            * This is important so that cookies will be visible for all routes in case html5mode is enabled
            *
            **/
           var defaults = this.defaults = {};

           function calcOptions(options) {
               return options ? angular.extend({}, defaults, options) : defaults;
           }

           /**
            * @ngdoc service
            * @name $cookies
            *
            * @description
            * Provides read/write access to browser's cookies.
            *
            * <div class="alert alert-info">
            * Up until Angular 1.3, `$cookies` exposed properties that represented the
            * current browser cookie values. In version 1.4, this behavior has changed, and
            * `$cookies` now provides a standard api of getters, setters etc.
            * </div>
            *
            * Requires the {@link ngCookies `ngCookies`} module to be installed.
            *
            * @example
            *
            * ```js
            * angular.module('cookiesExample', ['ngCookies'])
            *   .controller('ExampleController', ['$cookies', function($cookies) {
            *     // Retrieving a cookie
            *     var favoriteCookie = $cookies.get('myFavorite');
            *     // Setting a cookie
            *     $cookies.put('myFavorite', 'oatmeal');
            *   }]);
            * ```
            */
           this.$get = ['$$cookieReader', '$$cookieWriter', function ($$cookieReader, $$cookieWriter) {
               return {
                   /**
                    * @ngdoc method
                    * @name $cookies#get
                    *
                    * @description
                    * Returns the value of given cookie key
                    *
                    * @param {string} key Id to use for lookup.
                    * @returns {string} Raw cookie value.
                    */
                   get: function (key) {
                       return $$cookieReader()[key];
                   },

                   /**
                    * @ngdoc method
                    * @name $cookies#getObject
                    *
                    * @description
                    * Returns the deserialized value of given cookie key
                    *
                    * @param {string} key Id to use for lookup.
                    * @returns {Object} Deserialized cookie value.
                    */
                   getObject: function (key) {
                       var value = this.get(key);
                       return value ? angular.fromJson(value) : value;
                   },

                   /**
                    * @ngdoc method
                    * @name $cookies#getAll
                    *
                    * @description
                    * Returns a key value object with all the cookies
                    *
                    * @returns {Object} All cookies
                    */
                   getAll: function () {
                       return $$cookieReader();
                   },

                   /**
                    * @ngdoc method
                    * @name $cookies#put
                    *
                    * @description
                    * Sets a value for given cookie key
                    *
                    * @param {string} key Id for the `value`.
                    * @param {string} value Raw value to be stored.
                    * @param {Object=} options Options object.
                    *    See {@link ngCookies.$cookiesProvider#defaults $cookiesProvider.defaults}
                    */
                   put: function (key, value, options) {
                       $$cookieWriter(key, value, calcOptions(options));
                   },

                   /**
                    * @ngdoc method
                    * @name $cookies#putObject
                    *
                    * @description
                    * Serializes and sets a value for given cookie key
                    *
                    * @param {string} key Id for the `value`.
                    * @param {Object} value Value to be stored.
                    * @param {Object=} options Options object.
                    *    See {@link ngCookies.$cookiesProvider#defaults $cookiesProvider.defaults}
                    */
                   putObject: function (key, value, options) {
                       this.put(key, angular.toJson(value), options);
                   },

                   /**
                    * @ngdoc method
                    * @name $cookies#remove
                    *
                    * @description
                    * Remove given cookie
                    *
                    * @param {string} key Id of the key-value pair to delete.
                    * @param {Object=} options Options object.
                    *    See {@link ngCookies.$cookiesProvider#defaults $cookiesProvider.defaults}
                    */
                   remove: function (key, options) {
                       $$cookieWriter(key, undefined, calcOptions(options));
                   }
               };
           }];
       }]);

    angular.module('ngCookies').
    /**
     * @ngdoc service
     * @name $cookieStore
     * @deprecated
     * @requires $cookies
     *
     * @description
     * Provides a key-value (string-object) storage, that is backed by session cookies.
     * Objects put or retrieved from this storage are automatically serialized or
     * deserialized by angular's toJson/fromJson.
     *
     * Requires the {@link ngCookies `ngCookies`} module to be installed.
     *
     * <div class="alert alert-danger">
     * **Note:** The $cookieStore service is **deprecated**.
     * Please use the {@link ngCookies.$cookies `$cookies`} service instead.
     * </div>
     *
     * @example
     *
     * ```js
     * angular.module('cookieStoreExample', ['ngCookies'])
     *   .controller('ExampleController', ['$cookieStore', function($cookieStore) {
     *     // Put cookie
     *     $cookieStore.put('myFavorite','oatmeal');
     *     // Get cookie
     *     var favoriteCookie = $cookieStore.get('myFavorite');
     *     // Removing a cookie
     *     $cookieStore.remove('myFavorite');
     *   }]);
     * ```
     */
     factory('$cookieStore', ['$cookies', function ($cookies) {

         return {
             /**
              * @ngdoc method
              * @name $cookieStore#get
              *
              * @description
              * Returns the value of given cookie key
              *
              * @param {string} key Id to use for lookup.
              * @returns {Object} Deserialized cookie value, undefined if the cookie does not exist.
              */
             get: function (key) {
                 return $cookies.getObject(key);
             },

             /**
              * @ngdoc method
              * @name $cookieStore#put
              *
              * @description
              * Sets a value for given cookie key
              *
              * @param {string} key Id for the `value`.
              * @param {Object} value Value to be stored.
              */
             put: function (key, value) {
                 $cookies.putObject(key, value);
             },

             /**
              * @ngdoc method
              * @name $cookieStore#remove
              *
              * @description
              * Remove given cookie
              *
              * @param {string} key Id of the key-value pair to delete.
              */
             remove: function (key) {
                 $cookies.remove(key);
             }
         };

     }]);

    /**
     * @name $$cookieWriter
     * @requires $document
     *
     * @description
     * This is a private service for writing cookies
     *
     * @param {string} name Cookie name
     * @param {string=} value Cookie value (if undefined, cookie will be deleted)
     * @param {Object=} options Object with options that need to be stored for the cookie.
     */
    function $$CookieWriter($document, $log, $browser) {
        var cookiePath = $browser.baseHref();
        var rawDocument = $document[0];

        function buildCookieString(name, value, options) {
            var path, expires;
            options = options || {};
            expires = options.expires;
            path = angular.isDefined(options.path) ? options.path : cookiePath;
            if (angular.isUndefined(value)) {
                expires = 'Thu, 01 Jan 1970 00:00:00 GMT';
                value = '';
            }
            if (angular.isString(expires)) {
                expires = new Date(expires);
            }

            var str = encodeURIComponent(name) + '=' + encodeURIComponent(value);
            str += path ? ';path=' + path : '';
            str += options.domain ? ';domain=' + options.domain : '';
            str += expires ? ';expires=' + expires.toUTCString() : '';
            str += options.secure ? ';secure' : '';

            // per http://www.ietf.org/rfc/rfc2109.txt browser must allow at minimum:
            // - 300 cookies
            // - 20 cookies per unique domain
            // - 4096 bytes per cookie
            var cookieLength = str.length + 1;
            if (cookieLength > 4096) {
                $log.warn("Cookie '" + name +
                  "' possibly not set or overflowed because it was too large (" +
                  cookieLength + " > 4096 bytes)!");
            }

            return str;
        }

        return function (name, value, options) {
            rawDocument.cookie = buildCookieString(name, value, options);
        };
    }

    $$CookieWriter.$inject = ['$document', '$log', '$browser'];

    angular.module('ngCookies').provider('$$cookieWriter', function $$CookieWriterProvider() {
        this.$get = $$CookieWriter;
    });


})(window, window.angular);;
/**
 * @license AngularJS v1.4.8
 * (c) 2010-2015 Google, Inc. http://angularjs.org
 * License: MIT
 */
(function (window, angular, undefined) {
    'use strict';

    /* jshint ignore:start */
    var noop = angular.noop;
    var extend = angular.extend;
    var jqLite = angular.element;
    var forEach = angular.forEach;
    var isArray = angular.isArray;
    var isString = angular.isString;
    var isObject = angular.isObject;
    var isUndefined = angular.isUndefined;
    var isDefined = angular.isDefined;
    var isFunction = angular.isFunction;
    var isElement = angular.isElement;

    var ELEMENT_NODE = 1;
    var COMMENT_NODE = 8;

    var ADD_CLASS_SUFFIX = '-add';
    var REMOVE_CLASS_SUFFIX = '-remove';
    var EVENT_CLASS_PREFIX = 'ng-';
    var ACTIVE_CLASS_SUFFIX = '-active';

    var NG_ANIMATE_CLASSNAME = 'ng-animate';
    var NG_ANIMATE_CHILDREN_DATA = '$$ngAnimateChildren';

    // Detect proper transitionend/animationend event names.
    var CSS_PREFIX = '', TRANSITION_PROP, TRANSITIONEND_EVENT, ANIMATION_PROP, ANIMATIONEND_EVENT;

    // If unprefixed events are not supported but webkit-prefixed are, use the latter.
    // Otherwise, just use W3C names, browsers not supporting them at all will just ignore them.
    // Note: Chrome implements `window.onwebkitanimationend` and doesn't implement `window.onanimationend`
    // but at the same time dispatches the `animationend` event and not `webkitAnimationEnd`.
    // Register both events in case `window.onanimationend` is not supported because of that,
    // do the same for `transitionend` as Safari is likely to exhibit similar behavior.
    // Also, the only modern browser that uses vendor prefixes for transitions/keyframes is webkit
    // therefore there is no reason to test anymore for other vendor prefixes:
    // http://caniuse.com/#search=transition
    if (isUndefined(window.ontransitionend) && isDefined(window.onwebkittransitionend)) {
        CSS_PREFIX = '-webkit-';
        TRANSITION_PROP = 'WebkitTransition';
        TRANSITIONEND_EVENT = 'webkitTransitionEnd transitionend';
    } else {
        TRANSITION_PROP = 'transition';
        TRANSITIONEND_EVENT = 'transitionend';
    }

    if (isUndefined(window.onanimationend) && isDefined(window.onwebkitanimationend)) {
        CSS_PREFIX = '-webkit-';
        ANIMATION_PROP = 'WebkitAnimation';
        ANIMATIONEND_EVENT = 'webkitAnimationEnd animationend';
    } else {
        ANIMATION_PROP = 'animation';
        ANIMATIONEND_EVENT = 'animationend';
    }

    var DURATION_KEY = 'Duration';
    var PROPERTY_KEY = 'Property';
    var DELAY_KEY = 'Delay';
    var TIMING_KEY = 'TimingFunction';
    var ANIMATION_ITERATION_COUNT_KEY = 'IterationCount';
    var ANIMATION_PLAYSTATE_KEY = 'PlayState';
    var SAFE_FAST_FORWARD_DURATION_VALUE = 9999;

    var ANIMATION_DELAY_PROP = ANIMATION_PROP + DELAY_KEY;
    var ANIMATION_DURATION_PROP = ANIMATION_PROP + DURATION_KEY;
    var TRANSITION_DELAY_PROP = TRANSITION_PROP + DELAY_KEY;
    var TRANSITION_DURATION_PROP = TRANSITION_PROP + DURATION_KEY;

    var isPromiseLike = function (p) {
        return p && p.then ? true : false;
    };

    function assertArg(arg, name, reason) {
        if (!arg) {
            throw ngMinErr('areq', "Argument '{0}' is {1}", (name || '?'), (reason || "required"));
        }
        return arg;
    }

    function mergeClasses(a, b) {
        if (!a && !b) return '';
        if (!a) return b;
        if (!b) return a;
        if (isArray(a)) a = a.join(' ');
        if (isArray(b)) b = b.join(' ');
        return a + ' ' + b;
    }

    function packageStyles(options) {
        var styles = {};
        if (options && (options.to || options.from)) {
            styles.to = options.to;
            styles.from = options.from;
        }
        return styles;
    }

    function pendClasses(classes, fix, isPrefix) {
        var className = '';
        classes = isArray(classes)
            ? classes
            : classes && isString(classes) && classes.length
                ? classes.split(/\s+/)
                : [];
        forEach(classes, function (klass, i) {
            if (klass && klass.length > 0) {
                className += (i > 0) ? ' ' : '';
                className += isPrefix ? fix + klass
                                      : klass + fix;
            }
        });
        return className;
    }

    function removeFromArray(arr, val) {
        var index = arr.indexOf(val);
        if (val >= 0) {
            arr.splice(index, 1);
        }
    }

    function stripCommentsFromElement(element) {
        if (element instanceof jqLite) {
            switch (element.length) {
                case 0:
                    return [];
                    break;

                case 1:
                    // there is no point of stripping anything if the element
                    // is the only element within the jqLite wrapper.
                    // (it's important that we retain the element instance.)
                    if (element[0].nodeType === ELEMENT_NODE) {
                        return element;
                    }
                    break;

                default:
                    return jqLite(extractElementNode(element));
                    break;
            }
        }

        if (element.nodeType === ELEMENT_NODE) {
            return jqLite(element);
        }
    }

    function extractElementNode(element) {
        if (!element[0]) return element;
        for (var i = 0; i < element.length; i++) {
            var elm = element[i];
            if (elm.nodeType == ELEMENT_NODE) {
                return elm;
            }
        }
    }

    function $$addClass($$jqLite, element, className) {
        forEach(element, function (elm) {
            $$jqLite.addClass(elm, className);
        });
    }

    function $$removeClass($$jqLite, element, className) {
        forEach(element, function (elm) {
            $$jqLite.removeClass(elm, className);
        });
    }

    function applyAnimationClassesFactory($$jqLite) {
        return function (element, options) {
            if (options.addClass) {
                $$addClass($$jqLite, element, options.addClass);
                options.addClass = null;
            }
            if (options.removeClass) {
                $$removeClass($$jqLite, element, options.removeClass);
                options.removeClass = null;
            }
        }
    }

    function prepareAnimationOptions(options) {
        options = options || {};
        if (!options.$$prepared) {
            var domOperation = options.domOperation || noop;
            options.domOperation = function () {
                options.$$domOperationFired = true;
                domOperation();
                domOperation = noop;
            };
            options.$$prepared = true;
        }
        return options;
    }

    function applyAnimationStyles(element, options) {
        applyAnimationFromStyles(element, options);
        applyAnimationToStyles(element, options);
    }

    function applyAnimationFromStyles(element, options) {
        if (options.from) {
            element.css(options.from);
            options.from = null;
        }
    }

    function applyAnimationToStyles(element, options) {
        if (options.to) {
            element.css(options.to);
            options.to = null;
        }
    }

    function mergeAnimationOptions(element, target, newOptions) {
        var toAdd = (target.addClass || '') + ' ' + (newOptions.addClass || '');
        var toRemove = (target.removeClass || '') + ' ' + (newOptions.removeClass || '');
        var classes = resolveElementClasses(element.attr('class'), toAdd, toRemove);

        if (newOptions.preparationClasses) {
            target.preparationClasses = concatWithSpace(newOptions.preparationClasses, target.preparationClasses);
            delete newOptions.preparationClasses;
        }

        // noop is basically when there is no callback; otherwise something has been set
        var realDomOperation = target.domOperation !== noop ? target.domOperation : null;

        extend(target, newOptions);

        // TODO(matsko or sreeramu): proper fix is to maintain all animation callback in array and call at last,but now only leave has the callback so no issue with this.
        if (realDomOperation) {
            target.domOperation = realDomOperation;
        }

        if (classes.addClass) {
            target.addClass = classes.addClass;
        } else {
            target.addClass = null;
        }

        if (classes.removeClass) {
            target.removeClass = classes.removeClass;
        } else {
            target.removeClass = null;
        }

        return target;
    }

    function resolveElementClasses(existing, toAdd, toRemove) {
        var ADD_CLASS = 1;
        var REMOVE_CLASS = -1;

        var flags = {};
        existing = splitClassesToLookup(existing);

        toAdd = splitClassesToLookup(toAdd);
        forEach(toAdd, function (value, key) {
            flags[key] = ADD_CLASS;
        });

        toRemove = splitClassesToLookup(toRemove);
        forEach(toRemove, function (value, key) {
            flags[key] = flags[key] === ADD_CLASS ? null : REMOVE_CLASS;
        });

        var classes = {
            addClass: '',
            removeClass: ''
        };

        forEach(flags, function (val, klass) {
            var prop, allow;
            if (val === ADD_CLASS) {
                prop = 'addClass';
                allow = !existing[klass];
            } else if (val === REMOVE_CLASS) {
                prop = 'removeClass';
                allow = existing[klass];
            }
            if (allow) {
                if (classes[prop].length) {
                    classes[prop] += ' ';
                }
                classes[prop] += klass;
            }
        });

        function splitClassesToLookup(classes) {
            if (isString(classes)) {
                classes = classes.split(' ');
            }

            var obj = {};
            forEach(classes, function (klass) {
                // sometimes the split leaves empty string values
                // incase extra spaces were applied to the options
                if (klass.length) {
                    obj[klass] = true;
                }
            });
            return obj;
        }

        return classes;
    }

    function getDomNode(element) {
        return (element instanceof angular.element) ? element[0] : element;
    }

    function applyGeneratedPreparationClasses(element, event, options) {
        var classes = '';
        if (event) {
            classes = pendClasses(event, EVENT_CLASS_PREFIX, true);
        }
        if (options.addClass) {
            classes = concatWithSpace(classes, pendClasses(options.addClass, ADD_CLASS_SUFFIX));
        }
        if (options.removeClass) {
            classes = concatWithSpace(classes, pendClasses(options.removeClass, REMOVE_CLASS_SUFFIX));
        }
        if (classes.length) {
            options.preparationClasses = classes;
            element.addClass(classes);
        }
    }

    function clearGeneratedClasses(element, options) {
        if (options.preparationClasses) {
            element.removeClass(options.preparationClasses);
            options.preparationClasses = null;
        }
        if (options.activeClasses) {
            element.removeClass(options.activeClasses);
            options.activeClasses = null;
        }
    }

    function blockTransitions(node, duration) {
        // we use a negative delay value since it performs blocking
        // yet it doesn't kill any existing transitions running on the
        // same element which makes this safe for class-based animations
        var value = duration ? '-' + duration + 's' : '';
        applyInlineStyle(node, [TRANSITION_DELAY_PROP, value]);
        return [TRANSITION_DELAY_PROP, value];
    }

    function blockKeyframeAnimations(node, applyBlock) {
        var value = applyBlock ? 'paused' : '';
        var key = ANIMATION_PROP + ANIMATION_PLAYSTATE_KEY;
        applyInlineStyle(node, [key, value]);
        return [key, value];
    }

    function applyInlineStyle(node, styleTuple) {
        var prop = styleTuple[0];
        var value = styleTuple[1];
        node.style[prop] = value;
    }

    function concatWithSpace(a, b) {
        if (!a) return b;
        if (!b) return a;
        return a + ' ' + b;
    }

    var $$rAFSchedulerFactory = ['$$rAF', function ($$rAF) {
        var queue, cancelFn;

        function scheduler(tasks) {
            // we make a copy since RAFScheduler mutates the state
            // of the passed in array variable and this would be difficult
            // to track down on the outside code
            queue = queue.concat(tasks);
            nextTick();
        }

        queue = scheduler.queue = [];

        /* waitUntilQuiet does two things:
         * 1. It will run the FINAL `fn` value only when an uncancelled RAF has passed through
         * 2. It will delay the next wave of tasks from running until the quiet `fn` has run.
         *
         * The motivation here is that animation code can request more time from the scheduler
         * before the next wave runs. This allows for certain DOM properties such as classes to
         * be resolved in time for the next animation to run.
         */
        scheduler.waitUntilQuiet = function (fn) {
            if (cancelFn) cancelFn();

            cancelFn = $$rAF(function () {
                cancelFn = null;
                fn();
                nextTick();
            });
        };

        return scheduler;

        function nextTick() {
            if (!queue.length) return;

            var items = queue.shift();
            for (var i = 0; i < items.length; i++) {
                items[i]();
            }

            if (!cancelFn) {
                $$rAF(function () {
                    if (!cancelFn) nextTick();
                });
            }
        }
    }];

    var $$AnimateChildrenDirective = [function () {
        return function (scope, element, attrs) {
            var val = attrs.ngAnimateChildren;
            if (angular.isString(val) && val.length === 0) { //empty attribute
                element.data(NG_ANIMATE_CHILDREN_DATA, true);
            } else {
                attrs.$observe('ngAnimateChildren', function (value) {
                    value = value === 'on' || value === 'true';
                    element.data(NG_ANIMATE_CHILDREN_DATA, value);
                });
            }
        };
    }];

    var ANIMATE_TIMER_KEY = '$$animateCss';

    /**
     * @ngdoc service
     * @name $animateCss
     * @kind object
     *
     * @description
     * The `$animateCss` service is a useful utility to trigger customized CSS-based transitions/keyframes
     * from a JavaScript-based animation or directly from a directive. The purpose of `$animateCss` is NOT
     * to side-step how `$animate` and ngAnimate work, but the goal is to allow pre-existing animations or
     * directives to create more complex animations that can be purely driven using CSS code.
     *
     * Note that only browsers that support CSS transitions and/or keyframe animations are capable of
     * rendering animations triggered via `$animateCss` (bad news for IE9 and lower).
     *
     * ## Usage
     * Once again, `$animateCss` is designed to be used inside of a registered JavaScript animation that
     * is powered by ngAnimate. It is possible to use `$animateCss` directly inside of a directive, however,
     * any automatic control over cancelling animations and/or preventing animations from being run on
     * child elements will not be handled by Angular. For this to work as expected, please use `$animate` to
     * trigger the animation and then setup a JavaScript animation that injects `$animateCss` to trigger
     * the CSS animation.
     *
     * The example below shows how we can create a folding animation on an element using `ng-if`:
     *
     * ```html
     * <!-- notice the `fold-animation` CSS class -->
     * <div ng-if="onOff" class="fold-animation">
     *   This element will go BOOM
     * </div>
     * <button ng-click="onOff=true">Fold In</button>
     * ```
     *
     * Now we create the **JavaScript animation** that will trigger the CSS transition:
     *
     * ```js
     * ngModule.animation('.fold-animation', ['$animateCss', function($animateCss) {
     *   return {
     *     enter: function(element, doneFn) {
     *       var height = element[0].offsetHeight;
     *       return $animateCss(element, {
     *         from: { height:'0px' },
     *         to: { height:height + 'px' },
     *         duration: 1 // one second
     *       });
     *     }
     *   }
     * }]);
     * ```
     *
     * ## More Advanced Uses
     *
     * `$animateCss` is the underlying code that ngAnimate uses to power **CSS-based animations** behind the scenes. Therefore CSS hooks
     * like `.ng-EVENT`, `.ng-EVENT-active`, `.ng-EVENT-stagger` are all features that can be triggered using `$animateCss` via JavaScript code.
     *
     * This also means that just about any combination of adding classes, removing classes, setting styles, dynamically setting a keyframe animation,
     * applying a hardcoded duration or delay value, changing the animation easing or applying a stagger animation are all options that work with
     * `$animateCss`. The service itself is smart enough to figure out the combination of options and examine the element styling properties in order
     * to provide a working animation that will run in CSS.
     *
     * The example below showcases a more advanced version of the `.fold-animation` from the example above:
     *
     * ```js
     * ngModule.animation('.fold-animation', ['$animateCss', function($animateCss) {
     *   return {
     *     enter: function(element, doneFn) {
     *       var height = element[0].offsetHeight;
     *       return $animateCss(element, {
     *         addClass: 'red large-text pulse-twice',
     *         easing: 'ease-out',
     *         from: { height:'0px' },
     *         to: { height:height + 'px' },
     *         duration: 1 // one second
     *       });
     *     }
     *   }
     * }]);
     * ```
     *
     * Since we're adding/removing CSS classes then the CSS transition will also pick those up:
     *
     * ```css
     * /&#42; since a hardcoded duration value of 1 was provided in the JavaScript animation code,
     * the CSS classes below will be transitioned despite them being defined as regular CSS classes &#42;/
     * .red { background:red; }
     * .large-text { font-size:20px; }
     *
     * /&#42; we can also use a keyframe animation and $animateCss will make it work alongside the transition &#42;/
     * .pulse-twice {
     *   animation: 0.5s pulse linear 2;
     *   -webkit-animation: 0.5s pulse linear 2;
     * }
     *
     * @keyframes pulse {
     *   from { transform: scale(0.5); }
     *   to { transform: scale(1.5); }
     * }
     *
     * @-webkit-keyframes pulse {
     *   from { -webkit-transform: scale(0.5); }
     *   to { -webkit-transform: scale(1.5); }
     * }
     * ```
     *
     * Given this complex combination of CSS classes, styles and options, `$animateCss` will figure everything out and make the animation happen.
     *
     * ## How the Options are handled
     *
     * `$animateCss` is very versatile and intelligent when it comes to figuring out what configurations to apply to the element to ensure the animation
     * works with the options provided. Say for example we were adding a class that contained a keyframe value and we wanted to also animate some inline
     * styles using the `from` and `to` properties.
     *
     * ```js
     * var animator = $animateCss(element, {
     *   from: { background:'red' },
     *   to: { background:'blue' }
     * });
     * animator.start();
     * ```
     *
     * ```css
     * .rotating-animation {
     *   animation:0.5s rotate linear;
     *   -webkit-animation:0.5s rotate linear;
     * }
     *
     * @keyframes rotate {
     *   from { transform: rotate(0deg); }
     *   to { transform: rotate(360deg); }
     * }
     *
     * @-webkit-keyframes rotate {
     *   from { -webkit-transform: rotate(0deg); }
     *   to { -webkit-transform: rotate(360deg); }
     * }
     * ```
     *
     * The missing pieces here are that we do not have a transition set (within the CSS code nor within the `$animateCss` options) and the duration of the animation is
     * going to be detected from what the keyframe styles on the CSS class are. In this event, `$animateCss` will automatically create an inline transition
     * style matching the duration detected from the keyframe style (which is present in the CSS class that is being added) and then prepare both the transition
     * and keyframe animations to run in parallel on the element. Then when the animation is underway the provided `from` and `to` CSS styles will be applied
     * and spread across the transition and keyframe animation.
     *
     * ## What is returned
     *
     * `$animateCss` works in two stages: a preparation phase and an animation phase. Therefore when `$animateCss` is first called it will NOT actually
     * start the animation. All that is going on here is that the element is being prepared for the animation (which means that the generated CSS classes are
     * added and removed on the element). Once `$animateCss` is called it will return an object with the following properties:
     *
     * ```js
     * var animator = $animateCss(element, { ... });
     * ```
     *
     * Now what do the contents of our `animator` variable look like:
     *
     * ```js
     * {
     *   // starts the animation
     *   start: Function,
     *
     *   // ends (aborts) the animation
     *   end: Function
     * }
     * ```
     *
     * To actually start the animation we need to run `animation.start()` which will then return a promise that we can hook into to detect when the animation ends.
     * If we choose not to run the animation then we MUST run `animation.end()` to perform a cleanup on the element (since some CSS classes and stlyes may have been
     * applied to the element during the preparation phase). Note that all other properties such as duration, delay, transitions and keyframes are just properties
     * and that changing them will not reconfigure the parameters of the animation.
     *
     * ### runner.done() vs runner.then()
     * It is documented that `animation.start()` will return a promise object and this is true, however, there is also an additional method available on the
     * runner called `.done(callbackFn)`. The done method works the same as `.finally(callbackFn)`, however, it does **not trigger a digest to occur**.
     * Therefore, for performance reasons, it's always best to use `runner.done(callback)` instead of `runner.then()`, `runner.catch()` or `runner.finally()`
     * unless you really need a digest to kick off afterwards.
     *
     * Keep in mind that, to make this easier, ngAnimate has tweaked the JS animations API to recognize when a runner instance is returned from $animateCss
     * (so there is no need to call `runner.done(doneFn)` inside of your JavaScript animation code).
     * Check the {@link ngAnimate.$animateCss#usage animation code above} to see how this works.
     *
     * @param {DOMElement} element the element that will be animated
     * @param {object} options the animation-related options that will be applied during the animation
     *
     * * `event` - The DOM event (e.g. enter, leave, move). When used, a generated CSS class of `ng-EVENT` and `ng-EVENT-active` will be applied
     * to the element during the animation. Multiple events can be provided when spaces are used as a separator. (Note that this will not perform any DOM operation.)
     * * `structural` - Indicates that the `ng-` prefix will be added to the event class. Setting to `false` or omitting will turn `ng-EVENT` and
     * `ng-EVENT-active` in `EVENT` and `EVENT-active`. Unused if `event` is omitted.
     * * `easing` - The CSS easing value that will be applied to the transition or keyframe animation (or both).
     * * `transitionStyle` - The raw CSS transition style that will be used (e.g. `1s linear all`).
     * * `keyframeStyle` - The raw CSS keyframe animation style that will be used (e.g. `1s my_animation linear`).
     * * `from` - The starting CSS styles (a key/value object) that will be applied at the start of the animation.
     * * `to` - The ending CSS styles (a key/value object) that will be applied across the animation via a CSS transition.
     * * `addClass` - A space separated list of CSS classes that will be added to the element and spread across the animation.
     * * `removeClass` - A space separated list of CSS classes that will be removed from the element and spread across the animation.
     * * `duration` - A number value representing the total duration of the transition and/or keyframe (note that a value of 1 is 1000ms). If a value of `0`
     * is provided then the animation will be skipped entirely.
     * * `delay` - A number value representing the total delay of the transition and/or keyframe (note that a value of 1 is 1000ms). If a value of `true` is
     * used then whatever delay value is detected from the CSS classes will be mirrored on the elements styles (e.g. by setting delay true then the style value
     * of the element will be `transition-delay: DETECTED_VALUE`). Using `true` is useful when you want the CSS classes and inline styles to all share the same
     * CSS delay value.
     * * `stagger` - A numeric time value representing the delay between successively animated elements
     * ({@link ngAnimate#css-staggering-animations Click here to learn how CSS-based staggering works in ngAnimate.})
     * * `staggerIndex` - The numeric index representing the stagger item (e.g. a value of 5 is equal to the sixth item in the stagger; therefore when a
     * * `stagger` option value of `0.1` is used then there will be a stagger delay of `600ms`)
     * * `applyClassesEarly` - Whether or not the classes being added or removed will be used when detecting the animation. This is set by `$animate` when enter/leave/move animations are fired to ensure that the CSS classes are resolved in time. (Note that this will prevent any transitions from occuring on the classes being added and removed.)
     * * `cleanupStyles` - Whether or not the provided `from` and `to` styles will be removed once
     *    the animation is closed. This is useful for when the styles are used purely for the sake of
     *    the animation and do not have a lasting visual effect on the element (e.g. a colapse and open animation).
     *    By default this value is set to `false`.
     *
     * @return {object} an object with start and end methods and details about the animation.
     *
     * * `start` - The method to start the animation. This will return a `Promise` when called.
     * * `end` - This method will cancel the animation and remove all applied CSS classes and styles.
     */
    var ONE_SECOND = 1000;
    var BASE_TEN = 10;

    var ELAPSED_TIME_MAX_DECIMAL_PLACES = 3;
    var CLOSING_TIME_BUFFER = 1.5;

    var DETECT_CSS_PROPERTIES = {
        transitionDuration: TRANSITION_DURATION_PROP,
        transitionDelay: TRANSITION_DELAY_PROP,
        transitionProperty: TRANSITION_PROP + PROPERTY_KEY,
        animationDuration: ANIMATION_DURATION_PROP,
        animationDelay: ANIMATION_DELAY_PROP,
        animationIterationCount: ANIMATION_PROP + ANIMATION_ITERATION_COUNT_KEY
    };

    var DETECT_STAGGER_CSS_PROPERTIES = {
        transitionDuration: TRANSITION_DURATION_PROP,
        transitionDelay: TRANSITION_DELAY_PROP,
        animationDuration: ANIMATION_DURATION_PROP,
        animationDelay: ANIMATION_DELAY_PROP
    };

    function getCssKeyframeDurationStyle(duration) {
        return [ANIMATION_DURATION_PROP, duration + 's'];
    }

    function getCssDelayStyle(delay, isKeyframeAnimation) {
        var prop = isKeyframeAnimation ? ANIMATION_DELAY_PROP : TRANSITION_DELAY_PROP;
        return [prop, delay + 's'];
    }

    function computeCssStyles($window, element, properties) {
        var styles = Object.create(null);
        var detectedStyles = $window.getComputedStyle(element) || {};
        forEach(properties, function (formalStyleName, actualStyleName) {
            var val = detectedStyles[formalStyleName];
            if (val) {
                var c = val.charAt(0);

                // only numerical-based values have a negative sign or digit as the first value
                if (c === '-' || c === '+' || c >= 0) {
                    val = parseMaxTime(val);
                }

                // by setting this to null in the event that the delay is not set or is set directly as 0
                // then we can still allow for zegative values to be used later on and not mistake this
                // value for being greater than any other negative value.
                if (val === 0) {
                    val = null;
                }
                styles[actualStyleName] = val;
            }
        });

        return styles;
    }

    function parseMaxTime(str) {
        var maxValue = 0;
        var values = str.split(/\s*,\s*/);
        forEach(values, function (value) {
            // it's always safe to consider only second values and omit `ms` values since
            // getComputedStyle will always handle the conversion for us
            if (value.charAt(value.length - 1) == 's') {
                value = value.substring(0, value.length - 1);
            }
            value = parseFloat(value) || 0;
            maxValue = maxValue ? Math.max(value, maxValue) : value;
        });
        return maxValue;
    }

    function truthyTimingValue(val) {
        return val === 0 || val != null;
    }

    function getCssTransitionDurationStyle(duration, applyOnlyDuration) {
        var style = TRANSITION_PROP;
        var value = duration + 's';
        if (applyOnlyDuration) {
            style += DURATION_KEY;
        } else {
            value += ' linear all';
        }
        return [style, value];
    }

    function createLocalCacheLookup() {
        var cache = Object.create(null);
        return {
            flush: function () {
                cache = Object.create(null);
            },

            count: function (key) {
                var entry = cache[key];
                return entry ? entry.total : 0;
            },

            get: function (key) {
                var entry = cache[key];
                return entry && entry.value;
            },

            put: function (key, value) {
                if (!cache[key]) {
                    cache[key] = { total: 1, value: value };
                } else {
                    cache[key].total++;
                }
            }
        };
    }

    // we do not reassign an already present style value since
    // if we detect the style property value again we may be
    // detecting styles that were added via the `from` styles.
    // We make use of `isDefined` here since an empty string
    // or null value (which is what getPropertyValue will return
    // for a non-existing style) will still be marked as a valid
    // value for the style (a falsy value implies that the style
    // is to be removed at the end of the animation). If we had a simple
    // "OR" statement then it would not be enough to catch that.
    function registerRestorableStyles(backup, node, properties) {
        forEach(properties, function (prop) {
            backup[prop] = isDefined(backup[prop])
                ? backup[prop]
                : node.style.getPropertyValue(prop);
        });
    }

    var $AnimateCssProvider = ['$animateProvider', function ($animateProvider) {
        var gcsLookup = createLocalCacheLookup();
        var gcsStaggerLookup = createLocalCacheLookup();

        this.$get = ['$window', '$$jqLite', '$$AnimateRunner', '$timeout',
                     '$$forceReflow', '$sniffer', '$$rAFScheduler', '$animate',
             function ($window, $$jqLite, $$AnimateRunner, $timeout,
                      $$forceReflow, $sniffer, $$rAFScheduler, $animate) {

                 var applyAnimationClasses = applyAnimationClassesFactory($$jqLite);

                 var parentCounter = 0;
                 function gcsHashFn(node, extraClasses) {
                     var KEY = "$$ngAnimateParentKey";
                     var parentNode = node.parentNode;
                     var parentID = parentNode[KEY] || (parentNode[KEY] = ++parentCounter);
                     return parentID + '-' + node.getAttribute('class') + '-' + extraClasses;
                 }

                 function computeCachedCssStyles(node, className, cacheKey, properties) {
                     var timings = gcsLookup.get(cacheKey);

                     if (!timings) {
                         timings = computeCssStyles($window, node, properties);
                         if (timings.animationIterationCount === 'infinite') {
                             timings.animationIterationCount = 1;
                         }
                     }

                     // we keep putting this in multiple times even though the value and the cacheKey are the same
                     // because we're keeping an interal tally of how many duplicate animations are detected.
                     gcsLookup.put(cacheKey, timings);
                     return timings;
                 }

                 function computeCachedCssStaggerStyles(node, className, cacheKey, properties) {
                     var stagger;

                     // if we have one or more existing matches of matching elements
                     // containing the same parent + CSS styles (which is how cacheKey works)
                     // then staggering is possible
                     if (gcsLookup.count(cacheKey) > 0) {
                         stagger = gcsStaggerLookup.get(cacheKey);

                         if (!stagger) {
                             var staggerClassName = pendClasses(className, '-stagger');

                             $$jqLite.addClass(node, staggerClassName);

                             stagger = computeCssStyles($window, node, properties);

                             // force the conversion of a null value to zero incase not set
                             stagger.animationDuration = Math.max(stagger.animationDuration, 0);
                             stagger.transitionDuration = Math.max(stagger.transitionDuration, 0);

                             $$jqLite.removeClass(node, staggerClassName);

                             gcsStaggerLookup.put(cacheKey, stagger);
                         }
                     }

                     return stagger || {};
                 }

                 var cancelLastRAFRequest;
                 var rafWaitQueue = [];
                 function waitUntilQuiet(callback) {
                     rafWaitQueue.push(callback);
                     $$rAFScheduler.waitUntilQuiet(function () {
                         gcsLookup.flush();
                         gcsStaggerLookup.flush();

                         // DO NOT REMOVE THIS LINE OR REFACTOR OUT THE `pageWidth` variable.
                         // PLEASE EXAMINE THE `$$forceReflow` service to understand why.
                         var pageWidth = $$forceReflow();

                         // we use a for loop to ensure that if the queue is changed
                         // during this looping then it will consider new requests
                         for (var i = 0; i < rafWaitQueue.length; i++) {
                             rafWaitQueue[i](pageWidth);
                         }
                         rafWaitQueue.length = 0;
                     });
                 }

                 function computeTimings(node, className, cacheKey) {
                     var timings = computeCachedCssStyles(node, className, cacheKey, DETECT_CSS_PROPERTIES);
                     var aD = timings.animationDelay;
                     var tD = timings.transitionDelay;
                     timings.maxDelay = aD && tD
                         ? Math.max(aD, tD)
                         : (aD || tD);
                     timings.maxDuration = Math.max(
                         timings.animationDuration * timings.animationIterationCount,
                         timings.transitionDuration);

                     return timings;
                 }

                 return function init(element, options) {
                     var restoreStyles = {};
                     var node = getDomNode(element);
                     if (!node
                         || !node.parentNode
                         || !$animate.enabled()) {
                         return closeAndReturnNoopAnimator();
                     }

                     options = prepareAnimationOptions(options);

                     var temporaryStyles = [];
                     var classes = element.attr('class');
                     var styles = packageStyles(options);
                     var animationClosed;
                     var animationPaused;
                     var animationCompleted;
                     var runner;
                     var runnerHost;
                     var maxDelay;
                     var maxDelayTime;
                     var maxDuration;
                     var maxDurationTime;

                     if (options.duration === 0 || (!$sniffer.animations && !$sniffer.transitions)) {
                         return closeAndReturnNoopAnimator();
                     }

                     var method = options.event && isArray(options.event)
                           ? options.event.join(' ')
                           : options.event;

                     var isStructural = method && options.structural;
                     var structuralClassName = '';
                     var addRemoveClassName = '';

                     if (isStructural) {
                         structuralClassName = pendClasses(method, EVENT_CLASS_PREFIX, true);
                     } else if (method) {
                         structuralClassName = method;
                     }

                     if (options.addClass) {
                         addRemoveClassName += pendClasses(options.addClass, ADD_CLASS_SUFFIX);
                     }

                     if (options.removeClass) {
                         if (addRemoveClassName.length) {
                             addRemoveClassName += ' ';
                         }
                         addRemoveClassName += pendClasses(options.removeClass, REMOVE_CLASS_SUFFIX);
                     }

                     // there may be a situation where a structural animation is combined together
                     // with CSS classes that need to resolve before the animation is computed.
                     // However this means that there is no explicit CSS code to block the animation
                     // from happening (by setting 0s none in the class name). If this is the case
                     // we need to apply the classes before the first rAF so we know to continue if
                     // there actually is a detected transition or keyframe animation
                     if (options.applyClassesEarly && addRemoveClassName.length) {
                         applyAnimationClasses(element, options);
                     }

                     var preparationClasses = [structuralClassName, addRemoveClassName].join(' ').trim();
                     var fullClassName = classes + ' ' + preparationClasses;
                     var activeClasses = pendClasses(preparationClasses, ACTIVE_CLASS_SUFFIX);
                     var hasToStyles = styles.to && Object.keys(styles.to).length > 0;
                     var containsKeyframeAnimation = (options.keyframeStyle || '').length > 0;

                     // there is no way we can trigger an animation if no styles and
                     // no classes are being applied which would then trigger a transition,
                     // unless there a is raw keyframe value that is applied to the element.
                     if (!containsKeyframeAnimation
                          && !hasToStyles
                          && !preparationClasses) {
                         return closeAndReturnNoopAnimator();
                     }

                     var cacheKey, stagger;
                     if (options.stagger > 0) {
                         var staggerVal = parseFloat(options.stagger);
                         stagger = {
                             transitionDelay: staggerVal,
                             animationDelay: staggerVal,
                             transitionDuration: 0,
                             animationDuration: 0
                         };
                     } else {
                         cacheKey = gcsHashFn(node, fullClassName);
                         stagger = computeCachedCssStaggerStyles(node, preparationClasses, cacheKey, DETECT_STAGGER_CSS_PROPERTIES);
                     }

                     if (!options.$$skipPreparationClasses) {
                         $$jqLite.addClass(element, preparationClasses);
                     }

                     var applyOnlyDuration;

                     if (options.transitionStyle) {
                         var transitionStyle = [TRANSITION_PROP, options.transitionStyle];
                         applyInlineStyle(node, transitionStyle);
                         temporaryStyles.push(transitionStyle);
                     }

                     if (options.duration >= 0) {
                         applyOnlyDuration = node.style[TRANSITION_PROP].length > 0;
                         var durationStyle = getCssTransitionDurationStyle(options.duration, applyOnlyDuration);

                         // we set the duration so that it will be picked up by getComputedStyle later
                         applyInlineStyle(node, durationStyle);
                         temporaryStyles.push(durationStyle);
                     }

                     if (options.keyframeStyle) {
                         var keyframeStyle = [ANIMATION_PROP, options.keyframeStyle];
                         applyInlineStyle(node, keyframeStyle);
                         temporaryStyles.push(keyframeStyle);
                     }

                     var itemIndex = stagger
                         ? options.staggerIndex >= 0
                             ? options.staggerIndex
                             : gcsLookup.count(cacheKey)
                         : 0;

                     var isFirst = itemIndex === 0;

                     // this is a pre-emptive way of forcing the setup classes to be added and applied INSTANTLY
                     // without causing any combination of transitions to kick in. By adding a negative delay value
                     // it forces the setup class' transition to end immediately. We later then remove the negative
                     // transition delay to allow for the transition to naturally do it's thing. The beauty here is
                     // that if there is no transition defined then nothing will happen and this will also allow
                     // other transitions to be stacked on top of each other without any chopping them out.
                     if (isFirst && !options.skipBlocking) {
                         blockTransitions(node, SAFE_FAST_FORWARD_DURATION_VALUE);
                     }

                     var timings = computeTimings(node, fullClassName, cacheKey);
                     var relativeDelay = timings.maxDelay;
                     maxDelay = Math.max(relativeDelay, 0);
                     maxDuration = timings.maxDuration;

                     var flags = {};
                     flags.hasTransitions = timings.transitionDuration > 0;
                     flags.hasAnimations = timings.animationDuration > 0;
                     flags.hasTransitionAll = flags.hasTransitions && timings.transitionProperty == 'all';
                     flags.applyTransitionDuration = hasToStyles && (
                                                       (flags.hasTransitions && !flags.hasTransitionAll)
                                                        || (flags.hasAnimations && !flags.hasTransitions));
                     flags.applyAnimationDuration = options.duration && flags.hasAnimations;
                     flags.applyTransitionDelay = truthyTimingValue(options.delay) && (flags.applyTransitionDuration || flags.hasTransitions);
                     flags.applyAnimationDelay = truthyTimingValue(options.delay) && flags.hasAnimations;
                     flags.recalculateTimingStyles = addRemoveClassName.length > 0;

                     if (flags.applyTransitionDuration || flags.applyAnimationDuration) {
                         maxDuration = options.duration ? parseFloat(options.duration) : maxDuration;

                         if (flags.applyTransitionDuration) {
                             flags.hasTransitions = true;
                             timings.transitionDuration = maxDuration;
                             applyOnlyDuration = node.style[TRANSITION_PROP + PROPERTY_KEY].length > 0;
                             temporaryStyles.push(getCssTransitionDurationStyle(maxDuration, applyOnlyDuration));
                         }

                         if (flags.applyAnimationDuration) {
                             flags.hasAnimations = true;
                             timings.animationDuration = maxDuration;
                             temporaryStyles.push(getCssKeyframeDurationStyle(maxDuration));
                         }
                     }

                     if (maxDuration === 0 && !flags.recalculateTimingStyles) {
                         return closeAndReturnNoopAnimator();
                     }

                     if (options.delay != null) {
                         var delayStyle = parseFloat(options.delay);

                         if (flags.applyTransitionDelay) {
                             temporaryStyles.push(getCssDelayStyle(delayStyle));
                         }

                         if (flags.applyAnimationDelay) {
                             temporaryStyles.push(getCssDelayStyle(delayStyle, true));
                         }
                     }

                     // we need to recalculate the delay value since we used a pre-emptive negative
                     // delay value and the delay value is required for the final event checking. This
                     // property will ensure that this will happen after the RAF phase has passed.
                     if (options.duration == null && timings.transitionDuration > 0) {
                         flags.recalculateTimingStyles = flags.recalculateTimingStyles || isFirst;
                     }

                     maxDelayTime = maxDelay * ONE_SECOND;
                     maxDurationTime = maxDuration * ONE_SECOND;
                     if (!options.skipBlocking) {
                         flags.blockTransition = timings.transitionDuration > 0;
                         flags.blockKeyframeAnimation = timings.animationDuration > 0 &&
                                                        stagger.animationDelay > 0 &&
                                                        stagger.animationDuration === 0;
                     }

                     if (options.from) {
                         if (options.cleanupStyles) {
                             registerRestorableStyles(restoreStyles, node, Object.keys(options.from));
                         }
                         applyAnimationFromStyles(element, options);
                     }

                     if (flags.blockTransition || flags.blockKeyframeAnimation) {
                         applyBlocking(maxDuration);
                     } else if (!options.skipBlocking) {
                         blockTransitions(node, false);
                     }

                     // TODO(matsko): for 1.5 change this code to have an animator object for better debugging
                     return {
                         $$willAnimate: true,
                         end: endFn,
                         start: function () {
                             if (animationClosed) return;

                             runnerHost = {
                                 end: endFn,
                                 cancel: cancelFn,
                                 resume: null, //this will be set during the start() phase
                                 pause: null
                             };

                             runner = new $$AnimateRunner(runnerHost);

                             waitUntilQuiet(start);

                             // we don't have access to pause/resume the animation
                             // since it hasn't run yet. AnimateRunner will therefore
                             // set noop functions for resume and pause and they will
                             // later be overridden once the animation is triggered
                             return runner;
                         }
                     };

                     function endFn() {
                         close();
                     }

                     function cancelFn() {
                         close(true);
                     }

                     function close(rejected) { // jshint ignore:line
                         // if the promise has been called already then we shouldn't close
                         // the animation again
                         if (animationClosed || (animationCompleted && animationPaused)) return;
                         animationClosed = true;
                         animationPaused = false;

                         if (!options.$$skipPreparationClasses) {
                             $$jqLite.removeClass(element, preparationClasses);
                         }
                         $$jqLite.removeClass(element, activeClasses);

                         blockKeyframeAnimations(node, false);
                         blockTransitions(node, false);

                         forEach(temporaryStyles, function (entry) {
                             // There is only one way to remove inline style properties entirely from elements.
                             // By using `removeProperty` this works, but we need to convert camel-cased CSS
                             // styles down to hyphenated values.
                             node.style[entry[0]] = '';
                         });

                         applyAnimationClasses(element, options);
                         applyAnimationStyles(element, options);

                         if (Object.keys(restoreStyles).length) {
                             forEach(restoreStyles, function (value, prop) {
                                 value ? node.style.setProperty(prop, value)
                                       : node.style.removeProperty(prop);
                             });
                         }

                         // the reason why we have this option is to allow a synchronous closing callback
                         // that is fired as SOON as the animation ends (when the CSS is removed) or if
                         // the animation never takes off at all. A good example is a leave animation since
                         // the element must be removed just after the animation is over or else the element
                         // will appear on screen for one animation frame causing an overbearing flicker.
                         if (options.onDone) {
                             options.onDone();
                         }

                         // if the preparation function fails then the promise is not setup
                         if (runner) {
                             runner.complete(!rejected);
                         }
                     }

                     function applyBlocking(duration) {
                         if (flags.blockTransition) {
                             blockTransitions(node, duration);
                         }

                         if (flags.blockKeyframeAnimation) {
                             blockKeyframeAnimations(node, !!duration);
                         }
                     }

                     function closeAndReturnNoopAnimator() {
                         runner = new $$AnimateRunner({
                             end: endFn,
                             cancel: cancelFn
                         });

                         // should flush the cache animation
                         waitUntilQuiet(noop);
                         close();

                         return {
                             $$willAnimate: false,
                             start: function () {
                                 return runner;
                             },
                             end: endFn
                         };
                     }

                     function start() {
                         if (animationClosed) return;
                         if (!node.parentNode) {
                             close();
                             return;
                         }

                         var startTime, events = [];

                         // even though we only pause keyframe animations here the pause flag
                         // will still happen when transitions are used. Only the transition will
                         // not be paused since that is not possible. If the animation ends when
                         // paused then it will not complete until unpaused or cancelled.
                         var playPause = function (playAnimation) {
                             if (!animationCompleted) {
                                 animationPaused = !playAnimation;
                                 if (timings.animationDuration) {
                                     var value = blockKeyframeAnimations(node, animationPaused);
                                     animationPaused
                                         ? temporaryStyles.push(value)
                                         : removeFromArray(temporaryStyles, value);
                                 }
                             } else if (animationPaused && playAnimation) {
                                 animationPaused = false;
                                 close();
                             }
                         };

                         // checking the stagger duration prevents an accidently cascade of the CSS delay style
                         // being inherited from the parent. If the transition duration is zero then we can safely
                         // rely that the delay value is an intential stagger delay style.
                         var maxStagger = itemIndex > 0
                                          && ((timings.transitionDuration && stagger.transitionDuration === 0) ||
                                             (timings.animationDuration && stagger.animationDuration === 0))
                                          && Math.max(stagger.animationDelay, stagger.transitionDelay);
                         if (maxStagger) {
                             $timeout(triggerAnimationStart,
                                      Math.floor(maxStagger * itemIndex * ONE_SECOND),
                                      false);
                         } else {
                             triggerAnimationStart();
                         }

                         // this will decorate the existing promise runner with pause/resume methods
                         runnerHost.resume = function () {
                             playPause(true);
                         };

                         runnerHost.pause = function () {
                             playPause(false);
                         };

                         function triggerAnimationStart() {
                             // just incase a stagger animation kicks in when the animation
                             // itself was cancelled entirely
                             if (animationClosed) return;

                             applyBlocking(false);

                             forEach(temporaryStyles, function (entry) {
                                 var key = entry[0];
                                 var value = entry[1];
                                 node.style[key] = value;
                             });

                             applyAnimationClasses(element, options);
                             $$jqLite.addClass(element, activeClasses);

                             if (flags.recalculateTimingStyles) {
                                 fullClassName = node.className + ' ' + preparationClasses;
                                 cacheKey = gcsHashFn(node, fullClassName);

                                 timings = computeTimings(node, fullClassName, cacheKey);
                                 relativeDelay = timings.maxDelay;
                                 maxDelay = Math.max(relativeDelay, 0);
                                 maxDuration = timings.maxDuration;

                                 if (maxDuration === 0) {
                                     close();
                                     return;
                                 }

                                 flags.hasTransitions = timings.transitionDuration > 0;
                                 flags.hasAnimations = timings.animationDuration > 0;
                             }

                             if (flags.applyAnimationDelay) {
                                 relativeDelay = typeof options.delay !== "boolean" && truthyTimingValue(options.delay)
                                       ? parseFloat(options.delay)
                                       : relativeDelay;

                                 maxDelay = Math.max(relativeDelay, 0);
                                 timings.animationDelay = relativeDelay;
                                 delayStyle = getCssDelayStyle(relativeDelay, true);
                                 temporaryStyles.push(delayStyle);
                                 node.style[delayStyle[0]] = delayStyle[1];
                             }

                             maxDelayTime = maxDelay * ONE_SECOND;
                             maxDurationTime = maxDuration * ONE_SECOND;

                             if (options.easing) {
                                 var easeProp, easeVal = options.easing;
                                 if (flags.hasTransitions) {
                                     easeProp = TRANSITION_PROP + TIMING_KEY;
                                     temporaryStyles.push([easeProp, easeVal]);
                                     node.style[easeProp] = easeVal;
                                 }
                                 if (flags.hasAnimations) {
                                     easeProp = ANIMATION_PROP + TIMING_KEY;
                                     temporaryStyles.push([easeProp, easeVal]);
                                     node.style[easeProp] = easeVal;
                                 }
                             }

                             if (timings.transitionDuration) {
                                 events.push(TRANSITIONEND_EVENT);
                             }

                             if (timings.animationDuration) {
                                 events.push(ANIMATIONEND_EVENT);
                             }

                             startTime = Date.now();
                             var timerTime = maxDelayTime + CLOSING_TIME_BUFFER * maxDurationTime;
                             var endTime = startTime + timerTime;

                             var animationsData = element.data(ANIMATE_TIMER_KEY) || [];
                             var setupFallbackTimer = true;
                             if (animationsData.length) {
                                 var currentTimerData = animationsData[0];
                                 setupFallbackTimer = endTime > currentTimerData.expectedEndTime;
                                 if (setupFallbackTimer) {
                                     $timeout.cancel(currentTimerData.timer);
                                 } else {
                                     animationsData.push(close);
                                 }
                             }

                             if (setupFallbackTimer) {
                                 var timer = $timeout(onAnimationExpired, timerTime, false);
                                 animationsData[0] = {
                                     timer: timer,
                                     expectedEndTime: endTime
                                 };
                                 animationsData.push(close);
                                 element.data(ANIMATE_TIMER_KEY, animationsData);
                             }

                             element.on(events.join(' '), onAnimationProgress);
                             if (options.to) {
                                 if (options.cleanupStyles) {
                                     registerRestorableStyles(restoreStyles, node, Object.keys(options.to));
                                 }
                                 applyAnimationToStyles(element, options);
                             }
                         }

                         function onAnimationExpired() {
                             var animationsData = element.data(ANIMATE_TIMER_KEY);

                             // this will be false in the event that the element was
                             // removed from the DOM (via a leave animation or something
                             // similar)
                             if (animationsData) {
                                 for (var i = 1; i < animationsData.length; i++) {
                                     animationsData[i]();
                                 }
                                 element.removeData(ANIMATE_TIMER_KEY);
                             }
                         }

                         function onAnimationProgress(event) {
                             event.stopPropagation();
                             var ev = event.originalEvent || event;
                             var timeStamp = ev.$manualTimeStamp || ev.timeStamp || Date.now();

                             /* Firefox (or possibly just Gecko) likes to not round values up
                              * when a ms measurement is used for the animation */
                             var elapsedTime = parseFloat(ev.elapsedTime.toFixed(ELAPSED_TIME_MAX_DECIMAL_PLACES));

                             /* $manualTimeStamp is a mocked timeStamp value which is set
                              * within browserTrigger(). This is only here so that tests can
                              * mock animations properly. Real events fallback to event.timeStamp,
                              * or, if they don't, then a timeStamp is automatically created for them.
                              * We're checking to see if the timeStamp surpasses the expected delay,
                              * but we're using elapsedTime instead of the timeStamp on the 2nd
                              * pre-condition since animations sometimes close off early */
                             if (Math.max(timeStamp - startTime, 0) >= maxDelayTime && elapsedTime >= maxDuration) {
                                 // we set this flag to ensure that if the transition is paused then, when resumed,
                                 // the animation will automatically close itself since transitions cannot be paused.
                                 animationCompleted = true;
                                 close();
                             }
                         }
                     }
                 };
             }];
    }];

    var $$AnimateCssDriverProvider = ['$$animationProvider', function ($$animationProvider) {
        $$animationProvider.drivers.push('$$animateCssDriver');

        var NG_ANIMATE_SHIM_CLASS_NAME = 'ng-animate-shim';
        var NG_ANIMATE_ANCHOR_CLASS_NAME = 'ng-anchor';

        var NG_OUT_ANCHOR_CLASS_NAME = 'ng-anchor-out';
        var NG_IN_ANCHOR_CLASS_NAME = 'ng-anchor-in';

        function isDocumentFragment(node) {
            return node.parentNode && node.parentNode.nodeType === 11;
        }

        this.$get = ['$animateCss', '$rootScope', '$$AnimateRunner', '$rootElement', '$sniffer', '$$jqLite', '$document',
             function ($animateCss, $rootScope, $$AnimateRunner, $rootElement, $sniffer, $$jqLite, $document) {

                 // only browsers that support these properties can render animations
                 if (!$sniffer.animations && !$sniffer.transitions) return noop;

                 var bodyNode = $document[0].body;
                 var rootNode = getDomNode($rootElement);

                 var rootBodyElement = jqLite(
                   // this is to avoid using something that exists outside of the body
                   // we also special case the doc fragement case because our unit test code
                   // appends the $rootElement to the body after the app has been bootstrapped
                   isDocumentFragment(rootNode) || bodyNode.contains(rootNode) ? rootNode : bodyNode
                 );

                 var applyAnimationClasses = applyAnimationClassesFactory($$jqLite);

                 return function initDriverFn(animationDetails) {
                     return animationDetails.from && animationDetails.to
                         ? prepareFromToAnchorAnimation(animationDetails.from,
                                                        animationDetails.to,
                                                        animationDetails.classes,
                                                        animationDetails.anchors)
                         : prepareRegularAnimation(animationDetails);
                 };

                 function filterCssClasses(classes) {
                     //remove all the `ng-` stuff
                     return classes.replace(/\bng-\S+\b/g, '');
                 }

                 function getUniqueValues(a, b) {
                     if (isString(a)) a = a.split(' ');
                     if (isString(b)) b = b.split(' ');
                     return a.filter(function (val) {
                         return b.indexOf(val) === -1;
                     }).join(' ');
                 }

                 function prepareAnchoredAnimation(classes, outAnchor, inAnchor) {
                     var clone = jqLite(getDomNode(outAnchor).cloneNode(true));
                     var startingClasses = filterCssClasses(getClassVal(clone));

                     outAnchor.addClass(NG_ANIMATE_SHIM_CLASS_NAME);
                     inAnchor.addClass(NG_ANIMATE_SHIM_CLASS_NAME);

                     clone.addClass(NG_ANIMATE_ANCHOR_CLASS_NAME);

                     rootBodyElement.append(clone);

                     var animatorIn, animatorOut = prepareOutAnimation();

                     // the user may not end up using the `out` animation and
                     // only making use of the `in` animation or vice-versa.
                     // In either case we should allow this and not assume the
                     // animation is over unless both animations are not used.
                     if (!animatorOut) {
                         animatorIn = prepareInAnimation();
                         if (!animatorIn) {
                             return end();
                         }
                     }

                     var startingAnimator = animatorOut || animatorIn;

                     return {
                         start: function () {
                             var runner;

                             var currentAnimation = startingAnimator.start();
                             currentAnimation.done(function () {
                                 currentAnimation = null;
                                 if (!animatorIn) {
                                     animatorIn = prepareInAnimation();
                                     if (animatorIn) {
                                         currentAnimation = animatorIn.start();
                                         currentAnimation.done(function () {
                                             currentAnimation = null;
                                             end();
                                             runner.complete();
                                         });
                                         return currentAnimation;
                                     }
                                 }
                                 // in the event that there is no `in` animation
                                 end();
                                 runner.complete();
                             });

                             runner = new $$AnimateRunner({
                                 end: endFn,
                                 cancel: endFn
                             });

                             return runner;

                             function endFn() {
                                 if (currentAnimation) {
                                     currentAnimation.end();
                                 }
                             }
                         }
                     };

                     function calculateAnchorStyles(anchor) {
                         var styles = {};

                         var coords = getDomNode(anchor).getBoundingClientRect();

                         // we iterate directly since safari messes up and doesn't return
                         // all the keys for the coods object when iterated
                         forEach(['width', 'height', 'top', 'left'], function (key) {
                             var value = coords[key];
                             switch (key) {
                                 case 'top':
                                     value += bodyNode.scrollTop;
                                     break;
                                 case 'left':
                                     value += bodyNode.scrollLeft;
                                     break;
                             }
                             styles[key] = Math.floor(value) + 'px';
                         });
                         return styles;
                     }

                     function prepareOutAnimation() {
                         var animator = $animateCss(clone, {
                             addClass: NG_OUT_ANCHOR_CLASS_NAME,
                             delay: true,
                             from: calculateAnchorStyles(outAnchor)
                         });

                         // read the comment within `prepareRegularAnimation` to understand
                         // why this check is necessary
                         return animator.$$willAnimate ? animator : null;
                     }

                     function getClassVal(element) {
                         return element.attr('class') || '';
                     }

                     function prepareInAnimation() {
                         var endingClasses = filterCssClasses(getClassVal(inAnchor));
                         var toAdd = getUniqueValues(endingClasses, startingClasses);
                         var toRemove = getUniqueValues(startingClasses, endingClasses);

                         var animator = $animateCss(clone, {
                             to: calculateAnchorStyles(inAnchor),
                             addClass: NG_IN_ANCHOR_CLASS_NAME + ' ' + toAdd,
                             removeClass: NG_OUT_ANCHOR_CLASS_NAME + ' ' + toRemove,
                             delay: true
                         });

                         // read the comment within `prepareRegularAnimation` to understand
                         // why this check is necessary
                         return animator.$$willAnimate ? animator : null;
                     }

                     function end() {
                         clone.remove();
                         outAnchor.removeClass(NG_ANIMATE_SHIM_CLASS_NAME);
                         inAnchor.removeClass(NG_ANIMATE_SHIM_CLASS_NAME);
                     }
                 }

                 function prepareFromToAnchorAnimation(from, to, classes, anchors) {
                     var fromAnimation = prepareRegularAnimation(from, noop);
                     var toAnimation = prepareRegularAnimation(to, noop);

                     var anchorAnimations = [];
                     forEach(anchors, function (anchor) {
                         var outElement = anchor['out'];
                         var inElement = anchor['in'];
                         var animator = prepareAnchoredAnimation(classes, outElement, inElement);
                         if (animator) {
                             anchorAnimations.push(animator);
                         }
                     });

                     // no point in doing anything when there are no elements to animate
                     if (!fromAnimation && !toAnimation && anchorAnimations.length === 0) return;

                     return {
                         start: function () {
                             var animationRunners = [];

                             if (fromAnimation) {
                                 animationRunners.push(fromAnimation.start());
                             }

                             if (toAnimation) {
                                 animationRunners.push(toAnimation.start());
                             }

                             forEach(anchorAnimations, function (animation) {
                                 animationRunners.push(animation.start());
                             });

                             var runner = new $$AnimateRunner({
                                 end: endFn,
                                 cancel: endFn // CSS-driven animations cannot be cancelled, only ended
                             });

                             $$AnimateRunner.all(animationRunners, function (status) {
                                 runner.complete(status);
                             });

                             return runner;

                             function endFn() {
                                 forEach(animationRunners, function (runner) {
                                     runner.end();
                                 });
                             }
                         }
                     };
                 }

                 function prepareRegularAnimation(animationDetails) {
                     var element = animationDetails.element;
                     var options = animationDetails.options || {};

                     if (animationDetails.structural) {
                         options.event = animationDetails.event;
                         options.structural = true;
                         options.applyClassesEarly = true;

                         // we special case the leave animation since we want to ensure that
                         // the element is removed as soon as the animation is over. Otherwise
                         // a flicker might appear or the element may not be removed at all
                         if (animationDetails.event === 'leave') {
                             options.onDone = options.domOperation;
                         }
                     }

                     // We assign the preparationClasses as the actual animation event since
                     // the internals of $animateCss will just suffix the event token values
                     // with `-active` to trigger the animation.
                     if (options.preparationClasses) {
                         options.event = concatWithSpace(options.event, options.preparationClasses);
                     }

                     var animator = $animateCss(element, options);

                     // the driver lookup code inside of $$animation attempts to spawn a
                     // driver one by one until a driver returns a.$$willAnimate animator object.
                     // $animateCss will always return an object, however, it will pass in
                     // a flag as a hint as to whether an animation was detected or not
                     return animator.$$willAnimate ? animator : null;
                 }
             }];
    }];

    // TODO(matsko): use caching here to speed things up for detection
    // TODO(matsko): add documentation
    //  by the time...

    var $$AnimateJsProvider = ['$animateProvider', function ($animateProvider) {
        this.$get = ['$injector', '$$AnimateRunner', '$$jqLite',
             function ($injector, $$AnimateRunner, $$jqLite) {

                 var applyAnimationClasses = applyAnimationClassesFactory($$jqLite);
                 // $animateJs(element, 'enter');
                 return function (element, event, classes, options) {
                     // the `classes` argument is optional and if it is not used
                     // then the classes will be resolved from the element's className
                     // property as well as options.addClass/options.removeClass.
                     if (arguments.length === 3 && isObject(classes)) {
                         options = classes;
                         classes = null;
                     }

                     options = prepareAnimationOptions(options);
                     if (!classes) {
                         classes = element.attr('class') || '';
                         if (options.addClass) {
                             classes += ' ' + options.addClass;
                         }
                         if (options.removeClass) {
                             classes += ' ' + options.removeClass;
                         }
                     }

                     var classesToAdd = options.addClass;
                     var classesToRemove = options.removeClass;

                     // the lookupAnimations function returns a series of animation objects that are
                     // matched up with one or more of the CSS classes. These animation objects are
                     // defined via the module.animation factory function. If nothing is detected then
                     // we don't return anything which then makes $animation query the next driver.
                     var animations = lookupAnimations(classes);
                     var before, after;
                     if (animations.length) {
                         var afterFn, beforeFn;
                         if (event == 'leave') {
                             beforeFn = 'leave';
                             afterFn = 'afterLeave'; // TODO(matsko): get rid of this
                         } else {
                             beforeFn = 'before' + event.charAt(0).toUpperCase() + event.substr(1);
                             afterFn = event;
                         }

                         if (event !== 'enter' && event !== 'move') {
                             before = packageAnimations(element, event, options, animations, beforeFn);
                         }
                         after = packageAnimations(element, event, options, animations, afterFn);
                     }

                     // no matching animations
                     if (!before && !after) return;

                     function applyOptions() {
                         options.domOperation();
                         applyAnimationClasses(element, options);
                     }

                     return {
                         start: function () {
                             var closeActiveAnimations;
                             var chain = [];

                             if (before) {
                                 chain.push(function (fn) {
                                     closeActiveAnimations = before(fn);
                                 });
                             }

                             if (chain.length) {
                                 chain.push(function (fn) {
                                     applyOptions();
                                     fn(true);
                                 });
                             } else {
                                 applyOptions();
                             }

                             if (after) {
                                 chain.push(function (fn) {
                                     closeActiveAnimations = after(fn);
                                 });
                             }

                             var animationClosed = false;
                             var runner = new $$AnimateRunner({
                                 end: function () {
                                     endAnimations();
                                 },
                                 cancel: function () {
                                     endAnimations(true);
                                 }
                             });

                             $$AnimateRunner.chain(chain, onComplete);
                             return runner;

                             function onComplete(success) {
                                 animationClosed = true;
                                 applyOptions();
                                 applyAnimationStyles(element, options);
                                 runner.complete(success);
                             }

                             function endAnimations(cancelled) {
                                 if (!animationClosed) {
                                     (closeActiveAnimations || noop)(cancelled);
                                     onComplete(cancelled);
                                 }
                             }
                         }
                     };

                     function executeAnimationFn(fn, element, event, options, onDone) {
                         var args;
                         switch (event) {
                             case 'animate':
                                 args = [element, options.from, options.to, onDone];
                                 break;

                             case 'setClass':
                                 args = [element, classesToAdd, classesToRemove, onDone];
                                 break;

                             case 'addClass':
                                 args = [element, classesToAdd, onDone];
                                 break;

                             case 'removeClass':
                                 args = [element, classesToRemove, onDone];
                                 break;

                             default:
                                 args = [element, onDone];
                                 break;
                         }

                         args.push(options);

                         var value = fn.apply(fn, args);
                         if (value) {
                             if (isFunction(value.start)) {
                                 value = value.start();
                             }

                             if (value instanceof $$AnimateRunner) {
                                 value.done(onDone);
                             } else if (isFunction(value)) {
                                 // optional onEnd / onCancel callback
                                 return value;
                             }
                         }

                         return noop;
                     }

                     function groupEventedAnimations(element, event, options, animations, fnName) {
                         var operations = [];
                         forEach(animations, function (ani) {
                             var animation = ani[fnName];
                             if (!animation) return;

                             // note that all of these animations will run in parallel
                             operations.push(function () {
                                 var runner;
                                 var endProgressCb;

                                 var resolved = false;
                                 var onAnimationComplete = function (rejected) {
                                     if (!resolved) {
                                         resolved = true;
                                         (endProgressCb || noop)(rejected);
                                         runner.complete(!rejected);
                                     }
                                 };

                                 runner = new $$AnimateRunner({
                                     end: function () {
                                         onAnimationComplete();
                                     },
                                     cancel: function () {
                                         onAnimationComplete(true);
                                     }
                                 });

                                 endProgressCb = executeAnimationFn(animation, element, event, options, function (result) {
                                     var cancelled = result === false;
                                     onAnimationComplete(cancelled);
                                 });

                                 return runner;
                             });
                         });

                         return operations;
                     }

                     function packageAnimations(element, event, options, animations, fnName) {
                         var operations = groupEventedAnimations(element, event, options, animations, fnName);
                         if (operations.length === 0) {
                             var a, b;
                             if (fnName === 'beforeSetClass') {
                                 a = groupEventedAnimations(element, 'removeClass', options, animations, 'beforeRemoveClass');
                                 b = groupEventedAnimations(element, 'addClass', options, animations, 'beforeAddClass');
                             } else if (fnName === 'setClass') {
                                 a = groupEventedAnimations(element, 'removeClass', options, animations, 'removeClass');
                                 b = groupEventedAnimations(element, 'addClass', options, animations, 'addClass');
                             }

                             if (a) {
                                 operations = operations.concat(a);
                             }
                             if (b) {
                                 operations = operations.concat(b);
                             }
                         }

                         if (operations.length === 0) return;

                         // TODO(matsko): add documentation
                         return function startAnimation(callback) {
                             var runners = [];
                             if (operations.length) {
                                 forEach(operations, function (animateFn) {
                                     runners.push(animateFn());
                                 });
                             }

                             runners.length ? $$AnimateRunner.all(runners, callback) : callback();

                             return function endFn(reject) {
                                 forEach(runners, function (runner) {
                                     reject ? runner.cancel() : runner.end();
                                 });
                             };
                         };
                     }
                 };

                 function lookupAnimations(classes) {
                     classes = isArray(classes) ? classes : classes.split(' ');
                     var matches = [], flagMap = {};
                     for (var i = 0; i < classes.length; i++) {
                         var klass = classes[i],
                             animationFactory = $animateProvider.$$registeredAnimations[klass];
                         if (animationFactory && !flagMap[klass]) {
                             matches.push($injector.get(animationFactory));
                             flagMap[klass] = true;
                         }
                     }
                     return matches;
                 }
             }];
    }];

    var $$AnimateJsDriverProvider = ['$$animationProvider', function ($$animationProvider) {
        $$animationProvider.drivers.push('$$animateJsDriver');
        this.$get = ['$$animateJs', '$$AnimateRunner', function ($$animateJs, $$AnimateRunner) {
            return function initDriverFn(animationDetails) {
                if (animationDetails.from && animationDetails.to) {
                    var fromAnimation = prepareAnimation(animationDetails.from);
                    var toAnimation = prepareAnimation(animationDetails.to);
                    if (!fromAnimation && !toAnimation) return;

                    return {
                        start: function () {
                            var animationRunners = [];

                            if (fromAnimation) {
                                animationRunners.push(fromAnimation.start());
                            }

                            if (toAnimation) {
                                animationRunners.push(toAnimation.start());
                            }

                            $$AnimateRunner.all(animationRunners, done);

                            var runner = new $$AnimateRunner({
                                end: endFnFactory(),
                                cancel: endFnFactory()
                            });

                            return runner;

                            function endFnFactory() {
                                return function () {
                                    forEach(animationRunners, function (runner) {
                                        // at this point we cannot cancel animations for groups just yet. 1.5+
                                        runner.end();
                                    });
                                };
                            }

                            function done(status) {
                                runner.complete(status);
                            }
                        }
                    };
                } else {
                    return prepareAnimation(animationDetails);
                }
            };

            function prepareAnimation(animationDetails) {
                // TODO(matsko): make sure to check for grouped animations and delegate down to normal animations
                var element = animationDetails.element;
                var event = animationDetails.event;
                var options = animationDetails.options;
                var classes = animationDetails.classes;
                return $$animateJs(element, event, classes, options);
            }
        }];
    }];

    var NG_ANIMATE_ATTR_NAME = 'data-ng-animate';
    var NG_ANIMATE_PIN_DATA = '$ngAnimatePin';
    var $$AnimateQueueProvider = ['$animateProvider', function ($animateProvider) {
        var PRE_DIGEST_STATE = 1;
        var RUNNING_STATE = 2;

        var rules = this.rules = {
            skip: [],
            cancel: [],
            join: []
        };

        function isAllowed(ruleType, element, currentAnimation, previousAnimation) {
            return rules[ruleType].some(function (fn) {
                return fn(element, currentAnimation, previousAnimation);
            });
        }

        function hasAnimationClasses(options, and) {
            options = options || {};
            var a = (options.addClass || '').length > 0;
            var b = (options.removeClass || '').length > 0;
            return and ? a && b : a || b;
        }

        rules.join.push(function (element, newAnimation, currentAnimation) {
            // if the new animation is class-based then we can just tack that on
            return !newAnimation.structural && hasAnimationClasses(newAnimation.options);
        });

        rules.skip.push(function (element, newAnimation, currentAnimation) {
            // there is no need to animate anything if no classes are being added and
            // there is no structural animation that will be triggered
            return !newAnimation.structural && !hasAnimationClasses(newAnimation.options);
        });

        rules.skip.push(function (element, newAnimation, currentAnimation) {
            // why should we trigger a new structural animation if the element will
            // be removed from the DOM anyway?
            return currentAnimation.event == 'leave' && newAnimation.structural;
        });

        rules.skip.push(function (element, newAnimation, currentAnimation) {
            // if there is an ongoing current animation then don't even bother running the class-based animation
            return currentAnimation.structural && currentAnimation.state === RUNNING_STATE && !newAnimation.structural;
        });

        rules.cancel.push(function (element, newAnimation, currentAnimation) {
            // there can never be two structural animations running at the same time
            return currentAnimation.structural && newAnimation.structural;
        });

        rules.cancel.push(function (element, newAnimation, currentAnimation) {
            // if the previous animation is already running, but the new animation will
            // be triggered, but the new animation is structural
            return currentAnimation.state === RUNNING_STATE && newAnimation.structural;
        });

        rules.cancel.push(function (element, newAnimation, currentAnimation) {
            var nO = newAnimation.options;
            var cO = currentAnimation.options;

            // if the exact same CSS class is added/removed then it's safe to cancel it
            return (nO.addClass && nO.addClass === cO.removeClass) || (nO.removeClass && nO.removeClass === cO.addClass);
        });

        this.$get = ['$$rAF', '$rootScope', '$rootElement', '$document', '$$HashMap',
                     '$$animation', '$$AnimateRunner', '$templateRequest', '$$jqLite', '$$forceReflow',
             function ($$rAF, $rootScope, $rootElement, $document, $$HashMap,
                      $$animation, $$AnimateRunner, $templateRequest, $$jqLite, $$forceReflow) {

                 var activeAnimationsLookup = new $$HashMap();
                 var disabledElementsLookup = new $$HashMap();
                 var animationsEnabled = null;

                 function postDigestTaskFactory() {
                     var postDigestCalled = false;
                     return function (fn) {
                         // we only issue a call to postDigest before
                         // it has first passed. This prevents any callbacks
                         // from not firing once the animation has completed
                         // since it will be out of the digest cycle.
                         if (postDigestCalled) {
                             fn();
                         } else {
                             $rootScope.$$postDigest(function () {
                                 postDigestCalled = true;
                                 fn();
                             });
                         }
                     };
                 }

                 // Wait until all directive and route-related templates are downloaded and
                 // compiled. The $templateRequest.totalPendingRequests variable keeps track of
                 // all of the remote templates being currently downloaded. If there are no
                 // templates currently downloading then the watcher will still fire anyway.
                 var deregisterWatch = $rootScope.$watch(
                   function () { return $templateRequest.totalPendingRequests === 0; },
                   function (isEmpty) {
                       if (!isEmpty) return;
                       deregisterWatch();

                       // Now that all templates have been downloaded, $animate will wait until
                       // the post digest queue is empty before enabling animations. By having two
                       // calls to $postDigest calls we can ensure that the flag is enabled at the
                       // very end of the post digest queue. Since all of the animations in $animate
                       // use $postDigest, it's important that the code below executes at the end.
                       // This basically means that the page is fully downloaded and compiled before
                       // any animations are triggered.
                       $rootScope.$$postDigest(function () {
                           $rootScope.$$postDigest(function () {
                               // we check for null directly in the event that the application already called
                               // .enabled() with whatever arguments that it provided it with
                               if (animationsEnabled === null) {
                                   animationsEnabled = true;
                               }
                           });
                       });
                   }
                 );

                 var callbackRegistry = {};

                 // remember that the classNameFilter is set during the provider/config
                 // stage therefore we can optimize here and setup a helper function
                 var classNameFilter = $animateProvider.classNameFilter();
                 var isAnimatableClassName = !classNameFilter
                           ? function () { return true; }
                           : function (className) {
                               return classNameFilter.test(className);
                           };

                 var applyAnimationClasses = applyAnimationClassesFactory($$jqLite);

                 function normalizeAnimationOptions(element, options) {
                     return mergeAnimationOptions(element, options, {});
                 }

                 function findCallbacks(parent, element, event) {
                     var targetNode = getDomNode(element);
                     var targetParentNode = getDomNode(parent);

                     var matches = [];
                     var entries = callbackRegistry[event];
                     if (entries) {
                         forEach(entries, function (entry) {
                             if (entry.node.contains(targetNode)) {
                                 matches.push(entry.callback);
                             } else if (event === 'leave' && entry.node.contains(targetParentNode)) {
                                 matches.push(entry.callback);
                             }
                         });
                     }

                     return matches;
                 }

                 return {
                     on: function (event, container, callback) {
                         var node = extractElementNode(container);
                         callbackRegistry[event] = callbackRegistry[event] || [];
                         callbackRegistry[event].push({
                             node: node,
                             callback: callback
                         });
                     },

                     off: function (event, container, callback) {
                         var entries = callbackRegistry[event];
                         if (!entries) return;

                         callbackRegistry[event] = arguments.length === 1
                             ? null
                             : filterFromRegistry(entries, container, callback);

                         function filterFromRegistry(list, matchContainer, matchCallback) {
                             var containerNode = extractElementNode(matchContainer);
                             return list.filter(function (entry) {
                                 var isMatch = entry.node === containerNode &&
                                                 (!matchCallback || entry.callback === matchCallback);
                                 return !isMatch;
                             });
                         }
                     },

                     pin: function (element, parentElement) {
                         assertArg(isElement(element), 'element', 'not an element');
                         assertArg(isElement(parentElement), 'parentElement', 'not an element');
                         element.data(NG_ANIMATE_PIN_DATA, parentElement);
                     },

                     push: function (element, event, options, domOperation) {
                         options = options || {};
                         options.domOperation = domOperation;
                         return queueAnimation(element, event, options);
                     },

                     // this method has four signatures:
                     //  () - global getter
                     //  (bool) - global setter
                     //  (element) - element getter
                     //  (element, bool) - element setter<F37>
                     enabled: function (element, bool) {
                         var argCount = arguments.length;

                         if (argCount === 0) {
                             // () - Global getter
                             bool = !!animationsEnabled;
                         } else {
                             var hasElement = isElement(element);

                             if (!hasElement) {
                                 // (bool) - Global setter
                                 bool = animationsEnabled = !!element;
                             } else {
                                 var node = getDomNode(element);
                                 var recordExists = disabledElementsLookup.get(node);

                                 if (argCount === 1) {
                                     // (element) - Element getter
                                     bool = !recordExists;
                                 } else {
                                     // (element, bool) - Element setter
                                     bool = !!bool;
                                     if (!bool) {
                                         disabledElementsLookup.put(node, true);
                                     } else if (recordExists) {
                                         disabledElementsLookup.remove(node);
                                     }
                                 }
                             }
                         }

                         return bool;
                     }
                 };

                 function queueAnimation(element, event, options) {
                     var node, parent;
                     element = stripCommentsFromElement(element);
                     if (element) {
                         node = getDomNode(element);
                         parent = element.parent();
                     }

                     options = prepareAnimationOptions(options);

                     // we create a fake runner with a working promise.
                     // These methods will become available after the digest has passed
                     var runner = new $$AnimateRunner();

                     // this is used to trigger callbacks in postDigest mode
                     var runInNextPostDigestOrNow = postDigestTaskFactory();

                     if (isArray(options.addClass)) {
                         options.addClass = options.addClass.join(' ');
                     }

                     if (options.addClass && !isString(options.addClass)) {
                         options.addClass = null;
                     }

                     if (isArray(options.removeClass)) {
                         options.removeClass = options.removeClass.join(' ');
                     }

                     if (options.removeClass && !isString(options.removeClass)) {
                         options.removeClass = null;
                     }

                     if (options.from && !isObject(options.from)) {
                         options.from = null;
                     }

                     if (options.to && !isObject(options.to)) {
                         options.to = null;
                     }

                     // there are situations where a directive issues an animation for
                     // a jqLite wrapper that contains only comment nodes... If this
                     // happens then there is no way we can perform an animation
                     if (!node) {
                         close();
                         return runner;
                     }

                     var className = [node.className, options.addClass, options.removeClass].join(' ');
                     if (!isAnimatableClassName(className)) {
                         close();
                         return runner;
                     }

                     var isStructural = ['enter', 'move', 'leave'].indexOf(event) >= 0;

                     // this is a hard disable of all animations for the application or on
                     // the element itself, therefore  there is no need to continue further
                     // past this point if not enabled
                     var skipAnimations = !animationsEnabled || disabledElementsLookup.get(node);
                     var existingAnimation = (!skipAnimations && activeAnimationsLookup.get(node)) || {};
                     var hasExistingAnimation = !!existingAnimation.state;

                     // there is no point in traversing the same collection of parent ancestors if a followup
                     // animation will be run on the same element that already did all that checking work
                     if (!skipAnimations && (!hasExistingAnimation || existingAnimation.state != PRE_DIGEST_STATE)) {
                         skipAnimations = !areAnimationsAllowed(element, parent, event);
                     }

                     if (skipAnimations) {
                         close();
                         return runner;
                     }

                     if (isStructural) {
                         closeChildAnimations(element);
                     }

                     var newAnimation = {
                         structural: isStructural,
                         element: element,
                         event: event,
                         close: close,
                         options: options,
                         runner: runner
                     };

                     if (hasExistingAnimation) {
                         var skipAnimationFlag = isAllowed('skip', element, newAnimation, existingAnimation);
                         if (skipAnimationFlag) {
                             if (existingAnimation.state === RUNNING_STATE) {
                                 close();
                                 return runner;
                             } else {
                                 mergeAnimationOptions(element, existingAnimation.options, options);
                                 return existingAnimation.runner;
                             }
                         }

                         var cancelAnimationFlag = isAllowed('cancel', element, newAnimation, existingAnimation);
                         if (cancelAnimationFlag) {
                             if (existingAnimation.state === RUNNING_STATE) {
                                 // this will end the animation right away and it is safe
                                 // to do so since the animation is already running and the
                                 // runner callback code will run in async
                                 existingAnimation.runner.end();
                             } else if (existingAnimation.structural) {
                                 // this means that the animation is queued into a digest, but
                                 // hasn't started yet. Therefore it is safe to run the close
                                 // method which will call the runner methods in async.
                                 existingAnimation.close();
                             } else {
                                 // this will merge the new animation options into existing animation options
                                 mergeAnimationOptions(element, existingAnimation.options, newAnimation.options);
                                 return existingAnimation.runner;
                             }
                         } else {
                             // a joined animation means that this animation will take over the existing one
                             // so an example would involve a leave animation taking over an enter. Then when
                             // the postDigest kicks in the enter will be ignored.
                             var joinAnimationFlag = isAllowed('join', element, newAnimation, existingAnimation);
                             if (joinAnimationFlag) {
                                 if (existingAnimation.state === RUNNING_STATE) {
                                     normalizeAnimationOptions(element, options);
                                 } else {
                                     applyGeneratedPreparationClasses(element, isStructural ? event : null, options);

                                     event = newAnimation.event = existingAnimation.event;
                                     options = mergeAnimationOptions(element, existingAnimation.options, newAnimation.options);

                                     //we return the same runner since only the option values of this animation will
                                     //be fed into the `existingAnimation`.
                                     return existingAnimation.runner;
                                 }
                             }
                         }
                     } else {
                         // normalization in this case means that it removes redundant CSS classes that
                         // already exist (addClass) or do not exist (removeClass) on the element
                         normalizeAnimationOptions(element, options);
                     }

                     // when the options are merged and cleaned up we may end up not having to do
                     // an animation at all, therefore we should check this before issuing a post
                     // digest callback. Structural animations will always run no matter what.
                     var isValidAnimation = newAnimation.structural;
                     if (!isValidAnimation) {
                         // animate (from/to) can be quickly checked first, otherwise we check if any classes are present
                         isValidAnimation = (newAnimation.event === 'animate' && Object.keys(newAnimation.options.to || {}).length > 0)
                                             || hasAnimationClasses(newAnimation.options);
                     }

                     if (!isValidAnimation) {
                         close();
                         clearElementAnimationState(element);
                         return runner;
                     }

                     // the counter keeps track of cancelled animations
                     var counter = (existingAnimation.counter || 0) + 1;
                     newAnimation.counter = counter;

                     markElementAnimationState(element, PRE_DIGEST_STATE, newAnimation);

                     $rootScope.$$postDigest(function () {
                         var animationDetails = activeAnimationsLookup.get(node);
                         var animationCancelled = !animationDetails;
                         animationDetails = animationDetails || {};

                         // if addClass/removeClass is called before something like enter then the
                         // registered parent element may not be present. The code below will ensure
                         // that a final value for parent element is obtained
                         var parentElement = element.parent() || [];

                         // animate/structural/class-based animations all have requirements. Otherwise there
                         // is no point in performing an animation. The parent node must also be set.
                         var isValidAnimation = parentElement.length > 0
                                                 && (animationDetails.event === 'animate'
                                                     || animationDetails.structural
                                                     || hasAnimationClasses(animationDetails.options));

                         // this means that the previous animation was cancelled
                         // even if the follow-up animation is the same event
                         if (animationCancelled || animationDetails.counter !== counter || !isValidAnimation) {
                             // if another animation did not take over then we need
                             // to make sure that the domOperation and options are
                             // handled accordingly
                             if (animationCancelled) {
                                 applyAnimationClasses(element, options);
                                 applyAnimationStyles(element, options);
                             }

                             // if the event changed from something like enter to leave then we do
                             // it, otherwise if it's the same then the end result will be the same too
                             if (animationCancelled || (isStructural && animationDetails.event !== event)) {
                                 options.domOperation();
                                 runner.end();
                             }

                             // in the event that the element animation was not cancelled or a follow-up animation
                             // isn't allowed to animate from here then we need to clear the state of the element
                             // so that any future animations won't read the expired animation data.
                             if (!isValidAnimation) {
                                 clearElementAnimationState(element);
                             }

                             return;
                         }

                         // this combined multiple class to addClass / removeClass into a setClass event
                         // so long as a structural event did not take over the animation
                         event = !animationDetails.structural && hasAnimationClasses(animationDetails.options, true)
                             ? 'setClass'
                             : animationDetails.event;

                         markElementAnimationState(element, RUNNING_STATE);
                         var realRunner = $$animation(element, event, animationDetails.options);

                         realRunner.done(function (status) {
                             close(!status);
                             var animationDetails = activeAnimationsLookup.get(node);
                             if (animationDetails && animationDetails.counter === counter) {
                                 clearElementAnimationState(getDomNode(element));
                             }
                             notifyProgress(runner, event, 'close', {});
                         });

                         // this will update the runner's flow-control events based on
                         // the `realRunner` object.
                         runner.setHost(realRunner);
                         notifyProgress(runner, event, 'start', {});
                     });

                     return runner;

                     function notifyProgress(runner, event, phase, data) {
                         runInNextPostDigestOrNow(function () {
                             var callbacks = findCallbacks(parent, element, event);
                             if (callbacks.length) {
                                 // do not optimize this call here to RAF because
                                 // we don't know how heavy the callback code here will
                                 // be and if this code is buffered then this can
                                 // lead to a performance regression.
                                 $$rAF(function () {
                                     forEach(callbacks, function (callback) {
                                         callback(element, phase, data);
                                     });
                                 });
                             }
                         });
                         runner.progress(event, phase, data);
                     }

                     function close(reject) { // jshint ignore:line
                         clearGeneratedClasses(element, options);
                         applyAnimationClasses(element, options);
                         applyAnimationStyles(element, options);
                         options.domOperation();
                         runner.complete(!reject);
                     }
                 }

                 function closeChildAnimations(element) {
                     var node = getDomNode(element);
                     var children = node.querySelectorAll('[' + NG_ANIMATE_ATTR_NAME + ']');
                     forEach(children, function (child) {
                         var state = parseInt(child.getAttribute(NG_ANIMATE_ATTR_NAME));
                         var animationDetails = activeAnimationsLookup.get(child);
                         switch (state) {
                             case RUNNING_STATE:
                                 animationDetails.runner.end();
                                 /* falls through */
                             case PRE_DIGEST_STATE:
                                 if (animationDetails) {
                                     activeAnimationsLookup.remove(child);
                                 }
                                 break;
                         }
                     });
                 }

                 function clearElementAnimationState(element) {
                     var node = getDomNode(element);
                     node.removeAttribute(NG_ANIMATE_ATTR_NAME);
                     activeAnimationsLookup.remove(node);
                 }

                 function isMatchingElement(nodeOrElmA, nodeOrElmB) {
                     return getDomNode(nodeOrElmA) === getDomNode(nodeOrElmB);
                 }

                 function areAnimationsAllowed(element, parentElement, event) {
                     var bodyElement = jqLite($document[0].body);
                     var bodyElementDetected = isMatchingElement(element, bodyElement) || element[0].nodeName === 'HTML';
                     var rootElementDetected = isMatchingElement(element, $rootElement);
                     var parentAnimationDetected = false;
                     var animateChildren;

                     var parentHost = element.data(NG_ANIMATE_PIN_DATA);
                     if (parentHost) {
                         parentElement = parentHost;
                     }

                     while (parentElement && parentElement.length) {
                         if (!rootElementDetected) {
                             // angular doesn't want to attempt to animate elements outside of the application
                             // therefore we need to ensure that the rootElement is an ancestor of the current element
                             rootElementDetected = isMatchingElement(parentElement, $rootElement);
                         }

                         var parentNode = parentElement[0];
                         if (parentNode.nodeType !== ELEMENT_NODE) {
                             // no point in inspecting the #document element
                             break;
                         }

                         var details = activeAnimationsLookup.get(parentNode) || {};
                         // either an enter, leave or move animation will commence
                         // therefore we can't allow any animations to take place
                         // but if a parent animation is class-based then that's ok
                         if (!parentAnimationDetected) {
                             parentAnimationDetected = details.structural || disabledElementsLookup.get(parentNode);
                         }

                         if (isUndefined(animateChildren) || animateChildren === true) {
                             var value = parentElement.data(NG_ANIMATE_CHILDREN_DATA);
                             if (isDefined(value)) {
                                 animateChildren = value;
                             }
                         }

                         // there is no need to continue traversing at this point
                         if (parentAnimationDetected && animateChildren === false) break;

                         if (!rootElementDetected) {
                             // angular doesn't want to attempt to animate elements outside of the application
                             // therefore we need to ensure that the rootElement is an ancestor of the current element
                             rootElementDetected = isMatchingElement(parentElement, $rootElement);
                             if (!rootElementDetected) {
                                 parentHost = parentElement.data(NG_ANIMATE_PIN_DATA);
                                 if (parentHost) {
                                     parentElement = parentHost;
                                 }
                             }
                         }

                         if (!bodyElementDetected) {
                             // we also need to ensure that the element is or will be apart of the body element
                             // otherwise it is pointless to even issue an animation to be rendered
                             bodyElementDetected = isMatchingElement(parentElement, bodyElement);
                         }

                         parentElement = parentElement.parent();
                     }

                     var allowAnimation = !parentAnimationDetected || animateChildren;
                     return allowAnimation && rootElementDetected && bodyElementDetected;
                 }

                 function markElementAnimationState(element, state, details) {
                     details = details || {};
                     details.state = state;

                     var node = getDomNode(element);
                     node.setAttribute(NG_ANIMATE_ATTR_NAME, state);

                     var oldValue = activeAnimationsLookup.get(node);
                     var newValue = oldValue
                         ? extend(oldValue, details)
                         : details;
                     activeAnimationsLookup.put(node, newValue);
                 }
             }];
    }];

    var $$AnimateAsyncRunFactory = ['$$rAF', function ($$rAF) {
        var waitQueue = [];

        function waitForTick(fn) {
            waitQueue.push(fn);
            if (waitQueue.length > 1) return;
            $$rAF(function () {
                for (var i = 0; i < waitQueue.length; i++) {
                    waitQueue[i]();
                }
                waitQueue = [];
            });
        }

        return function () {
            var passed = false;
            waitForTick(function () {
                passed = true;
            });
            return function (callback) {
                passed ? callback() : waitForTick(callback);
            };
        };
    }];

    var $$AnimateRunnerFactory = ['$q', '$sniffer', '$$animateAsyncRun',
                          function ($q, $sniffer, $$animateAsyncRun) {

                              var INITIAL_STATE = 0;
                              var DONE_PENDING_STATE = 1;
                              var DONE_COMPLETE_STATE = 2;

                              AnimateRunner.chain = function (chain, callback) {
                                  var index = 0;

                                  next();
                                  function next() {
                                      if (index === chain.length) {
                                          callback(true);
                                          return;
                                      }

                                      chain[index](function (response) {
                                          if (response === false) {
                                              callback(false);
                                              return;
                                          }
                                          index++;
                                          next();
                                      });
                                  }
                              };

                              AnimateRunner.all = function (runners, callback) {
                                  var count = 0;
                                  var status = true;
                                  forEach(runners, function (runner) {
                                      runner.done(onProgress);
                                  });

                                  function onProgress(response) {
                                      status = status && response;
                                      if (++count === runners.length) {
                                          callback(status);
                                      }
                                  }
                              };

                              function AnimateRunner(host) {
                                  this.setHost(host);

                                  this._doneCallbacks = [];
                                  this._runInAnimationFrame = $$animateAsyncRun();
                                  this._state = 0;
                              }

                              AnimateRunner.prototype = {
                                  setHost: function (host) {
                                      this.host = host || {};
                                  },

                                  done: function (fn) {
                                      if (this._state === DONE_COMPLETE_STATE) {
                                          fn();
                                      } else {
                                          this._doneCallbacks.push(fn);
                                      }
                                  },

                                  progress: noop,

                                  getPromise: function () {
                                      if (!this.promise) {
                                          var self = this;
                                          this.promise = $q(function (resolve, reject) {
                                              self.done(function (status) {
                                                  status === false ? reject() : resolve();
                                              });
                                          });
                                      }
                                      return this.promise;
                                  },

                                  then: function (resolveHandler, rejectHandler) {
                                      return this.getPromise().then(resolveHandler, rejectHandler);
                                  },

                                  'catch': function (handler) {
                                      return this.getPromise()['catch'](handler);
                                  },

                                  'finally': function (handler) {
                                      return this.getPromise()['finally'](handler);
                                  },

                                  pause: function () {
                                      if (this.host.pause) {
                                          this.host.pause();
                                      }
                                  },

                                  resume: function () {
                                      if (this.host.resume) {
                                          this.host.resume();
                                      }
                                  },

                                  end: function () {
                                      if (this.host.end) {
                                          this.host.end();
                                      }
                                      this._resolve(true);
                                  },

                                  cancel: function () {
                                      if (this.host.cancel) {
                                          this.host.cancel();
                                      }
                                      this._resolve(false);
                                  },

                                  complete: function (response) {
                                      var self = this;
                                      if (self._state === INITIAL_STATE) {
                                          self._state = DONE_PENDING_STATE;
                                          self._runInAnimationFrame(function () {
                                              self._resolve(response);
                                          });
                                      }
                                  },

                                  _resolve: function (response) {
                                      if (this._state !== DONE_COMPLETE_STATE) {
                                          forEach(this._doneCallbacks, function (fn) {
                                              fn(response);
                                          });
                                          this._doneCallbacks.length = 0;
                                          this._state = DONE_COMPLETE_STATE;
                                      }
                                  }
                              };

                              return AnimateRunner;
                          }];

    var $$AnimationProvider = ['$animateProvider', function ($animateProvider) {
        var NG_ANIMATE_REF_ATTR = 'ng-animate-ref';

        var drivers = this.drivers = [];

        var RUNNER_STORAGE_KEY = '$$animationRunner';

        function setRunner(element, runner) {
            element.data(RUNNER_STORAGE_KEY, runner);
        }

        function removeRunner(element) {
            element.removeData(RUNNER_STORAGE_KEY);
        }

        function getRunner(element) {
            return element.data(RUNNER_STORAGE_KEY);
        }

        this.$get = ['$$jqLite', '$rootScope', '$injector', '$$AnimateRunner', '$$HashMap', '$$rAFScheduler',
             function ($$jqLite, $rootScope, $injector, $$AnimateRunner, $$HashMap, $$rAFScheduler) {

                 var animationQueue = [];
                 var applyAnimationClasses = applyAnimationClassesFactory($$jqLite);

                 function sortAnimations(animations) {
                     var tree = { children: [] };
                     var i, lookup = new $$HashMap();

                     // this is done first beforehand so that the hashmap
                     // is filled with a list of the elements that will be animated
                     for (i = 0; i < animations.length; i++) {
                         var animation = animations[i];
                         lookup.put(animation.domNode, animations[i] = {
                             domNode: animation.domNode,
                             fn: animation.fn,
                             children: []
                         });
                     }

                     for (i = 0; i < animations.length; i++) {
                         processNode(animations[i]);
                     }

                     return flatten(tree);

                     function processNode(entry) {
                         if (entry.processed) return entry;
                         entry.processed = true;

                         var elementNode = entry.domNode;
                         var parentNode = elementNode.parentNode;
                         lookup.put(elementNode, entry);

                         var parentEntry;
                         while (parentNode) {
                             parentEntry = lookup.get(parentNode);
                             if (parentEntry) {
                                 if (!parentEntry.processed) {
                                     parentEntry = processNode(parentEntry);
                                 }
                                 break;
                             }
                             parentNode = parentNode.parentNode;
                         }

                         (parentEntry || tree).children.push(entry);
                         return entry;
                     }

                     function flatten(tree) {
                         var result = [];
                         var queue = [];
                         var i;

                         for (i = 0; i < tree.children.length; i++) {
                             queue.push(tree.children[i]);
                         }

                         var remainingLevelEntries = queue.length;
                         var nextLevelEntries = 0;
                         var row = [];

                         for (i = 0; i < queue.length; i++) {
                             var entry = queue[i];
                             if (remainingLevelEntries <= 0) {
                                 remainingLevelEntries = nextLevelEntries;
                                 nextLevelEntries = 0;
                                 result.push(row);
                                 row = [];
                             }
                             row.push(entry.fn);
                             entry.children.forEach(function (childEntry) {
                                 nextLevelEntries++;
                                 queue.push(childEntry);
                             });
                             remainingLevelEntries--;
                         }

                         if (row.length) {
                             result.push(row);
                         }

                         return result;
                     }
                 }

                 // TODO(matsko): document the signature in a better way
                 return function (element, event, options) {
                     options = prepareAnimationOptions(options);
                     var isStructural = ['enter', 'move', 'leave'].indexOf(event) >= 0;

                     // there is no animation at the current moment, however
                     // these runner methods will get later updated with the
                     // methods leading into the driver's end/cancel methods
                     // for now they just stop the animation from starting
                     var runner = new $$AnimateRunner({
                         end: function () { close(); },
                         cancel: function () { close(true); }
                     });

                     if (!drivers.length) {
                         close();
                         return runner;
                     }

                     setRunner(element, runner);

                     var classes = mergeClasses(element.attr('class'), mergeClasses(options.addClass, options.removeClass));
                     var tempClasses = options.tempClasses;
                     if (tempClasses) {
                         classes += ' ' + tempClasses;
                         options.tempClasses = null;
                     }

                     animationQueue.push({
                         // this data is used by the postDigest code and passed into
                         // the driver step function
                         element: element,
                         classes: classes,
                         event: event,
                         structural: isStructural,
                         options: options,
                         beforeStart: beforeStart,
                         close: close
                     });

                     element.on('$destroy', handleDestroyedElement);

                     // we only want there to be one function called within the post digest
                     // block. This way we can group animations for all the animations that
                     // were apart of the same postDigest flush call.
                     if (animationQueue.length > 1) return runner;

                     $rootScope.$$postDigest(function () {
                         var animations = [];
                         forEach(animationQueue, function (entry) {
                             // the element was destroyed early on which removed the runner
                             // form its storage. This means we can't animate this element
                             // at all and it already has been closed due to destruction.
                             if (getRunner(entry.element)) {
                                 animations.push(entry);
                             } else {
                                 entry.close();
                             }
                         });

                         // now any future animations will be in another postDigest
                         animationQueue.length = 0;

                         var groupedAnimations = groupAnimations(animations);
                         var toBeSortedAnimations = [];

                         forEach(groupedAnimations, function (animationEntry) {
                             toBeSortedAnimations.push({
                                 domNode: getDomNode(animationEntry.from ? animationEntry.from.element : animationEntry.element),
                                 fn: function triggerAnimationStart() {
                                     // it's important that we apply the `ng-animate` CSS class and the
                                     // temporary classes before we do any driver invoking since these
                                     // CSS classes may be required for proper CSS detection.
                                     animationEntry.beforeStart();

                                     var startAnimationFn, closeFn = animationEntry.close;

                                     // in the event that the element was removed before the digest runs or
                                     // during the RAF sequencing then we should not trigger the animation.
                                     var targetElement = animationEntry.anchors
                                         ? (animationEntry.from.element || animationEntry.to.element)
                                         : animationEntry.element;

                                     if (getRunner(targetElement)) {
                                         var operation = invokeFirstDriver(animationEntry);
                                         if (operation) {
                                             startAnimationFn = operation.start;
                                         }
                                     }

                                     if (!startAnimationFn) {
                                         closeFn();
                                     } else {
                                         var animationRunner = startAnimationFn();
                                         animationRunner.done(function (status) {
                                             closeFn(!status);
                                         });
                                         updateAnimationRunners(animationEntry, animationRunner);
                                     }
                                 }
                             });
                         });

                         // we need to sort each of the animations in order of parent to child
                         // relationships. This ensures that the child classes are applied at the
                         // right time.
                         $$rAFScheduler(sortAnimations(toBeSortedAnimations));
                     });

                     return runner;

                     // TODO(matsko): change to reference nodes
                     function getAnchorNodes(node) {
                         var SELECTOR = '[' + NG_ANIMATE_REF_ATTR + ']';
                         var items = node.hasAttribute(NG_ANIMATE_REF_ATTR)
                               ? [node]
                               : node.querySelectorAll(SELECTOR);
                         var anchors = [];
                         forEach(items, function (node) {
                             var attr = node.getAttribute(NG_ANIMATE_REF_ATTR);
                             if (attr && attr.length) {
                                 anchors.push(node);
                             }
                         });
                         return anchors;
                     }

                     function groupAnimations(animations) {
                         var preparedAnimations = [];
                         var refLookup = {};
                         forEach(animations, function (animation, index) {
                             var element = animation.element;
                             var node = getDomNode(element);
                             var event = animation.event;
                             var enterOrMove = ['enter', 'move'].indexOf(event) >= 0;
                             var anchorNodes = animation.structural ? getAnchorNodes(node) : [];

                             if (anchorNodes.length) {
                                 var direction = enterOrMove ? 'to' : 'from';

                                 forEach(anchorNodes, function (anchor) {
                                     var key = anchor.getAttribute(NG_ANIMATE_REF_ATTR);
                                     refLookup[key] = refLookup[key] || {};
                                     refLookup[key][direction] = {
                                         animationID: index,
                                         element: jqLite(anchor)
                                     };
                                 });
                             } else {
                                 preparedAnimations.push(animation);
                             }
                         });

                         var usedIndicesLookup = {};
                         var anchorGroups = {};
                         forEach(refLookup, function (operations, key) {
                             var from = operations.from;
                             var to = operations.to;

                             if (!from || !to) {
                                 // only one of these is set therefore we can't have an
                                 // anchor animation since all three pieces are required
                                 var index = from ? from.animationID : to.animationID;
                                 var indexKey = index.toString();
                                 if (!usedIndicesLookup[indexKey]) {
                                     usedIndicesLookup[indexKey] = true;
                                     preparedAnimations.push(animations[index]);
                                 }
                                 return;
                             }

                             var fromAnimation = animations[from.animationID];
                             var toAnimation = animations[to.animationID];
                             var lookupKey = from.animationID.toString();
                             if (!anchorGroups[lookupKey]) {
                                 var group = anchorGroups[lookupKey] = {
                                     structural: true,
                                     beforeStart: function () {
                                         fromAnimation.beforeStart();
                                         toAnimation.beforeStart();
                                     },
                                     close: function () {
                                         fromAnimation.close();
                                         toAnimation.close();
                                     },
                                     classes: cssClassesIntersection(fromAnimation.classes, toAnimation.classes),
                                     from: fromAnimation,
                                     to: toAnimation,
                                     anchors: [] // TODO(matsko): change to reference nodes
                                 };

                                 // the anchor animations require that the from and to elements both have at least
                                 // one shared CSS class which effictively marries the two elements together to use
                                 // the same animation driver and to properly sequence the anchor animation.
                                 if (group.classes.length) {
                                     preparedAnimations.push(group);
                                 } else {
                                     preparedAnimations.push(fromAnimation);
                                     preparedAnimations.push(toAnimation);
                                 }
                             }

                             anchorGroups[lookupKey].anchors.push({
                                 'out': from.element, 'in': to.element
                             });
                         });

                         return preparedAnimations;
                     }

                     function cssClassesIntersection(a, b) {
                         a = a.split(' ');
                         b = b.split(' ');
                         var matches = [];

                         for (var i = 0; i < a.length; i++) {
                             var aa = a[i];
                             if (aa.substring(0, 3) === 'ng-') continue;

                             for (var j = 0; j < b.length; j++) {
                                 if (aa === b[j]) {
                                     matches.push(aa);
                                     break;
                                 }
                             }
                         }

                         return matches.join(' ');
                     }

                     function invokeFirstDriver(animationDetails) {
                         // we loop in reverse order since the more general drivers (like CSS and JS)
                         // may attempt more elements, but custom drivers are more particular
                         for (var i = drivers.length - 1; i >= 0; i--) {
                             var driverName = drivers[i];
                             if (!$injector.has(driverName)) continue; // TODO(matsko): remove this check

                             var factory = $injector.get(driverName);
                             var driver = factory(animationDetails);
                             if (driver) {
                                 return driver;
                             }
                         }
                     }

                     function beforeStart() {
                         element.addClass(NG_ANIMATE_CLASSNAME);
                         if (tempClasses) {
                             $$jqLite.addClass(element, tempClasses);
                         }
                     }

                     function updateAnimationRunners(animation, newRunner) {
                         if (animation.from && animation.to) {
                             update(animation.from.element);
                             update(animation.to.element);
                         } else {
                             update(animation.element);
                         }

                         function update(element) {
                             getRunner(element).setHost(newRunner);
                         }
                     }

                     function handleDestroyedElement() {
                         var runner = getRunner(element);
                         if (runner && (event !== 'leave' || !options.$$domOperationFired)) {
                             runner.end();
                         }
                     }

                     function close(rejected) { // jshint ignore:line
                         element.off('$destroy', handleDestroyedElement);
                         removeRunner(element);

                         applyAnimationClasses(element, options);
                         applyAnimationStyles(element, options);
                         options.domOperation();

                         if (tempClasses) {
                             $$jqLite.removeClass(element, tempClasses);
                         }

                         element.removeClass(NG_ANIMATE_CLASSNAME);
                         runner.complete(!rejected);
                     }
                 };
             }];
    }];

    /* global angularAnimateModule: true,
    
       $$AnimateAsyncRunFactory,
       $$rAFSchedulerFactory,
       $$AnimateChildrenDirective,
       $$AnimateRunnerFactory,
       $$AnimateQueueProvider,
       $$AnimationProvider,
       $AnimateCssProvider,
       $$AnimateCssDriverProvider,
       $$AnimateJsProvider,
       $$AnimateJsDriverProvider,
    */

    /**
     * @ngdoc module
     * @name ngAnimate
     * @description
     *
     * The `ngAnimate` module provides support for CSS-based animations (keyframes and transitions) as well as JavaScript-based animations via
     * callback hooks. Animations are not enabled by default, however, by including `ngAnimate` the animation hooks are enabled for an Angular app.
     *
     * <div doc-module-components="ngAnimate"></div>
     *
     * # Usage
     * Simply put, there are two ways to make use of animations when ngAnimate is used: by using **CSS** and **JavaScript**. The former works purely based
     * using CSS (by using matching CSS selectors/styles) and the latter triggers animations that are registered via `module.animation()`. For
     * both CSS and JS animations the sole requirement is to have a matching `CSS class` that exists both in the registered animation and within
     * the HTML element that the animation will be triggered on.
     *
     * ## Directive Support
     * The following directives are "animation aware":
     *
     * | Directive                                                                                                | Supported Animations                                                     |
     * |----------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------|
     * | {@link ng.directive:ngRepeat#animations ngRepeat}                                                        | enter, leave and move                                                    |
     * | {@link ngRoute.directive:ngView#animations ngView}                                                       | enter and leave                                                          |
     * | {@link ng.directive:ngInclude#animations ngInclude}                                                      | enter and leave                                                          |
     * | {@link ng.directive:ngSwitch#animations ngSwitch}                                                        | enter and leave                                                          |
     * | {@link ng.directive:ngIf#animations ngIf}                                                                | enter and leave                                                          |
     * | {@link ng.directive:ngClass#animations ngClass}                                                          | add and remove (the CSS class(es) present)                               |
     * | {@link ng.directive:ngShow#animations ngShow} & {@link ng.directive:ngHide#animations ngHide}            | add and remove (the ng-hide class value)                                 |
     * | {@link ng.directive:form#animation-hooks form} & {@link ng.directive:ngModel#animation-hooks ngModel}    | add and remove (dirty, pristine, valid, invalid & all other validations) |
     * | {@link module:ngMessages#animations ngMessages}                                                          | add and remove (ng-active & ng-inactive)                                 |
     * | {@link module:ngMessages#animations ngMessage}                                                           | enter and leave                                                          |
     *
     * (More information can be found by visiting each the documentation associated with each directive.)
     *
     * ## CSS-based Animations
     *
     * CSS-based animations with ngAnimate are unique since they require no JavaScript code at all. By using a CSS class that we reference between our HTML
     * and CSS code we can create an animation that will be picked up by Angular when an the underlying directive performs an operation.
     *
     * The example below shows how an `enter` animation can be made possible on an element using `ng-if`:
     *
     * ```html
     * <div ng-if="bool" class="fade">
     *    Fade me in out
     * </div>
     * <button ng-click="bool=true">Fade In!</button>
     * <button ng-click="bool=false">Fade Out!</button>
     * ```
     *
     * Notice the CSS class **fade**? We can now create the CSS transition code that references this class:
     *
     * ```css
     * /&#42; The starting CSS styles for the enter animation &#42;/
     * .fade.ng-enter {
     *   transition:0.5s linear all;
     *   opacity:0;
     * }
     *
     * /&#42; The finishing CSS styles for the enter animation &#42;/
     * .fade.ng-enter.ng-enter-active {
     *   opacity:1;
     * }
     * ```
     *
     * The key thing to remember here is that, depending on the animation event (which each of the directives above trigger depending on what's going on) two
     * generated CSS classes will be applied to the element; in the example above we have `.ng-enter` and `.ng-enter-active`. For CSS transitions, the transition
     * code **must** be defined within the starting CSS class (in this case `.ng-enter`). The destination class is what the transition will animate towards.
     *
     * If for example we wanted to create animations for `leave` and `move` (ngRepeat triggers move) then we can do so using the same CSS naming conventions:
     *
     * ```css
     * /&#42; now the element will fade out before it is removed from the DOM &#42;/
     * .fade.ng-leave {
     *   transition:0.5s linear all;
     *   opacity:1;
     * }
     * .fade.ng-leave.ng-leave-active {
     *   opacity:0;
     * }
     * ```
     *
     * We can also make use of **CSS Keyframes** by referencing the keyframe animation within the starting CSS class:
     *
     * ```css
     * /&#42; there is no need to define anything inside of the destination
     * CSS class since the keyframe will take charge of the animation &#42;/
     * .fade.ng-leave {
     *   animation: my_fade_animation 0.5s linear;
     *   -webkit-animation: my_fade_animation 0.5s linear;
     * }
     *
     * @keyframes my_fade_animation {
     *   from { opacity:1; }
     *   to { opacity:0; }
     * }
     *
     * @-webkit-keyframes my_fade_animation {
     *   from { opacity:1; }
     *   to { opacity:0; }
     * }
     * ```
     *
     * Feel free also mix transitions and keyframes together as well as any other CSS classes on the same element.
     *
     * ### CSS Class-based Animations
     *
     * Class-based animations (animations that are triggered via `ngClass`, `ngShow`, `ngHide` and some other directives) have a slightly different
     * naming convention. Class-based animations are basic enough that a standard transition or keyframe can be referenced on the class being added
     * and removed.
     *
     * For example if we wanted to do a CSS animation for `ngHide` then we place an animation on the `.ng-hide` CSS class:
     *
     * ```html
     * <div ng-show="bool" class="fade">
     *   Show and hide me
     * </div>
     * <button ng-click="bool=true">Toggle</button>
     *
     * <style>
     * .fade.ng-hide {
     *   transition:0.5s linear all;
     *   opacity:0;
     * }
     * </style>
     * ```
     *
     * All that is going on here with ngShow/ngHide behind the scenes is the `.ng-hide` class is added/removed (when the hidden state is valid). Since
     * ngShow and ngHide are animation aware then we can match up a transition and ngAnimate handles the rest.
     *
     * In addition the addition and removal of the CSS class, ngAnimate also provides two helper methods that we can use to further decorate the animation
     * with CSS styles.
     *
     * ```html
     * <div ng-class="{on:onOff}" class="highlight">
     *   Highlight this box
     * </div>
     * <button ng-click="onOff=!onOff">Toggle</button>
     *
     * <style>
     * .highlight {
     *   transition:0.5s linear all;
     * }
     * .highlight.on-add {
     *   background:white;
     * }
     * .highlight.on {
     *   background:yellow;
     * }
     * .highlight.on-remove {
     *   background:black;
     * }
     * </style>
     * ```
     *
     * We can also make use of CSS keyframes by placing them within the CSS classes.
     *
     *
     * ### CSS Staggering Animations
     * A Staggering animation is a collection of animations that are issued with a slight delay in between each successive operation resulting in a
     * curtain-like effect. The ngAnimate module (versions >=1.2) supports staggering animations and the stagger effect can be
     * performed by creating a **ng-EVENT-stagger** CSS class and attaching that class to the base CSS class used for
     * the animation. The style property expected within the stagger class can either be a **transition-delay** or an
     * **animation-delay** property (or both if your animation contains both transitions and keyframe animations).
     *
     * ```css
     * .my-animation.ng-enter {
     *   /&#42; standard transition code &#42;/
     *   transition: 1s linear all;
     *   opacity:0;
     * }
     * .my-animation.ng-enter-stagger {
     *   /&#42; this will have a 100ms delay between each successive leave animation &#42;/
     *   transition-delay: 0.1s;
     *
     *   /&#42; As of 1.4.4, this must always be set: it signals ngAnimate
     *     to not accidentally inherit a delay property from another CSS class &#42;/
     *   transition-duration: 0s;
     * }
     * .my-animation.ng-enter.ng-enter-active {
     *   /&#42; standard transition styles &#42;/
     *   opacity:1;
     * }
     * ```
     *
     * Staggering animations work by default in ngRepeat (so long as the CSS class is defined). Outside of ngRepeat, to use staggering animations
     * on your own, they can be triggered by firing multiple calls to the same event on $animate. However, the restrictions surrounding this
     * are that each of the elements must have the same CSS className value as well as the same parent element. A stagger operation
     * will also be reset if one or more animation frames have passed since the multiple calls to `$animate` were fired.
     *
     * The following code will issue the **ng-leave-stagger** event on the element provided:
     *
     * ```js
     * var kids = parent.children();
     *
     * $animate.leave(kids[0]); //stagger index=0
     * $animate.leave(kids[1]); //stagger index=1
     * $animate.leave(kids[2]); //stagger index=2
     * $animate.leave(kids[3]); //stagger index=3
     * $animate.leave(kids[4]); //stagger index=4
     *
     * window.requestAnimationFrame(function() {
     *   //stagger has reset itself
     *   $animate.leave(kids[5]); //stagger index=0
     *   $animate.leave(kids[6]); //stagger index=1
     *
     *   $scope.$digest();
     * });
     * ```
     *
     * Stagger animations are currently only supported within CSS-defined animations.
     *
     * ### The `ng-animate` CSS class
     *
     * When ngAnimate is animating an element it will apply the `ng-animate` CSS class to the element for the duration of the animation.
     * This is a temporary CSS class and it will be removed once the animation is over (for both JavaScript and CSS-based animations).
     *
     * Therefore, animations can be applied to an element using this temporary class directly via CSS.
     *
     * ```css
     * .zipper.ng-animate {
     *   transition:0.5s linear all;
     * }
     * .zipper.ng-enter {
     *   opacity:0;
     * }
     * .zipper.ng-enter.ng-enter-active {
     *   opacity:1;
     * }
     * .zipper.ng-leave {
     *   opacity:1;
     * }
     * .zipper.ng-leave.ng-leave-active {
     *   opacity:0;
     * }
     * ```
     *
     * (Note that the `ng-animate` CSS class is reserved and it cannot be applied on an element directly since ngAnimate will always remove
     * the CSS class once an animation has completed.)
     *
     *
     * ## JavaScript-based Animations
     *
     * ngAnimate also allows for animations to be consumed by JavaScript code. The approach is similar to CSS-based animations (where there is a shared
     * CSS class that is referenced in our HTML code) but in addition we need to register the JavaScript animation on the module. By making use of the
     * `module.animation()` module function we can register the ainmation.
     *
     * Let's see an example of a enter/leave animation using `ngRepeat`:
     *
     * ```html
     * <div ng-repeat="item in items" class="slide">
     *   {{ item }}
     * </div>
     * ```
     *
     * See the **slide** CSS class? Let's use that class to define an animation that we'll structure in our module code by using `module.animation`:
     *
     * ```js
     * myModule.animation('.slide', [function() {
     *   return {
     *     // make note that other events (like addClass/removeClass)
     *     // have different function input parameters
     *     enter: function(element, doneFn) {
     *       jQuery(element).fadeIn(1000, doneFn);
     *
     *       // remember to call doneFn so that angular
     *       // knows that the animation has concluded
     *     },
     *
     *     move: function(element, doneFn) {
     *       jQuery(element).fadeIn(1000, doneFn);
     *     },
     *
     *     leave: function(element, doneFn) {
     *       jQuery(element).fadeOut(1000, doneFn);
     *     }
     *   }
     * }]);
     * ```
     *
     * The nice thing about JS-based animations is that we can inject other services and make use of advanced animation libraries such as
     * greensock.js and velocity.js.
     *
     * If our animation code class-based (meaning that something like `ngClass`, `ngHide` and `ngShow` triggers it) then we can still define
     * our animations inside of the same registered animation, however, the function input arguments are a bit different:
     *
     * ```html
     * <div ng-class="color" class="colorful">
     *   this box is moody
     * </div>
     * <button ng-click="color='red'">Change to red</button>
     * <button ng-click="color='blue'">Change to blue</button>
     * <button ng-click="color='green'">Change to green</button>
     * ```
     *
     * ```js
     * myModule.animation('.colorful', [function() {
     *   return {
     *     addClass: function(element, className, doneFn) {
     *       // do some cool animation and call the doneFn
     *     },
     *     removeClass: function(element, className, doneFn) {
     *       // do some cool animation and call the doneFn
     *     },
     *     setClass: function(element, addedClass, removedClass, doneFn) {
     *       // do some cool animation and call the doneFn
     *     }
     *   }
     * }]);
     * ```
     *
     * ## CSS + JS Animations Together
     *
     * AngularJS 1.4 and higher has taken steps to make the amalgamation of CSS and JS animations more flexible. However, unlike earlier versions of Angular,
     * defining CSS and JS animations to work off of the same CSS class will not work anymore. Therefore the example below will only result in **JS animations taking
     * charge of the animation**:
     *
     * ```html
     * <div ng-if="bool" class="slide">
     *   Slide in and out
     * </div>
     * ```
     *
     * ```js
     * myModule.animation('.slide', [function() {
     *   return {
     *     enter: function(element, doneFn) {
     *       jQuery(element).slideIn(1000, doneFn);
     *     }
     *   }
     * }]);
     * ```
     *
     * ```css
     * .slide.ng-enter {
     *   transition:0.5s linear all;
     *   transform:translateY(-100px);
     * }
     * .slide.ng-enter.ng-enter-active {
     *   transform:translateY(0);
     * }
     * ```
     *
     * Does this mean that CSS and JS animations cannot be used together? Do JS-based animations always have higher priority? We can make up for the
     * lack of CSS animations by using the `$animateCss` service to trigger our own tweaked-out, CSS-based animations directly from
     * our own JS-based animation code:
     *
     * ```js
     * myModule.animation('.slide', ['$animateCss', function($animateCss) {
     *   return {
     *     enter: function(element) {
    *        // this will trigger `.slide.ng-enter` and `.slide.ng-enter-active`.
     *       return $animateCss(element, {
     *         event: 'enter',
     *         structural: true
     *       });
     *     }
     *   }
     * }]);
     * ```
     *
     * The nice thing here is that we can save bandwidth by sticking to our CSS-based animation code and we don't need to rely on a 3rd-party animation framework.
     *
     * The `$animateCss` service is very powerful since we can feed in all kinds of extra properties that will be evaluated and fed into a CSS transition or
     * keyframe animation. For example if we wanted to animate the height of an element while adding and removing classes then we can do so by providing that
     * data into `$animateCss` directly:
     *
     * ```js
     * myModule.animation('.slide', ['$animateCss', function($animateCss) {
     *   return {
     *     enter: function(element) {
     *       return $animateCss(element, {
     *         event: 'enter',
     *         structural: true,
     *         addClass: 'maroon-setting',
     *         from: { height:0 },
     *         to: { height: 200 }
     *       });
     *     }
     *   }
     * }]);
     * ```
     *
     * Now we can fill in the rest via our transition CSS code:
     *
     * ```css
     * /&#42; the transition tells ngAnimate to make the animation happen &#42;/
     * .slide.ng-enter { transition:0.5s linear all; }
     *
     * /&#42; this extra CSS class will be absorbed into the transition
     * since the $animateCss code is adding the class &#42;/
     * .maroon-setting { background:red; }
     * ```
     *
     * And `$animateCss` will figure out the rest. Just make sure to have the `done()` callback fire the `doneFn` function to signal when the animation is over.
     *
     * To learn more about what's possible be sure to visit the {@link ngAnimate.$animateCss $animateCss service}.
     *
     * ## Animation Anchoring (via `ng-animate-ref`)
     *
     * ngAnimate in AngularJS 1.4 comes packed with the ability to cross-animate elements between
     * structural areas of an application (like views) by pairing up elements using an attribute
     * called `ng-animate-ref`.
     *
     * Let's say for example we have two views that are managed by `ng-view` and we want to show
     * that there is a relationship between two components situated in within these views. By using the
     * `ng-animate-ref` attribute we can identify that the two components are paired together and we
     * can then attach an animation, which is triggered when the view changes.
     *
     * Say for example we have the following template code:
     *
     * ```html
     * <!-- index.html -->
     * <div ng-view class="view-animation">
     * </div>
     *
     * <!-- home.html -->
     * <a href="#/banner-page">
     *   <img src="./banner.jpg" class="banner" ng-animate-ref="banner">
     * </a>
     *
     * <!-- banner-page.html -->
     * <img src="./banner.jpg" class="banner" ng-animate-ref="banner">
     * ```
     *
     * Now, when the view changes (once the link is clicked), ngAnimate will examine the
     * HTML contents to see if there is a match reference between any components in the view
     * that is leaving and the view that is entering. It will scan both the view which is being
     * removed (leave) and inserted (enter) to see if there are any paired DOM elements that
     * contain a matching ref value.
     *
     * The two images match since they share the same ref value. ngAnimate will now create a
     * transport element (which is a clone of the first image element) and it will then attempt
     * to animate to the position of the second image element in the next view. For the animation to
     * work a special CSS class called `ng-anchor` will be added to the transported element.
     *
     * We can now attach a transition onto the `.banner.ng-anchor` CSS class and then
     * ngAnimate will handle the entire transition for us as well as the addition and removal of
     * any changes of CSS classes between the elements:
     *
     * ```css
     * .banner.ng-anchor {
     *   /&#42; this animation will last for 1 second since there are
     *          two phases to the animation (an `in` and an `out` phase) &#42;/
     *   transition:0.5s linear all;
     * }
     * ```
     *
     * We also **must** include animations for the views that are being entered and removed
     * (otherwise anchoring wouldn't be possible since the new view would be inserted right away).
     *
     * ```css
     * .view-animation.ng-enter, .view-animation.ng-leave {
     *   transition:0.5s linear all;
     *   position:fixed;
     *   left:0;
     *   top:0;
     *   width:100%;
     * }
     * .view-animation.ng-enter {
     *   transform:translateX(100%);
     * }
     * .view-animation.ng-leave,
     * .view-animation.ng-enter.ng-enter-active {
     *   transform:translateX(0%);
     * }
     * .view-animation.ng-leave.ng-leave-active {
     *   transform:translateX(-100%);
     * }
     * ```
     *
     * Now we can jump back to the anchor animation. When the animation happens, there are two stages that occur:
     * an `out` and an `in` stage. The `out` stage happens first and that is when the element is animated away
     * from its origin. Once that animation is over then the `in` stage occurs which animates the
     * element to its destination. The reason why there are two animations is to give enough time
     * for the enter animation on the new element to be ready.
     *
     * The example above sets up a transition for both the in and out phases, but we can also target the out or
     * in phases directly via `ng-anchor-out` and `ng-anchor-in`.
     *
     * ```css
     * .banner.ng-anchor-out {
     *   transition: 0.5s linear all;
     *
     *   /&#42; the scale will be applied during the out animation,
     *          but will be animated away when the in animation runs &#42;/
     *   transform: scale(1.2);
     * }
     *
     * .banner.ng-anchor-in {
     *   transition: 1s linear all;
     * }
     * ```
     *
     *
     *
     *
     * ### Anchoring Demo
     *
      <example module="anchoringExample"
               name="anchoringExample"
               id="anchoringExample"
               deps="angular-animate.js;angular-route.js"
               animations="true">
        <file name="index.html">
          <a href="#/">Home</a>
          <hr />
          <div class="view-container">
            <div ng-view class="view"></div>
          </div>
        </file>
        <file name="script.js">
          angular.module('anchoringExample', ['ngAnimate', 'ngRoute'])
            .config(['$routeProvider', function($routeProvider) {
              $routeProvider.when('/', {
                templateUrl: 'home.html',
                controller: 'HomeController as home'
              });
              $routeProvider.when('/profile/:id', {
                templateUrl: 'profile.html',
                controller: 'ProfileController as profile'
              });
            }])
            .run(['$rootScope', function($rootScope) {
              $rootScope.records = [
                { id:1, title: "Miss Beulah Roob" },
                { id:2, title: "Trent Morissette" },
                { id:3, title: "Miss Ava Pouros" },
                { id:4, title: "Rod Pouros" },
                { id:5, title: "Abdul Rice" },
                { id:6, title: "Laurie Rutherford Sr." },
                { id:7, title: "Nakia McLaughlin" },
                { id:8, title: "Jordon Blanda DVM" },
                { id:9, title: "Rhoda Hand" },
                { id:10, title: "Alexandrea Sauer" }
              ];
            }])
            .controller('HomeController', [function() {
              //empty
            }])
            .controller('ProfileController', ['$rootScope', '$routeParams', function($rootScope, $routeParams) {
              var index = parseInt($routeParams.id, 10);
              var record = $rootScope.records[index - 1];
    
              this.title = record.title;
              this.id = record.id;
            }]);
        </file>
        <file name="home.html">
          <h2>Welcome to the home page</h1>
          <p>Please click on an element</p>
          <a class="record"
             ng-href="#/profile/{{ record.id }}"
             ng-animate-ref="{{ record.id }}"
             ng-repeat="record in records">
            {{ record.title }}
          </a>
        </file>
        <file name="profile.html">
          <div class="profile record" ng-animate-ref="{{ profile.id }}">
            {{ profile.title }}
          </div>
        </file>
        <file name="animations.css">
          .record {
            display:block;
            font-size:20px;
          }
          .profile {
            background:black;
            color:white;
            font-size:100px;
          }
          .view-container {
            position:relative;
          }
          .view-container > .view.ng-animate {
            position:absolute;
            top:0;
            left:0;
            width:100%;
            min-height:500px;
          }
          .view.ng-enter, .view.ng-leave,
          .record.ng-anchor {
            transition:0.5s linear all;
          }
          .view.ng-enter {
            transform:translateX(100%);
          }
          .view.ng-enter.ng-enter-active, .view.ng-leave {
            transform:translateX(0%);
          }
          .view.ng-leave.ng-leave-active {
            transform:translateX(-100%);
          }
          .record.ng-anchor-out {
            background:red;
          }
        </file>
      </example>
     *
     * ### How is the element transported?
     *
     * When an anchor animation occurs, ngAnimate will clone the starting element and position it exactly where the starting
     * element is located on screen via absolute positioning. The cloned element will be placed inside of the root element
     * of the application (where ng-app was defined) and all of the CSS classes of the starting element will be applied. The
     * element will then animate into the `out` and `in` animations and will eventually reach the coordinates and match
     * the dimensions of the destination element. During the entire animation a CSS class of `.ng-animate-shim` will be applied
     * to both the starting and destination elements in order to hide them from being visible (the CSS styling for the class
     * is: `visibility:hidden`). Once the anchor reaches its destination then it will be removed and the destination element
     * will become visible since the shim class will be removed.
     *
     * ### How is the morphing handled?
     *
     * CSS Anchoring relies on transitions and keyframes and the internal code is intelligent enough to figure out
     * what CSS classes differ between the starting element and the destination element. These different CSS classes
     * will be added/removed on the anchor element and a transition will be applied (the transition that is provided
     * in the anchor class). Long story short, ngAnimate will figure out what classes to add and remove which will
     * make the transition of the element as smooth and automatic as possible. Be sure to use simple CSS classes that
     * do not rely on DOM nesting structure so that the anchor element appears the same as the starting element (since
     * the cloned element is placed inside of root element which is likely close to the body element).
     *
     * Note that if the root element is on the `<html>` element then the cloned node will be placed inside of body.
     *
     *
     * ## Using $animate in your directive code
     *
     * So far we've explored how to feed in animations into an Angular application, but how do we trigger animations within our own directives in our application?
     * By injecting the `$animate` service into our directive code, we can trigger structural and class-based hooks which can then be consumed by animations. Let's
     * imagine we have a greeting box that shows and hides itself when the data changes
     *
     * ```html
     * <greeting-box active="onOrOff">Hi there</greeting-box>
     * ```
     *
     * ```js
     * ngModule.directive('greetingBox', ['$animate', function($animate) {
     *   return function(scope, element, attrs) {
     *     attrs.$observe('active', function(value) {
     *       value ? $animate.addClass(element, 'on') : $animate.removeClass(element, 'on');
     *     });
     *   });
     * }]);
     * ```
     *
     * Now the `on` CSS class is added and removed on the greeting box component. Now if we add a CSS class on top of the greeting box element
     * in our HTML code then we can trigger a CSS or JS animation to happen.
     *
     * ```css
     * /&#42; normally we would create a CSS class to reference on the element &#42;/
     * greeting-box.on { transition:0.5s linear all; background:green; color:white; }
     * ```
     *
     * The `$animate` service contains a variety of other methods like `enter`, `leave`, `animate` and `setClass`. To learn more about what's
     * possible be sure to visit the {@link ng.$animate $animate service API page}.
     *
     *
     * ### Preventing Collisions With Third Party Libraries
     *
     * Some third-party frameworks place animation duration defaults across many element or className
     * selectors in order to make their code small and reuseable. This can lead to issues with ngAnimate, which
     * is expecting actual animations on these elements and has to wait for their completion.
     *
     * You can prevent this unwanted behavior by using a prefix on all your animation classes:
     *
     * ```css
     * /&#42; prefixed with animate- &#42;/
     * .animate-fade-add.animate-fade-add-active {
     *   transition:1s linear all;
     *   opacity:0;
     * }
     * ```
     *
     * You then configure `$animate` to enforce this prefix:
     *
     * ```js
     * $animateProvider.classNameFilter(/animate-/);
     * ```
     *
     * This also may provide your application with a speed boost since only specific elements containing CSS class prefix
     * will be evaluated for animation when any DOM changes occur in the application.
     *
     * ## Callbacks and Promises
     *
     * When `$animate` is called it returns a promise that can be used to capture when the animation has ended. Therefore if we were to trigger
     * an animation (within our directive code) then we can continue performing directive and scope related activities after the animation has
     * ended by chaining onto the returned promise that animation method returns.
     *
     * ```js
     * // somewhere within the depths of the directive
     * $animate.enter(element, parent).then(function() {
     *   //the animation has completed
     * });
     * ```
     *
     * (Note that earlier versions of Angular prior to v1.4 required the promise code to be wrapped using `$scope.$apply(...)`. This is not the case
     * anymore.)
     *
     * In addition to the animation promise, we can also make use of animation-related callbacks within our directives and controller code by registering
     * an event listener using the `$animate` service. Let's say for example that an animation was triggered on our view
     * routing controller to hook into that:
     *
     * ```js
     * ngModule.controller('HomePageController', ['$animate', function($animate) {
     *   $animate.on('enter', ngViewElement, function(element) {
     *     // the animation for this route has completed
     *   }]);
     * }])
     * ```
     *
     * (Note that you will need to trigger a digest within the callback to get angular to notice any scope-related changes.)
     */

    /**
     * @ngdoc service
     * @name $animate
     * @kind object
     *
     * @description
     * The ngAnimate `$animate` service documentation is the same for the core `$animate` service.
     *
     * Click here {@link ng.$animate to learn more about animations with `$animate`}.
     */
    angular.module('ngAnimate', [])
      .directive('ngAnimateChildren', $$AnimateChildrenDirective)
      .factory('$$rAFScheduler', $$rAFSchedulerFactory)

      .factory('$$AnimateRunner', $$AnimateRunnerFactory)
      .factory('$$animateAsyncRun', $$AnimateAsyncRunFactory)

      .provider('$$animateQueue', $$AnimateQueueProvider)
      .provider('$$animation', $$AnimationProvider)

      .provider('$animateCss', $AnimateCssProvider)
      .provider('$$animateCssDriver', $$AnimateCssDriverProvider)

      .provider('$$animateJs', $$AnimateJsProvider)
      .provider('$$animateJsDriver', $$AnimateJsDriverProvider);


})(window, window.angular);;
/*
 * angular-ui-bootstrap
 * http://angular-ui.github.io/bootstrap/

 * Version: 0.14.3 - 2015-10-23
 * License: MIT
 */
angular.module("ui.bootstrap", ["ui.bootstrap.tpls", "ui.bootstrap.collapse", "ui.bootstrap.buttons", "ui.bootstrap.position", "ui.bootstrap.stackedMap", "ui.bootstrap.tooltip", "ui.bootstrap.rating", "ui.bootstrap.typeahead", "ui.bootstrap.tabs", "ui.bootstrap.popover", "ui.bootstrap.modal", "ui.bootstrap.datepicker", "ui.bootstrap.dateparser", "ui.bootstrap.accordion", "ui.bootstrap.alert", "ui.bootstrap.dropdown", "ui.bootstrap.pagination", "ui.bootstrap.progressbar", "ui.bootstrap.timepicker"]);
angular.module("ui.bootstrap.tpls", ["template/tooltip/tooltip-html-popup.html", "template/tooltip/tooltip-popup.html", "template/tooltip/tooltip-template-popup.html", "template/rating/rating.html", "template/typeahead/typeahead-match.html", "template/typeahead/typeahead-popup.html", "template/tabs/tab.html", "template/tabs/tabset.html", "template/popover/popover-html.html", "template/popover/popover-template.html", "template/popover/popover.html", "template/modal/backdrop.html", "template/modal/window.html", "template/datepicker/datepicker.html", "template/datepicker/day.html", "template/datepicker/month.html", "template/datepicker/popup.html", "template/datepicker/year.html", "template/accordion/accordion-group.html", "template/accordion/accordion.html", "template/alert/alert.html", "template/pagination/pager.html", "template/pagination/pagination.html", "template/progressbar/bar.html", "template/progressbar/progress.html", "template/progressbar/progressbar.html", "template/timepicker/timepicker.html"]);
angular.module('ui.bootstrap.collapse', [])

  .directive('uibCollapse', ['$animate', '$injector', function ($animate, $injector) {
      var $animateCss = $injector.has('$animateCss') ? $injector.get('$animateCss') : null;
      return {
          link: function (scope, element, attrs) {
              function expand() {
                  element.removeClass('collapse')
                    .addClass('collapsing')
                    .attr('aria-expanded', true)
                    .attr('aria-hidden', false);

                  if ($animateCss) {
                      $animateCss(element, {
                          addClass: 'in',
                          easing: 'ease',
                          to: { height: element[0].scrollHeight + 'px' }
                      }).start().finally(expandDone);
                  } else {
                      $animate.addClass(element, 'in', {
                          to: { height: element[0].scrollHeight + 'px' }
                      }).then(expandDone);
                  }
              }

              function expandDone() {
                  element.removeClass('collapsing')
                    .addClass('collapse')
                    .css({ height: 'auto' });
              }

              function collapse() {
                  if (!element.hasClass('collapse') && !element.hasClass('in')) {
                      return collapseDone();
                  }

                  element
                    // IMPORTANT: The height must be set before adding "collapsing" class.
                    // Otherwise, the browser attempts to animate from height 0 (in
                    // collapsing class) to the given height here.
                    .css({ height: element[0].scrollHeight + 'px' })
                    // initially all panel collapse have the collapse class, this removal
                    // prevents the animation from jumping to collapsed state
                    .removeClass('collapse')
                    .addClass('collapsing')
                    .attr('aria-expanded', false)
                    .attr('aria-hidden', true);

                  if ($animateCss) {
                      $animateCss(element, {
                          removeClass: 'in',
                          to: { height: '0' }
                      }).start().finally(collapseDone);
                  } else {
                      $animate.removeClass(element, 'in', {
                          to: { height: '0' }
                      }).then(collapseDone);
                  }
              }

              function collapseDone() {
                  element.css({ height: '0' }); // Required so that collapse works when animation is disabled
                  element.removeClass('collapsing')
                    .addClass('collapse');
              }

              scope.$watch(attrs.uibCollapse, function (shouldCollapse) {
                  if (shouldCollapse) {
                      collapse();
                  } else {
                      expand();
                  }
              });
          }
      };
  }]);

/* Deprecated collapse below */

angular.module('ui.bootstrap.collapse')

  .value('$collapseSuppressWarning', false)

  .directive('collapse', ['$animate', '$injector', '$log', '$collapseSuppressWarning', function ($animate, $injector, $log, $collapseSuppressWarning) {
      var $animateCss = $injector.has('$animateCss') ? $injector.get('$animateCss') : null;
      return {
          link: function (scope, element, attrs) {
              if (!$collapseSuppressWarning) {
                  $log.warn('collapse is now deprecated. Use uib-collapse instead.');
              }

              function expand() {
                  element.removeClass('collapse')
                    .addClass('collapsing')
                    .attr('aria-expanded', true)
                    .attr('aria-hidden', false);

                  if ($animateCss) {
                      $animateCss(element, {
                          easing: 'ease',
                          to: { height: element[0].scrollHeight + 'px' }
                      }).start().done(expandDone);
                  } else {
                      $animate.animate(element, {}, {
                          height: element[0].scrollHeight + 'px'
                      }).then(expandDone);
                  }
              }

              function expandDone() {
                  element.removeClass('collapsing')
                    .addClass('collapse in')
                    .css({ height: 'auto' });
              }

              function collapse() {
                  if (!element.hasClass('collapse') && !element.hasClass('in')) {
                      return collapseDone();
                  }

                  element
                    // IMPORTANT: The height must be set before adding "collapsing" class.
                    // Otherwise, the browser attempts to animate from height 0 (in
                    // collapsing class) to the given height here.
                    .css({ height: element[0].scrollHeight + 'px' })
                    // initially all panel collapse have the collapse class, this removal
                    // prevents the animation from jumping to collapsed state
                    .removeClass('collapse in')
                    .addClass('collapsing')
                    .attr('aria-expanded', false)
                    .attr('aria-hidden', true);

                  if ($animateCss) {
                      $animateCss(element, {
                          to: { height: '0' }
                      }).start().done(collapseDone);
                  } else {
                      $animate.animate(element, {}, {
                          height: '0'
                      }).then(collapseDone);
                  }
              }

              function collapseDone() {
                  element.css({ height: '0' }); // Required so that collapse works when animation is disabled
                  element.removeClass('collapsing')
                    .addClass('collapse');
              }

              scope.$watch(attrs.collapse, function (shouldCollapse) {
                  if (shouldCollapse) {
                      collapse();
                  } else {
                      expand();
                  }
              });
          }
      };
  }]);

angular.module('ui.bootstrap.buttons', [])

.constant('uibButtonConfig', {
    activeClass: 'active',
    toggleEvent: 'click'
})

.controller('UibButtonsController', ['uibButtonConfig', function (buttonConfig) {
    this.activeClass = buttonConfig.activeClass || 'active';
    this.toggleEvent = buttonConfig.toggleEvent || 'click';
}])

.directive('uibBtnRadio', function () {
    return {
        require: ['uibBtnRadio', 'ngModel'],
        controller: 'UibButtonsController',
        controllerAs: 'buttons',
        link: function (scope, element, attrs, ctrls) {
            var buttonsCtrl = ctrls[0], ngModelCtrl = ctrls[1];

            element.find('input').css({ display: 'none' });

            //model -> UI
            ngModelCtrl.$render = function () {
                element.toggleClass(buttonsCtrl.activeClass, angular.equals(ngModelCtrl.$modelValue, scope.$eval(attrs.uibBtnRadio)));
            };

            //ui->model
            element.on(buttonsCtrl.toggleEvent, function () {
                if (attrs.disabled) {
                    return;
                }

                var isActive = element.hasClass(buttonsCtrl.activeClass);

                if (!isActive || angular.isDefined(attrs.uncheckable)) {
                    scope.$apply(function () {
                        ngModelCtrl.$setViewValue(isActive ? null : scope.$eval(attrs.uibBtnRadio));
                        ngModelCtrl.$render();
                    });
                }
            });
        }
    };
})

.directive('uibBtnCheckbox', function () {
    return {
        require: ['uibBtnCheckbox', 'ngModel'],
        controller: 'UibButtonsController',
        controllerAs: 'button',
        link: function (scope, element, attrs, ctrls) {
            var buttonsCtrl = ctrls[0], ngModelCtrl = ctrls[1];

            element.find('input').css({ display: 'none' });

            function getTrueValue() {
                return getCheckboxValue(attrs.btnCheckboxTrue, true);
            }

            function getFalseValue() {
                return getCheckboxValue(attrs.btnCheckboxFalse, false);
            }

            function getCheckboxValue(attribute, defaultValue) {
                return angular.isDefined(attribute) ? scope.$eval(attribute) : defaultValue;
            }

            //model -> UI
            ngModelCtrl.$render = function () {
                element.toggleClass(buttonsCtrl.activeClass, angular.equals(ngModelCtrl.$modelValue, getTrueValue()));
            };

            //ui->model
            element.on(buttonsCtrl.toggleEvent, function () {
                if (attrs.disabled) {
                    return;
                }

                scope.$apply(function () {
                    ngModelCtrl.$setViewValue(element.hasClass(buttonsCtrl.activeClass) ? getFalseValue() : getTrueValue());
                    ngModelCtrl.$render();
                });
            });
        }
    };
});

/* Deprecated buttons below */

angular.module('ui.bootstrap.buttons')

  .value('$buttonsSuppressWarning', false)

  .controller('ButtonsController', ['$controller', '$log', '$buttonsSuppressWarning', function ($controller, $log, $buttonsSuppressWarning) {
      if (!$buttonsSuppressWarning) {
          $log.warn('ButtonsController is now deprecated. Use UibButtonsController instead.');
      }

      angular.extend(this, $controller('UibButtonsController'));
  }])

  .directive('btnRadio', ['$log', '$buttonsSuppressWarning', function ($log, $buttonsSuppressWarning) {
      return {
          require: ['btnRadio', 'ngModel'],
          controller: 'ButtonsController',
          controllerAs: 'buttons',
          link: function (scope, element, attrs, ctrls) {
              if (!$buttonsSuppressWarning) {
                  $log.warn('btn-radio is now deprecated. Use uib-btn-radio instead.');
              }

              var buttonsCtrl = ctrls[0], ngModelCtrl = ctrls[1];

              element.find('input').css({ display: 'none' });

              //model -> UI
              ngModelCtrl.$render = function () {
                  element.toggleClass(buttonsCtrl.activeClass, angular.equals(ngModelCtrl.$modelValue, scope.$eval(attrs.btnRadio)));
              };

              //ui->model
              element.bind(buttonsCtrl.toggleEvent, function () {
                  if (attrs.disabled) {
                      return;
                  }

                  var isActive = element.hasClass(buttonsCtrl.activeClass);

                  if (!isActive || angular.isDefined(attrs.uncheckable)) {
                      scope.$apply(function () {
                          ngModelCtrl.$setViewValue(isActive ? null : scope.$eval(attrs.btnRadio));
                          ngModelCtrl.$render();
                      });
                  }
              });
          }
      };
  }])

  .directive('btnCheckbox', ['$document', '$log', '$buttonsSuppressWarning', function ($document, $log, $buttonsSuppressWarning) {
      return {
          require: ['btnCheckbox', 'ngModel'],
          controller: 'ButtonsController',
          controllerAs: 'button',
          link: function (scope, element, attrs, ctrls) {
              if (!$buttonsSuppressWarning) {
                  $log.warn('btn-checkbox is now deprecated. Use uib-btn-checkbox instead.');
              }

              var buttonsCtrl = ctrls[0], ngModelCtrl = ctrls[1];

              element.find('input').css({ display: 'none' });

              function getTrueValue() {
                  return getCheckboxValue(attrs.btnCheckboxTrue, true);
              }

              function getFalseValue() {
                  return getCheckboxValue(attrs.btnCheckboxFalse, false);
              }

              function getCheckboxValue(attributeValue, defaultValue) {
                  var val = scope.$eval(attributeValue);
                  return angular.isDefined(val) ? val : defaultValue;
              }

              //model -> UI
              ngModelCtrl.$render = function () {
                  element.toggleClass(buttonsCtrl.activeClass, angular.equals(ngModelCtrl.$modelValue, getTrueValue()));
              };

              //ui->model
              element.bind(buttonsCtrl.toggleEvent, function () {
                  if (attrs.disabled) {
                      return;
                  }

                  scope.$apply(function () {
                      ngModelCtrl.$setViewValue(element.hasClass(buttonsCtrl.activeClass) ? getFalseValue() : getTrueValue());
                      ngModelCtrl.$render();
                  });
              });

              //accessibility
              element.on('keypress', function (e) {
                  if (attrs.disabled || e.which !== 32 || $document[0].activeElement !== element[0]) {
                      return;
                  }

                  scope.$apply(function () {
                      ngModelCtrl.$setViewValue(element.hasClass(buttonsCtrl.activeClass) ? getFalseValue() : getTrueValue());
                      ngModelCtrl.$render();
                  });
              });
          }
      };
  }]);


angular.module('ui.bootstrap.position', [])

/**
 * A set of utility methods that can be use to retrieve position of DOM elements.
 * It is meant to be used where we need to absolute-position DOM elements in
 * relation to other, existing elements (this is the case for tooltips, popovers,
 * typeahead suggestions etc.).
 */
  .factory('$uibPosition', ['$document', '$window', function ($document, $window) {
      function getStyle(el, cssprop) {
          if (el.currentStyle) { //IE
              return el.currentStyle[cssprop];
          } else if ($window.getComputedStyle) {
              return $window.getComputedStyle(el)[cssprop];
          }
          // finally try and get inline style
          return el.style[cssprop];
      }

      /**
       * Checks if a given element is statically positioned
       * @param element - raw DOM element
       */
      function isStaticPositioned(element) {
          return (getStyle(element, 'position') || 'static') === 'static';
      }

      /**
       * returns the closest, non-statically positioned parentOffset of a given element
       * @param element
       */
      var parentOffsetEl = function (element) {
          var docDomEl = $document[0];
          var offsetParent = element.offsetParent || docDomEl;
          while (offsetParent && offsetParent !== docDomEl && isStaticPositioned(offsetParent)) {
              offsetParent = offsetParent.offsetParent;
          }
          return offsetParent || docDomEl;
      };

      return {
          /**
           * Provides read-only equivalent of jQuery's position function:
           * http://api.jquery.com/position/
           */
          position: function (element) {
              var elBCR = this.offset(element);
              var offsetParentBCR = { top: 0, left: 0 };
              var offsetParentEl = parentOffsetEl(element[0]);
              if (offsetParentEl != $document[0]) {
                  offsetParentBCR = this.offset(angular.element(offsetParentEl));
                  offsetParentBCR.top += offsetParentEl.clientTop - offsetParentEl.scrollTop;
                  offsetParentBCR.left += offsetParentEl.clientLeft - offsetParentEl.scrollLeft;
              }

              var boundingClientRect = element[0].getBoundingClientRect();
              return {
                  width: boundingClientRect.width || element.prop('offsetWidth'),
                  height: boundingClientRect.height || element.prop('offsetHeight'),
                  top: elBCR.top - offsetParentBCR.top,
                  left: elBCR.left - offsetParentBCR.left
              };
          },

          /**
           * Provides read-only equivalent of jQuery's offset function:
           * http://api.jquery.com/offset/
           */
          offset: function (element) {
              var boundingClientRect = element[0].getBoundingClientRect();
              return {
                  width: boundingClientRect.width || element.prop('offsetWidth'),
                  height: boundingClientRect.height || element.prop('offsetHeight'),
                  top: boundingClientRect.top + ($window.pageYOffset || $document[0].documentElement.scrollTop),
                  left: boundingClientRect.left + ($window.pageXOffset || $document[0].documentElement.scrollLeft)
              };
          },

          /**
           * Provides coordinates for the targetEl in relation to hostEl
           */
          positionElements: function (hostEl, targetEl, positionStr, appendToBody) {
              var positionStrParts = positionStr.split('-');
              var pos0 = positionStrParts[0], pos1 = positionStrParts[1] || 'center';

              var hostElPos,
                targetElWidth,
                targetElHeight,
                targetElPos;

              hostElPos = appendToBody ? this.offset(hostEl) : this.position(hostEl);

              targetElWidth = targetEl.prop('offsetWidth');
              targetElHeight = targetEl.prop('offsetHeight');

              var shiftWidth = {
                  center: function () {
                      return hostElPos.left + hostElPos.width / 2 - targetElWidth / 2;
                  },
                  left: function () {
                      return hostElPos.left;
                  },
                  right: function () {
                      return hostElPos.left + hostElPos.width;
                  }
              };

              var shiftHeight = {
                  center: function () {
                      return hostElPos.top + hostElPos.height / 2 - targetElHeight / 2;
                  },
                  top: function () {
                      return hostElPos.top;
                  },
                  bottom: function () {
                      return hostElPos.top + hostElPos.height;
                  }
              };

              switch (pos0) {
                  case 'right':
                      targetElPos = {
                          top: shiftHeight[pos1](),
                          left: shiftWidth[pos0]()
                      };
                      break;
                  case 'left':
                      targetElPos = {
                          top: shiftHeight[pos1](),
                          left: hostElPos.left - targetElWidth
                      };
                      break;
                  case 'bottom':
                      targetElPos = {
                          top: shiftHeight[pos0](),
                          left: shiftWidth[pos1]()
                      };
                      break;
                  default:
                      targetElPos = {
                          top: hostElPos.top - targetElHeight,
                          left: shiftWidth[pos1]()
                      };
                      break;
              }

              return targetElPos;
          }
      };
  }]);

/* Deprecated position below */

angular.module('ui.bootstrap.position')

.value('$positionSuppressWarning', false)

.service('$position', ['$log', '$positionSuppressWarning', '$uibPosition', function ($log, $positionSuppressWarning, $uibPosition) {
    if (!$positionSuppressWarning) {
        $log.warn('$position is now deprecated. Use $uibPosition instead.');
    }

    angular.extend(this, $uibPosition);
}]);

angular.module('ui.bootstrap.stackedMap', [])
/**
 * A helper, internal data structure that acts as a map but also allows getting / removing
 * elements in the LIFO order
 */
  .factory('$$stackedMap', function () {
      return {
          createNew: function () {
              var stack = [];

              return {
                  add: function (key, value) {
                      stack.push({
                          key: key,
                          value: value
                      });
                  },
                  get: function (key) {
                      for (var i = 0; i < stack.length; i++) {
                          if (key == stack[i].key) {
                              return stack[i];
                          }
                      }
                  },
                  keys: function () {
                      var keys = [];
                      for (var i = 0; i < stack.length; i++) {
                          keys.push(stack[i].key);
                      }
                      return keys;
                  },
                  top: function () {
                      return stack[stack.length - 1];
                  },
                  remove: function (key) {
                      var idx = -1;
                      for (var i = 0; i < stack.length; i++) {
                          if (key == stack[i].key) {
                              idx = i;
                              break;
                          }
                      }
                      return stack.splice(idx, 1)[0];
                  },
                  removeTop: function () {
                      return stack.splice(stack.length - 1, 1)[0];
                  },
                  length: function () {
                      return stack.length;
                  }
              };
          }
      };
  });
/**
 * The following features are still outstanding: animation as a
 * function, placement as a function, inside, support for more triggers than
 * just mouse enter/leave, html tooltips, and selector delegation.
 */
angular.module('ui.bootstrap.tooltip', ['ui.bootstrap.position', 'ui.bootstrap.stackedMap'])

/**
 * The $tooltip service creates tooltip- and popover-like directives as well as
 * houses global options for them.
 */
.provider('$uibTooltip', function () {
    // The default options tooltip and popover.
    var defaultOptions = {
        placement: 'top',
        animation: true,
        popupDelay: 0,
        popupCloseDelay: 0,
        useContentExp: false
    };

    // Default hide triggers for each show trigger
    var triggerMap = {
        'mouseenter': 'mouseleave',
        'click': 'click',
        'focus': 'blur',
        'none': ''
    };

    // The options specified to the provider globally.
    var globalOptions = {};

    /**
     * `options({})` allows global configuration of all tooltips in the
     * application.
     *
     *   var app = angular.module( 'App', ['ui.bootstrap.tooltip'], function( $tooltipProvider ) {
     *     // place tooltips left instead of top by default
     *     $tooltipProvider.options( { placement: 'left' } );
     *   });
     */
    this.options = function (value) {
        angular.extend(globalOptions, value);
    };

    /**
     * This allows you to extend the set of trigger mappings available. E.g.:
     *
     *   $tooltipProvider.setTriggers( 'openTrigger': 'closeTrigger' );
     */
    this.setTriggers = function setTriggers(triggers) {
        angular.extend(triggerMap, triggers);
    };

    /**
     * This is a helper function for translating camel-case to snake-case.
     */
    function snake_case(name) {
        var regexp = /[A-Z]/g;
        var separator = '-';
        return name.replace(regexp, function (letter, pos) {
            return (pos ? separator : '') + letter.toLowerCase();
        });
    }

    /**
     * Returns the actual instance of the $tooltip service.
     * TODO support multiple triggers
     */
    this.$get = ['$window', '$compile', '$timeout', '$document', '$uibPosition', '$interpolate', '$rootScope', '$parse', '$$stackedMap', function ($window, $compile, $timeout, $document, $position, $interpolate, $rootScope, $parse, $$stackedMap) {
        var openedTooltips = $$stackedMap.createNew();
        $document.on('keypress', function (e) {
            if (e.which === 27) {
                var last = openedTooltips.top();
                if (last) {
                    last.value.close();
                    openedTooltips.removeTop();
                    last = null;
                }
            }
        });

        return function $tooltip(ttType, prefix, defaultTriggerShow, options) {
            options = angular.extend({}, defaultOptions, globalOptions, options);

            /**
             * Returns an object of show and hide triggers.
             *
             * If a trigger is supplied,
             * it is used to show the tooltip; otherwise, it will use the `trigger`
             * option passed to the `$tooltipProvider.options` method; else it will
             * default to the trigger supplied to this directive factory.
             *
             * The hide trigger is based on the show trigger. If the `trigger` option
             * was passed to the `$tooltipProvider.options` method, it will use the
             * mapped trigger from `triggerMap` or the passed trigger if the map is
             * undefined; otherwise, it uses the `triggerMap` value of the show
             * trigger; else it will just use the show trigger.
             */
            function getTriggers(trigger) {
                var show = (trigger || options.trigger || defaultTriggerShow).split(' ');
                var hide = show.map(function (trigger) {
                    return triggerMap[trigger] || trigger;
                });
                return {
                    show: show,
                    hide: hide
                };
            }

            var directiveName = snake_case(ttType);

            var startSym = $interpolate.startSymbol();
            var endSym = $interpolate.endSymbol();
            var template =
              '<div ' + directiveName + '-popup ' +
                'title="' + startSym + 'title' + endSym + '" ' +
                (options.useContentExp ?
                  'content-exp="contentExp()" ' :
                  'content="' + startSym + 'content' + endSym + '" ') +
                'placement="' + startSym + 'placement' + endSym + '" ' +
                'popup-class="' + startSym + 'popupClass' + endSym + '" ' +
                'animation="animation" ' +
                'is-open="isOpen" ' +
                'origin-scope="origScope" ' +
                'style="visibility: hidden; display: block; top: -9999px; left: -9999px;"' +
                '>' +
              '</div>';

            return {
                compile: function (tElem, tAttrs) {
                    var tooltipLinker = $compile(template);

                    return function link(scope, element, attrs, tooltipCtrl) {
                        var tooltip;
                        var tooltipLinkedScope;
                        var transitionTimeout;
                        var showTimeout;
                        var hideTimeout;
                        var positionTimeout;
                        var appendToBody = angular.isDefined(options.appendToBody) ? options.appendToBody : false;
                        var triggers = getTriggers(undefined);
                        var hasEnableExp = angular.isDefined(attrs[prefix + 'Enable']);
                        var ttScope = scope.$new(true);
                        var repositionScheduled = false;
                        var isOpenParse = angular.isDefined(attrs[prefix + 'IsOpen']) ? $parse(attrs[prefix + 'IsOpen']) : false;
                        var contentParse = options.useContentExp ? $parse(attrs[ttType]) : false;
                        var observers = [];

                        var positionTooltip = function () {
                            // check if tooltip exists and is not empty
                            if (!tooltip || !tooltip.html()) { return; }

                            if (!positionTimeout) {
                                positionTimeout = $timeout(function () {
                                    // Reset the positioning.
                                    tooltip.css({ top: 0, left: 0 });

                                    // Now set the calculated positioning.
                                    var ttCss = $position.positionElements(element, tooltip, ttScope.placement, appendToBody);
                                    ttCss.top += 'px';
                                    ttCss.left += 'px';
                                    ttCss.visibility = 'visible';
                                    tooltip.css(ttCss);

                                    positionTimeout = null;
                                }, 0, false);
                            }
                        };

                        // Set up the correct scope to allow transclusion later
                        ttScope.origScope = scope;

                        // By default, the tooltip is not open.
                        // TODO add ability to start tooltip opened
                        ttScope.isOpen = false;
                        openedTooltips.add(ttScope, {
                            close: hide
                        });

                        function toggleTooltipBind() {
                            if (!ttScope.isOpen) {
                                showTooltipBind();
                            } else {
                                hideTooltipBind();
                            }
                        }

                        // Show the tooltip with delay if specified, otherwise show it immediately
                        function showTooltipBind() {
                            if (hasEnableExp && !scope.$eval(attrs[prefix + 'Enable'])) {
                                return;
                            }

                            cancelHide();
                            prepareTooltip();

                            if (ttScope.popupDelay) {
                                // Do nothing if the tooltip was already scheduled to pop-up.
                                // This happens if show is triggered multiple times before any hide is triggered.
                                if (!showTimeout) {
                                    showTimeout = $timeout(show, ttScope.popupDelay, false);
                                }
                            } else {
                                show();
                            }
                        }

                        function hideTooltipBind() {
                            cancelShow();

                            if (ttScope.popupCloseDelay) {
                                if (!hideTimeout) {
                                    hideTimeout = $timeout(hide, ttScope.popupCloseDelay, false);
                                }
                            } else {
                                hide();
                            }
                        }

                        // Show the tooltip popup element.
                        function show() {
                            cancelShow();
                            cancelHide();

                            // Don't show empty tooltips.
                            if (!ttScope.content) {
                                return angular.noop;
                            }

                            createTooltip();

                            // And show the tooltip.
                            ttScope.$evalAsync(function () {
                                ttScope.isOpen = true;
                                assignIsOpen(true);
                                positionTooltip();
                            });
                        }

                        function cancelShow() {
                            if (showTimeout) {
                                $timeout.cancel(showTimeout);
                                showTimeout = null;
                            }

                            if (positionTimeout) {
                                $timeout.cancel(positionTimeout);
                                positionTimeout = null;
                            }
                        }

                        // Hide the tooltip popup element.
                        function hide() {
                            cancelShow();
                            cancelHide();

                            if (!ttScope) {
                                return;
                            }

                            // First things first: we don't show it anymore.
                            ttScope.$evalAsync(function () {
                                ttScope.isOpen = false;
                                assignIsOpen(false);
                                // And now we remove it from the DOM. However, if we have animation, we
                                // need to wait for it to expire beforehand.
                                // FIXME: this is a placeholder for a port of the transitions library.
                                // The fade transition in TWBS is 150ms.
                                if (ttScope.animation) {
                                    if (!transitionTimeout) {
                                        transitionTimeout = $timeout(removeTooltip, 150, false);
                                    }
                                } else {
                                    removeTooltip();
                                }
                            });
                        }

                        function cancelHide() {
                            if (hideTimeout) {
                                $timeout.cancel(hideTimeout);
                                hideTimeout = null;
                            }
                            if (transitionTimeout) {
                                $timeout.cancel(transitionTimeout);
                                transitionTimeout = null;
                            }
                        }

                        function createTooltip() {
                            // There can only be one tooltip element per directive shown at once.
                            if (tooltip) {
                                return;
                            }

                            tooltipLinkedScope = ttScope.$new();
                            tooltip = tooltipLinker(tooltipLinkedScope, function (tooltip) {
                                if (appendToBody) {
                                    $document.find('body').append(tooltip);
                                } else {
                                    element.after(tooltip);
                                }
                            });

                            prepObservers();
                        }

                        function removeTooltip() {
                            unregisterObservers();

                            transitionTimeout = null;
                            if (tooltip) {
                                tooltip.remove();
                                tooltip = null;
                            }
                            if (tooltipLinkedScope) {
                                tooltipLinkedScope.$destroy();
                                tooltipLinkedScope = null;
                            }
                        }

                        /**
                         * Set the inital scope values. Once
                         * the tooltip is created, the observers
                         * will be added to keep things in synch.
                         */
                        function prepareTooltip() {
                            ttScope.title = attrs[prefix + 'Title'];
                            if (contentParse) {
                                ttScope.content = contentParse(scope);
                            } else {
                                ttScope.content = attrs[ttType];
                            }

                            ttScope.content = attrs[ttType]; // fbe fix

                            ttScope.popupClass = attrs[prefix + 'Class'];
                            ttScope.placement = angular.isDefined(attrs[prefix + 'Placement']) ? attrs[prefix + 'Placement'] : options.placement;

                            var delay = parseInt(attrs[prefix + 'PopupDelay'], 10);
                            var closeDelay = parseInt(attrs[prefix + 'PopupCloseDelay'], 10);
                            ttScope.popupDelay = !isNaN(delay) ? delay : options.popupDelay;
                            ttScope.popupCloseDelay = !isNaN(closeDelay) ? closeDelay : options.popupCloseDelay;
                        }

                        function assignIsOpen(isOpen) {
                            if (isOpenParse && angular.isFunction(isOpenParse.assign)) {
                                isOpenParse.assign(scope, isOpen);
                            }
                        }

                        ttScope.contentExp = function () {
                            return ttScope.content;
                        };

                        /**
                         * Observe the relevant attributes.
                         */
                        attrs.$observe('disabled', function (val) {
                            if (val) {
                                cancelShow();
                            }

                            if (val && ttScope.isOpen) {
                                hide();
                            }
                        });

                        if (isOpenParse) {
                            scope.$watch(isOpenParse, function (val) {
                                /*jshint -W018 */
                                if (ttScope && !val === ttScope.isOpen) {
                                    toggleTooltipBind();
                                }
                                /*jshint +W018 */
                            });
                        }

                        function prepObservers() {
                            observers.length = 0;

                            if (contentParse) {
                                observers.push(
                                  scope.$watch(contentParse, function (val) {
                                      ttScope.content = val;
                                      if (!val && ttScope.isOpen) {
                                          hide();
                                      }
                                  })
                                );

                                observers.push(
                                  tooltipLinkedScope.$watch(function () {
                                      if (!repositionScheduled) {
                                          repositionScheduled = true;
                                          tooltipLinkedScope.$$postDigest(function () {
                                              repositionScheduled = false;
                                              if (ttScope && ttScope.isOpen) {
                                                  positionTooltip();
                                              }
                                          });
                                      }
                                  })
                                );
                            } else {
                                observers.push(
                                  attrs.$observe(ttType, function (val) {
                                      ttScope.content = val;
                                      if (!val && ttScope.isOpen) {
                                          hide();
                                      } else {
                                          positionTooltip();
                                      }
                                  })
                                );
                            }

                            observers.push(
                              attrs.$observe(prefix + 'Title', function (val) {
                                  ttScope.title = val;
                                  if (ttScope.isOpen) {
                                      positionTooltip();
                                  }
                              })
                            );

                            observers.push(
                              attrs.$observe(prefix + 'Placement', function (val) {
                                  ttScope.placement = val ? val : options.placement;
                                  if (ttScope.isOpen) {
                                      positionTooltip();
                                  }
                              })
                            );
                        }

                        function unregisterObservers() {
                            if (observers.length) {
                                angular.forEach(observers, function (observer) {
                                    observer();
                                });
                                observers.length = 0;
                            }
                        }

                        var unregisterTriggers = function () {
                            triggers.show.forEach(function (trigger) {
                                element.unbind(trigger, showTooltipBind);
                            });
                            triggers.hide.forEach(function (trigger) {
                                trigger.split(' ').forEach(function (hideTrigger) {
                                    element[0].removeEventListener(hideTrigger, hideTooltipBind);
                                });
                            });
                        };

                        function prepTriggers() {
                            var val = attrs[prefix + 'Trigger'];
                            unregisterTriggers();

                            triggers = getTriggers(val);

                            if (triggers.show !== 'none') {
                                triggers.show.forEach(function (trigger, idx) {
                                    // Using raw addEventListener due to jqLite/jQuery bug - #4060
                                    if (trigger === triggers.hide[idx]) {
                                        element[0].addEventListener(trigger, toggleTooltipBind);
                                    } else if (trigger) {
                                        element[0].addEventListener(trigger, showTooltipBind);
                                        triggers.hide[idx].split(' ').forEach(function (trigger) {
                                            element[0].addEventListener(trigger, hideTooltipBind);
                                        });
                                    }

                                    element.on('keypress', function (e) {
                                        if (e.which === 27) {
                                            hideTooltipBind();
                                        }
                                    });
                                });
                            }
                        }

                        prepTriggers();

                        var animation = scope.$eval(attrs[prefix + 'Animation']);
                        ttScope.animation = angular.isDefined(animation) ? !!animation : options.animation;

                        var appendToBodyVal = scope.$eval(attrs[prefix + 'AppendToBody']);
                        appendToBody = angular.isDefined(appendToBodyVal) ? appendToBodyVal : appendToBody;

                        // if a tooltip is attached to <body> we need to remove it on
                        // location change as its parent scope will probably not be destroyed
                        // by the change.
                        if (appendToBody) {
                            scope.$on('$locationChangeSuccess', function closeTooltipOnLocationChangeSuccess() {
                                if (ttScope.isOpen) {
                                    hide();
                                }
                            });
                        }

                        // Make sure tooltip is destroyed and removed.
                        scope.$on('$destroy', function onDestroyTooltip() {
                            cancelShow();
                            cancelHide();
                            unregisterTriggers();
                            removeTooltip();
                            openedTooltips.remove(ttScope);
                            ttScope = null;
                        });
                    };
                }
            };
        };
    }];
})

// This is mostly ngInclude code but with a custom scope
.directive('uibTooltipTemplateTransclude', [
         '$animate', '$sce', '$compile', '$templateRequest',
function ($animate, $sce, $compile, $templateRequest) {
    return {
        link: function (scope, elem, attrs) {
            var origScope = scope.$eval(attrs.tooltipTemplateTranscludeScope);

            var changeCounter = 0,
              currentScope,
              previousElement,
              currentElement;

            var cleanupLastIncludeContent = function () {
                if (previousElement) {
                    previousElement.remove();
                    previousElement = null;
                }

                if (currentScope) {
                    currentScope.$destroy();
                    currentScope = null;
                }

                if (currentElement) {
                    $animate.leave(currentElement).then(function () {
                        previousElement = null;
                    });
                    previousElement = currentElement;
                    currentElement = null;
                }
            };

            scope.$watch($sce.parseAsResourceUrl(attrs.uibTooltipTemplateTransclude), function (src) {
                var thisChangeId = ++changeCounter;

                if (src) {
                    //set the 2nd param to true to ignore the template request error so that the inner
                    //contents and scope can be cleaned up.
                    $templateRequest(src, true).then(function (response) {
                        if (thisChangeId !== changeCounter) { return; }
                        var newScope = origScope.$new();
                        var template = response;

                        var clone = $compile(template)(newScope, function (clone) {
                            cleanupLastIncludeContent();
                            $animate.enter(clone, elem);
                        });

                        currentScope = newScope;
                        currentElement = clone;

                        currentScope.$emit('$includeContentLoaded', src);
                    }, function () {
                        if (thisChangeId === changeCounter) {
                            cleanupLastIncludeContent();
                            scope.$emit('$includeContentError', src);
                        }
                    });
                    scope.$emit('$includeContentRequested', src);
                } else {
                    cleanupLastIncludeContent();
                }
            });

            scope.$on('$destroy', cleanupLastIncludeContent);
        }
    };
}])

/**
 * Note that it's intentional that these classes are *not* applied through $animate.
 * They must not be animated as they're expected to be present on the tooltip on
 * initialization.
 */
.directive('uibTooltipClasses', function () {
    return {
        restrict: 'A',
        link: function (scope, element, attrs) {
            if (scope.placement) {
                element.addClass(scope.placement);
            }

            if (scope.popupClass) {
                element.addClass(scope.popupClass);
            }

            if (scope.animation()) {
                element.addClass(attrs.tooltipAnimationClass);
            }
        }
    };
})

.directive('uibTooltipPopup', function () {
    return {
        replace: true,
        scope: { content: '@', placement: '@', popupClass: '@', animation: '&', isOpen: '&' },
        templateUrl: 'template/tooltip/tooltip-popup.html',
        link: function (scope, element) {
            element.addClass('tooltip');
        }
    };
})

.directive('uibTooltip', ['$uibTooltip', function ($uibTooltip) {
    return $uibTooltip('uibTooltip', 'tooltip', 'mouseenter');
}])

.directive('uibTooltipTemplatePopup', function () {
    return {
        replace: true,
        scope: {
            contentExp: '&', placement: '@', popupClass: '@', animation: '&', isOpen: '&',
            originScope: '&'
        },
        templateUrl: 'template/tooltip/tooltip-template-popup.html',
        link: function (scope, element) {
            element.addClass('tooltip');
        }
    };
})

.directive('uibTooltipTemplate', ['$uibTooltip', function ($uibTooltip) {
    return $uibTooltip('uibTooltipTemplate', 'tooltip', 'mouseenter', {
        useContentExp: true
    });
}])

.directive('uibTooltipHtmlPopup', function () {
    return {
        replace: true,
        scope: { contentExp: '&', placement: '@', popupClass: '@', animation: '&', isOpen: '&' },
        templateUrl: 'template/tooltip/tooltip-html-popup.html',
        link: function (scope, element) {
            element.addClass('tooltip');
        }
    };
})

.directive('uibTooltipHtml', ['$uibTooltip', function ($uibTooltip) {
    return $uibTooltip('uibTooltipHtml', 'tooltip', 'mouseenter', {
        useContentExp: true
    });
}]);

/* Deprecated tooltip below */

angular.module('ui.bootstrap.tooltip')

.value('$tooltipSuppressWarning', false)

.provider('$tooltip', ['$uibTooltipProvider', function ($uibTooltipProvider) {
    angular.extend(this, $uibTooltipProvider);

    this.$get = ['$log', '$tooltipSuppressWarning', '$injector', function ($log, $tooltipSuppressWarning, $injector) {
        if (!$tooltipSuppressWarning) {
            $log.warn('$tooltip is now deprecated. Use $uibTooltip instead.');
        }

        return $injector.invoke($uibTooltipProvider.$get);
    }];
}])

// This is mostly ngInclude code but with a custom scope
.directive('tooltipTemplateTransclude', [
         '$animate', '$sce', '$compile', '$templateRequest', '$log', '$tooltipSuppressWarning',
function ($animate, $sce, $compile, $templateRequest, $log, $tooltipSuppressWarning) {
    return {
        link: function (scope, elem, attrs) {
            if (!$tooltipSuppressWarning) {
                $log.warn('tooltip-template-transclude is now deprecated. Use uib-tooltip-template-transclude instead.');
            }

            var origScope = scope.$eval(attrs.tooltipTemplateTranscludeScope);

            var changeCounter = 0,
              currentScope,
              previousElement,
              currentElement;

            var cleanupLastIncludeContent = function () {
                if (previousElement) {
                    previousElement.remove();
                    previousElement = null;
                }
                if (currentScope) {
                    currentScope.$destroy();
                    currentScope = null;
                }
                if (currentElement) {
                    $animate.leave(currentElement).then(function () {
                        previousElement = null;
                    });
                    previousElement = currentElement;
                    currentElement = null;
                }
            };

            scope.$watch($sce.parseAsResourceUrl(attrs.tooltipTemplateTransclude), function (src) {
                var thisChangeId = ++changeCounter;

                if (src) {
                    //set the 2nd param to true to ignore the template request error so that the inner
                    //contents and scope can be cleaned up.
                    $templateRequest(src, true).then(function (response) {
                        if (thisChangeId !== changeCounter) { return; }
                        var newScope = origScope.$new();
                        var template = response;

                        var clone = $compile(template)(newScope, function (clone) {
                            cleanupLastIncludeContent();
                            $animate.enter(clone, elem);
                        });

                        currentScope = newScope;
                        currentElement = clone;

                        currentScope.$emit('$includeContentLoaded', src);
                    }, function () {
                        if (thisChangeId === changeCounter) {
                            cleanupLastIncludeContent();
                            scope.$emit('$includeContentError', src);
                        }
                    });
                    scope.$emit('$includeContentRequested', src);
                } else {
                    cleanupLastIncludeContent();
                }
            });

            scope.$on('$destroy', cleanupLastIncludeContent);
        }
    };
}])

.directive('tooltipClasses', ['$log', '$tooltipSuppressWarning', function ($log, $tooltipSuppressWarning) {
    return {
        restrict: 'A',
        link: function (scope, element, attrs) {
            if (!$tooltipSuppressWarning) {
                $log.warn('tooltip-classes is now deprecated. Use uib-tooltip-classes instead.');
            }

            if (scope.placement) {
                element.addClass(scope.placement);
            }
            if (scope.popupClass) {
                element.addClass(scope.popupClass);
            }
            if (scope.animation()) {
                element.addClass(attrs.tooltipAnimationClass);
            }
        }
    };
}])

.directive('tooltipPopup', ['$log', '$tooltipSuppressWarning', function ($log, $tooltipSuppressWarning) {
    return {
        replace: true,
        scope: { content: '@', placement: '@', popupClass: '@', animation: '&', isOpen: '&' },
        templateUrl: 'template/tooltip/tooltip-popup.html',
        link: function (scope, element) {
            if (!$tooltipSuppressWarning) {
                $log.warn('tooltip-popup is now deprecated. Use uib-tooltip-popup instead.');
            }

            element.addClass('tooltip');
        }
    };
}])

.directive('tooltip', ['$tooltip', function ($tooltip) {
    return $tooltip('tooltip', 'tooltip', 'mouseenter');
}])

.directive('tooltipTemplatePopup', ['$log', '$tooltipSuppressWarning', function ($log, $tooltipSuppressWarning) {
    return {
        replace: true,
        scope: {
            contentExp: '&', placement: '@', popupClass: '@', animation: '&', isOpen: '&',
            originScope: '&'
        },
        templateUrl: 'template/tooltip/tooltip-template-popup.html',
        link: function (scope, element) {
            if (!$tooltipSuppressWarning) {
                $log.warn('tooltip-template-popup is now deprecated. Use uib-tooltip-template-popup instead.');
            }

            element.addClass('tooltip');
        }
    };
}])

.directive('tooltipTemplate', ['$tooltip', function ($tooltip) {
    return $tooltip('tooltipTemplate', 'tooltip', 'mouseenter', {
        useContentExp: true
    });
}])

.directive('tooltipHtmlPopup', ['$log', '$tooltipSuppressWarning', function ($log, $tooltipSuppressWarning) {
    return {
        replace: true,
        scope: { contentExp: '&', placement: '@', popupClass: '@', animation: '&', isOpen: '&' },
        templateUrl: 'template/tooltip/tooltip-html-popup.html',
        link: function (scope, element) {
            if (!$tooltipSuppressWarning) {
                $log.warn('tooltip-html-popup is now deprecated. Use uib-tooltip-html-popup instead.');
            }

            element.addClass('tooltip');
        }
    };
}])

.directive('tooltipHtml', ['$tooltip', function ($tooltip) {
    return $tooltip('tooltipHtml', 'tooltip', 'mouseenter', {
        useContentExp: true
    });
}]);

angular.module('ui.bootstrap.rating', [])

.constant('uibRatingConfig', {
    max: 5,
    stateOn: null,
    stateOff: null,
    titles: ['one', 'two', 'three', 'four', 'five']
})

.controller('UibRatingController', ['$scope', '$attrs', 'uibRatingConfig', function ($scope, $attrs, ratingConfig) {
    var ngModelCtrl = { $setViewValue: angular.noop };

    this.init = function (ngModelCtrl_) {
        ngModelCtrl = ngModelCtrl_;
        ngModelCtrl.$render = this.render;

        ngModelCtrl.$formatters.push(function (value) {
            if (angular.isNumber(value) && value << 0 !== value) {
                value = Math.round(value);
            }
            return value;
        });

        this.stateOn = angular.isDefined($attrs.stateOn) ? $scope.$parent.$eval($attrs.stateOn) : ratingConfig.stateOn;
        this.stateOff = angular.isDefined($attrs.stateOff) ? $scope.$parent.$eval($attrs.stateOff) : ratingConfig.stateOff;
        var tmpTitles = angular.isDefined($attrs.titles) ? $scope.$parent.$eval($attrs.titles) : ratingConfig.titles;
        this.titles = angular.isArray(tmpTitles) && tmpTitles.length > 0 ?
            tmpTitles : ratingConfig.titles;

        var ratingStates = angular.isDefined($attrs.ratingStates) ?
          $scope.$parent.$eval($attrs.ratingStates) :
          new Array(angular.isDefined($attrs.max) ? $scope.$parent.$eval($attrs.max) : ratingConfig.max);
        $scope.range = this.buildTemplateObjects(ratingStates);
    };

    this.buildTemplateObjects = function (states) {
        for (var i = 0, n = states.length; i < n; i++) {
            states[i] = angular.extend({ index: i }, { stateOn: this.stateOn, stateOff: this.stateOff, title: this.getTitle(i) }, states[i]);
        }
        return states;
    };

    this.getTitle = function (index) {
        if (index >= this.titles.length) {
            return index + 1;
        } else {
            return this.titles[index];
        }
    };

    $scope.rate = function (value) {
        if (!$scope.readonly && value >= 0 && value <= $scope.range.length) {
            ngModelCtrl.$setViewValue(ngModelCtrl.$viewValue === value ? 0 : value);
            ngModelCtrl.$render();
        }
    };

    $scope.enter = function (value) {
        if (!$scope.readonly) {
            $scope.value = value;
        }
        $scope.onHover({ value: value });
    };

    $scope.reset = function () {
        $scope.value = ngModelCtrl.$viewValue;
        $scope.onLeave();
    };

    $scope.onKeydown = function (evt) {
        if (/(37|38|39|40)/.test(evt.which)) {
            evt.preventDefault();
            evt.stopPropagation();
            $scope.rate($scope.value + (evt.which === 38 || evt.which === 39 ? 1 : -1));
        }
    };

    this.render = function () {
        $scope.value = ngModelCtrl.$viewValue;
    };
}])

.directive('uibRating', function () {
    return {
        require: ['uibRating', 'ngModel'],
        scope: {
            readonly: '=?',
            onHover: '&',
            onLeave: '&'
        },
        controller: 'UibRatingController',
        templateUrl: 'template/rating/rating.html',
        replace: true,
        link: function (scope, element, attrs, ctrls) {
            var ratingCtrl = ctrls[0], ngModelCtrl = ctrls[1];
            ratingCtrl.init(ngModelCtrl);
        }
    };
});

/* Deprecated rating below */

angular.module('ui.bootstrap.rating')

.value('$ratingSuppressWarning', false)

.controller('RatingController', ['$scope', '$attrs', '$controller', '$log', '$ratingSuppressWarning', function ($scope, $attrs, $controller, $log, $ratingSuppressWarning) {
    if (!$ratingSuppressWarning) {
        $log.warn('RatingController is now deprecated. Use UibRatingController instead.');
    }

    angular.extend(this, $controller('UibRatingController', {
        $scope: $scope,
        $attrs: $attrs
    }));
}])

.directive('rating', ['$log', '$ratingSuppressWarning', function ($log, $ratingSuppressWarning) {
    return {
        require: ['rating', 'ngModel'],
        scope: {
            readonly: '=?',
            onHover: '&',
            onLeave: '&'
        },
        controller: 'RatingController',
        templateUrl: 'template/rating/rating.html',
        replace: true,
        link: function (scope, element, attrs, ctrls) {
            if (!$ratingSuppressWarning) {
                $log.warn('rating is now deprecated. Use uib-rating instead.');
            }
            var ratingCtrl = ctrls[0], ngModelCtrl = ctrls[1];
            ratingCtrl.init(ngModelCtrl);
        }
    };
}]);

angular.module('ui.bootstrap.typeahead', ['ui.bootstrap.position'])

/**
 * A helper service that can parse typeahead's syntax (string provided by users)
 * Extracted to a separate service for ease of unit testing
 */
  .factory('uibTypeaheadParser', ['$parse', function ($parse) {
      //                      00000111000000000000022200000000000000003333333333333330000000000044000
      var TYPEAHEAD_REGEXP = /^\s*([\s\S]+?)(?:\s+as\s+([\s\S]+?))?\s+for\s+(?:([\$\w][\$\w\d]*))\s+in\s+([\s\S]+?)$/;
      return {
          parse: function (input) {
              var match = input.match(TYPEAHEAD_REGEXP);
              if (!match) {
                  throw new Error(
                    'Expected typeahead specification in form of "_modelValue_ (as _label_)? for _item_ in _collection_"' +
                      ' but got "' + input + '".');
              }

              return {
                  itemName: match[3],
                  source: $parse(match[4]),
                  viewMapper: $parse(match[2] || match[1]),
                  modelMapper: $parse(match[1])
              };
          }
      };
  }])

  .controller('UibTypeaheadController', ['$scope', '$element', '$attrs', '$compile', '$parse', '$q', '$timeout', '$document', '$window', '$rootScope', '$uibPosition', 'uibTypeaheadParser',
    function (originalScope, element, attrs, $compile, $parse, $q, $timeout, $document, $window, $rootScope, $position, typeaheadParser) {
        var HOT_KEYS = [9, 13, 27, 38, 40];
        var eventDebounceTime = 200;
        var modelCtrl, ngModelOptions;
        //SUPPORTED ATTRIBUTES (OPTIONS)

        //minimal no of characters that needs to be entered before typeahead kicks-in
        var minLength = originalScope.$eval(attrs.typeaheadMinLength);
        if (!minLength && minLength !== 0) {
            minLength = 1;
        }

        //minimal wait time after last character typed before typeahead kicks-in
        var waitTime = originalScope.$eval(attrs.typeaheadWaitMs) || 0;

        //should it restrict model values to the ones selected from the popup only?
        var isEditable = originalScope.$eval(attrs.typeaheadEditable) !== false;

        //binding to a variable that indicates if matches are being retrieved asynchronously
        var isLoadingSetter = $parse(attrs.typeaheadLoading).assign || angular.noop;

        //a callback executed when a match is selected
        var onSelectCallback = $parse(attrs.typeaheadOnSelect);

        //should it select highlighted popup value when losing focus?
        var isSelectOnBlur = angular.isDefined(attrs.typeaheadSelectOnBlur) ? originalScope.$eval(attrs.typeaheadSelectOnBlur) : false;

        //binding to a variable that indicates if there were no results after the query is completed
        var isNoResultsSetter = $parse(attrs.typeaheadNoResults).assign || angular.noop;

        var inputFormatter = attrs.typeaheadInputFormatter ? $parse(attrs.typeaheadInputFormatter) : undefined;

        var appendToBody = attrs.typeaheadAppendToBody ? originalScope.$eval(attrs.typeaheadAppendToBody) : false;

        var appendToElementId = attrs.typeaheadAppendToElementId || false;

        var focusFirst = originalScope.$eval(attrs.typeaheadFocusFirst) !== false;

        //If input matches an item of the list exactly, select it automatically
        var selectOnExact = attrs.typeaheadSelectOnExact ? originalScope.$eval(attrs.typeaheadSelectOnExact) : false;

        //INTERNAL VARIABLES

        //model setter executed upon match selection
        var parsedModel = $parse(attrs.ngModel);
        var invokeModelSetter = $parse(attrs.ngModel + '($$$p)');
        var $setModelValue = function (scope, newValue) {
            if (angular.isFunction(parsedModel(originalScope)) &&
              ngModelOptions && ngModelOptions.$options && ngModelOptions.$options.getterSetter) {
                return invokeModelSetter(scope, { $$$p: newValue });
            } else {
                return parsedModel.assign(scope, newValue);
            }
        };

        //expressions used by typeahead
        var parserResult = typeaheadParser.parse(attrs.uibTypeahead);

        var hasFocus;

        //Used to avoid bug in iOS webview where iOS keyboard does not fire
        //mousedown & mouseup events
        //Issue #3699
        var selected;

        //create a child scope for the typeahead directive so we are not polluting original scope
        //with typeahead-specific data (matches, query etc.)
        var scope = originalScope.$new();
        var offDestroy = originalScope.$on('$destroy', function () {
            scope.$destroy();
        });
        scope.$on('$destroy', offDestroy);

        // WAI-ARIA
        var popupId = 'typeahead-' + scope.$id + '-' + Math.floor(Math.random() * 10000);
        element.attr({
            'aria-autocomplete': 'list',
            'aria-expanded': false,
            'aria-owns': popupId
        });

        //pop-up element used to display matches
        var popUpEl = angular.element('<div uib-typeahead-popup></div>');
        popUpEl.attr({
            id: popupId,
            matches: 'matches',
            active: 'activeIdx',
            select: 'select(activeIdx)',
            'move-in-progress': 'moveInProgress',
            query: 'query',
            position: 'position'
        });
        //custom item template
        if (angular.isDefined(attrs.typeaheadTemplateUrl)) {
            popUpEl.attr('template-url', attrs.typeaheadTemplateUrl);
        }

        if (angular.isDefined(attrs.typeaheadPopupTemplateUrl)) {
            popUpEl.attr('popup-template-url', attrs.typeaheadPopupTemplateUrl);
        }

        var resetMatches = function () {
            scope.matches = [];
            scope.activeIdx = -1;
            element.attr('aria-expanded', false);
        };

        var getMatchId = function (index) {
            return popupId + '-option-' + index;
        };

        // Indicate that the specified match is the active (pre-selected) item in the list owned by this typeahead.
        // This attribute is added or removed automatically when the `activeIdx` changes.
        scope.$watch('activeIdx', function (index) {
            if (index < 0) {
                element.removeAttr('aria-activedescendant');
            } else {
                element.attr('aria-activedescendant', getMatchId(index));
            }
        });

        var inputIsExactMatch = function (inputValue, index) {
            if (scope.matches.length > index && inputValue) {
                return inputValue.toUpperCase() === scope.matches[index].label.toUpperCase();
            }

            return false;
        };

        var getMatchesAsync = function (inputValue) {
            var locals = { $viewValue: inputValue };
            isLoadingSetter(originalScope, true);
            isNoResultsSetter(originalScope, false);
            $q.when(parserResult.source(originalScope, locals)).then(function (matches) {
                //it might happen that several async queries were in progress if a user were typing fast
                //but we are interested only in responses that correspond to the current view value
                var onCurrentRequest = (inputValue === modelCtrl.$viewValue);
                if (onCurrentRequest && hasFocus) {
                    if (matches && matches.length > 0) {
                        scope.activeIdx = focusFirst ? 0 : -1;
                        isNoResultsSetter(originalScope, false);
                        scope.matches.length = 0;

                        //transform labels
                        for (var i = 0; i < matches.length; i++) {
                            locals[parserResult.itemName] = matches[i];
                            scope.matches.push({
                                id: getMatchId(i),
                                label: parserResult.viewMapper(scope, locals),
                                model: matches[i]
                            });
                        }

                        scope.query = inputValue;
                        //position pop-up with matches - we need to re-calculate its position each time we are opening a window
                        //with matches as a pop-up might be absolute-positioned and position of an input might have changed on a page
                        //due to other elements being rendered
                        recalculatePosition();

                        element.attr('aria-expanded', true);

                        //Select the single remaining option if user input matches
                        if (selectOnExact && scope.matches.length === 1 && inputIsExactMatch(inputValue, 0)) {
                            scope.select(0);
                        }
                    } else {
                        resetMatches();
                        isNoResultsSetter(originalScope, true);
                    }
                }
                if (onCurrentRequest) {
                    isLoadingSetter(originalScope, false);
                }
            }, function () {
                resetMatches();
                isLoadingSetter(originalScope, false);
                isNoResultsSetter(originalScope, true);
            });
        };

        // bind events only if appendToBody params exist - performance feature
        if (appendToBody) {
            angular.element($window).bind('resize', fireRecalculating);
            $document.find('body').bind('scroll', fireRecalculating);
        }

        // Declare the timeout promise var outside the function scope so that stacked calls can be cancelled later
        var timeoutEventPromise;

        // Default progress type
        scope.moveInProgress = false;

        function fireRecalculating() {
            if (!scope.moveInProgress) {
                scope.moveInProgress = true;
                scope.$digest();
            }

            // Cancel previous timeout
            if (timeoutEventPromise) {
                $timeout.cancel(timeoutEventPromise);
            }

            // Debounced executing recalculate after events fired
            timeoutEventPromise = $timeout(function () {
                // if popup is visible
                if (scope.matches.length) {
                    recalculatePosition();
                }

                scope.moveInProgress = false;
            }, eventDebounceTime);
        }

        // recalculate actual position and set new values to scope
        // after digest loop is popup in right position
        function recalculatePosition() {
            scope.position = appendToBody ? $position.offset(element) : $position.position(element);
            scope.position.top += element.prop('offsetHeight');
        }

        //we need to propagate user's query so we can higlight matches
        scope.query = undefined;

        //Declare the timeout promise var outside the function scope so that stacked calls can be cancelled later
        var timeoutPromise;

        var scheduleSearchWithTimeout = function (inputValue) {
            timeoutPromise = $timeout(function () {
                getMatchesAsync(inputValue);
            }, waitTime);
        };

        var cancelPreviousTimeout = function () {
            if (timeoutPromise) {
                $timeout.cancel(timeoutPromise);
            }
        };

        resetMatches();

        scope.select = function (activeIdx) {
            //called from within the $digest() cycle
            var locals = {};
            var model, item;

            selected = true;
            locals[parserResult.itemName] = item = scope.matches[activeIdx].model;
            model = parserResult.modelMapper(originalScope, locals);
            $setModelValue(originalScope, model);
            modelCtrl.$setValidity('editable', true);
            modelCtrl.$setValidity('parse', true);

            onSelectCallback(originalScope, {
                $item: item,
                $model: model,
                $label: parserResult.viewMapper(originalScope, locals)
            });

            resetMatches();

            //return focus to the input element if a match was selected via a mouse click event
            // use timeout to avoid $rootScope:inprog error
            if (scope.$eval(attrs.typeaheadFocusOnSelect) !== false) {
                $timeout(function () { element[0].focus(); }, 0, false);
            }
        };

        //bind keyboard events: arrows up(38) / down(40), enter(13) and tab(9), esc(27)
        element.bind('keydown', function (evt) {
            //typeahead is open and an "interesting" key was pressed
            if (scope.matches.length === 0 || HOT_KEYS.indexOf(evt.which) === -1) {
                return;
            }

            // if there's nothing selected (i.e. focusFirst) and enter or tab is hit, clear the results
            if (scope.activeIdx === -1 && (evt.which === 9 || evt.which === 13)) {
                resetMatches();
                scope.$digest();
                return;
            }

            evt.preventDefault();

            if (evt.which === 40) {
                scope.activeIdx = (scope.activeIdx + 1) % scope.matches.length;
                scope.$digest();
            } else if (evt.which === 38) {
                scope.activeIdx = (scope.activeIdx > 0 ? scope.activeIdx : scope.matches.length) - 1;
                scope.$digest();
            } else if (evt.which === 13 || evt.which === 9) {
                scope.$apply(function () {
                    scope.select(scope.activeIdx);
                });
            } else if (evt.which === 27) {
                evt.stopPropagation();

                resetMatches();
                scope.$digest();
            }
        });

        element.bind('blur', function () {
            if (isSelectOnBlur && scope.matches.length && scope.activeIdx !== -1 && !selected) {
                selected = true;
                scope.$apply(function () {
                    scope.select(scope.activeIdx);
                });
            }
            hasFocus = false;
            selected = false;
        });

        // Keep reference to click handler to unbind it.
        var dismissClickHandler = function (evt) {
            // Issue #3973
            // Firefox treats right click as a click on document
            if (element[0] !== evt.target && evt.which !== 3 && scope.matches.length !== 0) {
                resetMatches();
                if (!$rootScope.$$phase) {
                    scope.$digest();
                }
            }
        };

        $document.bind('click', dismissClickHandler);

        originalScope.$on('$destroy', function () {
            $document.unbind('click', dismissClickHandler);
            if (appendToBody || appendToElementId) {
                $popup.remove();
            }

            if (appendToBody) {
                angular.element($window).unbind('resize', fireRecalculating);
                $document.find('body').unbind('scroll', fireRecalculating);
            }
            // Prevent jQuery cache memory leak
            popUpEl.remove();
        });

        var $popup = $compile(popUpEl)(scope);

        if (appendToBody) {
            $document.find('body').append($popup);
        } else if (appendToElementId !== false) {
            angular.element($document[0].getElementById(appendToElementId)).append($popup);
        } else {
            element.after($popup);
        }

        this.init = function (_modelCtrl, _ngModelOptions) {
            modelCtrl = _modelCtrl;
            ngModelOptions = _ngModelOptions;

            //plug into $parsers pipeline to open a typeahead on view changes initiated from DOM
            //$parsers kick-in on all the changes coming from the view as well as manually triggered by $setViewValue
            modelCtrl.$parsers.unshift(function (inputValue) {
                hasFocus = true;

                if (minLength === 0 || inputValue && inputValue.length >= minLength) {
                    if (waitTime > 0) {
                        cancelPreviousTimeout();
                        scheduleSearchWithTimeout(inputValue);
                    } else {
                        getMatchesAsync(inputValue);
                    }
                } else {
                    isLoadingSetter(originalScope, false);
                    cancelPreviousTimeout();
                    resetMatches();
                }

                if (isEditable) {
                    return inputValue;
                } else {
                    if (!inputValue) {
                        // Reset in case user had typed something previously.
                        modelCtrl.$setValidity('editable', true);
                        return null;
                    } else {
                        modelCtrl.$setValidity('editable', false);
                        return undefined;
                    }
                }
            });

            modelCtrl.$formatters.push(function (modelValue) {
                var candidateViewValue, emptyViewValue;
                var locals = {};

                // The validity may be set to false via $parsers (see above) if
                // the model is restricted to selected values. If the model
                // is set manually it is considered to be valid.
                if (!isEditable) {
                    modelCtrl.$setValidity('editable', true);
                }

                if (inputFormatter) {
                    locals.$model = modelValue;
                    return inputFormatter(originalScope, locals);
                } else {
                    //it might happen that we don't have enough info to properly render input value
                    //we need to check for this situation and simply return model value if we can't apply custom formatting
                    locals[parserResult.itemName] = modelValue;
                    candidateViewValue = parserResult.viewMapper(originalScope, locals);
                    locals[parserResult.itemName] = undefined;
                    emptyViewValue = parserResult.viewMapper(originalScope, locals);

                    return candidateViewValue !== emptyViewValue ? candidateViewValue : modelValue;
                }
            });
        };
    }])

  .directive('uibTypeahead', function () {
      return {
          controller: 'UibTypeaheadController',
          require: ['ngModel', '^?ngModelOptions', 'uibTypeahead'],
          link: function (originalScope, element, attrs, ctrls) {
              ctrls[2].init(ctrls[0], ctrls[1]);
          }
      };
  })

  .directive('uibTypeaheadPopup', function () {
      return {
          scope: {
              matches: '=',
              query: '=',
              active: '=',
              position: '&',
              moveInProgress: '=',
              select: '&'
          },
          replace: true,
          templateUrl: function (element, attrs) {
              return attrs.popupTemplateUrl || 'template/typeahead/typeahead-popup.html';
          },
          link: function (scope, element, attrs) {
              scope.templateUrl = attrs.templateUrl;

              scope.isOpen = function () {
                  return scope.matches.length > 0;
              };

              scope.isActive = function (matchIdx) {
                  return scope.active == matchIdx;
              };

              scope.selectActive = function (matchIdx) {
                  scope.active = matchIdx;
              };

              scope.selectMatch = function (activeIdx) {
                  scope.select({ activeIdx: activeIdx });
              };
          }
      };
  })

  .directive('uibTypeaheadMatch', ['$templateRequest', '$compile', '$parse', function ($templateRequest, $compile, $parse) {
      return {
          scope: {
              index: '=',
              match: '=',
              query: '='
          },
          link: function (scope, element, attrs) {
              var tplUrl = $parse(attrs.templateUrl)(scope.$parent) || 'template/typeahead/typeahead-match.html';
              $templateRequest(tplUrl).then(function (tplContent) {
                  $compile(tplContent.trim())(scope, function (clonedElement) {
                      element.replaceWith(clonedElement);
                  });
              });
          }
      };
  }])

  .filter('uibTypeaheadHighlight', ['$sce', '$injector', '$log', function ($sce, $injector, $log) {
      var isSanitizePresent;
      isSanitizePresent = $injector.has('$sanitize');

      function escapeRegexp(queryToEscape) {
          // Regex: capture the whole query string and replace it with the string that will be used to match
          // the results, for example if the capture is "a" the result will be \a
          return queryToEscape.replace(/([.?*+^$[\]\\(){}|-])/g, '\\$1');
      }

      function containsHtml(matchItem) {
          return /<.*>/g.test(matchItem);
      }

      return function (matchItem, query) {
          if (!isSanitizePresent && containsHtml(matchItem)) {
              $log.warn('Unsafe use of typeahead please use ngSanitize'); // Warn the user about the danger
          }
          matchItem = query ? ('' + matchItem).replace(new RegExp(escapeRegexp(query), 'gi'), '<strong>$&</strong>') : matchItem; // Replaces the capture string with a the same string inside of a "strong" tag
          if (!isSanitizePresent) {
              matchItem = $sce.trustAsHtml(matchItem); // If $sanitize is not present we pack the string in a $sce object for the ng-bind-html directive
          }
          return matchItem;
      };
  }]);

/* Deprecated typeahead below */

angular.module('ui.bootstrap.typeahead')
  .value('$typeaheadSuppressWarning', false)
  .service('typeaheadParser', ['$parse', 'uibTypeaheadParser', '$log', '$typeaheadSuppressWarning', function ($parse, uibTypeaheadParser, $log, $typeaheadSuppressWarning) {
      if (!$typeaheadSuppressWarning) {
          $log.warn('typeaheadParser is now deprecated. Use uibTypeaheadParser instead.');
      }

      return uibTypeaheadParser;
  }])

  .directive('typeahead', ['$compile', '$parse', '$q', '$timeout', '$document', '$window', '$rootScope', '$uibPosition', 'typeaheadParser', '$log', '$typeaheadSuppressWarning',
    function ($compile, $parse, $q, $timeout, $document, $window, $rootScope, $position, typeaheadParser, $log, $typeaheadSuppressWarning) {
        var HOT_KEYS = [9, 13, 27, 38, 40];
        var eventDebounceTime = 200;
        return {
            require: ['ngModel', '^?ngModelOptions'],
            link: function (originalScope, element, attrs, ctrls) {
                if (!$typeaheadSuppressWarning) {
                    $log.warn('typeahead is now deprecated. Use uib-typeahead instead.');
                }
                var modelCtrl = ctrls[0];
                var ngModelOptions = ctrls[1];
                //SUPPORTED ATTRIBUTES (OPTIONS)

                //minimal no of characters that needs to be entered before typeahead kicks-in
                var minLength = originalScope.$eval(attrs.typeaheadMinLength);
                if (!minLength && minLength !== 0) {
                    minLength = 1;
                }

                //minimal wait time after last character typed before typeahead kicks-in
                var waitTime = originalScope.$eval(attrs.typeaheadWaitMs) || 0;

                //should it restrict model values to the ones selected from the popup only?
                var isEditable = originalScope.$eval(attrs.typeaheadEditable) !== false;

                //binding to a variable that indicates if matches are being retrieved asynchronously
                var isLoadingSetter = $parse(attrs.typeaheadLoading).assign || angular.noop;

                //a callback executed when a match is selected
                var onSelectCallback = $parse(attrs.typeaheadOnSelect);

                //should it select highlighted popup value when losing focus?
                var isSelectOnBlur = angular.isDefined(attrs.typeaheadSelectOnBlur) ? originalScope.$eval(attrs.typeaheadSelectOnBlur) : false;

                //binding to a variable that indicates if there were no results after the query is completed
                var isNoResultsSetter = $parse(attrs.typeaheadNoResults).assign || angular.noop;

                var inputFormatter = attrs.typeaheadInputFormatter ? $parse(attrs.typeaheadInputFormatter) : undefined;

                var appendToBody = attrs.typeaheadAppendToBody ? originalScope.$eval(attrs.typeaheadAppendToBody) : false;

                var appendToElementId = attrs.typeaheadAppendToElementId || false;

                var focusFirst = originalScope.$eval(attrs.typeaheadFocusFirst) !== false;

                //If input matches an item of the list exactly, select it automatically
                var selectOnExact = attrs.typeaheadSelectOnExact ? originalScope.$eval(attrs.typeaheadSelectOnExact) : false;

                //INTERNAL VARIABLES

                //model setter executed upon match selection
                var parsedModel = $parse(attrs.ngModel);
                var invokeModelSetter = $parse(attrs.ngModel + '($$$p)');
                var $setModelValue = function (scope, newValue) {
                    if (angular.isFunction(parsedModel(originalScope)) &&
                      ngModelOptions && ngModelOptions.$options && ngModelOptions.$options.getterSetter) {
                        return invokeModelSetter(scope, { $$$p: newValue });
                    } else {
                        return parsedModel.assign(scope, newValue);
                    }
                };

                //expressions used by typeahead
                var parserResult = typeaheadParser.parse(attrs.typeahead);

                var hasFocus;

                //Used to avoid bug in iOS webview where iOS keyboard does not fire
                //mousedown & mouseup events
                //Issue #3699
                var selected;

                //create a child scope for the typeahead directive so we are not polluting original scope
                //with typeahead-specific data (matches, query etc.)
                var scope = originalScope.$new();
                var offDestroy = originalScope.$on('$destroy', function () {
                    scope.$destroy();
                });
                scope.$on('$destroy', offDestroy);

                // WAI-ARIA
                var popupId = 'typeahead-' + scope.$id + '-' + Math.floor(Math.random() * 10000);
                element.attr({
                    'aria-autocomplete': 'list',
                    'aria-expanded': false,
                    'aria-owns': popupId
                });

                //pop-up element used to display matches
                var popUpEl = angular.element('<div typeahead-popup></div>');
                popUpEl.attr({
                    id: popupId,
                    matches: 'matches',
                    active: 'activeIdx',
                    select: 'select(activeIdx)',
                    'move-in-progress': 'moveInProgress',
                    query: 'query',
                    position: 'position'
                });
                //custom item template
                if (angular.isDefined(attrs.typeaheadTemplateUrl)) {
                    popUpEl.attr('template-url', attrs.typeaheadTemplateUrl);
                }

                if (angular.isDefined(attrs.typeaheadPopupTemplateUrl)) {
                    popUpEl.attr('popup-template-url', attrs.typeaheadPopupTemplateUrl);
                }

                var resetMatches = function () {
                    scope.matches = [];
                    scope.activeIdx = -1;
                    element.attr('aria-expanded', false);
                };

                var getMatchId = function (index) {
                    return popupId + '-option-' + index;
                };

                // Indicate that the specified match is the active (pre-selected) item in the list owned by this typeahead.
                // This attribute is added or removed automatically when the `activeIdx` changes.
                scope.$watch('activeIdx', function (index) {
                    if (index < 0) {
                        element.removeAttr('aria-activedescendant');
                    } else {
                        element.attr('aria-activedescendant', getMatchId(index));
                    }
                });

                var inputIsExactMatch = function (inputValue, index) {
                    if (scope.matches.length > index && inputValue) {
                        return inputValue.toUpperCase() === scope.matches[index].label.toUpperCase();
                    }

                    return false;
                };

                var getMatchesAsync = function (inputValue) {
                    var locals = { $viewValue: inputValue };
                    isLoadingSetter(originalScope, true);
                    isNoResultsSetter(originalScope, false);
                    $q.when(parserResult.source(originalScope, locals)).then(function (matches) {
                        //it might happen that several async queries were in progress if a user were typing fast
                        //but we are interested only in responses that correspond to the current view value
                        var onCurrentRequest = (inputValue === modelCtrl.$viewValue);
                        if (onCurrentRequest && hasFocus) {
                            if (matches && matches.length > 0) {
                                scope.activeIdx = focusFirst ? 0 : -1;
                                isNoResultsSetter(originalScope, false);
                                scope.matches.length = 0;

                                //transform labels
                                for (var i = 0; i < matches.length; i++) {
                                    locals[parserResult.itemName] = matches[i];
                                    scope.matches.push({
                                        id: getMatchId(i),
                                        label: parserResult.viewMapper(scope, locals),
                                        model: matches[i]
                                    });
                                }

                                scope.query = inputValue;
                                //position pop-up with matches - we need to re-calculate its position each time we are opening a window
                                //with matches as a pop-up might be absolute-positioned and position of an input might have changed on a page
                                //due to other elements being rendered
                                recalculatePosition();

                                element.attr('aria-expanded', true);

                                //Select the single remaining option if user input matches
                                if (selectOnExact && scope.matches.length === 1 && inputIsExactMatch(inputValue, 0)) {
                                    scope.select(0);
                                }
                            } else {
                                resetMatches();
                                isNoResultsSetter(originalScope, true);
                            }
                        }
                        if (onCurrentRequest) {
                            isLoadingSetter(originalScope, false);
                        }
                    }, function () {
                        resetMatches();
                        isLoadingSetter(originalScope, false);
                        isNoResultsSetter(originalScope, true);
                    });
                };

                // bind events only if appendToBody params exist - performance feature
                if (appendToBody) {
                    angular.element($window).bind('resize', fireRecalculating);
                    $document.find('body').bind('scroll', fireRecalculating);
                }

                // Declare the timeout promise var outside the function scope so that stacked calls can be cancelled later
                var timeoutEventPromise;

                // Default progress type
                scope.moveInProgress = false;

                function fireRecalculating() {
                    if (!scope.moveInProgress) {
                        scope.moveInProgress = true;
                        scope.$digest();
                    }

                    // Cancel previous timeout
                    if (timeoutEventPromise) {
                        $timeout.cancel(timeoutEventPromise);
                    }

                    // Debounced executing recalculate after events fired
                    timeoutEventPromise = $timeout(function () {
                        // if popup is visible
                        if (scope.matches.length) {
                            recalculatePosition();
                        }

                        scope.moveInProgress = false;
                    }, eventDebounceTime);
                }

                // recalculate actual position and set new values to scope
                // after digest loop is popup in right position
                function recalculatePosition() {
                    scope.position = appendToBody ? $position.offset(element) : $position.position(element);
                    scope.position.top += element.prop('offsetHeight');
                }

                resetMatches();

                //we need to propagate user's query so we can higlight matches
                scope.query = undefined;

                //Declare the timeout promise var outside the function scope so that stacked calls can be cancelled later
                var timeoutPromise;

                var scheduleSearchWithTimeout = function (inputValue) {
                    timeoutPromise = $timeout(function () {
                        getMatchesAsync(inputValue);
                    }, waitTime);
                };

                var cancelPreviousTimeout = function () {
                    if (timeoutPromise) {
                        $timeout.cancel(timeoutPromise);
                    }
                };

                //plug into $parsers pipeline to open a typeahead on view changes initiated from DOM
                //$parsers kick-in on all the changes coming from the view as well as manually triggered by $setViewValue
                modelCtrl.$parsers.unshift(function (inputValue) {
                    hasFocus = true;

                    if (minLength === 0 || inputValue && inputValue.length >= minLength) {
                        if (waitTime > 0) {
                            cancelPreviousTimeout();
                            scheduleSearchWithTimeout(inputValue);
                        } else {
                            getMatchesAsync(inputValue);
                        }
                    } else {
                        isLoadingSetter(originalScope, false);
                        cancelPreviousTimeout();
                        resetMatches();
                    }

                    if (isEditable) {
                        return inputValue;
                    } else {
                        if (!inputValue) {
                            // Reset in case user had typed something previously.
                            modelCtrl.$setValidity('editable', true);
                            return null;
                        } else {
                            modelCtrl.$setValidity('editable', false);
                            return undefined;
                        }
                    }
                });

                modelCtrl.$formatters.push(function (modelValue) {
                    var candidateViewValue, emptyViewValue;
                    var locals = {};

                    // The validity may be set to false via $parsers (see above) if
                    // the model is restricted to selected values. If the model
                    // is set manually it is considered to be valid.
                    if (!isEditable) {
                        modelCtrl.$setValidity('editable', true);
                    }

                    if (inputFormatter) {
                        locals.$model = modelValue;
                        return inputFormatter(originalScope, locals);
                    } else {
                        //it might happen that we don't have enough info to properly render input value
                        //we need to check for this situation and simply return model value if we can't apply custom formatting
                        locals[parserResult.itemName] = modelValue;
                        candidateViewValue = parserResult.viewMapper(originalScope, locals);
                        locals[parserResult.itemName] = undefined;
                        emptyViewValue = parserResult.viewMapper(originalScope, locals);

                        return candidateViewValue !== emptyViewValue ? candidateViewValue : modelValue;
                    }
                });

                scope.select = function (activeIdx) {
                    //called from within the $digest() cycle
                    var locals = {};
                    var model, item;

                    selected = true;
                    locals[parserResult.itemName] = item = scope.matches[activeIdx].model;
                    model = parserResult.modelMapper(originalScope, locals);
                    $setModelValue(originalScope, model);
                    modelCtrl.$setValidity('editable', true);
                    modelCtrl.$setValidity('parse', true);

                    onSelectCallback(originalScope, {
                        $item: item,
                        $model: model,
                        $label: parserResult.viewMapper(originalScope, locals)
                    });

                    resetMatches();

                    //return focus to the input element if a match was selected via a mouse click event
                    // use timeout to avoid $rootScope:inprog error
                    if (scope.$eval(attrs.typeaheadFocusOnSelect) !== false) {
                        $timeout(function () { element[0].focus(); }, 0, false);
                    }
                };

                //bind keyboard events: arrows up(38) / down(40), enter(13) and tab(9), esc(27)
                element.bind('keydown', function (evt) {
                    //typeahead is open and an "interesting" key was pressed
                    if (scope.matches.length === 0 || HOT_KEYS.indexOf(evt.which) === -1) {
                        return;
                    }

                    // if there's nothing selected (i.e. focusFirst) and enter or tab is hit, clear the results
                    if (scope.activeIdx === -1 && (evt.which === 9 || evt.which === 13)) {
                        resetMatches();
                        scope.$digest();
                        return;
                    }

                    evt.preventDefault();

                    if (evt.which === 40) {
                        scope.activeIdx = (scope.activeIdx + 1) % scope.matches.length;
                        scope.$digest();
                    } else if (evt.which === 38) {
                        scope.activeIdx = (scope.activeIdx > 0 ? scope.activeIdx : scope.matches.length) - 1;
                        scope.$digest();
                    } else if (evt.which === 13 || evt.which === 9) {
                        scope.$apply(function () {
                            scope.select(scope.activeIdx);
                        });
                    } else if (evt.which === 27) {
                        evt.stopPropagation();

                        resetMatches();
                        scope.$digest();
                    }
                });

                element.bind('blur', function () {
                    if (isSelectOnBlur && scope.matches.length && scope.activeIdx !== -1 && !selected) {
                        selected = true;
                        scope.$apply(function () {
                            scope.select(scope.activeIdx);
                        });
                    }
                    hasFocus = false;
                    selected = false;
                });

                // Keep reference to click handler to unbind it.
                var dismissClickHandler = function (evt) {
                    // Issue #3973
                    // Firefox treats right click as a click on document
                    if (element[0] !== evt.target && evt.which !== 3 && scope.matches.length !== 0) {
                        resetMatches();
                        if (!$rootScope.$$phase) {
                            scope.$digest();
                        }
                    }
                };

                $document.bind('click', dismissClickHandler);

                originalScope.$on('$destroy', function () {
                    $document.unbind('click', dismissClickHandler);
                    if (appendToBody || appendToElementId) {
                        $popup.remove();
                    }

                    if (appendToBody) {
                        angular.element($window).unbind('resize', fireRecalculating);
                        $document.find('body').unbind('scroll', fireRecalculating);
                    }
                    // Prevent jQuery cache memory leak
                    popUpEl.remove();
                });

                var $popup = $compile(popUpEl)(scope);

                if (appendToBody) {
                    $document.find('body').append($popup);
                } else if (appendToElementId !== false) {
                    angular.element($document[0].getElementById(appendToElementId)).append($popup);
                } else {
                    element.after($popup);
                }
            }
        };
    }])

  .directive('typeaheadPopup', ['$typeaheadSuppressWarning', '$log', function ($typeaheadSuppressWarning, $log) {
      return {
          scope: {
              matches: '=',
              query: '=',
              active: '=',
              position: '&',
              moveInProgress: '=',
              select: '&'
          },
          replace: true,
          templateUrl: function (element, attrs) {
              return attrs.popupTemplateUrl || 'template/typeahead/typeahead-popup.html';
          },
          link: function (scope, element, attrs) {

              if (!$typeaheadSuppressWarning) {
                  $log.warn('typeahead-popup is now deprecated. Use uib-typeahead-popup instead.');
              }
              scope.templateUrl = attrs.templateUrl;

              scope.isOpen = function () {
                  return scope.matches.length > 0;
              };

              scope.isActive = function (matchIdx) {
                  return scope.active == matchIdx;
              };

              scope.selectActive = function (matchIdx) {
                  scope.active = matchIdx;
              };

              scope.selectMatch = function (activeIdx) {
                  scope.select({ activeIdx: activeIdx });
              };
          }
      };
  }])

  .directive('typeaheadMatch', ['$templateRequest', '$compile', '$parse', '$typeaheadSuppressWarning', '$log', function ($templateRequest, $compile, $parse, $typeaheadSuppressWarning, $log) {
      return {
          restrict: 'EA',
          scope: {
              index: '=',
              match: '=',
              query: '='
          },
          link: function (scope, element, attrs) {
              if (!$typeaheadSuppressWarning) {
                  $log.warn('typeahead-match is now deprecated. Use uib-typeahead-match instead.');
              }

              var tplUrl = $parse(attrs.templateUrl)(scope.$parent) || 'template/typeahead/typeahead-match.html';
              $templateRequest(tplUrl).then(function (tplContent) {
                  $compile(tplContent.trim())(scope, function (clonedElement) {
                      element.replaceWith(clonedElement);
                  });
              });
          }
      };
  }])

  .filter('typeaheadHighlight', ['$sce', '$injector', '$log', '$typeaheadSuppressWarning', function ($sce, $injector, $log, $typeaheadSuppressWarning) {
      var isSanitizePresent;
      isSanitizePresent = $injector.has('$sanitize');

      function escapeRegexp(queryToEscape) {
          // Regex: capture the whole query string and replace it with the string that will be used to match
          // the results, for example if the capture is "a" the result will be \a
          return queryToEscape.replace(/([.?*+^$[\]\\(){}|-])/g, '\\$1');
      }

      function containsHtml(matchItem) {
          return /<.*>/g.test(matchItem);
      }

      return function (matchItem, query) {
          if (!$typeaheadSuppressWarning) {
              $log.warn('typeaheadHighlight is now deprecated. Use uibTypeaheadHighlight instead.');
          }

          if (!isSanitizePresent && containsHtml(matchItem)) {
              $log.warn('Unsafe use of typeahead please use ngSanitize'); // Warn the user about the danger
          }

          matchItem = query ? ('' + matchItem).replace(new RegExp(escapeRegexp(query), 'gi'), '<strong>$&</strong>') : matchItem; // Replaces the capture string with a the same string inside of a "strong" tag
          if (!isSanitizePresent) {
              matchItem = $sce.trustAsHtml(matchItem); // If $sanitize is not present we pack the string in a $sce object for the ng-bind-html directive
          }

          return matchItem;
      };
  }]);


/**
 * @ngdoc overview
 * @name ui.bootstrap.tabs
 *
 * @description
 * AngularJS version of the tabs directive.
 */

angular.module('ui.bootstrap.tabs', [])

.controller('UibTabsetController', ['$scope', function ($scope) {
    var ctrl = this,
        tabs = ctrl.tabs = $scope.tabs = [];

    ctrl.select = function (selectedTab) {
        angular.forEach(tabs, function (tab) {
            if (tab.active && tab !== selectedTab) {
                tab.active = false;
                tab.onDeselect();
                selectedTab.selectCalled = false;
            }
        });
        selectedTab.active = true;
        // only call select if it has not already been called
        if (!selectedTab.selectCalled) {
            selectedTab.onSelect();
            selectedTab.selectCalled = true;
        }
    };

    ctrl.addTab = function addTab(tab) {
        tabs.push(tab);
        // we can't run the select function on the first tab
        // since that would select it twice
        if (tabs.length === 1 && tab.active !== false) {
            tab.active = true;
        } else if (tab.active) {
            ctrl.select(tab);
        } else {
            tab.active = false;
        }
    };

    ctrl.removeTab = function removeTab(tab) {
        var index = tabs.indexOf(tab);
        //Select a new tab if the tab to be removed is selected and not destroyed
        if (tab.active && tabs.length > 1 && !destroyed) {
            //If this is the last tab, select the previous tab. else, the next tab.
            var newActiveIndex = index == tabs.length - 1 ? index - 1 : index + 1;
            ctrl.select(tabs[newActiveIndex]);
        }
        tabs.splice(index, 1);
    };

    var destroyed;
    $scope.$on('$destroy', function () {
        destroyed = true;
    });
}])

/**
 * @ngdoc directive
 * @name ui.bootstrap.tabs.directive:tabset
 * @restrict EA
 *
 * @description
 * Tabset is the outer container for the tabs directive
 *
 * @param {boolean=} vertical Whether or not to use vertical styling for the tabs.
 * @param {boolean=} justified Whether or not to use justified styling for the tabs.
 *
 * @example
<example module="ui.bootstrap">
  <file name="index.html">
    <uib-tabset>
      <uib-tab heading="Tab 1"><b>First</b> Content!</uib-tab>
      <uib-tab heading="Tab 2"><i>Second</i> Content!</uib-tab>
    </uib-tabset>
    <hr />
    <uib-tabset vertical="true">
      <uib-tab heading="Vertical Tab 1"><b>First</b> Vertical Content!</uib-tab>
      <uib-tab heading="Vertical Tab 2"><i>Second</i> Vertical Content!</uib-tab>
    </uib-tabset>
    <uib-tabset justified="true">
      <uib-tab heading="Justified Tab 1"><b>First</b> Justified Content!</uib-tab>
      <uib-tab heading="Justified Tab 2"><i>Second</i> Justified Content!</uib-tab>
    </uib-tabset>
  </file>
</example>
 */
.directive('uibTabset', function () {
    return {
        restrict: 'EA',
        transclude: true,
        replace: true,
        scope: {
            type: '@'
        },
        controller: 'UibTabsetController',
        templateUrl: 'template/tabs/tabset.html',
        link: function (scope, element, attrs) {
            scope.vertical = angular.isDefined(attrs.vertical) ? scope.$parent.$eval(attrs.vertical) : false;
            scope.justified = angular.isDefined(attrs.justified) ? scope.$parent.$eval(attrs.justified) : false;
        }
    };
})

/**
 * @ngdoc directive
 * @name ui.bootstrap.tabs.directive:tab
 * @restrict EA
 *
 * @param {string=} heading The visible heading, or title, of the tab. Set HTML headings with {@link ui.bootstrap.tabs.directive:tabHeading tabHeading}.
 * @param {string=} select An expression to evaluate when the tab is selected.
 * @param {boolean=} active A binding, telling whether or not this tab is selected.
 * @param {boolean=} disabled A binding, telling whether or not this tab is disabled.
 *
 * @description
 * Creates a tab with a heading and content. Must be placed within a {@link ui.bootstrap.tabs.directive:tabset tabset}.
 *
 * @example
<example module="ui.bootstrap">
  <file name="index.html">
    <div ng-controller="TabsDemoCtrl">
      <button class="btn btn-small" ng-click="items[0].active = true">
        Select item 1, using active binding
      </button>
      <button class="btn btn-small" ng-click="items[1].disabled = !items[1].disabled">
        Enable/disable item 2, using disabled binding
      </button>
      <br />
      <uib-tabset>
        <uib-tab heading="Tab 1">First Tab</uib-tab>
        <uib-tab select="alertMe()">
          <uib-tab-heading><i class="icon-bell"></i> Alert me!</tab-heading>
          Second Tab, with alert callback and html heading!
        </uib-tab>
        <uib-tab ng-repeat="item in items"
          heading="{{item.title}}"
          disabled="item.disabled"
          active="item.active">
          {{item.content}}
        </uib-tab>
      </uib-tabset>
    </div>
  </file>
  <file name="script.js">
    function TabsDemoCtrl($scope) {
      $scope.items = [
        { title:"Dynamic Title 1", content:"Dynamic Item 0" },
        { title:"Dynamic Title 2", content:"Dynamic Item 1", disabled: true }
      ];

      $scope.alertMe = function() {
        setTimeout(function() {
          alert("You've selected the alert tab!");
        });
      };
    };
  </file>
</example>
 */

/**
 * @ngdoc directive
 * @name ui.bootstrap.tabs.directive:tabHeading
 * @restrict EA
 *
 * @description
 * Creates an HTML heading for a {@link ui.bootstrap.tabs.directive:tab tab}. Must be placed as a child of a tab element.
 *
 * @example
<example module="ui.bootstrap">
  <file name="index.html">
    <uib-tabset>
      <uib-tab>
        <uib-tab-heading><b>HTML</b> in my titles?!</tab-heading>
        And some content, too!
      </uib-tab>
      <uib-tab>
        <uib-tab-heading><i class="icon-heart"></i> Icon heading?!?</tab-heading>
        That's right.
      </uib-tab>
    </uib-tabset>
  </file>
</example>
 */
.directive('uibTab', ['$parse', function ($parse) {
    return {
        require: '^uibTabset',
        restrict: 'EA',
        replace: true,
        templateUrl: 'template/tabs/tab.html',
        transclude: true,
        scope: {
            active: '=?',
            heading: '@',
            onSelect: '&select', //This callback is called in contentHeadingTransclude
            //once it inserts the tab's content into the dom
            onDeselect: '&deselect'
        },
        controller: function () {
            //Empty controller so other directives can require being 'under' a tab
        },
        link: function (scope, elm, attrs, tabsetCtrl, transclude) {
            scope.$watch('active', function (active) {
                if (active) {
                    tabsetCtrl.select(scope);
                }
            });

            scope.disabled = false;
            if (attrs.disable) {
                scope.$parent.$watch($parse(attrs.disable), function (value) {
                    scope.disabled = !!value;
                });
            }

            scope.select = function () {
                if (!scope.disabled) {
                    scope.active = true;
                }
            };

            tabsetCtrl.addTab(scope);
            scope.$on('$destroy', function () {
                tabsetCtrl.removeTab(scope);
            });

            //We need to transclude later, once the content container is ready.
            //when this link happens, we're inside a tab heading.
            scope.$transcludeFn = transclude;
        }
    };
}])

.directive('uibTabHeadingTransclude', function () {
    return {
        restrict: 'A',
        require: ['?^uibTab', '?^tab'], // TODO: change to '^uibTab' after deprecation removal
        link: function (scope, elm) {
            scope.$watch('headingElement', function updateHeadingElement(heading) {
                if (heading) {
                    elm.html('');
                    elm.append(heading);
                }
            });
        }
    };
})

.directive('uibTabContentTransclude', function () {
    return {
        restrict: 'A',
        require: ['?^uibTabset', '?^tabset'], // TODO: change to '^uibTabset' after deprecation removal
        link: function (scope, elm, attrs) {
            var tab = scope.$eval(attrs.uibTabContentTransclude);

            //Now our tab is ready to be transcluded: both the tab heading area
            //and the tab content area are loaded.  Transclude 'em both.
            tab.$transcludeFn(tab.$parent, function (contents) {
                angular.forEach(contents, function (node) {
                    if (isTabHeading(node)) {
                        //Let tabHeadingTransclude know.
                        tab.headingElement = node;
                    } else {
                        elm.append(node);
                    }
                });
            });
        }
    };

    function isTabHeading(node) {
        return node.tagName && (
          node.hasAttribute('tab-heading') || // TODO: remove after deprecation removal
          node.hasAttribute('data-tab-heading') || // TODO: remove after deprecation removal
          node.hasAttribute('x-tab-heading') || // TODO: remove after deprecation removal
          node.hasAttribute('uib-tab-heading') ||
          node.hasAttribute('data-uib-tab-heading') ||
          node.hasAttribute('x-uib-tab-heading') ||
          node.tagName.toLowerCase() === 'tab-heading' || // TODO: remove after deprecation removal
          node.tagName.toLowerCase() === 'data-tab-heading' || // TODO: remove after deprecation removal
          node.tagName.toLowerCase() === 'x-tab-heading' || // TODO: remove after deprecation removal
          node.tagName.toLowerCase() === 'uib-tab-heading' ||
          node.tagName.toLowerCase() === 'data-uib-tab-heading' ||
          node.tagName.toLowerCase() === 'x-uib-tab-heading'
        );
    }
});

/* deprecated tabs below */

angular.module('ui.bootstrap.tabs')

  .value('$tabsSuppressWarning', false)

  .controller('TabsetController', ['$scope', '$controller', '$log', '$tabsSuppressWarning', function ($scope, $controller, $log, $tabsSuppressWarning) {
      if (!$tabsSuppressWarning) {
          $log.warn('TabsetController is now deprecated. Use UibTabsetController instead.');
      }

      angular.extend(this, $controller('UibTabsetController', {
          $scope: $scope
      }));
  }])

  .directive('tabset', ['$log', '$tabsSuppressWarning', function ($log, $tabsSuppressWarning) {
      return {
          restrict: 'EA',
          transclude: true,
          replace: true,
          scope: {
              type: '@'
          },
          controller: 'TabsetController',
          templateUrl: 'template/tabs/tabset.html',
          link: function (scope, element, attrs) {

              if (!$tabsSuppressWarning) {
                  $log.warn('tabset is now deprecated. Use uib-tabset instead.');
              }
              scope.vertical = angular.isDefined(attrs.vertical) ? scope.$parent.$eval(attrs.vertical) : false;
              scope.justified = angular.isDefined(attrs.justified) ? scope.$parent.$eval(attrs.justified) : false;
          }
      };
  }])

  .directive('tab', ['$parse', '$log', '$tabsSuppressWarning', function ($parse, $log, $tabsSuppressWarning) {
      return {
          require: '^tabset',
          restrict: 'EA',
          replace: true,
          templateUrl: 'template/tabs/tab.html',
          transclude: true,
          scope: {
              active: '=?',
              heading: '@',
              onSelect: '&select', //This callback is called in contentHeadingTransclude
              //once it inserts the tab's content into the dom
              onDeselect: '&deselect'
          },
          controller: function () {
              //Empty controller so other directives can require being 'under' a tab
          },
          link: function (scope, elm, attrs, tabsetCtrl, transclude) {
              if (!$tabsSuppressWarning) {
                  $log.warn('tab is now deprecated. Use uib-tab instead.');
              }

              scope.$watch('active', function (active) {
                  if (active) {
                      tabsetCtrl.select(scope);
                  }
              });

              scope.disabled = false;
              if (attrs.disable) {
                  scope.$parent.$watch($parse(attrs.disable), function (value) {
                      scope.disabled = !!value;
                  });
              }

              scope.select = function () {
                  if (!scope.disabled) {
                      scope.active = true;
                  }
              };

              tabsetCtrl.addTab(scope);
              scope.$on('$destroy', function () {
                  tabsetCtrl.removeTab(scope);
              });

              //We need to transclude later, once the content container is ready.
              //when this link happens, we're inside a tab heading.
              scope.$transcludeFn = transclude;
          }
      };
  }])

  .directive('tabHeadingTransclude', ['$log', '$tabsSuppressWarning', function ($log, $tabsSuppressWarning) {
      return {
          restrict: 'A',
          require: '^tab',
          link: function (scope, elm) {
              if (!$tabsSuppressWarning) {
                  $log.warn('tab-heading-transclude is now deprecated. Use uib-tab-heading-transclude instead.');
              }

              scope.$watch('headingElement', function updateHeadingElement(heading) {
                  if (heading) {
                      elm.html('');
                      elm.append(heading);
                  }
              });
          }
      };
  }])

  .directive('tabContentTransclude', ['$log', '$tabsSuppressWarning', function ($log, $tabsSuppressWarning) {
      return {
          restrict: 'A',
          require: '^tabset',
          link: function (scope, elm, attrs) {
              if (!$tabsSuppressWarning) {
                  $log.warn('tab-content-transclude is now deprecated. Use uib-tab-content-transclude instead.');
              }

              var tab = scope.$eval(attrs.tabContentTransclude);

              //Now our tab is ready to be transcluded: both the tab heading area
              //and the tab content area are loaded.  Transclude 'em both.
              tab.$transcludeFn(tab.$parent, function (contents) {
                  angular.forEach(contents, function (node) {
                      if (isTabHeading(node)) {
                          //Let tabHeadingTransclude know.
                          tab.headingElement = node;
                      }
                      else {
                          elm.append(node);
                      }
                  });
              });
          }
      };

      function isTabHeading(node) {
          return node.tagName && (
              node.hasAttribute('tab-heading') ||
              node.hasAttribute('data-tab-heading') ||
              node.hasAttribute('x-tab-heading') ||
              node.tagName.toLowerCase() === 'tab-heading' ||
              node.tagName.toLowerCase() === 'data-tab-heading' ||
              node.tagName.toLowerCase() === 'x-tab-heading'
            );
      }
  }]);

/**
 * The following features are still outstanding: popup delay, animation as a
 * function, placement as a function, inside, support for more triggers than
 * just mouse enter/leave, and selector delegatation.
 */
angular.module('ui.bootstrap.popover', ['ui.bootstrap.tooltip'])

.directive('uibPopoverTemplatePopup', function () {
    return {
        replace: true,
        scope: {
            title: '@', contentExp: '&', placement: '@', popupClass: '@', animation: '&', isOpen: '&',
            originScope: '&'
        },
        templateUrl: 'template/popover/popover-template.html',
        link: function (scope, element) {
            element.addClass('popover');
        }
    };
})

.directive('uibPopoverTemplate', ['$uibTooltip', function ($uibTooltip) {
    return $uibTooltip('uibPopoverTemplate', 'popover', 'click', {
        useContentExp: true
    });
}])

.directive('uibPopoverHtmlPopup', function () {
    return {
        replace: true,
        scope: { contentExp: '&', title: '@', placement: '@', popupClass: '@', animation: '&', isOpen: '&' },
        templateUrl: 'template/popover/popover-html.html',
        link: function (scope, element) {
            element.addClass('popover');
        }
    };
})

.directive('uibPopoverHtml', ['$uibTooltip', function ($uibTooltip) {
    return $uibTooltip('uibPopoverHtml', 'popover', 'click', {
        useContentExp: true
    });
}])

.directive('uibPopoverPopup', function () {
    return {
        replace: true,
        scope: { title: '@', content: '@', placement: '@', popupClass: '@', animation: '&', isOpen: '&' },
        templateUrl: 'template/popover/popover.html',
        link: function (scope, element) {
            element.addClass('popover');
        }
    };
})

.directive('uibPopover', ['$uibTooltip', function ($uibTooltip) {
    return $uibTooltip('uibPopover', 'popover', 'click');
}]);

/* Deprecated popover below */

angular.module('ui.bootstrap.popover')

.value('$popoverSuppressWarning', false)

.directive('popoverTemplatePopup', ['$log', '$popoverSuppressWarning', function ($log, $popoverSuppressWarning) {
    return {
        replace: true,
        scope: {
            title: '@', contentExp: '&', placement: '@', popupClass: '@', animation: '&', isOpen: '&',
            originScope: '&'
        },
        templateUrl: 'template/popover/popover-template.html',
        link: function (scope, element) {
            if (!$popoverSuppressWarning) {
                $log.warn('popover-template-popup is now deprecated. Use uib-popover-template-popup instead.');
            }

            element.addClass('popover');
        }
    };
}])

.directive('popoverTemplate', ['$tooltip', function ($tooltip) {
    return $tooltip('popoverTemplate', 'popover', 'click', {
        useContentExp: true
    });
}])

.directive('popoverHtmlPopup', ['$log', '$popoverSuppressWarning', function ($log, $popoverSuppressWarning) {
    return {
        replace: true,
        scope: { contentExp: '&', title: '@', placement: '@', popupClass: '@', animation: '&', isOpen: '&' },
        templateUrl: 'template/popover/popover-html.html',
        link: function (scope, element) {
            if (!$popoverSuppressWarning) {
                $log.warn('popover-html-popup is now deprecated. Use uib-popover-html-popup instead.');
            }

            element.addClass('popover');
        }
    };
}])

.directive('popoverHtml', ['$tooltip', function ($tooltip) {
    return $tooltip('popoverHtml', 'popover', 'click', {
        useContentExp: true
    });
}])

.directive('popoverPopup', ['$log', '$popoverSuppressWarning', function ($log, $popoverSuppressWarning) {
    return {
        replace: true,
        scope: { title: '@', content: '@', placement: '@', popupClass: '@', animation: '&', isOpen: '&' },
        templateUrl: 'template/popover/popover.html',
        link: function (scope, element) {
            if (!$popoverSuppressWarning) {
                $log.warn('popover-popup is now deprecated. Use uib-popover-popup instead.');
            }

            element.addClass('popover');
        }
    };
}])

.directive('popover', ['$tooltip', function ($tooltip) {

    return $tooltip('popover', 'popover', 'click');
}]);

angular.module('ui.bootstrap.modal', ['ui.bootstrap.stackedMap'])
/**
 * A helper, internal data structure that stores all references attached to key
 */
  .factory('$$multiMap', function () {
      return {
          createNew: function () {
              var map = {};

              return {
                  entries: function () {
                      return Object.keys(map).map(function (key) {
                          return {
                              key: key,
                              value: map[key]
                          };
                      });
                  },
                  get: function (key) {
                      return map[key];
                  },
                  hasKey: function (key) {
                      return !!map[key];
                  },
                  keys: function () {
                      return Object.keys(map);
                  },
                  put: function (key, value) {
                      if (!map[key]) {
                          map[key] = [];
                      }

                      map[key].push(value);
                  },
                  remove: function (key, value) {
                      var values = map[key];

                      if (!values) {
                          return;
                      }

                      var idx = values.indexOf(value);

                      if (idx !== -1) {
                          values.splice(idx, 1);
                      }

                      if (!values.length) {
                          delete map[key];
                      }
                  }
              };
          }
      };
  })

/**
 * A helper directive for the $modal service. It creates a backdrop element.
 */
  .directive('uibModalBackdrop', [
           '$animate', '$injector', '$uibModalStack',
  function ($animate, $injector, $modalStack) {
      var $animateCss = null;

      if ($injector.has('$animateCss')) {
          $animateCss = $injector.get('$animateCss');
      }

      return {
          replace: true,
          templateUrl: 'template/modal/backdrop.html',
          compile: function (tElement, tAttrs) {
              tElement.addClass(tAttrs.backdropClass);
              return linkFn;
          }
      };

      function linkFn(scope, element, attrs) {
          // Temporary fix for prefixing
          element.addClass('modal-backdrop');

          if (attrs.modalInClass) {
              if ($animateCss) {
                  $animateCss(element, {
                      addClass: attrs.modalInClass
                  }).start();
              } else {
                  $animate.addClass(element, attrs.modalInClass);
              }

              scope.$on($modalStack.NOW_CLOSING_EVENT, function (e, setIsAsync) {
                  var done = setIsAsync();
                  if ($animateCss) {
                      $animateCss(element, {
                          removeClass: attrs.modalInClass
                      }).start().then(done);
                  } else {
                      $animate.removeClass(element, attrs.modalInClass).then(done);
                  }
              });
          }
      }
  }])

  .directive('uibModalWindow', [
           '$uibModalStack', '$q', '$animate', '$injector',
  function ($modalStack, $q, $animate, $injector) {
      var $animateCss = null;

      if ($injector.has('$animateCss')) {
          $animateCss = $injector.get('$animateCss');
      }

      return {
          scope: {
              index: '@'
          },
          replace: true,
          transclude: true,
          templateUrl: function (tElement, tAttrs) {
              return tAttrs.templateUrl || 'template/modal/window.html';
          },
          link: function (scope, element, attrs) {
              element.addClass(attrs.windowClass || '');
              element.addClass(attrs.windowTopClass || '');
              scope.size = attrs.size;

              scope.close = function (evt) {
                  var modal = $modalStack.getTop();
                  if (modal && modal.value.backdrop && modal.value.backdrop !== 'static' && (evt.target === evt.currentTarget)) {
                      evt.preventDefault();
                      evt.stopPropagation();
                      $modalStack.dismiss(modal.key, 'backdrop click');
                  }
              };

              // moved from template to fix issue #2280
              element.on('click', scope.close);

              // This property is only added to the scope for the purpose of detecting when this directive is rendered.
              // We can detect that by using this property in the template associated with this directive and then use
              // {@link Attribute#$observe} on it. For more details please see {@link TableColumnResize}.
              scope.$isRendered = true;

              // Deferred object that will be resolved when this modal is render.
              var modalRenderDeferObj = $q.defer();
              // Observe function will be called on next digest cycle after compilation, ensuring that the DOM is ready.
              // In order to use this way of finding whether DOM is ready, we need to observe a scope property used in modal's template.
              attrs.$observe('modalRender', function (value) {
                  if (value == 'true') {
                      modalRenderDeferObj.resolve();
                  }
              });

              modalRenderDeferObj.promise.then(function () {
                  var animationPromise = null;

                  if (attrs.modalInClass) {
                      if ($animateCss) {
                          animationPromise = $animateCss(element, {
                              addClass: attrs.modalInClass
                          }).start();
                      } else {
                          animationPromise = $animate.addClass(element, attrs.modalInClass);
                      }

                      scope.$on($modalStack.NOW_CLOSING_EVENT, function (e, setIsAsync) {
                          var done = setIsAsync();
                          if ($animateCss) {
                              $animateCss(element, {
                                  removeClass: attrs.modalInClass
                              }).start().then(done);
                          } else {
                              $animate.removeClass(element, attrs.modalInClass).then(done);
                          }
                      });
                  }


                  $q.when(animationPromise).then(function () {
                      var inputWithAutofocus = element[0].querySelector('[autofocus]');
                      /**
                       * Auto-focusing of a freshly-opened modal element causes any child elements
                       * with the autofocus attribute to lose focus. This is an issue on touch
                       * based devices which will show and then hide the onscreen keyboard.
                       * Attempts to refocus the autofocus element via JavaScript will not reopen
                       * the onscreen keyboard. Fixed by updated the focusing logic to only autofocus
                       * the modal element if the modal does not contain an autofocus element.
                       */
                      if (inputWithAutofocus) {
                          inputWithAutofocus.focus();
                      } else {
                          element[0].focus();
                      }
                  });

                  // Notify {@link $modalStack} that modal is rendered.
                  var modal = $modalStack.getTop();
                  if (modal) {
                      $modalStack.modalRendered(modal.key);
                  }
              });
          }
      };
  }])

  .directive('uibModalAnimationClass', function () {
      return {
          compile: function (tElement, tAttrs) {
              if (tAttrs.modalAnimation) {
                  tElement.addClass(tAttrs.uibModalAnimationClass);
              }
          }
      };
  })

  .directive('uibModalTransclude', function () {
      return {
          link: function ($scope, $element, $attrs, controller, $transclude) {
              $transclude($scope.$parent, function (clone) {
                  $element.empty();
                  $element.append(clone);
              });
          }
      };
  })

  .factory('$uibModalStack', [
             '$animate', '$timeout', '$document', '$compile', '$rootScope',
             '$q',
             '$injector',
             '$$multiMap',
             '$$stackedMap',
    function ($animate, $timeout, $document, $compile, $rootScope,
              $q,
              $injector,
              $$multiMap,
              $$stackedMap) {
        var $animateCss = null;

        if ($injector.has('$animateCss')) {
            $animateCss = $injector.get('$animateCss');
        }

        var OPENED_MODAL_CLASS = 'modal-open';

        var backdropDomEl, backdropScope;
        var openedWindows = $$stackedMap.createNew();
        var openedClasses = $$multiMap.createNew();
        var $modalStack = {
            NOW_CLOSING_EVENT: 'modal.stack.now-closing'
        };

        //Modal focus behavior
        var focusableElementList;
        var focusIndex = 0;
        var tababbleSelector = 'a[href], area[href], input:not([disabled]), ' +
          'button:not([disabled]),select:not([disabled]), textarea:not([disabled]), ' +
          'iframe, object, embed, *[tabindex], *[contenteditable=true]';

        function backdropIndex() {
            var topBackdropIndex = -1;
            var opened = openedWindows.keys();
            for (var i = 0; i < opened.length; i++) {
                if (openedWindows.get(opened[i]).value.backdrop) {
                    topBackdropIndex = i;
                }
            }
            return topBackdropIndex;
        }

        $rootScope.$watch(backdropIndex, function (newBackdropIndex) {
            if (backdropScope) {
                backdropScope.index = newBackdropIndex;
            }
        });

        function removeModalWindow(modalInstance, elementToReceiveFocus) {
            var body = $document.find('body').eq(0);
            var modalWindow = openedWindows.get(modalInstance).value;

            //clean up the stack
            openedWindows.remove(modalInstance);

            removeAfterAnimate(modalWindow.modalDomEl, modalWindow.modalScope, function () {
                var modalBodyClass = modalWindow.openedClass || OPENED_MODAL_CLASS;
                openedClasses.remove(modalBodyClass, modalInstance);
                body.toggleClass(modalBodyClass, openedClasses.hasKey(modalBodyClass));
                toggleTopWindowClass(true);
            });
            checkRemoveBackdrop();

            //move focus to specified element if available, or else to body
            if (elementToReceiveFocus && elementToReceiveFocus.focus) {
                elementToReceiveFocus.focus();
            } else {
                body.focus();
            }
        }

        // Add or remove "windowTopClass" from the top window in the stack
        function toggleTopWindowClass(toggleSwitch) {
            var modalWindow;

            if (openedWindows.length() > 0) {
                modalWindow = openedWindows.top().value;
                modalWindow.modalDomEl.toggleClass(modalWindow.windowTopClass || '', toggleSwitch);
            }
        }

        function checkRemoveBackdrop() {
            //remove backdrop if no longer needed
            if (backdropDomEl && backdropIndex() == -1) {
                var backdropScopeRef = backdropScope;
                removeAfterAnimate(backdropDomEl, backdropScope, function () {
                    backdropScopeRef = null;
                });
                backdropDomEl = undefined;
                backdropScope = undefined;
            }
        }

        function removeAfterAnimate(domEl, scope, done) {
            var asyncDeferred;
            var asyncPromise = null;
            var setIsAsync = function () {
                if (!asyncDeferred) {
                    asyncDeferred = $q.defer();
                    asyncPromise = asyncDeferred.promise;
                }

                return function asyncDone() {
                    asyncDeferred.resolve();
                };
            };
            scope.$broadcast($modalStack.NOW_CLOSING_EVENT, setIsAsync);

            // Note that it's intentional that asyncPromise might be null.
            // That's when setIsAsync has not been called during the
            // NOW_CLOSING_EVENT broadcast.
            return $q.when(asyncPromise).then(afterAnimating);

            function afterAnimating() {
                if (afterAnimating.done) {
                    return;
                }
                afterAnimating.done = true;

                if ($animateCss) {
                    $animateCss(domEl, {
                        event: 'leave'
                    }).start().then(function () {
                        domEl.remove();
                    });
                } else {
                    $animate.leave(domEl);
                }
                scope.$destroy();
                if (done) {
                    done();
                }
            }
        }

        $document.bind('keydown', function (evt) {
            if (evt.isDefaultPrevented()) {
                return evt;
            }

            var modal = openedWindows.top();
            if (modal && modal.value.keyboard) {
                switch (evt.which) {
                    case 27: {
                        evt.preventDefault();
                        $rootScope.$apply(function () {
                            $modalStack.dismiss(modal.key, 'escape key press');
                        });
                        break;
                    }
                    case 9: {
                        $modalStack.loadFocusElementList(modal);
                        var focusChanged = false;
                        if (evt.shiftKey) {
                            if ($modalStack.isFocusInFirstItem(evt)) {
                                focusChanged = $modalStack.focusLastFocusableElement();
                            }
                        } else {
                            if ($modalStack.isFocusInLastItem(evt)) {
                                focusChanged = $modalStack.focusFirstFocusableElement();
                            }
                        }

                        if (focusChanged) {
                            evt.preventDefault();
                            evt.stopPropagation();
                        }
                        break;
                    }
                }
            }
        });

        $modalStack.open = function (modalInstance, modal) {
            var modalOpener = $document[0].activeElement,
              modalBodyClass = modal.openedClass || OPENED_MODAL_CLASS;

            toggleTopWindowClass(false);

            openedWindows.add(modalInstance, {
                deferred: modal.deferred,
                renderDeferred: modal.renderDeferred,
                modalScope: modal.scope,
                backdrop: modal.backdrop,
                keyboard: modal.keyboard,
                openedClass: modal.openedClass,
                windowTopClass: modal.windowTopClass
            });

            openedClasses.put(modalBodyClass, modalInstance);

            var body = $document.find('body').eq(0),
                currBackdropIndex = backdropIndex();

            if (currBackdropIndex >= 0 && !backdropDomEl) {
                backdropScope = $rootScope.$new(true);
                backdropScope.index = currBackdropIndex;
                var angularBackgroundDomEl = angular.element('<div uib-modal-backdrop="modal-backdrop"></div>');
                angularBackgroundDomEl.attr('backdrop-class', modal.backdropClass);
                if (modal.animation) {
                    angularBackgroundDomEl.attr('modal-animation', 'true');
                }
                backdropDomEl = $compile(angularBackgroundDomEl)(backdropScope);
                body.append(backdropDomEl);
            }

            var angularDomEl = angular.element('<div uib-modal-window="modal-window"></div>');
            angularDomEl.attr({
                'template-url': modal.windowTemplateUrl,
                'window-class': modal.windowClass,
                'window-top-class': modal.windowTopClass,
                'size': modal.size,
                'index': openedWindows.length() - 1,
                'animate': 'animate'
            }).html(modal.content);
            if (modal.animation) {
                angularDomEl.attr('modal-animation', 'true');
            }

            var modalDomEl = $compile(angularDomEl)(modal.scope);
            openedWindows.top().value.modalDomEl = modalDomEl;
            openedWindows.top().value.modalOpener = modalOpener;
            body.append(modalDomEl);
            body.addClass(modalBodyClass);

            $modalStack.clearFocusListCache();
        };

        function broadcastClosing(modalWindow, resultOrReason, closing) {
            return !modalWindow.value.modalScope.$broadcast('modal.closing', resultOrReason, closing).defaultPrevented;
        }

        $modalStack.close = function (modalInstance, result) {
            var modalWindow = openedWindows.get(modalInstance);
            if (modalWindow && broadcastClosing(modalWindow, result, true)) {
                modalWindow.value.modalScope.$$uibDestructionScheduled = true;
                modalWindow.value.deferred.resolve(result);
                removeModalWindow(modalInstance, modalWindow.value.modalOpener);
                return true;
            }
            return !modalWindow;
        };

        $modalStack.dismiss = function (modalInstance, reason) {
            var modalWindow = openedWindows.get(modalInstance);
            if (modalWindow && broadcastClosing(modalWindow, reason, false)) {
                modalWindow.value.modalScope.$$uibDestructionScheduled = true;
                modalWindow.value.deferred.reject(reason);
                removeModalWindow(modalInstance, modalWindow.value.modalOpener);
                return true;
            }
            return !modalWindow;
        };

        $modalStack.dismissAll = function (reason) {
            var topModal = this.getTop();
            while (topModal && this.dismiss(topModal.key, reason)) {
                topModal = this.getTop();
            }
        };

        $modalStack.getTop = function () {
            return openedWindows.top();
        };

        $modalStack.modalRendered = function (modalInstance) {
            var modalWindow = openedWindows.get(modalInstance);
            if (modalWindow) {
                modalWindow.value.renderDeferred.resolve();
            }
        };

        $modalStack.focusFirstFocusableElement = function () {
            if (focusableElementList.length > 0) {
                focusableElementList[0].focus();
                return true;
            }
            return false;
        };
        $modalStack.focusLastFocusableElement = function () {
            if (focusableElementList.length > 0) {
                focusableElementList[focusableElementList.length - 1].focus();
                return true;
            }
            return false;
        };

        $modalStack.isFocusInFirstItem = function (evt) {
            if (focusableElementList.length > 0) {
                return (evt.target || evt.srcElement) == focusableElementList[0];
            }
            return false;
        };

        $modalStack.isFocusInLastItem = function (evt) {
            if (focusableElementList.length > 0) {
                return (evt.target || evt.srcElement) == focusableElementList[focusableElementList.length - 1];
            }
            return false;
        };

        $modalStack.clearFocusListCache = function () {
            focusableElementList = [];
            focusIndex = 0;
        };

        $modalStack.loadFocusElementList = function (modalWindow) {
            if (focusableElementList === undefined || !focusableElementList.length) {
                if (modalWindow) {
                    var modalDomE1 = modalWindow.value.modalDomEl;
                    if (modalDomE1 && modalDomE1.length) {
                        focusableElementList = modalDomE1[0].querySelectorAll(tababbleSelector);
                    }
                }
            }
        };

        return $modalStack;
    }])

  .provider('$uibModal', function () {
      var $modalProvider = {
          options: {
              animation: true,
              backdrop: true, //can also be false or 'static'
              keyboard: true
          },
          $get: ['$injector', '$rootScope', '$q', '$templateRequest', '$controller', '$uibModalStack', '$modalSuppressWarning', '$log',
            function ($injector, $rootScope, $q, $templateRequest, $controller, $modalStack, $modalSuppressWarning, $log) {
                var $modal = {};

                function getTemplatePromise(options) {
                    return options.template ? $q.when(options.template) :
                      $templateRequest(angular.isFunction(options.templateUrl) ? (options.templateUrl)() : options.templateUrl);
                }

                function getResolvePromises(resolves) {
                    var promisesArr = [];
                    angular.forEach(resolves, function (value) {
                        if (angular.isFunction(value) || angular.isArray(value)) {
                            promisesArr.push($q.when($injector.invoke(value)));
                        } else if (angular.isString(value)) {
                            promisesArr.push($q.when($injector.get(value)));
                        } else {
                            promisesArr.push($q.when(value));
                        }
                    });
                    return promisesArr;
                }

                var promiseChain = null;
                $modal.getPromiseChain = function () {
                    return promiseChain;
                };

                $modal.open = function (modalOptions) {
                    var modalResultDeferred = $q.defer();
                    var modalOpenedDeferred = $q.defer();
                    var modalRenderDeferred = $q.defer();

                    //prepare an instance of a modal to be injected into controllers and returned to a caller
                    var modalInstance = {
                        result: modalResultDeferred.promise,
                        opened: modalOpenedDeferred.promise,
                        rendered: modalRenderDeferred.promise,
                        close: function (result) {
                            return $modalStack.close(modalInstance, result);
                        },
                        dismiss: function (reason) {
                            return $modalStack.dismiss(modalInstance, reason);
                        }
                    };

                    //merge and clean up options
                    modalOptions = angular.extend({}, $modalProvider.options, modalOptions);
                    modalOptions.resolve = modalOptions.resolve || {};

                    //verify options
                    if (!modalOptions.template && !modalOptions.templateUrl) {
                        throw new Error('One of template or templateUrl options is required.');
                    }

                    var templateAndResolvePromise =
                      $q.all([getTemplatePromise(modalOptions)].concat(getResolvePromises(modalOptions.resolve)));

                    function resolveWithTemplate() {
                        return templateAndResolvePromise;
                    }

                    // Wait for the resolution of the existing promise chain.
                    // Then switch to our own combined promise dependency (regardless of how the previous modal fared).
                    // Then add to $modalStack and resolve opened.
                    // Finally clean up the chain variable if no subsequent modal has overwritten it.
                    var samePromise;
                    samePromise = promiseChain = $q.all([promiseChain])
                      .then(resolveWithTemplate, resolveWithTemplate)
                      .then(function resolveSuccess(tplAndVars) {

                          var modalScope = (modalOptions.scope || $rootScope).$new();
                          modalScope.$close = modalInstance.close;
                          modalScope.$dismiss = modalInstance.dismiss;

                          modalScope.$on('$destroy', function () {
                              if (!modalScope.$$uibDestructionScheduled) {
                                  modalScope.$dismiss('$uibUnscheduledDestruction');
                              }
                          });

                          var ctrlInstance, ctrlLocals = {};
                          var resolveIter = 1;

                          //controllers
                          if (modalOptions.controller) {
                              ctrlLocals.$scope = modalScope;
                              ctrlLocals.$uibModalInstance = modalInstance;
                              Object.defineProperty(ctrlLocals, '$modalInstance', {
                                  get: function () {
                                      if (!$modalSuppressWarning) {
                                          $log.warn('$modalInstance is now deprecated. Use $uibModalInstance instead.');
                                      }

                                      return modalInstance;
                                  }
                              });
                              angular.forEach(modalOptions.resolve, function (value, key) {
                                  ctrlLocals[key] = tplAndVars[resolveIter++];
                              });

                              ctrlInstance = $controller(modalOptions.controller, ctrlLocals);
                              if (modalOptions.controllerAs) {
                                  if (modalOptions.bindToController) {
                                      angular.extend(ctrlInstance, modalScope);
                                  }

                                  modalScope[modalOptions.controllerAs] = ctrlInstance;
                              }
                          }

                          $modalStack.open(modalInstance, {
                              scope: modalScope,
                              deferred: modalResultDeferred,
                              renderDeferred: modalRenderDeferred,
                              content: tplAndVars[0],
                              animation: modalOptions.animation,
                              backdrop: modalOptions.backdrop,
                              keyboard: modalOptions.keyboard,
                              backdropClass: modalOptions.backdropClass,
                              windowTopClass: modalOptions.windowTopClass,
                              windowClass: modalOptions.windowClass,
                              windowTemplateUrl: modalOptions.windowTemplateUrl,
                              size: modalOptions.size,
                              openedClass: modalOptions.openedClass
                          });
                          modalOpenedDeferred.resolve(true);

                      }, function resolveError(reason) {
                          modalOpenedDeferred.reject(reason);
                          modalResultDeferred.reject(reason);
                      })
                    .finally(function () {
                        if (promiseChain === samePromise) {
                            promiseChain = null;
                        }
                    });

                    return modalInstance;
                };

                return $modal;
            }
          ]
      };

      return $modalProvider;
  });

/* deprecated modal below */

angular.module('ui.bootstrap.modal')

  .value('$modalSuppressWarning', false)

  /**
   * A helper directive for the $modal service. It creates a backdrop element.
   */
  .directive('modalBackdrop', [
    '$animate', '$injector', '$modalStack', '$log', '$modalSuppressWarning',
    function ($animate, $injector, $modalStack, $log, $modalSuppressWarning) {
        var $animateCss = null;

        if ($injector.has('$animateCss')) {
            $animateCss = $injector.get('$animateCss');
        }

        return {
            replace: true,
            templateUrl: 'template/modal/backdrop.html',
            compile: function (tElement, tAttrs) {
                tElement.addClass(tAttrs.backdropClass);
                return linkFn;
            }
        };

        function linkFn(scope, element, attrs) {
            if (!$modalSuppressWarning) {
                $log.warn('modal-backdrop is now deprecated. Use uib-modal-backdrop instead.');
            }
            element.addClass('modal-backdrop');

            if (attrs.modalInClass) {
                if ($animateCss) {
                    $animateCss(element, {
                        addClass: attrs.modalInClass
                    }).start();
                } else {
                    $animate.addClass(element, attrs.modalInClass);
                }

                scope.$on($modalStack.NOW_CLOSING_EVENT, function (e, setIsAsync) {
                    var done = setIsAsync();
                    if ($animateCss) {
                        $animateCss(element, {
                            removeClass: attrs.modalInClass
                        }).start().then(done);
                    } else {
                        $animate.removeClass(element, attrs.modalInClass).then(done);
                    }
                });
            }
        }
    }])

  .directive('modalWindow', [
    '$modalStack', '$q', '$animate', '$injector', '$log', '$modalSuppressWarning',
    function ($modalStack, $q, $animate, $injector, $log, $modalSuppressWarning) {
        var $animateCss = null;

        if ($injector.has('$animateCss')) {
            $animateCss = $injector.get('$animateCss');
        }

        return {
            scope: {
                index: '@'
            },
            replace: true,
            transclude: true,
            templateUrl: function (tElement, tAttrs) {
                return tAttrs.templateUrl || 'template/modal/window.html';
            },
            link: function (scope, element, attrs) {
                if (!$modalSuppressWarning) {
                    $log.warn('modal-window is now deprecated. Use uib-modal-window instead.');
                }
                element.addClass(attrs.windowClass || '');
                element.addClass(attrs.windowTopClass || '');
                scope.size = attrs.size;

                scope.close = function (evt) {
                    var modal = $modalStack.getTop();
                    if (modal && modal.value.backdrop && modal.value.backdrop !== 'static' && (evt.target === evt.currentTarget)) {
                        evt.preventDefault();
                        evt.stopPropagation();
                        $modalStack.dismiss(modal.key, 'backdrop click');
                    }
                };

                // moved from template to fix issue #2280
                element.on('click', scope.close);

                // This property is only added to the scope for the purpose of detecting when this directive is rendered.
                // We can detect that by using this property in the template associated with this directive and then use
                // {@link Attribute#$observe} on it. For more details please see {@link TableColumnResize}.
                scope.$isRendered = true;

                // Deferred object that will be resolved when this modal is render.
                var modalRenderDeferObj = $q.defer();
                // Observe function will be called on next digest cycle after compilation, ensuring that the DOM is ready.
                // In order to use this way of finding whether DOM is ready, we need to observe a scope property used in modal's template.
                attrs.$observe('modalRender', function (value) {
                    if (value == 'true') {
                        modalRenderDeferObj.resolve();
                    }
                });

                modalRenderDeferObj.promise.then(function () {
                    var animationPromise = null;

                    if (attrs.modalInClass) {
                        if ($animateCss) {
                            animationPromise = $animateCss(element, {
                                addClass: attrs.modalInClass
                            }).start();
                        } else {
                            animationPromise = $animate.addClass(element, attrs.modalInClass);
                        }

                        scope.$on($modalStack.NOW_CLOSING_EVENT, function (e, setIsAsync) {
                            var done = setIsAsync();
                            if ($animateCss) {
                                $animateCss(element, {
                                    removeClass: attrs.modalInClass
                                }).start().then(done);
                            } else {
                                $animate.removeClass(element, attrs.modalInClass).then(done);
                            }
                        });
                    }


                    $q.when(animationPromise).then(function () {
                        var inputWithAutofocus = element[0].querySelector('[autofocus]');
                        /**
                         * Auto-focusing of a freshly-opened modal element causes any child elements
                         * with the autofocus attribute to lose focus. This is an issue on touch
                         * based devices which will show and then hide the onscreen keyboard.
                         * Attempts to refocus the autofocus element via JavaScript will not reopen
                         * the onscreen keyboard. Fixed by updated the focusing logic to only autofocus
                         * the modal element if the modal does not contain an autofocus element.
                         */
                        if (inputWithAutofocus) {
                            inputWithAutofocus.focus();
                        } else {
                            element[0].focus();
                        }
                    });

                    // Notify {@link $modalStack} that modal is rendered.
                    var modal = $modalStack.getTop();
                    if (modal) {
                        $modalStack.modalRendered(modal.key);
                    }
                });
            }
        };
    }])

  .directive('modalAnimationClass', [
    '$log', '$modalSuppressWarning',
    function ($log, $modalSuppressWarning) {
        return {
            compile: function (tElement, tAttrs) {
                if (!$modalSuppressWarning) {
                    $log.warn('modal-animation-class is now deprecated. Use uib-modal-animation-class instead.');
                }
                if (tAttrs.modalAnimation) {
                    tElement.addClass(tAttrs.modalAnimationClass);
                }
            }
        };
    }])

  .directive('modalTransclude', [
    '$log', '$modalSuppressWarning',
    function ($log, $modalSuppressWarning) {
        return {
            link: function ($scope, $element, $attrs, controller, $transclude) {
                if (!$modalSuppressWarning) {
                    $log.warn('modal-transclude is now deprecated. Use uib-modal-transclude instead.');
                }
                $transclude($scope.$parent, function (clone) {
                    $element.empty();
                    $element.append(clone);
                });
            }
        };
    }])

  .service('$modalStack', [
    '$animate', '$timeout', '$document', '$compile', '$rootScope',
    '$q',
    '$injector',
    '$$multiMap',
    '$$stackedMap',
    '$uibModalStack',
    '$log',
    '$modalSuppressWarning',
    function ($animate, $timeout, $document, $compile, $rootScope,
             $q,
             $injector,
             $$multiMap,
             $$stackedMap,
             $uibModalStack,
             $log,
             $modalSuppressWarning) {
        if (!$modalSuppressWarning) {
            $log.warn('$modalStack is now deprecated. Use $uibModalStack instead.');
        }

        angular.extend(this, $uibModalStack);
    }])

  .provider('$modal', ['$uibModalProvider', function ($uibModalProvider) {
      angular.extend(this, $uibModalProvider);

      this.$get = ['$injector', '$log', '$modalSuppressWarning',
        function ($injector, $log, $modalSuppressWarning) {
            if (!$modalSuppressWarning) {
                $log.warn('$modal is now deprecated. Use $uibModal instead.');
            }

            return $injector.invoke($uibModalProvider.$get);
        }];
  }]);

angular.module('ui.bootstrap.datepicker', ['ui.bootstrap.dateparser', 'ui.bootstrap.position'])

.value('$datepickerSuppressError', false)

.constant('uibDatepickerConfig', {
    formatDay: 'dd',
    formatMonth: 'MMMM',
    formatYear: 'yyyy',
    formatDayHeader: 'EEE',
    formatDayTitle: 'MMMM yyyy',
    formatMonthTitle: 'yyyy',
    datepickerMode: 'day',
    minMode: 'day',
    maxMode: 'year',
    showWeeks: true,
    startingDay: 0,
    yearRange: 20,
    minDate: null,
    maxDate: null,
    shortcutPropagation: false
})

.controller('UibDatepickerController', ['$scope', '$attrs', '$parse', '$interpolate', '$log', 'dateFilter', '$locale', 'uibDatepickerConfig', '$datepickerSuppressError', function ($scope, $attrs, $parse, $interpolate, $log, dateFilter, $locale, datepickerConfig, $datepickerSuppressError) {
    var self = this,
        ngModelCtrl = { $setViewValue: angular.noop }; // nullModelCtrl;

    // Modes chain
    this.modes = ['day', 'month', 'year'];

    // Change first day of week
    if ($locale.DATETIME_FORMATS.FIRSTDAYOFWEEK)
        datepickerConfig.startingDay = $locale.DATETIME_FORMATS.FIRSTDAYOFWEEK;

    // Configuration attributes
    angular.forEach(['formatDay', 'formatMonth', 'formatYear', 'formatDayHeader', 'formatDayTitle', 'formatMonthTitle',
                     'showWeeks', 'startingDay', 'yearRange', 'shortcutPropagation'], function (key, index) {
                         self[key] = angular.isDefined($attrs[key]) ? (index < 6 ? $interpolate($attrs[key])($scope.$parent) : $scope.$parent.$eval($attrs[key])) : datepickerConfig[key];
                     });

    // Watchable date attributes
    angular.forEach(['minDate', 'maxDate'], function (key) {
        if ($attrs[key]) {
            $scope.$parent.$watch($parse($attrs[key]), function (value) {
                self[key] = value ? new Date(value) : null;
                self.refreshView();
            });
        } else {
            self[key] = datepickerConfig[key] ? new Date(datepickerConfig[key]) : null;
        }
    });

    angular.forEach(['minMode', 'maxMode'], function (key) {
        if ($attrs[key]) {
            $scope.$parent.$watch($parse($attrs[key]), function (value) {
                self[key] = angular.isDefined(value) ? value : $attrs[key];
                $scope[key] = self[key];
                if ((key == 'minMode' && self.modes.indexOf($scope.datepickerMode) < self.modes.indexOf(self[key])) || (key == 'maxMode' && self.modes.indexOf($scope.datepickerMode) > self.modes.indexOf(self[key]))) {
                    $scope.datepickerMode = self[key];
                }
            });
        } else {
            self[key] = datepickerConfig[key] || null;
            $scope[key] = self[key];
        }
    });

    $scope.datepickerMode = $scope.datepickerMode || datepickerConfig.datepickerMode;
    $scope.uniqueId = 'datepicker-' + $scope.$id + '-' + Math.floor(Math.random() * 10000);

    if (angular.isDefined($attrs.initDate)) {
        this.activeDate = $scope.$parent.$eval($attrs.initDate) || new Date();
        $scope.$parent.$watch($attrs.initDate, function (initDate) {
            if (initDate && (ngModelCtrl.$isEmpty(ngModelCtrl.$modelValue) || ngModelCtrl.$invalid)) {
                self.activeDate = initDate;
                self.refreshView();
            }
        });
    } else {
        this.activeDate = new Date();
    }

    $scope.isActive = function (dateObject) {
        if (self.compare(dateObject.date, self.activeDate) === 0) {
            $scope.activeDateId = dateObject.uid;
            return true;
        }
        return false;
    };

    this.init = function (ngModelCtrl_) {
        ngModelCtrl = ngModelCtrl_;

        ngModelCtrl.$render = function () {
            self.render();
        };
    };

    this.render = function () {
        if (ngModelCtrl.$viewValue) {
            var date = new Date(ngModelCtrl.$viewValue),
                isValid = !isNaN(date);

            if (isValid) {
                this.activeDate = date;
            } else if (!$datepickerSuppressError) {
                $log.error('Datepicker directive: "ng-model" value must be a Date object, a number of milliseconds since 01.01.1970 or a string representing an RFC2822 or ISO 8601 date.');
            }
        }
        this.refreshView();
    };

    this.refreshView = function () {
        if (this.element) {
            this._refreshView();

            var date = ngModelCtrl.$viewValue ? new Date(ngModelCtrl.$viewValue) : null;
            ngModelCtrl.$setValidity('dateDisabled', !date || (this.element && !this.isDisabled(date)));
        }
    };

    this.createDateObject = function (date, format) {
        var model = ngModelCtrl.$viewValue ? new Date(ngModelCtrl.$viewValue) : null;
        return {
            date: date,
            label: dateFilter(date, format),
            selected: model && this.compare(date, model) === 0,
            disabled: this.isDisabled(date),
            current: this.compare(date, new Date()) === 0,
            customClass: this.customClass(date)
        };
    };

    this.isDisabled = function (date) {
        return ((this.minDate && this.compare(date, this.minDate) < 0) || (this.maxDate && this.compare(date, this.maxDate) > 0) || ($attrs.dateDisabled && $scope.dateDisabled({ date: date, mode: $scope.datepickerMode })));
    };

    this.customClass = function (date) {
        return $scope.customClass({ date: date, mode: $scope.datepickerMode });
    };

    // Split array into smaller arrays
    this.split = function (arr, size) {
        var arrays = [];
        while (arr.length > 0) {
            arrays.push(arr.splice(0, size));
        }
        return arrays;
    };

    $scope.select = function (date) {
        if ($scope.datepickerMode === self.minMode) {
            var dt = ngModelCtrl.$viewValue ? new Date(ngModelCtrl.$viewValue) : new Date(0, 0, 0, 0, 0, 0, 0);
            dt.setFullYear(date.getFullYear(), date.getMonth(), date.getDate());
            ngModelCtrl.$setViewValue(dt);
            ngModelCtrl.$render();
        } else {
            self.activeDate = date;
            $scope.datepickerMode = self.modes[self.modes.indexOf($scope.datepickerMode) - 1];
        }
    };

    $scope.move = function (direction) {
        var year = self.activeDate.getFullYear() + direction * (self.step.years || 0),
            month = self.activeDate.getMonth() + direction * (self.step.months || 0);
        self.activeDate.setFullYear(year, month, 1);
        self.refreshView();
    };

    $scope.toggleMode = function (direction) {
        direction = direction || 1;

        if (($scope.datepickerMode === self.maxMode && direction === 1) || ($scope.datepickerMode === self.minMode && direction === -1)) {
            return;
        }

        $scope.datepickerMode = self.modes[self.modes.indexOf($scope.datepickerMode) + direction];
    };

    // Key event mapper
    $scope.keys = { 13: 'enter', 32: 'space', 33: 'pageup', 34: 'pagedown', 35: 'end', 36: 'home', 37: 'left', 38: 'up', 39: 'right', 40: 'down' };

    var focusElement = function () {
        self.element[0].focus();
    };

    // Listen for focus requests from popup directive
    $scope.$on('uib:datepicker.focus', focusElement);

    $scope.keydown = function (evt) {
        var key = $scope.keys[evt.which];

        if (!key || evt.shiftKey || evt.altKey) {
            return;
        }

        evt.preventDefault();
        if (!self.shortcutPropagation) {
            evt.stopPropagation();
        }

        if (key === 'enter' || key === 'space') {
            if (self.isDisabled(self.activeDate)) {
                return; // do nothing
            }
            $scope.select(self.activeDate);
        } else if (evt.ctrlKey && (key === 'up' || key === 'down')) {
            $scope.toggleMode(key === 'up' ? 1 : -1);
        } else {
            self.handleKeyDown(key, evt);
            self.refreshView();
        }
    };
}])

.controller('UibDaypickerController', ['$scope', '$element', 'dateFilter', function (scope, $element, dateFilter) {
    var DAYS_IN_MONTH = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31];

    this.step = { months: 1 };
    this.element = $element;
    function getDaysInMonth(year, month) {
        return ((month === 1) && (year % 4 === 0) && ((year % 100 !== 0) || (year % 400 === 0))) ? 29 : DAYS_IN_MONTH[month];
    }

    this.init = function (ctrl) {
        angular.extend(ctrl, this);
        scope.showWeeks = ctrl.showWeeks;
        ctrl.refreshView();
    };

    this.getDates = function (startDate, n) {
        var dates = new Array(n), current = new Date(startDate), i = 0, date;
        while (i < n) {
            date = new Date(current);
            dates[i++] = date;
            current.setDate(current.getDate() + 1);
        }
        return dates;
    };

    this._refreshView = function () {
        var year = this.activeDate.getFullYear(),
          month = this.activeDate.getMonth(),
          firstDayOfMonth = new Date(this.activeDate);

        firstDayOfMonth.setFullYear(year, month, 1);

        var difference = this.startingDay - firstDayOfMonth.getDay(),
          numDisplayedFromPreviousMonth = (difference > 0) ? 7 - difference : -difference,
          firstDate = new Date(firstDayOfMonth);

        if (numDisplayedFromPreviousMonth > 0) {
            firstDate.setDate(-numDisplayedFromPreviousMonth + 1);
        }

        // 42 is the number of days on a six-month calendar
        var days = this.getDates(firstDate, 42);
        for (var i = 0; i < 42; i++) {
            days[i] = angular.extend(this.createDateObject(days[i], this.formatDay), {
                secondary: days[i].getMonth() !== month,
                uid: scope.uniqueId + '-' + i
            });
        }

        scope.labels = new Array(7);
        for (var j = 0; j < 7; j++) {
            scope.labels[j] = {
                abbr: dateFilter(days[j].date, this.formatDayHeader),
                full: dateFilter(days[j].date, 'EEEE')
            };
        }

        scope.title = dateFilter(this.activeDate, this.formatDayTitle);
        scope.rows = this.split(days, 7);

        if (scope.showWeeks) {
            scope.weekNumbers = [];
            var thursdayIndex = (4 + 7 - this.startingDay) % 7,
                numWeeks = scope.rows.length;
            for (var curWeek = 0; curWeek < numWeeks; curWeek++) {
                scope.weekNumbers.push(
                  getISO8601WeekNumber(scope.rows[curWeek][thursdayIndex].date));
            }
        }
    };

    this.compare = function (date1, date2) {
        return (new Date(date1.getFullYear(), date1.getMonth(), date1.getDate()) - new Date(date2.getFullYear(), date2.getMonth(), date2.getDate()));
    };

    function getISO8601WeekNumber(date) {
        var checkDate = new Date(date);
        checkDate.setDate(checkDate.getDate() + 4 - (checkDate.getDay() || 7)); // Thursday
        var time = checkDate.getTime();
        checkDate.setMonth(0); // Compare with Jan 1
        checkDate.setDate(1);
        return Math.floor(Math.round((time - checkDate) / 86400000) / 7) + 1;
    }

    this.handleKeyDown = function (key, evt) {
        var date = this.activeDate.getDate();

        if (key === 'left') {
            date = date - 1;   // up
        } else if (key === 'up') {
            date = date - 7;   // down
        } else if (key === 'right') {
            date = date + 1;   // down
        } else if (key === 'down') {
            date = date + 7;
        } else if (key === 'pageup' || key === 'pagedown') {
            var month = this.activeDate.getMonth() + (key === 'pageup' ? -1 : 1);
            this.activeDate.setMonth(month, 1);
            date = Math.min(getDaysInMonth(this.activeDate.getFullYear(), this.activeDate.getMonth()), date);
        } else if (key === 'home') {
            date = 1;
        } else if (key === 'end') {
            date = getDaysInMonth(this.activeDate.getFullYear(), this.activeDate.getMonth());
        }
        this.activeDate.setDate(date);
    };
}])

.controller('UibMonthpickerController', ['$scope', '$element', 'dateFilter', function (scope, $element, dateFilter) {
    this.step = { years: 1 };
    this.element = $element;

    this.init = function (ctrl) {
        angular.extend(ctrl, this);
        ctrl.refreshView();
    };

    this._refreshView = function () {
        var months = new Array(12),
            year = this.activeDate.getFullYear(),
            date;

        for (var i = 0; i < 12; i++) {
            date = new Date(this.activeDate);
            date.setFullYear(year, i, 1);
            months[i] = angular.extend(this.createDateObject(date, this.formatMonth), {
                uid: scope.uniqueId + '-' + i
            });
        }

        scope.title = dateFilter(this.activeDate, this.formatMonthTitle);
        scope.rows = this.split(months, 3);
    };

    this.compare = function (date1, date2) {
        return new Date(date1.getFullYear(), date1.getMonth()) - new Date(date2.getFullYear(), date2.getMonth());
    };

    this.handleKeyDown = function (key, evt) {
        var date = this.activeDate.getMonth();

        if (key === 'left') {
            date = date - 1;   // up
        } else if (key === 'up') {
            date = date - 3;   // down
        } else if (key === 'right') {
            date = date + 1;   // down
        } else if (key === 'down') {
            date = date + 3;
        } else if (key === 'pageup' || key === 'pagedown') {
            var year = this.activeDate.getFullYear() + (key === 'pageup' ? -1 : 1);
            this.activeDate.setFullYear(year);
        } else if (key === 'home') {
            date = 0;
        } else if (key === 'end') {
            date = 11;
        }
        this.activeDate.setMonth(date);
    };
}])

.controller('UibYearpickerController', ['$scope', '$element', 'dateFilter', function (scope, $element, dateFilter) {
    var range;
    this.element = $element;

    function getStartingYear(year) {
        return parseInt((year - 1) / range, 10) * range + 1;
    }

    this.yearpickerInit = function () {
        range = this.yearRange;
        this.step = { years: range };
    };

    this._refreshView = function () {
        var years = new Array(range), date;

        for (var i = 0, start = getStartingYear(this.activeDate.getFullYear()) ; i < range; i++) {
            date = new Date(this.activeDate);
            date.setFullYear(start + i, 0, 1);
            years[i] = angular.extend(this.createDateObject(date, this.formatYear), {
                uid: scope.uniqueId + '-' + i
            });
        }

        scope.title = [years[0].label, years[range - 1].label].join(' - ');
        scope.rows = this.split(years, 5);
    };

    this.compare = function (date1, date2) {
        return date1.getFullYear() - date2.getFullYear();
    };

    this.handleKeyDown = function (key, evt) {
        var date = this.activeDate.getFullYear();

        if (key === 'left') {
            date = date - 1;   // up
        } else if (key === 'up') {
            date = date - 5;   // down
        } else if (key === 'right') {
            date = date + 1;   // down
        } else if (key === 'down') {
            date = date + 5;
        } else if (key === 'pageup' || key === 'pagedown') {
            date += (key === 'pageup' ? -1 : 1) * this.step.years;
        } else if (key === 'home') {
            date = getStartingYear(this.activeDate.getFullYear());
        } else if (key === 'end') {
            date = getStartingYear(this.activeDate.getFullYear()) + range - 1;
        }
        this.activeDate.setFullYear(date);
    };
}])

.directive('uibDatepicker', function () {
    return {
        replace: true,
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/datepicker/datepicker.html';
        },
        scope: {
            datepickerMode: '=?',
            dateDisabled: '&',
            customClass: '&',
            shortcutPropagation: '&?'
        },
        require: ['uibDatepicker', '^ngModel'],
        controller: 'UibDatepickerController',
        controllerAs: 'datepicker',
        link: function (scope, element, attrs, ctrls) {
            var datepickerCtrl = ctrls[0], ngModelCtrl = ctrls[1];

            datepickerCtrl.init(ngModelCtrl);
        }
    };
})

.directive('uibDaypicker', function () {
    return {
        replace: true,
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/datepicker/day.html';
        },
        require: ['^?uibDatepicker', 'uibDaypicker', '^?datepicker'],
        controller: 'UibDaypickerController',
        link: function (scope, element, attrs, ctrls) {
            var datepickerCtrl = ctrls[0] || ctrls[2],
              daypickerCtrl = ctrls[1];

            daypickerCtrl.init(datepickerCtrl);
        }
    };
})

.directive('uibMonthpicker', function () {
    return {
        replace: true,
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/datepicker/month.html';
        },
        require: ['^?uibDatepicker', 'uibMonthpicker', '^?datepicker'],
        controller: 'UibMonthpickerController',
        link: function (scope, element, attrs, ctrls) {
            var datepickerCtrl = ctrls[0] || ctrls[2],
              monthpickerCtrl = ctrls[1];

            monthpickerCtrl.init(datepickerCtrl);
        }
    };
})

.directive('uibYearpicker', function () {
    return {
        replace: true,
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/datepicker/year.html';
        },
        require: ['^?uibDatepicker', 'uibYearpicker', '^?datepicker'],
        controller: 'UibYearpickerController',
        link: function (scope, element, attrs, ctrls) {
            var ctrl = ctrls[0] || ctrls[2];
            angular.extend(ctrl, ctrls[1]);
            ctrl.yearpickerInit();

            ctrl.refreshView();
        }
    };
})

.constant('uibDatepickerPopupConfig', {
    datepickerPopup: 'yyyy-MM-dd',
    datepickerPopupTemplateUrl: 'template/datepicker/popup.html',
    datepickerTemplateUrl: 'template/datepicker/datepicker.html',
    html5Types: {
        date: 'yyyy-MM-dd',
        'datetime-local': 'yyyy-MM-ddTHH:mm:ss.sss',
        'month': 'yyyy-MM'
    },
    currentText: 'Today',
    clearText: 'Clear',
    closeText: 'Done',
    closeOnDateSelection: true,
    appendToBody: false,
    showButtonBar: true,
    onOpenFocus: true
})

.controller('UibDatepickerPopupController', ['$scope', '$element', '$attrs', '$compile', '$parse', '$document', '$rootScope', '$uibPosition', 'dateFilter', 'uibDateParser', 'uibDatepickerPopupConfig', '$timeout',
function (scope, element, attrs, $compile, $parse, $document, $rootScope, $position, dateFilter, dateParser, datepickerPopupConfig, $timeout) {
    var self = this;
    var cache = {},
      isHtml5DateInput = false;
    var dateFormat, closeOnDateSelection, appendToBody, onOpenFocus,
      datepickerPopupTemplateUrl, datepickerTemplateUrl, popupEl, datepickerEl,
      ngModel, $popup;

    scope.watchData = {};

    this.init = function (_ngModel_) {
        ngModel = _ngModel_;
        closeOnDateSelection = angular.isDefined(attrs.closeOnDateSelection) ? scope.$parent.$eval(attrs.closeOnDateSelection) : datepickerPopupConfig.closeOnDateSelection;
        appendToBody = angular.isDefined(attrs.datepickerAppendToBody) ? scope.$parent.$eval(attrs.datepickerAppendToBody) : datepickerPopupConfig.appendToBody;
        onOpenFocus = angular.isDefined(attrs.onOpenFocus) ? scope.$parent.$eval(attrs.onOpenFocus) : datepickerPopupConfig.onOpenFocus;
        datepickerPopupTemplateUrl = angular.isDefined(attrs.datepickerPopupTemplateUrl) ? attrs.datepickerPopupTemplateUrl : datepickerPopupConfig.datepickerPopupTemplateUrl;
        datepickerTemplateUrl = angular.isDefined(attrs.datepickerTemplateUrl) ? attrs.datepickerTemplateUrl : datepickerPopupConfig.datepickerTemplateUrl;

        scope.showButtonBar = angular.isDefined(attrs.showButtonBar) ? scope.$parent.$eval(attrs.showButtonBar) : datepickerPopupConfig.showButtonBar;

        if (datepickerPopupConfig.html5Types[attrs.type]) {
            dateFormat = datepickerPopupConfig.html5Types[attrs.type];
            isHtml5DateInput = true;
        } else {
            dateFormat = attrs.datepickerPopup || attrs.uibDatepickerPopup || datepickerPopupConfig.datepickerPopup;
            attrs.$observe('uibDatepickerPopup', function (value, oldValue) {
                var newDateFormat = value || datepickerPopupConfig.datepickerPopup;
                // Invalidate the $modelValue to ensure that formatters re-run
                // FIXME: Refactor when PR is merged: https://github.com/angular/angular.js/pull/10764
                if (newDateFormat !== dateFormat) {
                    dateFormat = newDateFormat;
                    ngModel.$modelValue = null;

                    if (!dateFormat) {
                        throw new Error('uibDatepickerPopup must have a date format specified.');
                    }
                }
            });
        }

        if (!dateFormat) {
            throw new Error('uibDatepickerPopup must have a date format specified.');
        }

        if (isHtml5DateInput && attrs.datepickerPopup) {
            throw new Error('HTML5 date input types do not support custom formats.');
        }

        // popup element used to display calendar
        popupEl = angular.element('<div uib-datepicker-popup-wrap><div uib-datepicker></div></div>');
        popupEl.attr({
            'ng-model': 'date',
            'ng-change': 'dateSelection(date)',
            'template-url': datepickerPopupTemplateUrl
        });

        // datepicker element
        datepickerEl = angular.element(popupEl.children()[0]);
        datepickerEl.attr('template-url', datepickerTemplateUrl);

        if (isHtml5DateInput) {
            if (attrs.type === 'month') {
                datepickerEl.attr('datepicker-mode', '"month"');
                datepickerEl.attr('min-mode', 'month');
            }
        }

        if (attrs.datepickerOptions) {
            var options = scope.$parent.$eval(attrs.datepickerOptions);
            if (options && options.initDate) {
                scope.initDate = options.initDate;
                datepickerEl.attr('init-date', 'initDate');
                delete options.initDate;
            }
            angular.forEach(options, function (value, option) {
                datepickerEl.attr(cameltoDash(option), value);
            });
        }

        angular.forEach(['minMode', 'maxMode', 'minDate', 'maxDate', 'datepickerMode', 'initDate', 'shortcutPropagation'], function (key) {
            if (attrs[key]) {
                var getAttribute = $parse(attrs[key]);
                scope.$parent.$watch(getAttribute, function (value) {
                    scope.watchData[key] = value;
                    if (key === 'minDate' || key === 'maxDate') {
                        cache[key] = new Date(value);
                    }
                });
                datepickerEl.attr(cameltoDash(key), 'watchData.' + key);

                // Propagate changes from datepicker to outside
                if (key === 'datepickerMode') {
                    var setAttribute = getAttribute.assign;
                    scope.$watch('watchData.' + key, function (value, oldvalue) {
                        if (angular.isFunction(setAttribute) && value !== oldvalue) {
                            setAttribute(scope.$parent, value);
                        }
                    });
                }
            }
        });
        if (attrs.dateDisabled) {
            datepickerEl.attr('date-disabled', 'dateDisabled({ date: date, mode: mode })');
        }

        if (attrs.showWeeks) {
            datepickerEl.attr('show-weeks', attrs.showWeeks);
        }

        if (attrs.customClass) {
            datepickerEl.attr('custom-class', 'customClass({ date: date, mode: mode })');
        }

        if (!isHtml5DateInput) {
            // Internal API to maintain the correct ng-invalid-[key] class
            ngModel.$$parserName = 'date';
            ngModel.$validators.date = validator;
            ngModel.$parsers.unshift(parseDate);
            ngModel.$formatters.push(function (value) {
                scope.date = value;
                return ngModel.$isEmpty(value) ? value : dateFilter(value, dateFormat);
            });
        } else {
            ngModel.$formatters.push(function (value) {
                scope.date = value;
                return value;
            });
        }

        // Detect changes in the view from the text box
        ngModel.$viewChangeListeners.push(function () {
            scope.date = dateParser.parse(ngModel.$viewValue, dateFormat, scope.date);
        });

        element.bind('keydown', inputKeydownBind);

        $popup = $compile(popupEl)(scope);
        // Prevent jQuery cache memory leak (template is now redundant after linking)
        popupEl.remove();

        if (appendToBody) {
            $document.find('body').append($popup);
        } else {
            element.after($popup);
        }

        scope.$on('$destroy', function () {
            if (scope.isOpen === true) {
                if (!$rootScope.$$phase) {
                    scope.$apply(function () {
                        scope.isOpen = false;
                    });
                }
            }

            $popup.remove();
            element.unbind('keydown', inputKeydownBind);
            $document.unbind('click', documentClickBind);
        });
    };

    scope.getText = function (key) {
        return scope[key + 'Text'] || datepickerPopupConfig[key + 'Text'];
    };

    scope.isDisabled = function (date) {
        if (date === 'today') {
            date = new Date();
        }

        return ((scope.watchData.minDate && scope.compare(date, cache.minDate) < 0) ||
          (scope.watchData.maxDate && scope.compare(date, cache.maxDate) > 0));
    };

    scope.compare = function (date1, date2) {
        return (new Date(date1.getFullYear(), date1.getMonth(), date1.getDate()) - new Date(date2.getFullYear(), date2.getMonth(), date2.getDate()));
    };

    // Inner change
    scope.dateSelection = function (dt) {
        if (angular.isDefined(dt)) {
            scope.date = dt;
        }
        var date = scope.date ? dateFilter(scope.date, dateFormat) : null; // Setting to NULL is necessary for form validators to function
        element.val(date);
        ngModel.$setViewValue(date);

        if (closeOnDateSelection) {
            scope.isOpen = false;
            element[0].focus();
        }
    };

    scope.keydown = function (evt) {
        if (evt.which === 27) {
            scope.isOpen = false;
            element[0].focus();
        }
    };

    scope.select = function (date) {
        if (date === 'today') {
            var today = new Date();
            if (angular.isDate(scope.date)) {
                date = new Date(scope.date);
                date.setFullYear(today.getFullYear(), today.getMonth(), today.getDate());
            } else {
                date = new Date(today.setHours(0, 0, 0, 0));
            }
        }
        scope.dateSelection(date);
    };

    scope.close = function () {
        scope.isOpen = false;
        element[0].focus();
    };

    scope.$watch('isOpen', function (value) {
        if (value) {
            scope.position = appendToBody ? $position.offset(element) : $position.position(element);
            scope.position.top = scope.position.top + element.prop('offsetHeight');

            $timeout(function () {
                if (onOpenFocus) {
                    scope.$broadcast('uib:datepicker.focus');
                }
                $document.bind('click', documentClickBind);
            }, 0, false);
        } else {
            $document.unbind('click', documentClickBind);
        }
    });

    function cameltoDash(string) {
        return string.replace(/([A-Z])/g, function ($1) { return '-' + $1.toLowerCase(); });
    }

    function parseDate(viewValue) {
        if (angular.isNumber(viewValue)) {
            // presumably timestamp to date object
            viewValue = new Date(viewValue);
        }

        if (!viewValue) {
            return null;
        } else if (angular.isDate(viewValue) && !isNaN(viewValue)) {
            return viewValue;
        } else if (angular.isString(viewValue)) {
            var date = dateParser.parse(viewValue, dateFormat, scope.date);
            if (isNaN(date)) {
                return undefined;
            } else {
                return date;
            }
        } else {
            return undefined;
        }
    }

    function validator(modelValue, viewValue) {
        var value = modelValue || viewValue;

        if (!attrs.ngRequired && !value) {
            return true;
        }

        if (angular.isNumber(value)) {
            value = new Date(value);
        }
        if (!value) {
            return true;
        } else if (angular.isDate(value) && !isNaN(value)) {
            return true;
        } else if (angular.isString(value)) {
            var date = dateParser.parse(value, dateFormat);
            return !isNaN(date);
        } else {
            return false;
        }
    }

    function documentClickBind(event) {
        var popup = $popup[0];
        var dpContainsTarget = element[0].contains(event.target);
        // The popup node may not be an element node
        // In some browsers (IE) only element nodes have the 'contains' function
        var popupContainsTarget = popup.contains !== undefined && popup.contains(event.target);
        if (scope.isOpen && !(dpContainsTarget || popupContainsTarget)) {
            scope.$apply(function () {
                scope.isOpen = false;
            });
        }
    }

    function inputKeydownBind(evt) {
        if (evt.which === 27 && scope.isOpen) {
            evt.preventDefault();
            evt.stopPropagation();
            scope.$apply(function () {
                scope.isOpen = false;
            });
            element[0].focus();
        } else if (evt.which === 40 && !scope.isOpen) {
            evt.preventDefault();
            evt.stopPropagation();
            scope.$apply(function () {
                scope.isOpen = true;
            });
        }
    }
}])

.directive('uibDatepickerPopup', function () {
    return {
        require: ['ngModel', 'uibDatepickerPopup'],
        controller: 'UibDatepickerPopupController',
        scope: {
            isOpen: '=?',
            currentText: '@',
            clearText: '@',
            closeText: '@',
            dateDisabled: '&',
            customClass: '&'
        },
        link: function (scope, element, attrs, ctrls) {
            var ngModel = ctrls[0],
              ctrl = ctrls[1];

            ctrl.init(ngModel);
        }
    };
})

.directive('uibDatepickerPopupWrap', function () {
    return {
        replace: true,
        transclude: true,
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/datepicker/popup.html';
        }
    };
});

/* Deprecated datepicker below */

angular.module('ui.bootstrap.datepicker')

.value('$datepickerSuppressWarning', false)

.controller('DatepickerController', ['$scope', '$attrs', '$parse', '$interpolate', '$log', 'dateFilter', 'uibDatepickerConfig', '$datepickerSuppressError', '$datepickerSuppressWarning', function ($scope, $attrs, $parse, $interpolate, $log, dateFilter, datepickerConfig, $datepickerSuppressError, $datepickerSuppressWarning) {
    if (!$datepickerSuppressWarning) {
        $log.warn('DatepickerController is now deprecated. Use UibDatepickerController instead.');
    }

    var self = this,
      ngModelCtrl = { $setViewValue: angular.noop }; // nullModelCtrl;

    this.modes = ['day', 'month', 'year'];

    angular.forEach(['formatDay', 'formatMonth', 'formatYear', 'formatDayHeader', 'formatDayTitle', 'formatMonthTitle',
      'showWeeks', 'startingDay', 'yearRange', 'shortcutPropagation'], function (key, index) {
          self[key] = angular.isDefined($attrs[key]) ? (index < 6 ? $interpolate($attrs[key])($scope.$parent) : $scope.$parent.$eval($attrs[key])) : datepickerConfig[key];
      });

    angular.forEach(['minDate', 'maxDate'], function (key) {
        if ($attrs[key]) {
            $scope.$parent.$watch($parse($attrs[key]), function (value) {
                self[key] = value ? new Date(value) : null;
                self.refreshView();
            });
        } else {
            self[key] = datepickerConfig[key] ? new Date(datepickerConfig[key]) : null;
        }
    });

    angular.forEach(['minMode', 'maxMode'], function (key) {
        if ($attrs[key]) {
            $scope.$parent.$watch($parse($attrs[key]), function (value) {
                self[key] = angular.isDefined(value) ? value : $attrs[key];
                $scope[key] = self[key];
                if ((key == 'minMode' && self.modes.indexOf($scope.datepickerMode) < self.modes.indexOf(self[key])) || (key == 'maxMode' && self.modes.indexOf($scope.datepickerMode) > self.modes.indexOf(self[key]))) {
                    $scope.datepickerMode = self[key];
                }
            });
        } else {
            self[key] = datepickerConfig[key] || null;
            $scope[key] = self[key];
        }
    });

    $scope.datepickerMode = $scope.datepickerMode || datepickerConfig.datepickerMode;
    $scope.uniqueId = 'datepicker-' + $scope.$id + '-' + Math.floor(Math.random() * 10000);

    if (angular.isDefined($attrs.initDate)) {
        this.activeDate = $scope.$parent.$eval($attrs.initDate) || new Date();
        $scope.$parent.$watch($attrs.initDate, function (initDate) {
            if (initDate && (ngModelCtrl.$isEmpty(ngModelCtrl.$modelValue) || ngModelCtrl.$invalid)) {
                self.activeDate = initDate;
                self.refreshView();
            }
        });
    } else {
        this.activeDate = new Date();
    }

    $scope.isActive = function (dateObject) {
        if (self.compare(dateObject.date, self.activeDate) === 0) {
            $scope.activeDateId = dateObject.uid;
            return true;
        }
        return false;
    };

    this.init = function (ngModelCtrl_) {
        ngModelCtrl = ngModelCtrl_;

        ngModelCtrl.$render = function () {
            self.render();
        };
    };

    this.render = function () {
        if (ngModelCtrl.$viewValue) {
            var date = new Date(ngModelCtrl.$viewValue),
              isValid = !isNaN(date);

            if (isValid) {
                this.activeDate = date;
            } else if (!$datepickerSuppressError) {
                $log.error('Datepicker directive: "ng-model" value must be a Date object, a number of milliseconds since 01.01.1970 or a string representing an RFC2822 or ISO 8601 date.');
            }
        }
        this.refreshView();
    };

    this.refreshView = function () {
        if (this.element) {
            this._refreshView();

            var date = ngModelCtrl.$viewValue ? new Date(ngModelCtrl.$viewValue) : null;
            ngModelCtrl.$setValidity('dateDisabled', !date || (this.element && !this.isDisabled(date)));
        }
    };

    this.createDateObject = function (date, format) {
        var model = ngModelCtrl.$viewValue ? new Date(ngModelCtrl.$viewValue) : null;
        return {
            date: date,
            label: dateFilter(date, format),
            selected: model && this.compare(date, model) === 0,
            disabled: this.isDisabled(date),
            current: this.compare(date, new Date()) === 0,
            customClass: this.customClass(date)
        };
    };

    this.isDisabled = function (date) {
        return ((this.minDate && this.compare(date, this.minDate) < 0) || (this.maxDate && this.compare(date, this.maxDate) > 0) || ($attrs.dateDisabled && $scope.dateDisabled({ date: date, mode: $scope.datepickerMode })));
    };

    this.customClass = function (date) {
        return $scope.customClass({ date: date, mode: $scope.datepickerMode });
    };

    // Split array into smaller arrays
    this.split = function (arr, size) {
        var arrays = [];
        while (arr.length > 0) {
            arrays.push(arr.splice(0, size));
        }
        return arrays;
    };

    this.fixTimeZone = function (date) {
        var hours = date.getHours();
        date.setHours(hours === 23 ? hours + 2 : 0);
    };

    $scope.select = function (date) {
        if ($scope.datepickerMode === self.minMode) {
            var dt = ngModelCtrl.$viewValue ? new Date(ngModelCtrl.$viewValue) : new Date(0, 0, 0, 0, 0, 0, 0);
            dt.setFullYear(date.getFullYear(), date.getMonth(), date.getDate());
            ngModelCtrl.$setViewValue(dt);
            ngModelCtrl.$render();
        } else {
            self.activeDate = date;
            $scope.datepickerMode = self.modes[self.modes.indexOf($scope.datepickerMode) - 1];
        }
    };

    $scope.move = function (direction) {
        var year = self.activeDate.getFullYear() + direction * (self.step.years || 0),
          month = self.activeDate.getMonth() + direction * (self.step.months || 0);
        self.activeDate.setFullYear(year, month, 1);
        self.refreshView();
    };

    $scope.toggleMode = function (direction) {
        direction = direction || 1;

        if (($scope.datepickerMode === self.maxMode && direction === 1) || ($scope.datepickerMode === self.minMode && direction === -1)) {
            return;
        }

        $scope.datepickerMode = self.modes[self.modes.indexOf($scope.datepickerMode) + direction];
    };

    // Key event mapper
    $scope.keys = { 13: 'enter', 32: 'space', 33: 'pageup', 34: 'pagedown', 35: 'end', 36: 'home', 37: 'left', 38: 'up', 39: 'right', 40: 'down' };

    var focusElement = function () {
        self.element[0].focus();
    };

    $scope.$on('uib:datepicker.focus', focusElement);

    $scope.keydown = function (evt) {
        var key = $scope.keys[evt.which];

        if (!key || evt.shiftKey || evt.altKey) {
            return;
        }

        evt.preventDefault();
        if (!self.shortcutPropagation) {
            evt.stopPropagation();
        }

        if (key === 'enter' || key === 'space') {
            if (self.isDisabled(self.activeDate)) {
                return; // do nothing
            }
            $scope.select(self.activeDate);
        } else if (evt.ctrlKey && (key === 'up' || key === 'down')) {
            $scope.toggleMode(key === 'up' ? 1 : -1);
        } else {
            self.handleKeyDown(key, evt);
            self.refreshView();
        }
    };
}])

.directive('datepicker', ['$log', '$datepickerSuppressWarning', function ($log, $datepickerSuppressWarning) {
    return {
        replace: true,
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/datepicker/datepicker.html';
        },
        scope: {
            datepickerMode: '=?',
            dateDisabled: '&',
            customClass: '&',
            shortcutPropagation: '&?'
        },
        require: ['datepicker', '^ngModel'],
        controller: 'DatepickerController',
        controllerAs: 'datepicker',
        link: function (scope, element, attrs, ctrls) {
            if (!$datepickerSuppressWarning) {
                $log.warn('datepicker is now deprecated. Use uib-datepicker instead.');
            }

            var datepickerCtrl = ctrls[0], ngModelCtrl = ctrls[1];

            datepickerCtrl.init(ngModelCtrl);
        }
    };
}])

.directive('daypicker', ['$log', '$datepickerSuppressWarning', function ($log, $datepickerSuppressWarning) {
    return {
        replace: true,
        templateUrl: 'template/datepicker/day.html',
        require: ['^datepicker', 'daypicker'],
        controller: 'UibDaypickerController',
        link: function (scope, element, attrs, ctrls) {
            if (!$datepickerSuppressWarning) {
                $log.warn('daypicker is now deprecated. Use uib-daypicker instead.');
            }

            var datepickerCtrl = ctrls[0],
              daypickerCtrl = ctrls[1];

            daypickerCtrl.init(datepickerCtrl);
        }
    };
}])

.directive('monthpicker', ['$log', '$datepickerSuppressWarning', function ($log, $datepickerSuppressWarning) {
    return {
        replace: true,
        templateUrl: 'template/datepicker/month.html',
        require: ['^datepicker', 'monthpicker'],
        controller: 'UibMonthpickerController',
        link: function (scope, element, attrs, ctrls) {
            if (!$datepickerSuppressWarning) {
                $log.warn('monthpicker is now deprecated. Use uib-monthpicker instead.');
            }

            var datepickerCtrl = ctrls[0],
              monthpickerCtrl = ctrls[1];

            monthpickerCtrl.init(datepickerCtrl);
        }
    };
}])

.directive('yearpicker', ['$log', '$datepickerSuppressWarning', function ($log, $datepickerSuppressWarning) {
    return {
        replace: true,
        templateUrl: 'template/datepicker/year.html',
        require: ['^datepicker', 'yearpicker'],
        controller: 'UibYearpickerController',
        link: function (scope, element, attrs, ctrls) {
            if (!$datepickerSuppressWarning) {
                $log.warn('yearpicker is now deprecated. Use uib-yearpicker instead.');
            }

            var ctrl = ctrls[0];
            angular.extend(ctrl, ctrls[1]);
            ctrl.yearpickerInit();

            ctrl.refreshView();
        }
    };
}])

.directive('datepickerPopup', ['$log', '$datepickerSuppressWarning', function ($log, $datepickerSuppressWarning) {
    return {
        require: ['ngModel', 'datepickerPopup'],
        controller: 'UibDatepickerPopupController',
        scope: {
            isOpen: '=?',
            currentText: '@',
            clearText: '@',
            closeText: '@',
            dateDisabled: '&',
            customClass: '&'
        },
        link: function (scope, element, attrs, ctrls) {
            if (!$datepickerSuppressWarning) {
                $log.warn('datepicker-popup is now deprecated. Use uib-datepicker-popup instead.');
            }

            var ngModel = ctrls[0],
              ctrl = ctrls[1];

            ctrl.init(ngModel);
        }
    };
}])

.directive('datepickerPopupWrap', ['$log', '$datepickerSuppressWarning', function ($log, $datepickerSuppressWarning) {
    return {
        replace: true,
        transclude: true,
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/datepicker/popup.html';
        },
        link: function () {
            if (!$datepickerSuppressWarning) {
                $log.warn('datepicker-popup-wrap is now deprecated. Use uib-datepicker-popup-wrap instead.');
            }
        }
    };
}]);

angular.module('ui.bootstrap.dateparser', [])

.service('uibDateParser', ['$log', '$locale', 'orderByFilter', function ($log, $locale, orderByFilter) {
    // Pulled from https://github.com/mbostock/d3/blob/master/src/format/requote.js
    var SPECIAL_CHARACTERS_REGEXP = /[\\\^\$\*\+\?\|\[\]\(\)\.\{\}]/g;

    var localeId;
    var formatCodeToRegex;

    this.init = function () {
        localeId = $locale.id;

        this.parsers = {};

        formatCodeToRegex = {
            'yyyy': {
                regex: '\\d{4}',
                apply: function (value) { this.year = +value; }
            },
            'yy': {
                regex: '\\d{2}',
                apply: function (value) { this.year = +value + 2000; }
            },
            'y': {
                regex: '\\d{1,4}',
                apply: function (value) { this.year = +value; }
            },
            'MMMM': {
                regex: $locale.DATETIME_FORMATS.MONTH.join('|'),
                apply: function (value) { this.month = $locale.DATETIME_FORMATS.MONTH.indexOf(value); }
            },
            'MMM': {
                regex: $locale.DATETIME_FORMATS.SHORTMONTH.join('|'),
                apply: function (value) { this.month = $locale.DATETIME_FORMATS.SHORTMONTH.indexOf(value); }
            },
            'MM': {
                regex: '0[1-9]|1[0-2]',
                apply: function (value) { this.month = value - 1; }
            },
            'M': {
                regex: '[1-9]|1[0-2]',
                apply: function (value) { this.month = value - 1; }
            },
            'dd': {
                regex: '[0-2][0-9]{1}|3[0-1]{1}',
                apply: function (value) { this.date = +value; }
            },
            'd': {
                regex: '[1-2]?[0-9]{1}|3[0-1]{1}',
                apply: function (value) { this.date = +value; }
            },
            'EEEE': {
                regex: $locale.DATETIME_FORMATS.DAY.join('|')
            },
            'EEE': {
                regex: $locale.DATETIME_FORMATS.SHORTDAY.join('|')
            },
            'HH': {
                regex: '(?:0|1)[0-9]|2[0-3]',
                apply: function (value) { this.hours = +value; }
            },
            'hh': {
                regex: '0[0-9]|1[0-2]',
                apply: function (value) { this.hours = +value; }
            },
            'H': {
                regex: '1?[0-9]|2[0-3]',
                apply: function (value) { this.hours = +value; }
            },
            'h': {
                regex: '[0-9]|1[0-2]',
                apply: function (value) { this.hours = +value; }
            },
            'mm': {
                regex: '[0-5][0-9]',
                apply: function (value) { this.minutes = +value; }
            },
            'm': {
                regex: '[0-9]|[1-5][0-9]',
                apply: function (value) { this.minutes = +value; }
            },
            'sss': {
                regex: '[0-9][0-9][0-9]',
                apply: function (value) { this.milliseconds = +value; }
            },
            'ss': {
                regex: '[0-5][0-9]',
                apply: function (value) { this.seconds = +value; }
            },
            's': {
                regex: '[0-9]|[1-5][0-9]',
                apply: function (value) { this.seconds = +value; }
            },
            'a': {
                regex: $locale.DATETIME_FORMATS.AMPMS.join('|'),
                apply: function (value) {
                    if (this.hours === 12) {
                        this.hours = 0;
                    }

                    if (value === 'PM') {
                        this.hours += 12;
                    }
                }
            }
        };
    };

    this.init();

    function createParser(format) {
        var map = [], regex = format.split('');

        angular.forEach(formatCodeToRegex, function (data, code) {
            var index = format.indexOf(code);

            if (index > -1) {
                format = format.split('');

                regex[index] = '(' + data.regex + ')';
                format[index] = '$'; // Custom symbol to define consumed part of format
                for (var i = index + 1, n = index + code.length; i < n; i++) {
                    regex[i] = '';
                    format[i] = '$';
                }
                format = format.join('');

                map.push({ index: index, apply: data.apply });
            }
        });

        return {
            regex: new RegExp('^' + regex.join('') + '$'),
            map: orderByFilter(map, 'index')
        };
    }

    this.parse = function (input, format, baseDate) {
        if (!angular.isString(input) || !format) {
            return input;
        }

        format = $locale.DATETIME_FORMATS[format] || format;
        format = format.replace(SPECIAL_CHARACTERS_REGEXP, '\\$&');

        if ($locale.id !== localeId) {
            this.init();
        }

        if (!this.parsers[format]) {
            this.parsers[format] = createParser(format);
        }

        var parser = this.parsers[format],
            regex = parser.regex,
            map = parser.map,
            results = input.match(regex);

        if (results && results.length) {
            var fields, dt;
            if (angular.isDate(baseDate) && !isNaN(baseDate.getTime())) {
                fields = {
                    year: baseDate.getFullYear(),
                    month: baseDate.getMonth(),
                    date: baseDate.getDate(),
                    hours: baseDate.getHours(),
                    minutes: baseDate.getMinutes(),
                    seconds: baseDate.getSeconds(),
                    milliseconds: baseDate.getMilliseconds()
                };
            } else {
                if (baseDate) {
                    $log.warn('dateparser:', 'baseDate is not a valid date');
                }
                fields = { year: 1900, month: 0, date: 1, hours: 0, minutes: 0, seconds: 0, milliseconds: 0 };
            }

            for (var i = 1, n = results.length; i < n; i++) {
                var mapper = map[i - 1];
                if (mapper.apply) {
                    mapper.apply.call(fields, results[i]);
                }
            }

            if (isValid(fields.year, fields.month, fields.date)) {
                if (angular.isDate(baseDate) && !isNaN(baseDate.getTime())) {
                    dt = new Date(baseDate);
                    dt.setFullYear(fields.year, fields.month, fields.date,
                      fields.hours, fields.minutes, fields.seconds,
                      fields.milliseconds || 0);
                } else {
                    dt = new Date(fields.year, fields.month, fields.date,
                      fields.hours, fields.minutes, fields.seconds,
                      fields.milliseconds || 0);
                }
            }

            return dt;
        }
    };

    // Check if date is valid for specific month (and year for February).
    // Month: 0 = Jan, 1 = Feb, etc
    function isValid(year, month, date) {
        if (date < 1) {
            return false;
        }

        if (month === 1 && date > 28) {
            return date === 29 && ((year % 4 === 0 && year % 100 !== 0) || year % 400 === 0);
        }

        if (month === 3 || month === 5 || month === 8 || month === 10) {
            return date < 31;
        }

        return true;
    }
}]);

/* Deprecated dateparser below */

angular.module('ui.bootstrap.dateparser')

.value('$dateParserSuppressWarning', false)

.service('dateParser', ['$log', '$dateParserSuppressWarning', 'uibDateParser', function ($log, $dateParserSuppressWarning, uibDateParser) {
    if (!$dateParserSuppressWarning) {
        $log.warn('dateParser is now deprecated. Use uibDateParser instead.');
    }

    angular.extend(this, uibDateParser);
}]);

angular.module('ui.bootstrap.accordion', ['ui.bootstrap.collapse'])

.constant('uibAccordionConfig', {
    closeOthers: true
})

.controller('UibAccordionController', ['$scope', '$attrs', 'uibAccordionConfig', function ($scope, $attrs, accordionConfig) {
    // This array keeps track of the accordion groups
    this.groups = [];

    // Ensure that all the groups in this accordion are closed, unless close-others explicitly says not to
    this.closeOthers = function (openGroup) {
        var closeOthers = angular.isDefined($attrs.closeOthers) ?
          $scope.$eval($attrs.closeOthers) : accordionConfig.closeOthers;
        if (closeOthers) {
            angular.forEach(this.groups, function (group) {
                if (group !== openGroup) {
                    group.isOpen = false;
                }
            });
        }
    };

    // This is called from the accordion-group directive to add itself to the accordion
    this.addGroup = function (groupScope) {
        var that = this;
        this.groups.push(groupScope);

        groupScope.$on('$destroy', function (event) {
            that.removeGroup(groupScope);
        });
    };

    // This is called from the accordion-group directive when to remove itself
    this.removeGroup = function (group) {
        var index = this.groups.indexOf(group);
        if (index !== -1) {
            this.groups.splice(index, 1);
        }
    };

}])

// The accordion directive simply sets up the directive controller
// and adds an accordion CSS class to itself element.
.directive('uibAccordion', function () {
    return {
        controller: 'UibAccordionController',
        controllerAs: 'accordion',
        transclude: true,
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/accordion/accordion.html';
        }
    };
})

// The accordion-group directive indicates a block of html that will expand and collapse in an accordion
.directive('uibAccordionGroup', function () {
    return {
        require: '^uibAccordion',         // We need this directive to be inside an accordion
        transclude: true,              // It transcludes the contents of the directive into the template
        replace: true,                // The element containing the directive will be replaced with the template
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/accordion/accordion-group.html';
        },
        scope: {
            heading: '@',               // Interpolate the heading attribute onto this scope
            isOpen: '=?',
            isDisabled: '=?'
        },
        controller: function () {
            this.setHeading = function (element) {
                this.heading = element;
            };
        },
        link: function (scope, element, attrs, accordionCtrl) {
            accordionCtrl.addGroup(scope);

            scope.openClass = attrs.openClass || 'panel-open';
            scope.panelClass = attrs.panelClass;
            scope.$watch('isOpen', function (value) {
                element.toggleClass(scope.openClass, !!value);
                if (value) {
                    accordionCtrl.closeOthers(scope);
                }
            });

            scope.toggleOpen = function ($event) {
                if (!scope.isDisabled) {
                    if (!$event || $event.which === 32) {
                        scope.isOpen = !scope.isOpen;
                    }
                }
            };
        }
    };
})

// Use accordion-heading below an accordion-group to provide a heading containing HTML
.directive('uibAccordionHeading', function () {
    return {
        transclude: true,   // Grab the contents to be used as the heading
        template: '',       // In effect remove this element!
        replace: true,
        require: '^uibAccordionGroup',
        link: function (scope, element, attrs, accordionGroupCtrl, transclude) {
            // Pass the heading to the accordion-group controller
            // so that it can be transcluded into the right place in the template
            // [The second parameter to transclude causes the elements to be cloned so that they work in ng-repeat]
            accordionGroupCtrl.setHeading(transclude(scope, angular.noop));
        }
    };
})

// Use in the accordion-group template to indicate where you want the heading to be transcluded
// You must provide the property on the accordion-group controller that will hold the transcluded element
.directive('uibAccordionTransclude', function () {
    return {
        require: ['?^uibAccordionGroup', '?^accordionGroup'],
        link: function (scope, element, attrs, controller) {
            controller = controller[0] ? controller[0] : controller[1]; // Delete after we remove deprecation
            scope.$watch(function () { return controller[attrs.uibAccordionTransclude]; }, function (heading) {
                if (heading) {
                    element.find('span').html('');
                    element.find('span').append(heading);
                }
            });
        }
    };
});

/* Deprecated accordion below */

angular.module('ui.bootstrap.accordion')

  .value('$accordionSuppressWarning', false)

  .controller('AccordionController', ['$scope', '$attrs', '$controller', '$log', '$accordionSuppressWarning', function ($scope, $attrs, $controller, $log, $accordionSuppressWarning) {
      if (!$accordionSuppressWarning) {
          $log.warn('AccordionController is now deprecated. Use UibAccordionController instead.');
      }

      angular.extend(this, $controller('UibAccordionController', {
          $scope: $scope,
          $attrs: $attrs
      }));
  }])

  .directive('accordion', ['$log', '$accordionSuppressWarning', function ($log, $accordionSuppressWarning) {
      return {
          restrict: 'EA',
          controller: 'AccordionController',
          controllerAs: 'accordion',
          transclude: true,
          replace: false,
          templateUrl: function (element, attrs) {
              return attrs.templateUrl || 'template/accordion/accordion.html';
          },
          link: function () {
              if (!$accordionSuppressWarning) {
                  $log.warn('accordion is now deprecated. Use uib-accordion instead.');
              }
          }
      };
  }])

  .directive('accordionGroup', ['$log', '$accordionSuppressWarning', function ($log, $accordionSuppressWarning) {
      return {
          require: '^accordion',         // We need this directive to be inside an accordion
          restrict: 'EA',
          transclude: true,              // It transcludes the contents of the directive into the template
          replace: true,                // The element containing the directive will be replaced with the template
          templateUrl: function (element, attrs) {
              return attrs.templateUrl || 'template/accordion/accordion-group.html';
          },
          scope: {
              heading: '@',               // Interpolate the heading attribute onto this scope
              isOpen: '=?',
              isDisabled: '=?'
          },
          controller: function () {
              this.setHeading = function (element) {
                  this.heading = element;
              };
          },
          link: function (scope, element, attrs, accordionCtrl) {
              if (!$accordionSuppressWarning) {
                  $log.warn('accordion-group is now deprecated. Use uib-accordion-group instead.');
              }

              accordionCtrl.addGroup(scope);

              scope.openClass = attrs.openClass || 'panel-open';
              scope.panelClass = attrs.panelClass;
              scope.$watch('isOpen', function (value) {
                  element.toggleClass(scope.openClass, !!value);
                  if (value) {
                      accordionCtrl.closeOthers(scope);
                  }
              });

              scope.toggleOpen = function ($event) {
                  if (!scope.isDisabled) {
                      if (!$event || $event.which === 32) {
                          scope.isOpen = !scope.isOpen;
                      }
                  }
              };
          }
      };
  }])

  .directive('accordionHeading', ['$log', '$accordionSuppressWarning', function ($log, $accordionSuppressWarning) {
      return {
          restrict: 'EA',
          transclude: true,   // Grab the contents to be used as the heading
          template: '',       // In effect remove this element!
          replace: true,
          require: '^accordionGroup',
          link: function (scope, element, attr, accordionGroupCtrl, transclude) {
              if (!$accordionSuppressWarning) {
                  $log.warn('accordion-heading is now deprecated. Use uib-accordion-heading instead.');
              }
              // Pass the heading to the accordion-group controller
              // so that it can be transcluded into the right place in the template
              // [The second parameter to transclude causes the elements to be cloned so that they work in ng-repeat]
              accordionGroupCtrl.setHeading(transclude(scope, angular.noop));
          }
      };
  }])

  .directive('accordionTransclude', ['$log', '$accordionSuppressWarning', function ($log, $accordionSuppressWarning) {
      return {
          require: '^accordionGroup',
          link: function (scope, element, attr, controller) {
              if (!$accordionSuppressWarning) {
                  $log.warn('accordion-transclude is now deprecated. Use uib-accordion-transclude instead.');
              }

              scope.$watch(function () { return controller[attr.accordionTransclude]; }, function (heading) {
                  if (heading) {
                      element.find('span').html('');
                      element.find('span').append(heading);
                  }
              });
          }
      };
  }]);


angular.module('ui.bootstrap.alert', [])

.controller('UibAlertController', ['$scope', '$attrs', '$interpolate', '$timeout', function ($scope, $attrs, $interpolate, $timeout) {
    $scope.closeable = !!$attrs.close;

    var dismissOnTimeout = angular.isDefined($attrs.dismissOnTimeout) ?
      $interpolate($attrs.dismissOnTimeout)($scope.$parent) : null;

    if (dismissOnTimeout) {
        $timeout(function () {
            $scope.close();
        }, parseInt(dismissOnTimeout, 10));
    }
}])

.directive('uibAlert', function () {
    return {
        controller: 'UibAlertController',
        controllerAs: 'alert',
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/alert/alert.html';
        },
        transclude: true,
        replace: true,
        scope: {
            type: '@',
            close: '&'
        }
    };
});

/* Deprecated alert below */

angular.module('ui.bootstrap.alert')

  .value('$alertSuppressWarning', false)

  .controller('AlertController', ['$scope', '$attrs', '$controller', '$log', '$alertSuppressWarning', function ($scope, $attrs, $controller, $log, $alertSuppressWarning) {
      if (!$alertSuppressWarning) {
          $log.warn('AlertController is now deprecated. Use UibAlertController instead.');
      }

      angular.extend(this, $controller('UibAlertController', {
          $scope: $scope,
          $attrs: $attrs
      }));
  }])

  .directive('alert', ['$log', '$alertSuppressWarning', function ($log, $alertSuppressWarning) {
      return {
          controller: 'AlertController',
          controllerAs: 'alert',
          templateUrl: function (element, attrs) {
              return attrs.templateUrl || 'template/alert/alert.html';
          },
          transclude: true,
          replace: true,
          scope: {
              type: '@',
              close: '&'
          },
          link: function () {
              if (!$alertSuppressWarning) {
                  $log.warn('alert is now deprecated. Use uib-alert instead.');
              }
          }
      };
  }]);

angular.module('ui.bootstrap.dropdown', ['ui.bootstrap.position'])

.constant('uibDropdownConfig', {
    openClass: 'open'
})

.service('uibDropdownService', ['$document', '$rootScope', function ($document, $rootScope) {
    var openScope = null;

    this.open = function (dropdownScope) {
        if (!openScope) {
            $document.bind('click', closeDropdown);
            $document.bind('keydown', keybindFilter);
        }

        if (openScope && openScope !== dropdownScope) {
            openScope.isOpen = false;
        }

        openScope = dropdownScope;
    };

    this.close = function (dropdownScope) {
        if (openScope === dropdownScope) {
            openScope = null;
            $document.unbind('click', closeDropdown);
            $document.unbind('keydown', keybindFilter);
        }
    };

    var closeDropdown = function (evt) {
        // This method may still be called during the same mouse event that
        // unbound this event handler. So check openScope before proceeding.
        if (!openScope) { return; }

        if (evt && openScope.getAutoClose() === 'disabled') { return; }

        var toggleElement = openScope.getToggleElement();
        if (evt && toggleElement && toggleElement[0].contains(evt.target)) {
            return;
        }

        var dropdownElement = openScope.getDropdownElement();
        if (evt && openScope.getAutoClose() === 'outsideClick' &&
          dropdownElement && dropdownElement[0].contains(evt.target)) {
            return;
        }

        openScope.isOpen = false;

        if (!$rootScope.$$phase) {
            openScope.$apply();
        }
    };

    var keybindFilter = function (evt) {
        if (evt.which === 27) {
            openScope.focusToggleElement();
            closeDropdown();
        } else if (openScope.isKeynavEnabled() && /(38|40)/.test(evt.which) && openScope.isOpen) {
            evt.preventDefault();
            evt.stopPropagation();
            openScope.focusDropdownEntry(evt.which);
        }
    };
}])

.controller('UibDropdownController', ['$scope', '$element', '$attrs', '$parse', 'uibDropdownConfig', 'uibDropdownService', '$animate', '$uibPosition', '$document', '$compile', '$templateRequest', function ($scope, $element, $attrs, $parse, dropdownConfig, uibDropdownService, $animate, $position, $document, $compile, $templateRequest) {
    var self = this,
      scope = $scope.$new(), // create a child scope so we are not polluting original one
      templateScope,
      openClass = dropdownConfig.openClass,
      getIsOpen,
      setIsOpen = angular.noop,
      toggleInvoker = $attrs.onToggle ? $parse($attrs.onToggle) : angular.noop,
      appendToBody = false,
      keynavEnabled = false,
      selectedOption = null;


    $element.addClass('dropdown');

    this.init = function () {
        if ($attrs.isOpen) {
            getIsOpen = $parse($attrs.isOpen);
            setIsOpen = getIsOpen.assign;

            $scope.$watch(getIsOpen, function (value) {
                scope.isOpen = !!value;
            });
        }

        appendToBody = angular.isDefined($attrs.dropdownAppendToBody);
        keynavEnabled = angular.isDefined($attrs.uibKeyboardNav);

        if (appendToBody && self.dropdownMenu) {
            $document.find('body').append(self.dropdownMenu);
            $element.on('$destroy', function handleDestroyEvent() {
                self.dropdownMenu.remove();
            });
        }
    };

    this.toggle = function (open) {
        return scope.isOpen = arguments.length ? !!open : !scope.isOpen;
    };

    // Allow other directives to watch status
    this.isOpen = function () {
        return scope.isOpen;
    };

    scope.getToggleElement = function () {
        return self.toggleElement;
    };

    scope.getAutoClose = function () {
        return $attrs.autoClose || 'always'; //or 'outsideClick' or 'disabled'
    };

    scope.getElement = function () {
        return $element;
    };

    scope.isKeynavEnabled = function () {
        return keynavEnabled;
    };

    scope.focusDropdownEntry = function (keyCode) {
        var elems = self.dropdownMenu ? //If append to body is used.
          (angular.element(self.dropdownMenu).find('a')) :
          (angular.element($element).find('ul').eq(0).find('a'));

        switch (keyCode) {
            case (40): {
                if (!angular.isNumber(self.selectedOption)) {
                    self.selectedOption = 0;
                } else {
                    self.selectedOption = (self.selectedOption === elems.length - 1 ?
                      self.selectedOption :
                      self.selectedOption + 1);
                }
                break;
            }
            case (38): {
                if (!angular.isNumber(self.selectedOption)) {
                    self.selectedOption = elems.length - 1;
                } else {
                    self.selectedOption = self.selectedOption === 0 ?
                      0 : self.selectedOption - 1;
                }
                break;
            }
        }
        elems[self.selectedOption].focus();
    };

    scope.getDropdownElement = function () {
        return self.dropdownMenu;
    };

    scope.focusToggleElement = function () {
        if (self.toggleElement) {
            self.toggleElement[0].focus();
        }
    };

    scope.$watch('isOpen', function (isOpen, wasOpen) {
        if (appendToBody && self.dropdownMenu) {
            var pos = $position.positionElements($element, self.dropdownMenu, 'bottom-left', true);
            var css = {
                top: pos.top + 'px',
                display: isOpen ? 'block' : 'none'
            };

            var rightalign = self.dropdownMenu.hasClass('dropdown-menu-right');
            if (!rightalign) {
                css.left = pos.left + 'px';
                css.right = 'auto';
            } else {
                css.left = 'auto';
                css.right = (window.innerWidth - (pos.left + $element.prop('offsetWidth'))) + 'px';
            }

            self.dropdownMenu.css(css);
        }

        $animate[isOpen ? 'addClass' : 'removeClass']($element, openClass).then(function () {
            if (angular.isDefined(isOpen) && isOpen !== wasOpen) {
                toggleInvoker($scope, { open: !!isOpen });
            }
        });

        if (isOpen) {
            if (self.dropdownMenuTemplateUrl) {
                $templateRequest(self.dropdownMenuTemplateUrl).then(function (tplContent) {
                    templateScope = scope.$new();
                    $compile(tplContent.trim())(templateScope, function (dropdownElement) {
                        var newEl = dropdownElement;
                        self.dropdownMenu.replaceWith(newEl);
                        self.dropdownMenu = newEl;
                    });
                });
            }

            scope.focusToggleElement();
            uibDropdownService.open(scope);
        } else {
            if (self.dropdownMenuTemplateUrl) {
                if (templateScope) {
                    templateScope.$destroy();
                }
                var newEl = angular.element('<ul class="dropdown-menu"></ul>');
                self.dropdownMenu.replaceWith(newEl);
                self.dropdownMenu = newEl;
            }

            uibDropdownService.close(scope);
            self.selectedOption = null;
        }

        if (angular.isFunction(setIsOpen)) {
            setIsOpen($scope, isOpen);
        }
    });

    $scope.$on('$locationChangeSuccess', function () {
        if (scope.getAutoClose() !== 'disabled') {
            scope.isOpen = false;
        }
    });

    var offDestroy = $scope.$on('$destroy', function () {
        scope.$destroy();
    });
    scope.$on('$destroy', offDestroy);
}])

.directive('uibDropdown', function () {
    return {
        controller: 'UibDropdownController',
        link: function (scope, element, attrs, dropdownCtrl) {
            dropdownCtrl.init();
        }
    };
})

.directive('uibDropdownMenu', function () {
    return {
        restrict: 'AC',
        require: '?^uibDropdown',
        link: function (scope, element, attrs, dropdownCtrl) {
            if (!dropdownCtrl || angular.isDefined(attrs.dropdownNested)) {
                return;
            }

            element.addClass('dropdown-menu');

            var tplUrl = attrs.templateUrl;
            if (tplUrl) {
                dropdownCtrl.dropdownMenuTemplateUrl = tplUrl;
            }

            if (!dropdownCtrl.dropdownMenu) {
                dropdownCtrl.dropdownMenu = element;
            }
        }
    };
})

.directive('uibKeyboardNav', function () {
    return {
        restrict: 'A',
        require: '?^uibDropdown',
        link: function (scope, element, attrs, dropdownCtrl) {
            element.bind('keydown', function (e) {
                if ([38, 40].indexOf(e.which) !== -1) {
                    e.preventDefault();
                    e.stopPropagation();

                    var elems = dropdownCtrl.dropdownMenu.find('a');

                    switch (e.which) {
                        case (40): { // Down
                            if (!angular.isNumber(dropdownCtrl.selectedOption)) {
                                dropdownCtrl.selectedOption = 0;
                            } else {
                                dropdownCtrl.selectedOption = dropdownCtrl.selectedOption === elems.length - 1 ?
                                  dropdownCtrl.selectedOption : dropdownCtrl.selectedOption + 1;
                            }
                            break;
                        }
                        case (38): { // Up
                            if (!angular.isNumber(dropdownCtrl.selectedOption)) {
                                dropdownCtrl.selectedOption = elems.length - 1;
                            } else {
                                dropdownCtrl.selectedOption = dropdownCtrl.selectedOption === 0 ?
                                  0 : dropdownCtrl.selectedOption - 1;
                            }
                            break;
                        }
                    }
                    elems[dropdownCtrl.selectedOption].focus();
                }
            });
        }
    };
})

.directive('uibDropdownToggle', function () {
    return {
        require: '?^uibDropdown',
        link: function (scope, element, attrs, dropdownCtrl) {
            if (!dropdownCtrl) {
                return;
            }

            element.addClass('dropdown-toggle');

            dropdownCtrl.toggleElement = element;

            var toggleDropdown = function (event) {
                event.preventDefault();

                if (!element.hasClass('disabled') && !attrs.disabled) {
                    scope.$apply(function () {
                        dropdownCtrl.toggle();
                    });
                }
            };

            element.bind('click', toggleDropdown);

            // WAI-ARIA
            element.attr({ 'aria-haspopup': true, 'aria-expanded': false });
            scope.$watch(dropdownCtrl.isOpen, function (isOpen) {
                element.attr('aria-expanded', !!isOpen);
            });

            scope.$on('$destroy', function () {
                element.unbind('click', toggleDropdown);
            });
        }
    };
});

/* Deprecated dropdown below */

angular.module('ui.bootstrap.dropdown')

.value('$dropdownSuppressWarning', false)

.service('dropdownService', ['$log', '$dropdownSuppressWarning', 'uibDropdownService', function ($log, $dropdownSuppressWarning, uibDropdownService) {
    if (!$dropdownSuppressWarning) {
        $log.warn('dropdownService is now deprecated. Use uibDropdownService instead.');
    }

    angular.extend(this, uibDropdownService);
}])

.controller('DropdownController', ['$scope', '$element', '$attrs', '$parse', 'uibDropdownConfig', 'uibDropdownService', '$animate', '$uibPosition', '$document', '$compile', '$templateRequest', '$log', '$dropdownSuppressWarning', function ($scope, $element, $attrs, $parse, dropdownConfig, uibDropdownService, $animate, $position, $document, $compile, $templateRequest, $log, $dropdownSuppressWarning) {
    if (!$dropdownSuppressWarning) {
        $log.warn('DropdownController is now deprecated. Use UibDropdownController instead.');
    }

    var self = this,
      scope = $scope.$new(), // create a child scope so we are not polluting original one
      templateScope,
      openClass = dropdownConfig.openClass,
      getIsOpen,
      setIsOpen = angular.noop,
      toggleInvoker = $attrs.onToggle ? $parse($attrs.onToggle) : angular.noop,
      appendToBody = false,
      keynavEnabled = false,
      selectedOption = null;


    $element.addClass('dropdown');

    this.init = function () {
        if ($attrs.isOpen) {
            getIsOpen = $parse($attrs.isOpen);
            setIsOpen = getIsOpen.assign;

            $scope.$watch(getIsOpen, function (value) {
                scope.isOpen = !!value;
            });
        }

        appendToBody = angular.isDefined($attrs.dropdownAppendToBody);
        keynavEnabled = angular.isDefined($attrs.uibKeyboardNav);

        if (appendToBody && self.dropdownMenu) {
            $document.find('body').append(self.dropdownMenu);
            $element.on('$destroy', function handleDestroyEvent() {
                self.dropdownMenu.remove();
            });
        }
    };

    this.toggle = function (open) {
        return scope.isOpen = arguments.length ? !!open : !scope.isOpen;
    };

    // Allow other directives to watch status
    this.isOpen = function () {
        return scope.isOpen;
    };

    scope.getToggleElement = function () {
        return self.toggleElement;
    };

    scope.getAutoClose = function () {
        return $attrs.autoClose || 'always'; //or 'outsideClick' or 'disabled'
    };

    scope.getElement = function () {
        return $element;
    };

    scope.isKeynavEnabled = function () {
        return keynavEnabled;
    };

    scope.focusDropdownEntry = function (keyCode) {
        var elems = self.dropdownMenu ? //If append to body is used.
          (angular.element(self.dropdownMenu).find('a')) :
          (angular.element($element).find('ul').eq(0).find('a'));

        switch (keyCode) {
            case (40): {
                if (!angular.isNumber(self.selectedOption)) {
                    self.selectedOption = 0;
                } else {
                    self.selectedOption = (self.selectedOption === elems.length - 1 ?
                      self.selectedOption :
                    self.selectedOption + 1);
                }
                break;
            }
            case (38): {
                if (!angular.isNumber(self.selectedOption)) {
                    self.selectedOption = elems.length - 1;
                } else {
                    self.selectedOption = self.selectedOption === 0 ?
                      0 : self.selectedOption - 1;
                }
                break;
            }
        }
        elems[self.selectedOption].focus();
    };

    scope.getDropdownElement = function () {
        return self.dropdownMenu;
    };

    scope.focusToggleElement = function () {
        if (self.toggleElement) {
            self.toggleElement[0].focus();
        }
    };

    scope.$watch('isOpen', function (isOpen, wasOpen) {
        if (appendToBody && self.dropdownMenu) {
            var pos = $position.positionElements($element, self.dropdownMenu, 'bottom-left', true);
            var css = {
                top: pos.top + 'px',
                display: isOpen ? 'block' : 'none'
            };

            var rightalign = self.dropdownMenu.hasClass('dropdown-menu-right');
            if (!rightalign) {
                css.left = pos.left + 'px';
                css.right = 'auto';
            } else {
                css.left = 'auto';
                css.right = (window.innerWidth - (pos.left + $element.prop('offsetWidth'))) + 'px';
            }

            self.dropdownMenu.css(css);
        }

        $animate[isOpen ? 'addClass' : 'removeClass']($element, openClass).then(function () {
            if (angular.isDefined(isOpen) && isOpen !== wasOpen) {
                toggleInvoker($scope, { open: !!isOpen });
            }
        });

        if (isOpen) {
            if (self.dropdownMenuTemplateUrl) {
                $templateRequest(self.dropdownMenuTemplateUrl).then(function (tplContent) {
                    templateScope = scope.$new();
                    $compile(tplContent.trim())(templateScope, function (dropdownElement) {
                        var newEl = dropdownElement;
                        self.dropdownMenu.replaceWith(newEl);
                        self.dropdownMenu = newEl;
                    });
                });
            }

            scope.focusToggleElement();
            uibDropdownService.open(scope);
        } else {
            if (self.dropdownMenuTemplateUrl) {
                if (templateScope) {
                    templateScope.$destroy();
                }
                var newEl = angular.element('<ul class="dropdown-menu"></ul>');
                self.dropdownMenu.replaceWith(newEl);
                self.dropdownMenu = newEl;
            }

            uibDropdownService.close(scope);
            self.selectedOption = null;
        }

        if (angular.isFunction(setIsOpen)) {
            setIsOpen($scope, isOpen);
        }
    });

    $scope.$on('$locationChangeSuccess', function () {
        if (scope.getAutoClose() !== 'disabled') {
            scope.isOpen = false;
        }
    });

    var offDestroy = $scope.$on('$destroy', function () {
        scope.$destroy();
    });
    scope.$on('$destroy', offDestroy);
}])

.directive('dropdown', ['$log', '$dropdownSuppressWarning', function ($log, $dropdownSuppressWarning) {
    return {
        controller: 'DropdownController',
        link: function (scope, element, attrs, dropdownCtrl) {
            if (!$dropdownSuppressWarning) {
                $log.warn('dropdown is now deprecated. Use uib-dropdown instead.');
            }

            dropdownCtrl.init();
        }
    };
}])

.directive('dropdownMenu', ['$log', '$dropdownSuppressWarning', function ($log, $dropdownSuppressWarning) {
    return {
        restrict: 'AC',
        require: '?^dropdown',
        link: function (scope, element, attrs, dropdownCtrl) {
            if (!dropdownCtrl || angular.isDefined(attrs.dropdownNested)) {
                return;
            }

            if (!$dropdownSuppressWarning) {
                $log.warn('dropdown-menu is now deprecated. Use uib-dropdown-menu instead.');
            }

            element.addClass('dropdown-menu');

            var tplUrl = attrs.templateUrl;
            if (tplUrl) {
                dropdownCtrl.dropdownMenuTemplateUrl = tplUrl;
            }

            if (!dropdownCtrl.dropdownMenu) {
                dropdownCtrl.dropdownMenu = element;
            }
        }
    };
}])

.directive('keyboardNav', ['$log', '$dropdownSuppressWarning', function ($log, $dropdownSuppressWarning) {
    return {
        restrict: 'A',
        require: '?^dropdown',
        link: function (scope, element, attrs, dropdownCtrl) {
            if (!$dropdownSuppressWarning) {
                $log.warn('keyboard-nav is now deprecated. Use uib-keyboard-nav instead.');
            }

            element.bind('keydown', function (e) {
                if ([38, 40].indexOf(e.which) !== -1) {
                    e.preventDefault();
                    e.stopPropagation();

                    var elems = dropdownCtrl.dropdownMenu.find('a');

                    switch (e.which) {
                        case (40): { // Down
                            if (!angular.isNumber(dropdownCtrl.selectedOption)) {
                                dropdownCtrl.selectedOption = 0;
                            } else {
                                dropdownCtrl.selectedOption = dropdownCtrl.selectedOption === elems.length - 1 ?
                                  dropdownCtrl.selectedOption : dropdownCtrl.selectedOption + 1;
                            }
                            break;
                        }
                        case (38): { // Up
                            if (!angular.isNumber(dropdownCtrl.selectedOption)) {
                                dropdownCtrl.selectedOption = elems.length - 1;
                            } else {
                                dropdownCtrl.selectedOption = dropdownCtrl.selectedOption === 0 ?
                                  0 : dropdownCtrl.selectedOption - 1;
                            }
                            break;
                        }
                    }
                    elems[dropdownCtrl.selectedOption].focus();
                }
            });
        }
    };
}])

.directive('dropdownToggle', ['$log', '$dropdownSuppressWarning', function ($log, $dropdownSuppressWarning) {
    return {
        require: '?^dropdown',
        link: function (scope, element, attrs, dropdownCtrl) {
            if (!$dropdownSuppressWarning) {
                $log.warn('dropdown-toggle is now deprecated. Use uib-dropdown-toggle instead.');
            }

            if (!dropdownCtrl) {
                return;
            }

            element.addClass('dropdown-toggle');

            dropdownCtrl.toggleElement = element;

            var toggleDropdown = function (event) {
                event.preventDefault();

                if (!element.hasClass('disabled') && !attrs.disabled) {
                    scope.$apply(function () {
                        dropdownCtrl.toggle();
                    });
                }
            };

            element.bind('click', toggleDropdown);

            // WAI-ARIA
            element.attr({ 'aria-haspopup': true, 'aria-expanded': false });
            scope.$watch(dropdownCtrl.isOpen, function (isOpen) {
                element.attr('aria-expanded', !!isOpen);
            });

            scope.$on('$destroy', function () {
                element.unbind('click', toggleDropdown);
            });
        }
    };
}]);

angular.module('ui.bootstrap.pagination', [])
.controller('UibPaginationController', ['$scope', '$attrs', '$parse', function ($scope, $attrs, $parse) {
    var self = this,
        ngModelCtrl = { $setViewValue: angular.noop }, // nullModelCtrl
        setNumPages = $attrs.numPages ? $parse($attrs.numPages).assign : angular.noop;

    this.init = function (ngModelCtrl_, config) {
        ngModelCtrl = ngModelCtrl_;
        this.config = config;

        ngModelCtrl.$render = function () {
            self.render();
        };

        if ($attrs.itemsPerPage) {
            $scope.$parent.$watch($parse($attrs.itemsPerPage), function (value) {
                self.itemsPerPage = parseInt(value, 10);
                $scope.totalPages = self.calculateTotalPages();
            });
        } else {
            this.itemsPerPage = config.itemsPerPage;
        }

        $scope.$watch('totalItems', function () {
            $scope.totalPages = self.calculateTotalPages();
        });

        $scope.$watch('totalPages', function (value) {
            setNumPages($scope.$parent, value); // Readonly variable

            if ($scope.page > value) {
                $scope.selectPage(value);
            } else {
                ngModelCtrl.$render();
            }
        });
    };

    this.calculateTotalPages = function () {
        var totalPages = this.itemsPerPage < 1 ? 1 : Math.ceil($scope.totalItems / this.itemsPerPage);
        return Math.max(totalPages || 0, 1);
    };

    this.render = function () {
        $scope.page = parseInt(ngModelCtrl.$viewValue, 10) || 1;
    };

    $scope.selectPage = function (page, evt) {
        if (evt) {
            evt.preventDefault();
        }

        var clickAllowed = !$scope.ngDisabled || !evt;
        if (clickAllowed && $scope.page !== page && page > 0 && page <= $scope.totalPages) {
            if (evt && evt.target) {
                evt.target.blur();
            }
            ngModelCtrl.$setViewValue(page);
            ngModelCtrl.$render();
        }
    };

    $scope.getText = function (key) {
        return $scope[key + 'Text'] || self.config[key + 'Text'];
    };

    $scope.noPrevious = function () {
        return $scope.page === 1;
    };

    $scope.noNext = function () {
        return $scope.page === $scope.totalPages;
    };
}])

.constant('uibPaginationConfig', {
    itemsPerPage: 10,
    boundaryLinks: false,
    directionLinks: true,
    firstText: 'First',
    previousText: 'Previous',
    nextText: 'Next',
    lastText: 'Last',
    rotate: true
})

.directive('uibPagination', ['$parse', 'uibPaginationConfig', function ($parse, paginationConfig) {
    return {
        restrict: 'EA',
        scope: {
            totalItems: '=',
            firstText: '@',
            previousText: '@',
            nextText: '@',
            lastText: '@',
            ngDisabled: '='
        },
        require: ['uibPagination', '?ngModel'],
        controller: 'UibPaginationController',
        controllerAs: 'pagination',
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/pagination/pagination.html';
        },
        replace: true,
        link: function (scope, element, attrs, ctrls) {
            var paginationCtrl = ctrls[0], ngModelCtrl = ctrls[1];

            if (!ngModelCtrl) {
                return; // do nothing if no ng-model
            }

            // Setup configuration parameters
            var maxSize = angular.isDefined(attrs.maxSize) ? scope.$parent.$eval(attrs.maxSize) : paginationConfig.maxSize,
                rotate = angular.isDefined(attrs.rotate) ? scope.$parent.$eval(attrs.rotate) : paginationConfig.rotate;
            scope.boundaryLinks = angular.isDefined(attrs.boundaryLinks) ? scope.$parent.$eval(attrs.boundaryLinks) : paginationConfig.boundaryLinks;
            scope.directionLinks = angular.isDefined(attrs.directionLinks) ? scope.$parent.$eval(attrs.directionLinks) : paginationConfig.directionLinks;

            paginationCtrl.init(ngModelCtrl, paginationConfig);

            if (attrs.maxSize) {
                scope.$parent.$watch($parse(attrs.maxSize), function (value) {
                    maxSize = parseInt(value, 10);
                    paginationCtrl.render();
                });
            }

            // Create page object used in template
            function makePage(number, text, isActive) {
                return {
                    number: number,
                    text: text,
                    active: isActive
                };
            }

            function getPages(currentPage, totalPages) {
                var pages = [];

                // Default page limits
                var startPage = 1, endPage = totalPages;
                var isMaxSized = angular.isDefined(maxSize) && maxSize < totalPages;

                // recompute if maxSize
                if (isMaxSized) {
                    if (rotate) {
                        // Current page is displayed in the middle of the visible ones
                        startPage = Math.max(currentPage - Math.floor(maxSize / 2), 1);
                        endPage = startPage + maxSize - 1;

                        // Adjust if limit is exceeded
                        if (endPage > totalPages) {
                            endPage = totalPages;
                            startPage = endPage - maxSize + 1;
                        }
                    } else {
                        // Visible pages are paginated with maxSize
                        startPage = ((Math.ceil(currentPage / maxSize) - 1) * maxSize) + 1;

                        // Adjust last page if limit is exceeded
                        endPage = Math.min(startPage + maxSize - 1, totalPages);
                    }
                }

                // Add page number links
                for (var number = startPage; number <= endPage; number++) {
                    var page = makePage(number, number, number === currentPage);
                    pages.push(page);
                }

                // Add links to move between page sets
                if (isMaxSized && !rotate) {
                    if (startPage > 1) {
                        var previousPageSet = makePage(startPage - 1, '...', false);
                        pages.unshift(previousPageSet);
                    }

                    if (endPage < totalPages) {
                        var nextPageSet = makePage(endPage + 1, '...', false);
                        pages.push(nextPageSet);
                    }
                }

                return pages;
            }

            var originalRender = paginationCtrl.render;
            paginationCtrl.render = function () {
                originalRender();
                if (scope.page > 0 && scope.page <= scope.totalPages) {
                    scope.pages = getPages(scope.page, scope.totalPages);
                }
            };
        }
    };
}])

.constant('uibPagerConfig', {
    itemsPerPage: 10,
    previousText: '« Previous',
    nextText: 'Next »',
    align: true
})

.directive('uibPager', ['uibPagerConfig', function (pagerConfig) {
    return {
        restrict: 'EA',
        scope: {
            totalItems: '=',
            previousText: '@',
            nextText: '@',
            ngDisabled: '='
        },
        require: ['uibPager', '?ngModel'],
        controller: 'UibPaginationController',
        controllerAs: 'pagination',
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/pagination/pager.html';
        },
        replace: true,
        link: function (scope, element, attrs, ctrls) {
            var paginationCtrl = ctrls[0], ngModelCtrl = ctrls[1];

            if (!ngModelCtrl) {
                return; // do nothing if no ng-model
            }

            scope.align = angular.isDefined(attrs.align) ? scope.$parent.$eval(attrs.align) : pagerConfig.align;
            paginationCtrl.init(ngModelCtrl, pagerConfig);
        }
    };
}]);

/* Deprecated Pagination Below */

angular.module('ui.bootstrap.pagination')
.value('$paginationSuppressWarning', false)
.controller('PaginationController', ['$scope', '$attrs', '$parse', '$log', '$paginationSuppressWarning', function ($scope, $attrs, $parse, $log, $paginationSuppressWarning) {
    if (!$paginationSuppressWarning) {
        $log.warn('PaginationController is now deprecated. Use UibPaginationController instead.');
    }

    var self = this,
      ngModelCtrl = { $setViewValue: angular.noop }, // nullModelCtrl
      setNumPages = $attrs.numPages ? $parse($attrs.numPages).assign : angular.noop;

    this.init = function (ngModelCtrl_, config) {
        ngModelCtrl = ngModelCtrl_;
        this.config = config;

        ngModelCtrl.$render = function () {
            self.render();
        };

        if ($attrs.itemsPerPage) {
            $scope.$parent.$watch($parse($attrs.itemsPerPage), function (value) {
                self.itemsPerPage = parseInt(value, 10);
                $scope.totalPages = self.calculateTotalPages();
            });
        } else {
            this.itemsPerPage = config.itemsPerPage;
        }

        $scope.$watch('totalItems', function () {
            $scope.totalPages = self.calculateTotalPages();
        });

        $scope.$watch('totalPages', function (value) {
            setNumPages($scope.$parent, value); // Readonly variable

            if ($scope.page > value) {
                $scope.selectPage(value);
            } else {
                ngModelCtrl.$render();
            }
        });
    };

    this.calculateTotalPages = function () {
        var totalPages = this.itemsPerPage < 1 ? 1 : Math.ceil($scope.totalItems / this.itemsPerPage);
        return Math.max(totalPages || 0, 1);
    };

    this.render = function () {
        $scope.page = parseInt(ngModelCtrl.$viewValue, 10) || 1;
    };

    $scope.selectPage = function (page, evt) {
        if (evt) {
            evt.preventDefault();
        }

        var clickAllowed = !$scope.ngDisabled || !evt;
        if (clickAllowed && $scope.page !== page && page > 0 && page <= $scope.totalPages) {
            if (evt && evt.target) {
                evt.target.blur();
            }
            ngModelCtrl.$setViewValue(page);
            ngModelCtrl.$render();
        }
    };

    $scope.getText = function (key) {
        return $scope[key + 'Text'] || self.config[key + 'Text'];
    };

    $scope.noPrevious = function () {
        return $scope.page === 1;
    };

    $scope.noNext = function () {
        return $scope.page === $scope.totalPages;
    };
}])
.directive('pagination', ['$parse', 'uibPaginationConfig', '$log', '$paginationSuppressWarning', function ($parse, paginationConfig, $log, $paginationSuppressWarning) {
    return {
        restrict: 'EA',
        scope: {
            totalItems: '=',
            firstText: '@',
            previousText: '@',
            nextText: '@',
            lastText: '@',
            ngDisabled: '='
        },
        require: ['pagination', '?ngModel'],
        controller: 'PaginationController',
        controllerAs: 'pagination',
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/pagination/pagination.html';
        },
        replace: true,
        link: function (scope, element, attrs, ctrls) {
            if (!$paginationSuppressWarning) {
                $log.warn('pagination is now deprecated. Use uib-pagination instead.');
            }
            var paginationCtrl = ctrls[0], ngModelCtrl = ctrls[1];

            if (!ngModelCtrl) {
                return; // do nothing if no ng-model
            }

            // Setup configuration parameters
            var maxSize = angular.isDefined(attrs.maxSize) ? scope.$parent.$eval(attrs.maxSize) : paginationConfig.maxSize,
                rotate = angular.isDefined(attrs.rotate) ? scope.$parent.$eval(attrs.rotate) : paginationConfig.rotate;
            scope.boundaryLinks = angular.isDefined(attrs.boundaryLinks) ? scope.$parent.$eval(attrs.boundaryLinks) : paginationConfig.boundaryLinks;
            scope.directionLinks = angular.isDefined(attrs.directionLinks) ? scope.$parent.$eval(attrs.directionLinks) : paginationConfig.directionLinks;

            paginationCtrl.init(ngModelCtrl, paginationConfig);

            if (attrs.maxSize) {
                scope.$parent.$watch($parse(attrs.maxSize), function (value) {
                    maxSize = parseInt(value, 10);
                    paginationCtrl.render();
                });
            }

            // Create page object used in template
            function makePage(number, text, isActive) {
                return {
                    number: number,
                    text: text,
                    active: isActive
                };
            }

            function getPages(currentPage, totalPages) {
                var pages = [];

                // Default page limits
                var startPage = 1, endPage = totalPages;
                var isMaxSized = angular.isDefined(maxSize) && maxSize < totalPages;

                // recompute if maxSize
                if (isMaxSized) {
                    if (rotate) {
                        // Current page is displayed in the middle of the visible ones
                        startPage = Math.max(currentPage - Math.floor(maxSize / 2), 1);
                        endPage = startPage + maxSize - 1;

                        // Adjust if limit is exceeded
                        if (endPage > totalPages) {
                            endPage = totalPages;
                            startPage = endPage - maxSize + 1;
                        }
                    } else {
                        // Visible pages are paginated with maxSize
                        startPage = ((Math.ceil(currentPage / maxSize) - 1) * maxSize) + 1;

                        // Adjust last page if limit is exceeded
                        endPage = Math.min(startPage + maxSize - 1, totalPages);
                    }
                }

                // Add page number links
                for (var number = startPage; number <= endPage; number++) {
                    var page = makePage(number, number, number === currentPage);
                    pages.push(page);
                }

                // Add links to move between page sets
                if (isMaxSized && !rotate) {
                    if (startPage > 1) {
                        var previousPageSet = makePage(startPage - 1, '...', false);
                        pages.unshift(previousPageSet);
                    }

                    if (endPage < totalPages) {
                        var nextPageSet = makePage(endPage + 1, '...', false);
                        pages.push(nextPageSet);
                    }
                }

                return pages;
            }

            var originalRender = paginationCtrl.render;
            paginationCtrl.render = function () {
                originalRender();
                if (scope.page > 0 && scope.page <= scope.totalPages) {
                    scope.pages = getPages(scope.page, scope.totalPages);
                }
            };
        }
    };
}])

.directive('pager', ['uibPagerConfig', '$log', '$paginationSuppressWarning', function (pagerConfig, $log, $paginationSuppressWarning) {
    return {
        restrict: 'EA',
        scope: {
            totalItems: '=',
            previousText: '@',
            nextText: '@',
            ngDisabled: '='
        },
        require: ['pager', '?ngModel'],
        controller: 'PaginationController',
        controllerAs: 'pagination',
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/pagination/pager.html';
        },
        replace: true,
        link: function (scope, element, attrs, ctrls) {
            if (!$paginationSuppressWarning) {
                $log.warn('pager is now deprecated. Use uib-pager instead.');
            }
            var paginationCtrl = ctrls[0], ngModelCtrl = ctrls[1];

            if (!ngModelCtrl) {
                return; // do nothing if no ng-model
            }

            scope.align = angular.isDefined(attrs.align) ? scope.$parent.$eval(attrs.align) : pagerConfig.align;
            paginationCtrl.init(ngModelCtrl, pagerConfig);
        }
    };
}]);

angular.module('ui.bootstrap.progressbar', [])

.constant('uibProgressConfig', {
    animate: true,
    max: 100
})

.controller('UibProgressController', ['$scope', '$attrs', 'uibProgressConfig', function ($scope, $attrs, progressConfig) {
    var self = this,
        animate = angular.isDefined($attrs.animate) ? $scope.$parent.$eval($attrs.animate) : progressConfig.animate;

    this.bars = [];
    $scope.max = angular.isDefined($scope.max) ? $scope.max : progressConfig.max;

    this.addBar = function (bar, element, attrs) {
        if (!animate) {
            element.css({ 'transition': 'none' });
        }

        this.bars.push(bar);

        bar.max = $scope.max;
        bar.title = attrs && angular.isDefined(attrs.title) ? attrs.title : 'progressbar';

        bar.$watch('value', function (value) {
            bar.recalculatePercentage();
        });

        bar.recalculatePercentage = function () {
            var totalPercentage = self.bars.reduce(function (total, bar) {
                bar.percent = +(100 * bar.value / bar.max).toFixed(2);
                return total + bar.percent;
            }, 0);

            if (totalPercentage > 100) {
                bar.percent -= totalPercentage - 100;
            }
        };

        bar.$on('$destroy', function () {
            element = null;
            self.removeBar(bar);
        });
    };

    this.removeBar = function (bar) {
        this.bars.splice(this.bars.indexOf(bar), 1);
        this.bars.forEach(function (bar) {
            bar.recalculatePercentage();
        });
    };

    $scope.$watch('max', function (max) {
        self.bars.forEach(function (bar) {
            bar.max = $scope.max;
            bar.recalculatePercentage();
        });
    });
}])

.directive('uibProgress', function () {
    return {
        replace: true,
        transclude: true,
        controller: 'UibProgressController',
        require: 'uibProgress',
        scope: {
            max: '=?'
        },
        templateUrl: 'template/progressbar/progress.html'
    };
})

.directive('uibBar', function () {
    return {
        replace: true,
        transclude: true,
        require: '^uibProgress',
        scope: {
            value: '=',
            type: '@'
        },
        templateUrl: 'template/progressbar/bar.html',
        link: function (scope, element, attrs, progressCtrl) {
            progressCtrl.addBar(scope, element, attrs);
        }
    };
})

.directive('uibProgressbar', function () {
    return {
        replace: true,
        transclude: true,
        controller: 'UibProgressController',
        scope: {
            value: '=',
            max: '=?',
            type: '@'
        },
        templateUrl: 'template/progressbar/progressbar.html',
        link: function (scope, element, attrs, progressCtrl) {
            progressCtrl.addBar(scope, angular.element(element.children()[0]), { title: attrs.title });
        }
    };
});

/* Deprecated progressbar below */

angular.module('ui.bootstrap.progressbar')

.value('$progressSuppressWarning', false)

.controller('ProgressController', ['$scope', '$attrs', 'uibProgressConfig', '$log', '$progressSuppressWarning', function ($scope, $attrs, progressConfig, $log, $progressSuppressWarning) {
    if (!$progressSuppressWarning) {
        $log.warn('ProgressController is now deprecated. Use UibProgressController instead.');
    }

    var self = this,
      animate = angular.isDefined($attrs.animate) ? $scope.$parent.$eval($attrs.animate) : progressConfig.animate;

    this.bars = [];
    $scope.max = angular.isDefined($scope.max) ? $scope.max : progressConfig.max;

    this.addBar = function (bar, element, attrs) {
        if (!animate) {
            element.css({ 'transition': 'none' });
        }

        this.bars.push(bar);

        bar.max = $scope.max;
        bar.title = attrs && angular.isDefined(attrs.title) ? attrs.title : 'progressbar';

        bar.$watch('value', function (value) {
            bar.recalculatePercentage();
        });

        bar.recalculatePercentage = function () {
            bar.percent = +(100 * bar.value / bar.max).toFixed(2);

            var totalPercentage = self.bars.reduce(function (total, bar) {
                return total + bar.percent;
            }, 0);

            if (totalPercentage > 100) {
                bar.percent -= totalPercentage - 100;
            }
        };

        bar.$on('$destroy', function () {
            element = null;
            self.removeBar(bar);
        });
    };

    this.removeBar = function (bar) {
        this.bars.splice(this.bars.indexOf(bar), 1);
    };

    $scope.$watch('max', function (max) {
        self.bars.forEach(function (bar) {
            bar.max = $scope.max;
            bar.recalculatePercentage();
        });
    });
}])

.directive('progress', ['$log', '$progressSuppressWarning', function ($log, $progressSuppressWarning) {
    return {
        replace: true,
        transclude: true,
        controller: 'ProgressController',
        require: 'progress',
        scope: {
            max: '=?',
            title: '@?'
        },
        templateUrl: 'template/progressbar/progress.html',
        link: function () {
            if (!$progressSuppressWarning) {
                $log.warn('progress is now deprecated. Use uib-progress instead.');
            }
        }
    };
}])

.directive('bar', ['$log', '$progressSuppressWarning', function ($log, $progressSuppressWarning) {
    return {
        replace: true,
        transclude: true,
        require: '^progress',
        scope: {
            value: '=',
            type: '@'
        },
        templateUrl: 'template/progressbar/bar.html',
        link: function (scope, element, attrs, progressCtrl) {
            if (!$progressSuppressWarning) {
                $log.warn('bar is now deprecated. Use uib-bar instead.');
            }
            progressCtrl.addBar(scope, element);
        }
    };
}])

.directive('progressbar', ['$log', '$progressSuppressWarning', function ($log, $progressSuppressWarning) {
    return {
        replace: true,
        transclude: true,
        controller: 'ProgressController',
        scope: {
            value: '=',
            max: '=?',
            type: '@'
        },
        templateUrl: 'template/progressbar/progressbar.html',
        link: function (scope, element, attrs, progressCtrl) {
            if (!$progressSuppressWarning) {
                $log.warn('progressbar is now deprecated. Use uib-progressbar instead.');
            }
            progressCtrl.addBar(scope, angular.element(element.children()[0]), { title: attrs.title });
        }
    };
}]);

angular.module('ui.bootstrap.timepicker', [])

.constant('uibTimepickerConfig', {
    hourStep: 1,
    minuteStep: 1,
    showMeridian: true,
    meridians: null,
    readonlyInput: false,
    mousewheel: true,
    arrowkeys: true,
    showSpinners: true
})

.controller('UibTimepickerController', ['$scope', '$element', '$attrs', '$parse', '$log', '$locale', 'uibTimepickerConfig', function ($scope, $element, $attrs, $parse, $log, $locale, timepickerConfig) {
    var selected = new Date(),
        ngModelCtrl = { $setViewValue: angular.noop }, // nullModelCtrl
        meridians = angular.isDefined($attrs.meridians) ? $scope.$parent.$eval($attrs.meridians) : timepickerConfig.meridians || $locale.DATETIME_FORMATS.AMPMS;

    $scope.tabindex = angular.isDefined($attrs.tabindex) ? $attrs.tabindex : 0;
    $element.removeAttr('tabindex');

    this.init = function (ngModelCtrl_, inputs) {
        ngModelCtrl = ngModelCtrl_;
        ngModelCtrl.$render = this.render;

        ngModelCtrl.$formatters.unshift(function (modelValue) {
            return modelValue ? new Date(modelValue) : null;
        });

        var hoursInputEl = inputs.eq(0),
            minutesInputEl = inputs.eq(1);

        var mousewheel = angular.isDefined($attrs.mousewheel) ? $scope.$parent.$eval($attrs.mousewheel) : timepickerConfig.mousewheel;
        if (mousewheel) {
            this.setupMousewheelEvents(hoursInputEl, minutesInputEl);
        }

        var arrowkeys = angular.isDefined($attrs.arrowkeys) ? $scope.$parent.$eval($attrs.arrowkeys) : timepickerConfig.arrowkeys;
        if (arrowkeys) {
            this.setupArrowkeyEvents(hoursInputEl, minutesInputEl);
        }

        $scope.readonlyInput = angular.isDefined($attrs.readonlyInput) ? $scope.$parent.$eval($attrs.readonlyInput) : timepickerConfig.readonlyInput;
        this.setupInputEvents(hoursInputEl, minutesInputEl);
    };

    var hourStep = timepickerConfig.hourStep;
    if ($attrs.hourStep) {
        $scope.$parent.$watch($parse($attrs.hourStep), function (value) {
            hourStep = parseInt(value, 10);
        });
    }

    var minuteStep = timepickerConfig.minuteStep;
    if ($attrs.minuteStep) {
        $scope.$parent.$watch($parse($attrs.minuteStep), function (value) {
            minuteStep = parseInt(value, 10);
        });
    }

    var min;
    $scope.$parent.$watch($parse($attrs.min), function (value) {
        var dt = new Date(value);
        min = isNaN(dt) ? undefined : dt;
    });

    var max;
    $scope.$parent.$watch($parse($attrs.max), function (value) {
        var dt = new Date(value);
        max = isNaN(dt) ? undefined : dt;
    });

    $scope.noIncrementHours = function () {
        var incrementedSelected = addMinutes(selected, hourStep * 60);
        return incrementedSelected > max ||
          (incrementedSelected < selected && incrementedSelected < min);
    };

    $scope.noDecrementHours = function () {
        var decrementedSelected = addMinutes(selected, -hourStep * 60);
        return decrementedSelected < min ||
          (decrementedSelected > selected && decrementedSelected > max);
    };

    $scope.noIncrementMinutes = function () {
        var incrementedSelected = addMinutes(selected, minuteStep);
        return incrementedSelected > max ||
          (incrementedSelected < selected && incrementedSelected < min);
    };

    $scope.noDecrementMinutes = function () {
        var decrementedSelected = addMinutes(selected, -minuteStep);
        return decrementedSelected < min ||
          (decrementedSelected > selected && decrementedSelected > max);
    };

    $scope.noToggleMeridian = function () {
        if (selected.getHours() < 13) {
            return addMinutes(selected, 12 * 60) > max;
        } else {
            return addMinutes(selected, -12 * 60) < min;
        }
    };

    // 12H / 24H mode
    $scope.showMeridian = timepickerConfig.showMeridian;
    if ($attrs.showMeridian) {
        $scope.$parent.$watch($parse($attrs.showMeridian), function (value) {
            $scope.showMeridian = !!value;

            if (ngModelCtrl.$error.time) {
                // Evaluate from template
                var hours = getHoursFromTemplate(), minutes = getMinutesFromTemplate();
                if (angular.isDefined(hours) && angular.isDefined(minutes)) {
                    selected.setHours(hours);
                    refresh();
                }
            } else {
                updateTemplate();
            }
        });
    }

    // Get $scope.hours in 24H mode if valid
    function getHoursFromTemplate() {
        var hours = parseInt($scope.hours, 10);
        var valid = $scope.showMeridian ? (hours > 0 && hours < 13) : (hours >= 0 && hours < 24);
        if (!valid) {
            return undefined;
        }

        if ($scope.showMeridian) {
            if (hours === 12) {
                hours = 0;
            }
            if ($scope.meridian === meridians[1]) {
                hours = hours + 12;
            }
        }
        return hours;
    }

    function getMinutesFromTemplate() {
        var minutes = parseInt($scope.minutes, 10);
        return (minutes >= 0 && minutes < 60) ? minutes : undefined;
    }

    function pad(value) {
        return (angular.isDefined(value) && value.toString().length < 2) ? '0' + value : value.toString();
    }

    // Respond on mousewheel spin
    this.setupMousewheelEvents = function (hoursInputEl, minutesInputEl) {
        var isScrollingUp = function (e) {
            if (e.originalEvent) {
                e = e.originalEvent;
            }
            //pick correct delta variable depending on event
            var delta = (e.wheelDelta) ? e.wheelDelta : -e.deltaY;
            return (e.detail || delta > 0);
        };

        hoursInputEl.bind('mousewheel wheel', function (e) {
            $scope.$apply(isScrollingUp(e) ? $scope.incrementHours() : $scope.decrementHours());
            e.preventDefault();
        });

        minutesInputEl.bind('mousewheel wheel', function (e) {
            $scope.$apply(isScrollingUp(e) ? $scope.incrementMinutes() : $scope.decrementMinutes());
            e.preventDefault();
        });

    };

    // Respond on up/down arrowkeys
    this.setupArrowkeyEvents = function (hoursInputEl, minutesInputEl) {
        hoursInputEl.bind('keydown', function (e) {
            if (e.which === 38) { // up
                e.preventDefault();
                $scope.incrementHours();
                $scope.$apply();
            } else if (e.which === 40) { // down
                e.preventDefault();
                $scope.decrementHours();
                $scope.$apply();
            }
        });

        minutesInputEl.bind('keydown', function (e) {
            if (e.which === 38) { // up
                e.preventDefault();
                $scope.incrementMinutes();
                $scope.$apply();
            } else if (e.which === 40) { // down
                e.preventDefault();
                $scope.decrementMinutes();
                $scope.$apply();
            }
        });
    };

    this.setupInputEvents = function (hoursInputEl, minutesInputEl) {
        if ($scope.readonlyInput) {
            $scope.updateHours = angular.noop;
            $scope.updateMinutes = angular.noop;
            return;
        }

        var invalidate = function (invalidHours, invalidMinutes) {
            ngModelCtrl.$setViewValue(null);
            ngModelCtrl.$setValidity('time', false);
            if (angular.isDefined(invalidHours)) {
                $scope.invalidHours = invalidHours;
            }
            if (angular.isDefined(invalidMinutes)) {
                $scope.invalidMinutes = invalidMinutes;
            }
        };

        $scope.updateHours = function () {
            var hours = getHoursFromTemplate(),
              minutes = getMinutesFromTemplate();

            if (angular.isDefined(hours) && angular.isDefined(minutes)) {
                selected.setHours(hours);
                if (selected < min || selected > max) {
                    invalidate(true);
                } else {
                    refresh('h');
                }
            } else {
                invalidate(true);
            }
        };

        hoursInputEl.bind('blur', function (e) {
            if (!$scope.invalidHours && $scope.hours < 10) {
                $scope.$apply(function () {
                    $scope.hours = pad($scope.hours);
                });
            }
        });

        $scope.updateMinutes = function () {
            var minutes = getMinutesFromTemplate(),
              hours = getHoursFromTemplate();

            if (angular.isDefined(minutes) && angular.isDefined(hours)) {
                selected.setMinutes(minutes);
                if (selected < min || selected > max) {
                    invalidate(undefined, true);
                } else {
                    refresh('m');
                }
            } else {
                invalidate(undefined, true);
            }
        };

        minutesInputEl.bind('blur', function (e) {
            if (!$scope.invalidMinutes && $scope.minutes < 10) {
                $scope.$apply(function () {
                    $scope.minutes = pad($scope.minutes);
                });
            }
        });

    };

    this.render = function () {
        var date = ngModelCtrl.$viewValue;

        if (isNaN(date)) {
            ngModelCtrl.$setValidity('time', false);
            $log.error('Timepicker directive: "ng-model" value must be a Date object, a number of milliseconds since 01.01.1970 or a string representing an RFC2822 or ISO 8601 date.');
        } else {
            if (date) {
                selected = date;
            }

            if (selected < min || selected > max) {
                ngModelCtrl.$setValidity('time', false);
                $scope.invalidHours = true;
                $scope.invalidMinutes = true;
            } else {
                makeValid();
            }
            updateTemplate();
        }
    };

    // Call internally when we know that model is valid.
    function refresh(keyboardChange) {
        makeValid();
        ngModelCtrl.$setViewValue(new Date(selected));
        updateTemplate(keyboardChange);
    }

    function makeValid() {
        ngModelCtrl.$setValidity('time', true);
        $scope.invalidHours = false;
        $scope.invalidMinutes = false;
    }

    function updateTemplate(keyboardChange) {
        var hours = selected.getHours(), minutes = selected.getMinutes();

        if ($scope.showMeridian) {
            hours = (hours === 0 || hours === 12) ? 12 : hours % 12; // Convert 24 to 12 hour system
        }

        $scope.hours = keyboardChange === 'h' ? hours : pad(hours);
        if (keyboardChange !== 'm') {
            $scope.minutes = pad(minutes);
        }
        $scope.meridian = selected.getHours() < 12 ? meridians[0] : meridians[1];
    }

    function addMinutes(date, minutes) {
        var dt = new Date(date.getTime() + minutes * 60000);
        var newDate = new Date(date);
        newDate.setHours(dt.getHours(), dt.getMinutes());
        return newDate;
    }

    function addMinutesToSelected(minutes) {
        selected = addMinutes(selected, minutes);
        refresh();
    }

    $scope.showSpinners = angular.isDefined($attrs.showSpinners) ?
      $scope.$parent.$eval($attrs.showSpinners) : timepickerConfig.showSpinners;

    $scope.incrementHours = function () {
        if (!$scope.noIncrementHours()) {
            addMinutesToSelected(hourStep * 60);
        }
    };

    $scope.decrementHours = function () {
        if (!$scope.noDecrementHours()) {
            addMinutesToSelected(-hourStep * 60);
        }
    };

    $scope.incrementMinutes = function () {
        if (!$scope.noIncrementMinutes()) {
            addMinutesToSelected(minuteStep);
        }
    };

    $scope.decrementMinutes = function () {
        if (!$scope.noDecrementMinutes()) {
            addMinutesToSelected(-minuteStep);
        }
    };

    $scope.toggleMeridian = function () {
        if (!$scope.noToggleMeridian()) {
            addMinutesToSelected(12 * 60 * (selected.getHours() < 12 ? 1 : -1));
        }
    };
}])

.directive('uibTimepicker', function () {
    return {
        restrict: 'EA',
        require: ['uibTimepicker', '?^ngModel'],
        controller: 'UibTimepickerController',
        controllerAs: 'timepicker',
        replace: true,
        scope: {},
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/timepicker/timepicker.html';
        },
        link: function (scope, element, attrs, ctrls) {
            var timepickerCtrl = ctrls[0], ngModelCtrl = ctrls[1];

            if (ngModelCtrl) {
                timepickerCtrl.init(ngModelCtrl, element.find('input'));
            }
        }
    };
});

/* Deprecated timepicker below */

angular.module('ui.bootstrap.timepicker')

.value('$timepickerSuppressWarning', false)

.controller('TimepickerController', ['$scope', '$element', '$attrs', '$controller', '$log', '$timepickerSuppressWarning', function ($scope, $element, $attrs, $controller, $log, $timepickerSuppressWarning) {
    if (!$timepickerSuppressWarning) {
        $log.warn('TimepickerController is now deprecated. Use UibTimepickerController instead.');
    }

    angular.extend(this, $controller('UibTimepickerController', {
        $scope: $scope,
        $element: $element,
        $attrs: $attrs
    }));
}])

.directive('timepicker', ['$log', '$timepickerSuppressWarning', function ($log, $timepickerSuppressWarning) {
    return {
        restrict: 'EA',
        require: ['timepicker', '?^ngModel'],
        controller: 'TimepickerController',
        controllerAs: 'timepicker',
        replace: true,
        scope: {},
        templateUrl: function (element, attrs) {
            return attrs.templateUrl || 'template/timepicker/timepicker.html';
        },
        link: function (scope, element, attrs, ctrls) {
            if (!$timepickerSuppressWarning) {
                $log.warn('timepicker is now deprecated. Use uib-timepicker instead.');
            }
            var timepickerCtrl = ctrls[0], ngModelCtrl = ctrls[1];

            if (ngModelCtrl) {
                timepickerCtrl.init(ngModelCtrl, element.find('input'));
            }
        }
    };
}]);

angular.module("template/tooltip/tooltip-html-popup.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/tooltip/tooltip-html-popup.html",
      "<div\n" +
      "  tooltip-animation-class=\"fade\"\n" +
      "  uib-tooltip-classes\n" +
      "  ng-class=\"{ in: isOpen() }\">\n" +
      "  <div class=\"tooltip-arrow\"></div>\n" +
      "  <div class=\"tooltip-inner\" ng-bind-html=\"contentExp()\"></div>\n" +
      "</div>\n" +
      "");
}]);

angular.module("template/tooltip/tooltip-popup.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/tooltip/tooltip-popup.html",
      "<div\n" +
      "  tooltip-animation-class=\"fade\"\n" +
      "  uib-tooltip-classes\n" +
      "  ng-class=\"{ in: isOpen() }\">\n" +
      "  <div class=\"tooltip-arrow\"></div>\n" +
      "  <div class=\"tooltip-inner\" ng-bind=\"content\"></div>\n" +
      "</div>\n" +
      "");
}]);

angular.module("template/tooltip/tooltip-template-popup.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/tooltip/tooltip-template-popup.html",
      "<div\n" +
      "  tooltip-animation-class=\"fade\"\n" +
      "  uib-tooltip-classes\n" +
      "  ng-class=\"{ in: isOpen() }\">\n" +
      "  <div class=\"tooltip-arrow\"></div>\n" +
      "  <div class=\"tooltip-inner\"\n" +
      "    uib-tooltip-template-transclude=\"contentExp()\"\n" +
      "    tooltip-template-transclude-scope=\"originScope()\"></div>\n" +
      "</div>\n" +
      "");
}]);

angular.module("template/rating/rating.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/rating/rating.html",
      "<span ng-mouseleave=\"reset()\" ng-keydown=\"onKeydown($event)\" tabindex=\"0\" role=\"slider\" aria-valuemin=\"0\" aria-valuemax=\"{{range.length}}\" aria-valuenow=\"{{value}}\">\n" +
      "    <span ng-repeat-start=\"r in range track by $index\" class=\"sr-only\">({{ $index < value ? '*' : ' ' }})</span>\n" +
      "    <i ng-repeat-end ng-mouseenter=\"enter($index + 1)\" ng-click=\"rate($index + 1)\" class=\"glyphicon\" ng-class=\"$index < value && (r.stateOn || 'glyphicon-star') || (r.stateOff || 'glyphicon-star-empty')\" ng-attr-title=\"{{r.title}}\" aria-valuetext=\"{{r.title}}\"></i>\n" +
      "</span>\n" +
      "");
}]);

angular.module("template/typeahead/typeahead-match.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/typeahead/typeahead-match.html",
      "<a href tabindex=\"-1\" ng-bind-html=\"match.label | uibTypeaheadHighlight:query\"></a>\n" +
      "");
}]);

angular.module("template/typeahead/typeahead-popup.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/typeahead/typeahead-popup.html",
      "<ul class=\"dropdown-menu\" ng-show=\"isOpen() && !moveInProgress\" ng-style=\"{top: position().top+'px', left: position().left+'px'}\" style=\"display: block;\" role=\"listbox\" aria-hidden=\"{{!isOpen()}}\">\n" +
      "    <li ng-repeat=\"match in matches track by $index\" ng-class=\"{active: isActive($index) }\" ng-mouseenter=\"selectActive($index)\" ng-click=\"selectMatch($index)\" role=\"option\" id=\"{{::match.id}}\">\n" +
      "        <div uib-typeahead-match index=\"$index\" match=\"match\" query=\"query\" template-url=\"templateUrl\"></div>\n" +
      "    </li>\n" +
      "</ul>\n" +
      "");
}]);

angular.module("template/tabs/tab.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/tabs/tab.html",
      "<li ng-class=\"{active: active, disabled: disabled}\">\n" +
      "  <a href ng-click=\"select()\" uib-tab-heading-transclude>{{heading}}</a>\n" +
      "</li>\n" +
      "");
}]);

angular.module("template/tabs/tabset.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/tabs/tabset.html",
      "<div>\n" +
      "  <ul class=\"nav nav-{{type || 'tabs'}}\" ng-class=\"{'nav-stacked': vertical, 'nav-justified': justified}\" ng-transclude></ul>\n" +
      "  <div class=\"tab-content\">\n" +
      "    <div class=\"tab-pane\" \n" +
      "         ng-repeat=\"tab in tabs\" \n" +
      "         ng-class=\"{active: tab.active}\"\n" +
      "         uib-tab-content-transclude=\"tab\">\n" +
      "    </div>\n" +
      "  </div>\n" +
      "</div>\n" +
      "");
}]);

angular.module("template/popover/popover-html.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/popover/popover-html.html",
      "<div tooltip-animation-class=\"fade\"\n" +
      "  uib-tooltip-classes\n" +
      "  ng-class=\"{ in: isOpen() }\">\n" +
      "  <div class=\"arrow\"></div>\n" +
      "\n" +
      "  <div class=\"popover-inner\">\n" +
      "      <h3 class=\"popover-title\" ng-bind=\"title\" ng-if=\"title\"></h3>\n" +
      "      <div class=\"popover-content\" ng-bind-html=\"contentExp()\"></div>\n" +
      "  </div>\n" +
      "</div>\n" +
      "");
}]);

angular.module("template/popover/popover-template.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/popover/popover-template.html",
      "<div tooltip-animation-class=\"fade\"\n" +
      "  uib-tooltip-classes\n" +
      "  ng-class=\"{ in: isOpen() }\">\n" +
      "  <div class=\"arrow\"></div>\n" +
      "\n" +
      "  <div class=\"popover-inner\">\n" +
      "      <h3 class=\"popover-title\" ng-bind=\"title\" ng-if=\"title\"></h3>\n" +
      "      <div class=\"popover-content\"\n" +
      "        uib-tooltip-template-transclude=\"contentExp()\"\n" +
      "        tooltip-template-transclude-scope=\"originScope()\"></div>\n" +
      "  </div>\n" +
      "</div>\n" +
      "");
}]);

angular.module("template/popover/popover.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/popover/popover.html",
      "<div tooltip-animation-class=\"fade\"\n" +
      "  uib-tooltip-classes\n" +
      "  ng-class=\"{ in: isOpen() }\">\n" +
      "  <div class=\"arrow\"></div>\n" +
      "\n" +
      "  <div class=\"popover-inner\">\n" +
      "      <h3 class=\"popover-title\" ng-bind=\"title\" ng-if=\"title\"></h3>\n" +
      "      <div class=\"popover-content\" ng-bind=\"content\"></div>\n" +
      "  </div>\n" +
      "</div>\n" +
      "");
}]);

angular.module("template/modal/backdrop.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/modal/backdrop.html",
      "<div uib-modal-animation-class=\"fade\"\n" +
      "     modal-in-class=\"in\"\n" +
      "     ng-style=\"{'z-index': 1040 + (index && 1 || 0) + index*10}\"\n" +
      "></div>\n" +
      "");
}]);

angular.module("template/modal/window.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/modal/window.html",
      "<div modal-render=\"{{$isRendered}}\" tabindex=\"-1\" role=\"dialog\" class=\"modal\"\n" +
      "    uib-modal-animation-class=\"fade\"\n" +
      "    modal-in-class=\"in\"\n" +
      "    ng-style=\"{'z-index': 1050 + index*10, display: 'block'}\">\n" +
      "    <div class=\"modal-dialog\" ng-class=\"size ? 'modal-' + size : ''\"><div class=\"modal-content\" uib-modal-transclude></div></div>\n" +
      "</div>\n" +
      "");
}]);

angular.module("template/datepicker/datepicker.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/datepicker/datepicker.html",
      "<div ng-switch=\"datepickerMode\" role=\"application\" ng-keydown=\"keydown($event)\">\n" +
      "  <uib-daypicker ng-switch-when=\"day\" tabindex=\"0\"></uib-daypicker>\n" +
      "  <uib-monthpicker ng-switch-when=\"month\" tabindex=\"0\"></uib-monthpicker>\n" +
      "  <uib-yearpicker ng-switch-when=\"year\" tabindex=\"0\"></uib-yearpicker>\n" +
      "</div>");
}]);

angular.module("template/datepicker/day.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/datepicker/day.html",
      "<table role=\"grid\" aria-labelledby=\"{{::uniqueId}}-title\" aria-activedescendant=\"{{activeDateId}}\">\n" +
      "  <thead>\n" +
      "    <tr>\n" +
      "      <th><button type=\"button\" class=\"btn btn-default btn-sm pull-left\" ng-click=\"move(-1)\" tabindex=\"-1\"><i class=\"glyphicon glyphicon-chevron-left\"></i></button></th>\n" +
      "      <th colspan=\"{{::5 + showWeeks}}\"><button id=\"{{::uniqueId}}-title\" role=\"heading\" aria-live=\"assertive\" aria-atomic=\"true\" type=\"button\" class=\"btn btn-default btn-sm\" ng-click=\"toggleMode()\" ng-disabled=\"datepickerMode === maxMode\" tabindex=\"-1\" style=\"width:100%;\"><strong>{{title}}</strong></button></th>\n" +
      "      <th><button type=\"button\" class=\"btn btn-default btn-sm pull-right\" ng-click=\"move(1)\" tabindex=\"-1\"><i class=\"glyphicon glyphicon-chevron-right\"></i></button></th>\n" +
      "    </tr>\n" +
      "    <tr>\n" +
      "      <th ng-if=\"showWeeks\" class=\"text-center\"></th>\n" +
      "      <th ng-repeat=\"label in ::labels track by $index\" class=\"text-center\"><small aria-label=\"{{::label.full}}\">{{::label.abbr}}</small></th>\n" +
      "    </tr>\n" +
      "  </thead>\n" +
      "  <tbody>\n" +
      "    <tr ng-repeat=\"row in rows track by $index\">\n" +
      "      <td ng-if=\"showWeeks\" class=\"text-center h6\"><em>{{ weekNumbers[$index] }}</em></td>\n" +
      "      <td ng-repeat=\"dt in row track by dt.date\" class=\"text-center\" role=\"gridcell\" id=\"{{::dt.uid}}\" ng-class=\"::dt.customClass\">\n" +
      "        <button type=\"button\" style=\"min-width:100%;\" class=\"btn btn-default btn-sm\" ng-class=\"{'btn-info': dt.selected, active: isActive(dt)}\" ng-click=\"select(dt.date)\" ng-disabled=\"dt.disabled\" tabindex=\"-1\"><span ng-class=\"::{'text-muted': dt.secondary, 'text-info': dt.current}\">{{::dt.label}}</span></button>\n" +
      "      </td>\n" +
      "    </tr>\n" +
      "  </tbody>\n" +
      "</table>\n" +
      "");
}]);

angular.module("template/datepicker/month.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/datepicker/month.html",
      "<table role=\"grid\" aria-labelledby=\"{{::uniqueId}}-title\" aria-activedescendant=\"{{activeDateId}}\">\n" +
      "  <thead>\n" +
      "    <tr>\n" +
      "      <th><button type=\"button\" class=\"btn btn-default btn-sm pull-left\" ng-click=\"move(-1)\" tabindex=\"-1\"><i class=\"glyphicon glyphicon-chevron-left\"></i></button></th>\n" +
      "      <th><button id=\"{{::uniqueId}}-title\" role=\"heading\" aria-live=\"assertive\" aria-atomic=\"true\" type=\"button\" class=\"btn btn-default btn-sm\" ng-click=\"toggleMode()\" ng-disabled=\"datepickerMode === maxMode\" tabindex=\"-1\" style=\"width:100%;\"><strong>{{title}}</strong></button></th>\n" +
      "      <th><button type=\"button\" class=\"btn btn-default btn-sm pull-right\" ng-click=\"move(1)\" tabindex=\"-1\"><i class=\"glyphicon glyphicon-chevron-right\"></i></button></th>\n" +
      "    </tr>\n" +
      "  </thead>\n" +
      "  <tbody>\n" +
      "    <tr ng-repeat=\"row in rows track by $index\">\n" +
      "      <td ng-repeat=\"dt in row track by dt.date\" class=\"text-center\" role=\"gridcell\" id=\"{{::dt.uid}}\" ng-class=\"::dt.customClass\">\n" +
      "        <button type=\"button\" style=\"min-width:100%;\" class=\"btn btn-default\" ng-class=\"{'btn-info': dt.selected, active: isActive(dt)}\" ng-click=\"select(dt.date)\" ng-disabled=\"dt.disabled\" tabindex=\"-1\"><span ng-class=\"::{'text-info': dt.current}\">{{::dt.label}}</span></button>\n" +
      "      </td>\n" +
      "    </tr>\n" +
      "  </tbody>\n" +
      "</table>\n" +
      "");
}]);

angular.module("template/datepicker/popup.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/datepicker/popup.html",
      "<ul class=\"dropdown-menu\" dropdown-nested ng-if=\"isOpen\" style=\"display: block\" ng-style=\"{top: position.top+'px', left: position.left+'px'}\" ng-keydown=\"keydown($event)\" ng-click=\"$event.stopPropagation()\">\n" +
      "	<li ng-transclude></li>\n" +
      "	<li ng-if=\"showButtonBar\" style=\"padding:10px 9px 2px\">\n" +
      "		<span class=\"btn-group pull-left\">\n" +
      "			<button type=\"button\" class=\"btn btn-sm btn-info\" ng-click=\"select('today')\" ng-disabled=\"isDisabled('today')\">{{ getText('current') }}</button>\n" +
      "			<button type=\"button\" class=\"btn btn-sm btn-danger\" ng-click=\"select(null)\">{{ getText('clear') }}</button>\n" +
      "		</span>\n" +
      "		<button type=\"button\" class=\"btn btn-sm btn-success pull-right\" ng-click=\"close()\">{{ getText('close') }}</button>\n" +
      "	</li>\n" +
      "</ul>\n" +
      "");
}]);

angular.module("template/datepicker/year.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/datepicker/year.html",
      "<table role=\"grid\" aria-labelledby=\"{{::uniqueId}}-title\" aria-activedescendant=\"{{activeDateId}}\">\n" +
      "  <thead>\n" +
      "    <tr>\n" +
      "      <th><button type=\"button\" class=\"btn btn-default btn-sm pull-left\" ng-click=\"move(-1)\" tabindex=\"-1\"><i class=\"glyphicon glyphicon-chevron-left\"></i></button></th>\n" +
      "      <th colspan=\"3\"><button id=\"{{::uniqueId}}-title\" role=\"heading\" aria-live=\"assertive\" aria-atomic=\"true\" type=\"button\" class=\"btn btn-default btn-sm\" ng-click=\"toggleMode()\" ng-disabled=\"datepickerMode === maxMode\" tabindex=\"-1\" style=\"width:100%;\"><strong>{{title}}</strong></button></th>\n" +
      "      <th><button type=\"button\" class=\"btn btn-default btn-sm pull-right\" ng-click=\"move(1)\" tabindex=\"-1\"><i class=\"glyphicon glyphicon-chevron-right\"></i></button></th>\n" +
      "    </tr>\n" +
      "  </thead>\n" +
      "  <tbody>\n" +
      "    <tr ng-repeat=\"row in rows track by $index\">\n" +
      "      <td ng-repeat=\"dt in row track by dt.date\" class=\"text-center\" role=\"gridcell\" id=\"{{::dt.uid}}\" ng-class=\"::dt.customClass\">\n" +
      "        <button type=\"button\" style=\"min-width:100%;\" class=\"btn btn-default\" ng-class=\"{'btn-info': dt.selected, active: isActive(dt)}\" ng-click=\"select(dt.date)\" ng-disabled=\"dt.disabled\" tabindex=\"-1\"><span ng-class=\"::{'text-info': dt.current}\">{{::dt.label}}</span></button>\n" +
      "      </td>\n" +
      "    </tr>\n" +
      "  </tbody>\n" +
      "</table>\n" +
      "");
}]);

angular.module("template/accordion/accordion-group.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/accordion/accordion-group.html",
      "<div class=\"panel {{panelClass || 'panel-default'}}\">\n" +
      "  <div class=\"panel-heading\" ng-keypress=\"toggleOpen($event)\">\n" +
      "    <h4 class=\"panel-title\">\n" +
      "      <a href tabindex=\"0\" class=\"accordion-toggle\" ng-click=\"toggleOpen()\" uib-accordion-transclude=\"heading\"><span ng-class=\"{'text-muted': isDisabled}\">{{heading}}</span></a>\n" +
      "    </h4>\n" +
      "  </div>\n" +
      "  <div class=\"panel-collapse collapse\" uib-collapse=\"!isOpen\">\n" +
      "	  <div class=\"panel-body\" ng-transclude></div>\n" +
      "  </div>\n" +
      "</div>\n" +
      "");
}]);

angular.module("template/accordion/accordion.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/accordion/accordion.html",
      "<div class=\"panel-group\" ng-transclude></div>");
}]);

angular.module("template/alert/alert.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/alert/alert.html",
      "<div class=\"alert\" ng-class=\"['alert-' + (type || 'warning'), closeable ? 'alert-dismissible' : null]\" role=\"alert\">\n" +
      "    <button ng-show=\"closeable\" type=\"button\" class=\"close\" ng-click=\"close({$event: $event})\">\n" +
      "        <span class=\"close\" aria-hidden=\"true\">&times;</span>\n" +
      "        <span class=\"close sr-only\">Close</span>\n" +
      "    </button>\n" +
      "    <div ng-transclude></div>\n" +
      "</div>\n" +
      "");
}]);

angular.module("template/pagination/pager.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/pagination/pager.html",
      "<ul class=\"pager\">\n" +
      "  <li ng-class=\"{disabled: noPrevious()||ngDisabled, previous: align}\"><a href ng-click=\"selectPage(page - 1, $event)\">{{::getText('previous')}}</a></li>\n" +
      "  <li ng-class=\"{disabled: noNext()||ngDisabled, next: align}\"><a href ng-click=\"selectPage(page + 1, $event)\">{{::getText('next')}}</a></li>\n" +
      "</ul>\n" +
      "");
}]);

angular.module("template/pagination/pagination.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/pagination/pagination.html",
      "<ul class=\"pagination\">\n" +
      "  <li ng-if=\"::boundaryLinks\" ng-class=\"{disabled: noPrevious()||ngDisabled}\" class=\"pagination-first\"><a href ng-click=\"selectPage(1, $event)\">{{::getText('first')}}</a></li>\n" +
      "  <li ng-if=\"::directionLinks\" ng-class=\"{disabled: noPrevious()||ngDisabled}\" class=\"pagination-prev\"><a href ng-click=\"selectPage(page - 1, $event)\">{{::getText('previous')}}</a></li>\n" +
      "  <li ng-repeat=\"page in pages track by $index\" ng-class=\"{active: page.active,disabled: ngDisabled&&!page.active}\" class=\"pagination-page\"><a href ng-click=\"selectPage(page.number, $event)\">{{page.text}}</a></li>\n" +
      "  <li ng-if=\"::directionLinks\" ng-class=\"{disabled: noNext()||ngDisabled}\" class=\"pagination-next\"><a href ng-click=\"selectPage(page + 1, $event)\">{{::getText('next')}}</a></li>\n" +
      "  <li ng-if=\"::boundaryLinks\" ng-class=\"{disabled: noNext()||ngDisabled}\" class=\"pagination-last\"><a href ng-click=\"selectPage(totalPages, $event)\">{{::getText('last')}}</a></li>\n" +
      "</ul>\n" +
      "");
}]);

angular.module("template/progressbar/bar.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/progressbar/bar.html",
      "<div class=\"progress-bar\" ng-class=\"type && 'progress-bar-' + type\" role=\"progressbar\" aria-valuenow=\"{{value}}\" aria-valuemin=\"0\" aria-valuemax=\"{{max}}\" ng-style=\"{width: (percent < 100 ? percent : 100) + '%'}\" aria-valuetext=\"{{percent | number:0}}%\" aria-labelledby=\"{{::title}}\" style=\"min-width: 0;\" ng-transclude></div>\n" +
      "");
}]);

angular.module("template/progressbar/progress.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/progressbar/progress.html",
      "<div class=\"progress\" ng-transclude aria-labelledby=\"{{::title}}\"></div>");
}]);

angular.module("template/progressbar/progressbar.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/progressbar/progressbar.html",
      "<div class=\"progress\">\n" +
      "  <div class=\"progress-bar\" ng-class=\"type && 'progress-bar-' + type\" role=\"progressbar\" aria-valuenow=\"{{value}}\" aria-valuemin=\"0\" aria-valuemax=\"{{max}}\" ng-style=\"{width: (percent < 100 ? percent : 100) + '%'}\" aria-valuetext=\"{{percent | number:0}}%\" aria-labelledby=\"{{::title}}\" style=\"min-width: 0;\" ng-transclude></div>\n" +
      "</div>\n" +
      "");
}]);

angular.module("template/timepicker/timepicker.html", []).run(["$templateCache", function ($templateCache) {
    $templateCache.put("template/timepicker/timepicker.html",
      "<table>\n" +
      "  <tbody>\n" +
      "    <tr class=\"text-center\" ng-show=\"::showSpinners\">\n" +
      "      <td><a ng-click=\"incrementHours()\" ng-class=\"{disabled: noIncrementHours()}\" class=\"btn btn-link\" ng-disabled=\"noIncrementHours()\" tabindex=\"{{::tabindex}}\"><span class=\"glyphicon glyphicon-chevron-up\"></span></a></td>\n" +
      "      <td>&nbsp;</td>\n" +
      "      <td><a ng-click=\"incrementMinutes()\" ng-class=\"{disabled: noIncrementMinutes()}\" class=\"btn btn-link\" ng-disabled=\"noIncrementMinutes()\" tabindex=\"{{::tabindex}}\"><span class=\"glyphicon glyphicon-chevron-up\"></span></a></td>\n" +
      "      <td ng-show=\"showMeridian\"></td>\n" +
      "    </tr>\n" +
      "    <tr>\n" +
      "      <td class=\"form-group\" ng-class=\"{'has-error': invalidHours}\">\n" +
      "        <input style=\"width:50px;\" type=\"text\" ng-model=\"hours\" ng-change=\"updateHours()\" class=\"form-control text-center\" ng-readonly=\"::readonlyInput\" maxlength=\"2\" tabindex=\"{{::tabindex}}\">\n" +
      "      </td>\n" +
      "      <td>:</td>\n" +
      "      <td class=\"form-group\" ng-class=\"{'has-error': invalidMinutes}\">\n" +
      "        <input style=\"width:50px;\" type=\"text\" ng-model=\"minutes\" ng-change=\"updateMinutes()\" class=\"form-control text-center\" ng-readonly=\"::readonlyInput\" maxlength=\"2\" tabindex=\"{{::tabindex}}\">\n" +
      "      </td>\n" +
      "      <td ng-show=\"showMeridian\"><button type=\"button\" ng-class=\"{disabled: noToggleMeridian()}\" class=\"btn btn-default text-center\" ng-click=\"toggleMeridian()\" ng-disabled=\"noToggleMeridian()\" tabindex=\"{{::tabindex}}\">{{meridian}}</button></td>\n" +
      "    </tr>\n" +
      "    <tr class=\"text-center\" ng-show=\"::showSpinners\">\n" +
      "      <td><a ng-click=\"decrementHours()\" ng-class=\"{disabled: noDecrementHours()}\" class=\"btn btn-link\" ng-disabled=\"noDecrementHours()\" tabindex=\"{{::tabindex}}\"><span class=\"glyphicon glyphicon-chevron-down\"></span></a></td>\n" +
      "      <td>&nbsp;</td>\n" +
      "      <td><a ng-click=\"decrementMinutes()\" ng-class=\"{disabled: noDecrementMinutes()}\" class=\"btn btn-link\" ng-disabled=\"noDecrementMinutes()\" tabindex=\"{{::tabindex}}\"><span class=\"glyphicon glyphicon-chevron-down\"></span></a></td>\n" +
      "      <td ng-show=\"showMeridian\"></td>\n" +
      "    </tr>\n" +
      "  </tbody>\n" +
      "</table>\n" +
      "");
}]);
;
/* ImageMapster
   Version: 1.2.14-beta1 (6/18/2013)

Copyright 2011-2012 James Treworgy

http://www.outsharked.com/imagemapster
https://github.com/jamietre/ImageMapster

A jQuery plugin to enhance image maps.

*/

;

/// LICENSE (MIT License)
///
/// Permission is hereby granted, free of charge, to any person obtaining
/// a copy of this software and associated documentation files (the
/// "Software"), to deal in the Software without restriction, including
/// without limitation the rights to use, copy, modify, merge, publish,
/// distribute, sublicense, and/or sell copies of the Software, and to
/// permit persons to whom the Software is furnished to do so, subject to
/// the following conditions:
///
/// The above copyright notice and this permission notice shall be
/// included in all copies or substantial portions of the Software.
///
/// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
/// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
/// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
/// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
/// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
/// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
/// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
///
/// January 19, 2011

/** @license MIT License (c) copyright B Cavalier & J Hann */

/**
* when
* A lightweight CommonJS Promises/A and when() implementation
*
* when is part of the cujo.js family of libraries (http://cujojs.com/)
*
* Licensed under the MIT License at:
* http://www.opensource.org/licenses/mit-license.php
*
* @version 1.2.0
*/

/*lint-ignore-start*/

(function (define) {
    define(function () {
        var freeze, reduceArray, slice, undef;

        //
        // Public API
        //

        when.defer = defer;
        when.reject = reject;
        when.isPromise = isPromise;

        when.all = all;
        when.some = some;
        when.any = any;

        when.map = map;
        when.reduce = reduce;

        when.chain = chain;

        /** Object.freeze */
        freeze = Object.freeze || function (o) { return o; };

        /**
        * Trusted Promise constructor.  A Promise created from this constructor is
        * a trusted when.js promise.  Any other duck-typed promise is considered
        * untrusted.
        *
        * @constructor
        */
        function Promise() { }

        Promise.prototype = freeze({
            always: function (alwaysback, progback) {
                return this.then(alwaysback, alwaysback, progback);
            },

            otherwise: function (errback) {
                return this.then(undef, errback);
            }
        });

        /**
        * Create an already-resolved promise for the supplied value
        * @private
        *
        * @param value anything
        * @return {Promise}
        */
        function resolved(value) {

            var p = new Promise();

            p.then = function (callback) {
                var nextValue;
                try {
                    if (callback) nextValue = callback(value);
                    return promise(nextValue === undef ? value : nextValue);
                } catch (e) {
                    return rejected(e);
                }
            };

            return freeze(p);
        }

        /**
        * Create an already-rejected {@link Promise} with the supplied
        * rejection reason.
        * @private
        *
        * @param reason rejection reason
        * @return {Promise}
        */
        function rejected(reason) {

            var p = new Promise();

            p.then = function (callback, errback) {
                var nextValue;
                try {
                    if (errback) {
                        nextValue = errback(reason);
                        return promise(nextValue === undef ? reason : nextValue)
                    }

                    return rejected(reason);

                } catch (e) {
                    return rejected(e);
                }
            };

            return freeze(p);
        }

        /**
        * Returns a rejected promise for the supplied promiseOrValue. If
        * promiseOrValue is a value, it will be the rejection value of the
        * returned promise.  If promiseOrValue is a promise, its
        * completion value will be the rejected value of the returned promise
        *
        * @param promiseOrValue {*} the rejected value of the returned {@link Promise}
        *
        * @return {Promise} rejected {@link Promise}
        */
        function reject(promiseOrValue) {
            return when(promiseOrValue, function (value) {
                return rejected(value);
            });
        }

        /**
        * Creates a new, CommonJS compliant, Deferred with fully isolated
        * resolver and promise parts, either or both of which may be given out
        * safely to consumers.
        * The Deferred itself has the full API: resolve, reject, progress, and
        * then. The resolver has resolve, reject, and progress.  The promise
        * only has then.
        *
        * @memberOf when
        * @function
        *
        * @returns {Deferred}
        */
        function defer() {
            var deferred, promise, listeners, progressHandlers, _then, _progress, complete;

            listeners = [];
            progressHandlers = [];

            /**
            * Pre-resolution then() that adds the supplied callback, errback, and progback
            * functions to the registered listeners
            *
            * @private
            *
            * @param [callback] {Function} resolution handler
            * @param [errback] {Function} rejection handler
            * @param [progback] {Function} progress handler
            *
            * @throws {Error} if any argument is not null, undefined, or a Function
            */
            _then = function unresolvedThen(callback, errback, progback) {
                var deferred = defer();

                listeners.push(function (promise) {
                    promise.then(callback, errback)
					.then(deferred.resolve, deferred.reject, deferred.progress);
                });

                progback && progressHandlers.push(progback);

                return deferred.promise;
            };

            /**
            * Registers a handler for this {@link Deferred}'s {@link Promise}.  Even though all arguments
            * are optional, each argument that *is* supplied must be null, undefined, or a Function.
            * Any other value will cause an Error to be thrown.
            *
            * @memberOf Promise
            *
            * @param [callback] {Function} resolution handler
            * @param [errback] {Function} rejection handler
            * @param [progback] {Function} progress handler
            *
            * @throws {Error} if any argument is not null, undefined, or a Function
            */
            function then(callback, errback, progback) {
                return _then(callback, errback, progback);
            }

            /**
            * Resolves this {@link Deferred}'s {@link Promise} with val as the
            * resolution value.
            *
            * @memberOf Resolver
            *
            * @param val anything
            */
            function resolve(val) {
                complete(resolved(val));
            }

            /**
            * Rejects this {@link Deferred}'s {@link Promise} with err as the
            * reason.
            *
            * @memberOf Resolver
            *
            * @param err anything
            */
            function reject(err) {
                complete(rejected(err));
            }

            /**
            * @private
            * @param update
            */
            _progress = function (update) {
                var progress, i = 0;
                while (progress = progressHandlers[i++]) progress(update);
            };

            /**
            * Emits a progress update to all progress observers registered with
            * this {@link Deferred}'s {@link Promise}
            *
            * @memberOf Resolver
            *
            * @param update anything
            */
            function progress(update) {
                _progress(update);
            }

            /**
            * Transition from pre-resolution state to post-resolution state, notifying
            * all listeners of the resolution or rejection
            *
            * @private
            *
            * @param completed {Promise} the completed value of this deferred
            */
            complete = function (completed) {
                var listener, i = 0;

                // Replace _then with one that directly notifies with the result.
                _then = completed.then;

                // Replace complete so that this Deferred can only be completed
                // once. Also Replace _progress, so that subsequent attempts to issue
                // progress throw.
                complete = _progress = function alreadyCompleted() {
                    // TODO: Consider silently returning here so that parties who
                    // have a reference to the resolver cannot tell that the promise
                    // has been resolved using try/catch
                    throw new Error("already completed");
                };

                // Free progressHandlers array since we'll never issue progress events
                // for this promise again now that it's completed
                progressHandlers = undef;

                // Notify listeners
                // Traverse all listeners registered directly with this Deferred

                while (listener = listeners[i++]) {
                    listener(completed);
                }

                listeners = [];
            };

            /**
            * The full Deferred object, with both {@link Promise} and {@link Resolver}
            * parts
            * @class Deferred
            * @name Deferred
            */
            deferred = {};

            // Promise and Resolver parts
            // Freeze Promise and Resolver APIs

            promise = new Promise();
            promise.then = deferred.then = then;

            /**
            * The {@link Promise} for this {@link Deferred}
            * @memberOf Deferred
            * @name promise
            * @type {Promise}
            */
            deferred.promise = freeze(promise);

            /**
            * The {@link Resolver} for this {@link Deferred}
            * @memberOf Deferred
            * @name resolver
            * @class Resolver
            */
            deferred.resolver = freeze({
                resolve: (deferred.resolve = resolve),
                reject: (deferred.reject = reject),
                progress: (deferred.progress = progress)
            });

            return deferred;
        }

        /**
        * Determines if promiseOrValue is a promise or not.  Uses the feature
        * test from http://wiki.commonjs.org/wiki/Promises/A to determine if
        * promiseOrValue is a promise.
        *
        * @param promiseOrValue anything
        *
        * @returns {Boolean} true if promiseOrValue is a {@link Promise}
        */
        function isPromise(promiseOrValue) {
            return promiseOrValue && typeof promiseOrValue.then === 'function';
        }

        /**
        * Register an observer for a promise or immediate value.
        *
        * @function
        * @name when
        * @namespace
        *
        * @param promiseOrValue anything
        * @param {Function} [callback] callback to be called when promiseOrValue is
        *   successfully resolved.  If promiseOrValue is an immediate value, callback
        *   will be invoked immediately.
        * @param {Function} [errback] callback to be called when promiseOrValue is
        *   rejected.
        * @param {Function} [progressHandler] callback to be called when progress updates
        *   are issued for promiseOrValue.
        *
        * @returns {Promise} a new {@link Promise} that will complete with the return
        *   value of callback or errback or the completion value of promiseOrValue if
        *   callback and/or errback is not supplied.
        */
        function when(promiseOrValue, callback, errback, progressHandler) {
            // Get a promise for the input promiseOrValue
            // See promise()
            var trustedPromise = promise(promiseOrValue);

            // Register promise handlers
            return trustedPromise.then(callback, errback, progressHandler);
        }

        /**
        * Returns promiseOrValue if promiseOrValue is a {@link Promise}, a new Promise if
        * promiseOrValue is a foreign promise, or a new, already-resolved {@link Promise}
        * whose resolution value is promiseOrValue if promiseOrValue is an immediate value.
        *
        * Note that this function is not safe to export since it will return its
        * input when promiseOrValue is a {@link Promise}
        *
        * @private
        *
        * @param promiseOrValue anything
        *
        * @returns Guaranteed to return a trusted Promise.  If promiseOrValue is a when.js {@link Promise}
        *   returns promiseOrValue, otherwise, returns a new, already-resolved, when.js {@link Promise}
        *   whose resolution value is:
        *   * the resolution value of promiseOrValue if it's a foreign promise, or
        *   * promiseOrValue if it's a value
        */
        function promise(promiseOrValue) {
            var promise, deferred;

            if (promiseOrValue instanceof Promise) {
                // It's a when.js promise, so we trust it
                promise = promiseOrValue;

            } else {
                // It's not a when.js promise.  Check to see if it's a foreign promise
                // or a value.

                deferred = defer();
                if (isPromise(promiseOrValue)) {
                    // It's a compliant promise, but we don't know where it came from,
                    // so we don't trust its implementation entirely.  Introduce a trusted
                    // middleman when.js promise

                    // IMPORTANT: This is the only place when.js should ever call .then() on
                    // an untrusted promise.
                    promiseOrValue.then(deferred.resolve, deferred.reject, deferred.progress);
                    promise = deferred.promise;

                } else {
                    // It's a value, not a promise.  Create an already-resolved promise
                    // for it.
                    deferred.resolve(promiseOrValue);
                    promise = deferred.promise;
                }
            }

            return promise;
        }

        /**
        * Return a promise that will resolve when howMany of the supplied promisesOrValues
        * have resolved. The resolution value of the returned promise will be an array of
        * length howMany containing the resolutions values of the triggering promisesOrValues.
        *
        * @memberOf when
        *
        * @param promisesOrValues {Array} array of anything, may contain a mix
        *      of {@link Promise}s and values
        * @param howMany
        * @param [callback]
        * @param [errback]
        * @param [progressHandler]
        *
        * @returns {Promise}
        */
        function some(promisesOrValues, howMany, callback, errback, progressHandler) {

            checkCallbacks(2, arguments);

            return when(promisesOrValues, function (promisesOrValues) {

                var toResolve, results, ret, deferred, resolver, rejecter, handleProgress, len, i;

                len = promisesOrValues.length >>> 0;

                toResolve = Math.max(0, Math.min(howMany, len));
                results = [];
                deferred = defer();
                ret = when(deferred, callback, errback, progressHandler);

                // Wrapper so that resolver can be replaced
                function resolve(val) {
                    resolver(val);
                }

                // Wrapper so that rejecter can be replaced
                function reject(err) {
                    rejecter(err);
                }

                // Wrapper so that progress can be replaced
                function progress(update) {
                    handleProgress(update);
                }

                function complete() {
                    resolver = rejecter = handleProgress = noop;
                }

                // No items in the input, resolve immediately
                if (!toResolve) {
                    deferred.resolve(results);

                } else {
                    // Resolver for promises.  Captures the value and resolves
                    // the returned promise when toResolve reaches zero.
                    // Overwrites resolver var with a noop once promise has
                    // be resolved to cover case where n < promises.length
                    resolver = function (val) {
                        // This orders the values based on promise resolution order
                        // Another strategy would be to use the original position of
                        // the corresponding promise.
                        results.push(val);

                        if (! --toResolve) {
                            complete();
                            deferred.resolve(results);
                        }
                    };

                    // Rejecter for promises.  Rejects returned promise
                    // immediately, and overwrites rejecter var with a noop
                    // once promise to cover case where n < promises.length.
                    // TODO: Consider rejecting only when N (or promises.length - N?)
                    // promises have been rejected instead of only one?
                    rejecter = function (err) {
                        complete();
                        deferred.reject(err);
                    };

                    handleProgress = deferred.progress;

                    // TODO: Replace while with forEach
                    for (i = 0; i < len; ++i) {
                        if (i in promisesOrValues) {
                            when(promisesOrValues[i], resolve, reject, progress);
                        }
                    }
                }

                return ret;
            });
        }

        /**
        * Return a promise that will resolve only once all the supplied promisesOrValues
        * have resolved. The resolution value of the returned promise will be an array
        * containing the resolution values of each of the promisesOrValues.
        *
        * @memberOf when
        *
        * @param promisesOrValues {Array|Promise} array of anything, may contain a mix
        *      of {@link Promise}s and values
        * @param [callback] {Function}
        * @param [errback] {Function}
        * @param [progressHandler] {Function}
        *
        * @returns {Promise}
        */
        function all(promisesOrValues, callback, errback, progressHandler) {

            checkCallbacks(1, arguments);

            return when(promisesOrValues, function (promisesOrValues) {
                return _reduce(promisesOrValues, reduceIntoArray, []);
            }).then(callback, errback, progressHandler);
        }

        function reduceIntoArray(current, val, i) {
            current[i] = val;
            return current;
        }

        /**
        * Return a promise that will resolve when any one of the supplied promisesOrValues
        * has resolved. The resolution value of the returned promise will be the resolution
        * value of the triggering promiseOrValue.
        *
        * @memberOf when
        *
        * @param promisesOrValues {Array|Promise} array of anything, may contain a mix
        *      of {@link Promise}s and values
        * @param [callback] {Function}
        * @param [errback] {Function}
        * @param [progressHandler] {Function}
        *
        * @returns {Promise}
        */
        function any(promisesOrValues, callback, errback, progressHandler) {

            function unwrapSingleResult(val) {
                return callback ? callback(val[0]) : val[0];
            }

            return some(promisesOrValues, 1, unwrapSingleResult, errback, progressHandler);
        }

        /**
        * Traditional map function, similar to `Array.prototype.map()`, but allows
        * input to contain {@link Promise}s and/or values, and mapFunc may return
        * either a value or a {@link Promise}
        *
        * @memberOf when
        *
        * @param promise {Array|Promise} array of anything, may contain a mix
        *      of {@link Promise}s and values
        * @param mapFunc {Function} mapping function mapFunc(value) which may return
        *      either a {@link Promise} or value
        *
        * @returns {Promise} a {@link Promise} that will resolve to an array containing
        *      the mapped output values.
        */
        function map(promise, mapFunc) {
            return when(promise, function (array) {
                return _map(array, mapFunc);
            });
        }

        /**
        * Private map helper to map an array of promises
        * @private
        *
        * @param promisesOrValues {Array}
        * @param mapFunc {Function}
        * @return {Promise}
        */
        function _map(promisesOrValues, mapFunc) {

            var results, len, i;

            // Since we know the resulting length, we can preallocate the results
            // array to avoid array expansions.
            len = promisesOrValues.length >>> 0;
            results = new Array(len);

            // Since mapFunc may be async, get all invocations of it into flight
            // asap, and then use reduce() to collect all the results
            for (i = 0; i < len; i++) {
                if (i in promisesOrValues)
                    results[i] = when(promisesOrValues[i], mapFunc);
            }

            // Could use all() here, but that would result in another array
            // being allocated, i.e. map() would end up allocating 2 arrays
            // of size len instead of just 1.  Since all() uses reduce()
            // anyway, avoid the additional allocation by calling reduce
            // directly.
            return _reduce(results, reduceIntoArray, results);
        }

        /**
        * Traditional reduce function, similar to `Array.prototype.reduce()`, but
        * input may contain {@link Promise}s and/or values, and reduceFunc
        * may return either a value or a {@link Promise}, *and* initialValue may
        * be a {@link Promise} for the starting value.
        *
        * @memberOf when
        *
        * @param promise {Array|Promise} array of anything, may contain a mix
        *      of {@link Promise}s and values.  May also be a {@link Promise} for
        *      an array.
        * @param reduceFunc {Function} reduce function reduce(currentValue, nextValue, index, total),
        *      where total is the total number of items being reduced, and will be the same
        *      in each call to reduceFunc.
        * @param initialValue starting value, or a {@link Promise} for the starting value
        *
        * @returns {Promise} that will resolve to the final reduced value
        */
        function reduce(promise, reduceFunc, initialValue) {
            var args = slice.call(arguments, 1);
            return when(promise, function (array) {
                return _reduce.apply(undef, [array].concat(args));
            });
        }

        /**
        * Private reduce to reduce an array of promises
        * @private
        *
        * @param promisesOrValues {Array}
        * @param reduceFunc {Function}
        * @param initialValue {*}
        * @return {Promise}
        */
        function _reduce(promisesOrValues, reduceFunc, initialValue) {

            var total, args;

            total = promisesOrValues.length;

            // Skip promisesOrValues, since it will be used as 'this' in the call
            // to the actual reduce engine below.

            // Wrap the supplied reduceFunc with one that handles promises and then
            // delegates to the supplied.

            args = [
			function (current, val, i) {
			    return when(current, function (c) {
			        return when(val, function (value) {
			            return reduceFunc(c, value, i, total);
			        });
			    });
			}
            ];

            if (arguments.length > 2) args.push(initialValue);

            return reduceArray.apply(promisesOrValues, args);
        }

        /**
        * Ensure that resolution of promiseOrValue will complete resolver with the completion
        * value of promiseOrValue, or instead with resolveValue if it is provided.
        *
        * @memberOf when
        *
        * @param promiseOrValue
        * @param resolver {Resolver}
        * @param [resolveValue] anything
        *
        * @returns {Promise}
        */
        function chain(promiseOrValue, resolver, resolveValue) {
            var useResolveValue = arguments.length > 2;

            return when(promiseOrValue,
			function (val) {
			    if (useResolveValue) val = resolveValue;
			    resolver.resolve(val);
			    return val;
			},
			function (e) {
			    resolver.reject(e);
			    return rejected(e);
			},
			resolver.progress
		);
        }

        //
        // Utility functions
        //

        /**
        * Helper that checks arrayOfCallbacks to ensure that each element is either
        * a function, or null or undefined.
        *
        * @private
        *
        * @param arrayOfCallbacks {Array} array to check
        * @throws {Error} if any element of arrayOfCallbacks is something other than
        * a Functions, null, or undefined.
        */
        function checkCallbacks(start, arrayOfCallbacks) {
            var arg, i = arrayOfCallbacks.length;
            while (i > start) {
                arg = arrayOfCallbacks[--i];
                if (arg != null && typeof arg != 'function') throw new Error('callback is not a function');
            }
        }

        /**
        * No-Op function used in method replacement
        * @private
        */
        function noop() { }

        slice = [].slice;

        // ES5 reduce implementation if native not available
        // See: http://es5.github.com/#x15.4.4.21 as there are many
        // specifics and edge cases.
        reduceArray = [].reduce ||
		function (reduceFunc /*, initialValue */) {
		    // ES5 dictates that reduce.length === 1

		    // This implementation deviates from ES5 spec in the following ways:
		    // 1. It does not check if reduceFunc is a Callable

		    var arr, args, reduced, len, i;

		    i = 0;
		    arr = Object(this);
		    len = arr.length >>> 0;
		    args = arguments;

		    // If no initialValue, use first item of array (we know length !== 0 here)
		    // and adjust i to start at second item
		    if (args.length <= 1) {
		        // Skip to the first real element in the array
		        for (; ;) {
		            if (i in arr) {
		                reduced = arr[i++];
		                break;
		            }

		            // If we reached the end of the array without finding any real
		            // elements, it's a TypeError
		            if (++i >= len) {
		                throw new TypeError();
		            }
		        }
		    } else {
		        // If initialValue provided, use it
		        reduced = args[1];
		    }

		    // Do the actual reduce
		    for (; i < len; ++i) {
		        // Skip holes
		        if (i in arr)
		            reduced = reduceFunc(reduced, arr[i], i, arr);
		    }

		    return reduced;
		};

        return when;
    });
})(typeof define == 'function'
	? define
	: function (factory) {
	    typeof module != 'undefined'
		? (module.exports = factory())
		: (jQuery.mapster_when = factory());
	}
// Boilerplate for AMD, Node, and browser global
);
/*lint-ignore-end*/
/* ImageMapster core */

/*jslint laxbreak: true, evil: true, unparam: true */

/*global jQuery: true, Zepto: true */


(function ($) {
    // all public functions in $.mapster.impl are methods
    $.fn.mapster = function (method) {
        var m = $.mapster.impl;
        if ($.isFunction(m[method])) {
            return m[method].apply(this, Array.prototype.slice.call(arguments, 1));
        } else if (typeof method === 'object' || !method) {
            return m.bind.apply(this, arguments);
        } else {
            $.error('Method ' + method + ' does not exist on jQuery.mapster');
        }
    };

    $.mapster = {
        version: "1.2.14-beta1",
        render_defaults: {
            isSelectable: true,
            isDeselectable: true,
            fade: false,
            fadeDuration: 150,
            fill: true,
            fillColor: '000000',
            fillColorMask: 'FFFFFF',
            fillOpacity: 0.7,
            highlight: true,
            stroke: false,
            strokeColor: 'ff0000',
            strokeOpacity: 1,
            strokeWidth: 1,
            includeKeys: '',
            altImage: null,
            altImageId: null, // used internally            
            altImages: {}
        },
        defaults: {
            clickNavigate: false,
            wrapClass: null,
            wrapCss: null,
            onGetList: null,
            sortList: false,
            listenToList: false,
            mapKey: '',
            mapValue: '',
            singleSelect: false,
            listKey: 'value',
            listSelectedAttribute: 'selected',
            listSelectedClass: null,
            onClick: null,
            onMouseover: null,
            onMouseout: null,
            mouseoutDelay: 0,
            onStateChange: null,
            boundList: null,
            onConfigured: null,
            configTimeout: 30000,
            noHrefIsMask: true,
            scaleMap: true,
            safeLoad: false,
            areas: []
        },
        shared_defaults: {
            render_highlight: { fade: true },
            render_select: { fade: false },
            staticState: null,
            selected: null
        },
        area_defaults:
        {
            includeKeys: '',
            isMask: false
        },
        canvas_style: {
            position: 'absolute',
            left: 0,
            top: 0,
            padding: 0,
            border: 0
        },
        hasCanvas: null,
        isTouch: null,
        map_cache: [],
        hooks: {},
        addHook: function (name, callback) {
            this.hooks[name] = (this.hooks[name] || []).push(callback);
        },
        callHooks: function (name, context) {
            $.each(this.hooks[name] || [], function (i, e) {
                e.apply(context);
            });
        },
        utils: {
            when: $.mapster_when,
            defer: $.mapster_when.defer,

            // extends the constructor, returns a new object prototype. Does not refer to the
            // original constructor so is protected if the original object is altered. This way you
            // can "extend" an object by replacing it with its subclass.
            subclass: function (BaseClass, constr) {
                var Subclass = function () {
                    var me = this,
                        args = Array.prototype.slice.call(arguments, 0);
                    me.base = BaseClass.prototype;
                    me.base.init = function () {
                        BaseClass.prototype.constructor.apply(me, args);
                    };
                    constr.apply(me, args);
                };
                Subclass.prototype = new BaseClass();
                Subclass.prototype.constructor = Subclass;
                return Subclass;
            },
            asArray: function (obj) {
                return obj.constructor === Array ?
                    obj : this.split(obj);
            },
            // clean split: no padding or empty elements
            split: function (text, cb) {
                var i, el, arr = text.split(',');
                for (i = 0; i < arr.length; i++) {
                    el = $.trim(arr[i]);
                    if (el === '') {
                        arr.splice(i, 1);
                    } else {
                        arr[i] = cb ? cb(el) : el;
                    }
                }
                return arr;
            },
            // similar to $.extend but does not add properties (only updates), unless the
            // first argument is an empty object, then all properties will be copied
            updateProps: function (_target, _template) {
                var onlyProps,
                    target = _target || {},
                    template = $.isEmptyObject(target) ? _template : _target;

                //if (template) {
                onlyProps = [];
                $.each(template, function (prop) {
                    onlyProps.push(prop);
                });
                //}

                $.each(Array.prototype.slice.call(arguments, 1), function (i, src) {
                    $.each(src || {}, function (prop) {
                        if (!onlyProps || $.inArray(prop, onlyProps) >= 0) {
                            var p = src[prop];

                            if ($.isPlainObject(p)) {
                                // not recursive - only copies 1 level of subobjects, and always merges
                                target[prop] = $.extend(target[prop] || {}, p);
                            } else if (p && p.constructor === Array) {
                                target[prop] = p.slice(0);
                            } else if (typeof p !== 'undefined') {
                                target[prop] = src[prop];
                            }

                        }
                    });
                });
                return target;
            },
            isElement: function (o) {
                return (typeof HTMLElement === "object" ? o instanceof HTMLElement :
                        o && typeof o === "object" && o.nodeType === 1 && typeof o.nodeName === "string");
            },
            /**
             * Basic indexOf implementation for IE7-8. Though we use $.inArray, some jQuery versions will try to 
             * use a prototpye on the calling object, defeating the purpose of using $.inArray in the first place.
             *
             * This will be replaced with the array prototype if it's available.
             * 
             * @param  {Array} arr The array to search
             * @param {Object} target The item to search for
             * @return {Number} The index of the item, or -1 if not found
             */
            indexOf: function (arr, target) {
                for (var i = 0; i < arr.length; i++) {
                    if (arr[i] === target) {
                        return i;
                    }
                }
                return -1;
            },

            // finds element of array or object with a property "prop" having value "val"
            // if prop is not defined, then just looks for property with value "val"
            indexOfProp: function (obj, prop, val) {
                var result = obj.constructor === Array ? -1 : null;
                $.each(obj, function (i, e) {
                    if (e && (prop ? e[prop] : e) === val) {
                        result = i;
                        return false;
                    }
                });
                return result;
            },
            // returns "obj" if true or false, or "def" if not true/false
            boolOrDefault: function (obj, def) {
                return this.isBool(obj) ?
                    obj : def || false;
            },
            isBool: function (obj) {
                return typeof obj === "boolean";
            },
            isUndef: function (obj) {
                return typeof obj === "undefined";
            },
            // evaluates "obj", if function, calls it with args
            // (todo - update this to handle variable lenght/more than one arg)
            ifFunction: function (obj, that, args) {
                if ($.isFunction(obj)) {
                    obj.call(that, args);
                }
            },
            size: function (image, raw) {
                var u = $.mapster.utils;
                return {
                    width: raw ? (image.width || image.naturalWidth) : u.imgWidth(image, true),
                    height: raw ? (image.height || image.naturalHeight) : u.imgHeight(image, true),
                    complete: function () { return !!this.height && !!this.width; }
                };
            },


            /**
             * Set the opacity of the element. This is an IE<8 specific function for handling VML.
             * When using VML we must override the "setOpacity" utility function (monkey patch ourselves).
             * jQuery does not deal with opacity correctly for VML elements. This deals with that.
             * 
             * @param {Element} el The DOM element
             * @param {double} opacity A value between 0 and 1 inclusive.
             */

            setOpacity: function (el, opacity) {
                if ($.mapster.hasCanvas()) {
                    el.style.opacity = opacity;
                } else {
                    $(el).each(function (i, e) {
                        if (typeof e.opacity !== 'undefined') {
                            e.opacity = opacity;
                        } else {
                            $(e).css("opacity", opacity);
                        }
                    });
                }
            },


            // fade "el" from opacity "op" to "endOp" over a period of time "duration"

            fader: (function () {
                var elements = {},
                        lastKey = 0,
                        fade_func = function (el, op, endOp, duration) {
                            var index,
                                cbIntervals = duration / 15,
                                obj, u = $.mapster.utils;

                            if (typeof el === 'number') {
                                obj = elements[el];
                                if (!obj) {
                                    return;
                                }
                            } else {
                                index = u.indexOfProp(elements, null, el);
                                if (index) {
                                    delete elements[index];
                                }
                                elements[++lastKey] = obj = el;
                                el = lastKey;
                            }

                            endOp = endOp || 1;

                            op = (op + (endOp / cbIntervals) > endOp - 0.01) ? endOp : op + (endOp / cbIntervals);

                            u.setOpacity(obj, op);
                            if (op < endOp) {
                                setTimeout(function () {
                                    fade_func(el, op, endOp, duration);
                                }, 15);
                            }
                        };
                return fade_func;
            }())
        },
        getBoundList: function (opts, key_list) {
            if (!opts.boundList) {
                return null;
            }
            var index, key, result = $(), list = $.mapster.utils.split(key_list);
            opts.boundList.each(function (i, e) {
                for (index = 0; index < list.length; index++) {
                    key = list[index];
                    if ($(e).is('[' + opts.listKey + '="' + key + '"]')) {
                        result = result.add(e);
                    }
                }
            });
            return result;
        },
        // Causes changes to the bound list based on the user action (select or deselect)
        // area: the jQuery area object
        // returns the matching elements from the bound list for the first area passed (normally only one should be passed, but
        // a list can be passed
        setBoundListProperties: function (opts, target, selected) {
            target.each(function (i, e) {
                if (opts.listSelectedClass) {
                    if (selected) {
                        $(e).addClass(opts.listSelectedClass);
                    } else {
                        $(e).removeClass(opts.listSelectedClass);
                    }
                }
                if (opts.listSelectedAttribute) {
                    $(e).attr(opts.listSelectedAttribute, selected);
                }
            });
        },
        getMapDataIndex: function (obj) {
            var img, id;
            switch (obj.tagName && obj.tagName.toLowerCase()) {
                case 'area':
                    id = $(obj).parent().attr('name');
                    img = $("img[usemap='#" + id + "']")[0];
                    break;
                case 'img':
                    img = obj;
                    break;
            }
            return img ?
                this.utils.indexOfProp(this.map_cache, 'image', img) : -1;
        },
        getMapData: function (obj) {
            var index = this.getMapDataIndex(obj.length ? obj[0] : obj);
            if (index >= 0) {
                return index >= 0 ? this.map_cache[index] : null;
            }
        },
        /**
         * Queue a command to be run after the active async operation has finished
         * @param  {MapData}  map_data    The target MapData object
         * @param  {jQuery}   that        jQuery object on which the command was invoked
         * @param  {string}   command     the ImageMapster method name
         * @param  {object[]} args        arguments passed to the method
         * @return {bool}                 true if the command was queued, false if not (e.g. there was no need to)
         */
        queueCommand: function (map_data, that, command, args) {
            if (!map_data) {
                return false;
            }
            if (!map_data.complete || map_data.currentAction) {
                map_data.commands.push(
                {
                    that: that,
                    command: command,
                    args: args
                });
                return true;
            }
            return false;
        },
        unload: function () {
            this.impl.unload();
            this.utils = null;
            this.impl = null;
            $.fn.mapster = null;
            $.mapster = null;
            $('*').unbind();
        }
    };

    // Config for object prototypes
    // first: use only first object (for things that should not apply to lists)
    /// calls back one of two fuinctions, depending on whether an area was obtained.
    // opts: {
    //    name: 'method name',
    //    key: 'key,
    //    args: 'args'
    //
    //}
    // name: name of method (required)
    // args: arguments to re-call with
    // Iterates through all the objects passed, and determines whether it's an area or an image, and calls the appropriate
    // callback for each. If anything is returned from that callback, the process is stopped and that data return. Otherwise,
    // the object itself is returned.

    var m = $.mapster,
        u = m.utils,
        ap = Array.prototype;


    // jQuery's width() and height() are broken on IE9 in some situations. This tries everything. 
    $.each(["width", "height"], function (i, e) {
        var capProp = e.substr(0, 1).toUpperCase() + e.substr(1);
        // when jqwidth parm is passed, it also checks the jQuery width()/height() property
        // the issue is that jQUery width() can report a valid size before the image is loaded in some browsers
        // without it, we can read zero even when image is loaded in other browsers if its not visible
        // we must still check because stuff like adblock can temporarily block it
        // what a goddamn headache
        u["img" + capProp] = function (img, jqwidth) {
            return (jqwidth ? $(img)[e]() : 0) ||
                img[e] || img["natural" + capProp] || img["client" + capProp] || img["offset" + capProp];
        };

    });

    /**
     * The Method object encapsulates the process of testing an ImageMapster method to see if it's being
     * invoked on an image, or an area; then queues the command if the MapData is in an active state.
     * 
     * @param {[jQuery]}    that        The target of the invocation
     * @param {[function]}  func_map    The callback if the target is an imagemap
     * @param {[function]}  func_area   The callback if the target is an area
     * @param {[object]}    opt         Options: { key: a map key if passed explicitly
     *                                             name: the command name, if it can be queued,
     *                                             args: arguments to the method
     *                                            }
     */

    m.Method = function (that, func_map, func_area, opts) {
        var me = this;
        me.name = opts.name;
        me.output = that;
        me.input = that;
        me.first = opts.first || false;
        me.args = opts.args ? ap.slice.call(opts.args, 0) : [];
        me.key = opts.key;
        me.func_map = func_map;
        me.func_area = func_area;
        //$.extend(me, opts);
        me.name = opts.name;
        me.allowAsync = opts.allowAsync || false;
    };
    m.Method.prototype = {
        constructor: m.Method,
        go: function () {
            var i, data, ar, len, result, src = this.input,
                    area_list = [],
                    me = this;

            len = src.length;
            for (i = 0; i < len; i++) {
                data = $.mapster.getMapData(src[i]);
                if (data) {
                    if (!me.allowAsync && m.queueCommand(data, me.input, me.name, me.args)) {
                        if (this.first) {
                            result = '';
                        }
                        continue;
                    }

                    ar = data.getData(src[i].nodeName === 'AREA' ? src[i] : this.key);
                    if (ar) {
                        if ($.inArray(ar, area_list) < 0) {
                            area_list.push(ar);
                        }
                    } else {
                        result = this.func_map.apply(data, me.args);
                    }
                    if (this.first || typeof result !== 'undefined') {
                        break;
                    }
                }
            }
            // if there were areas, call the area function for each unique group
            $(area_list).each(function (i, e) {
                result = me.func_area.apply(e, me.args);
            });

            if (typeof result !== 'undefined') {
                return result;
            } else {
                return this.output;
            }
        }
    };

    $.mapster.impl = (function () {
        var me = {},
        addMap = function (map_data) {
            return m.map_cache.push(map_data) - 1;
        },
        removeMap = function (map_data) {
            m.map_cache.splice(map_data.index, 1);
            for (var i = m.map_cache.length - 1; i >= this.index; i--) {
                m.map_cache[i].index--;
            }
        };


        /**
         * Test whether the browser supports VML. Credit: google.
         * http://stackoverflow.com/questions/654112/how-do-you-detect-support-for-vml-or-svg-in-a-browser
         * 
         * @return {bool} true if vml is supported, false if not
         */

        function hasVml() {
            var a = $('<div />').appendTo('body');
            a.html('<v:shape id="vml_flag1" adj="1" />');

            var b = a[0].firstChild;
            b.style.behavior = "url(#default#VML)";
            var has = b ? typeof b.adj === "object" : true;
            a.remove();
            return has;
        }

        /**
         * Return a reference to the IE namespaces object, if available, or an empty object otherwise
         * @return {obkect} The document.namespaces object.
         */
        function namespaces() {
            return typeof (document.namespaces) === 'object' ?
                document.namespaces :
                null;
        }

        /**
         * Test for the presence of HTML5 Canvas support. This also checks to see if excanvas.js has been 
         * loaded and is faking it; if so, we assume that canvas is not supported.
         *
         * @return {bool} true if HTML5 canvas support, false if not
         */

        function hasCanvas() {
            var d = namespaces();
            // when g_vml_ is present, then we can be sure excanvas is active, meaning there's not a real canvas.

            return d && d.g_vml_ ?
               false :
               $('<canvas />')[0].getContext ?
                   true :
                   false;
        }

        /**
         * Merge new area data into existing area options on a MapData object. Used for rebinding.
         * 
         * @param  {[MapData]} map_data     The MapData object
         * @param  {[object[]]} areas       areas array to merge
         */

        function merge_areas(map_data, areas) {
            var ar, index,
                map_areas = map_data.options.areas;

            if (areas) {
                $.each(areas, function (i, e) {

                    // Issue #68 - ignore invalid data in areas array

                    if (!e || !e.key) {
                        return;
                    }

                    index = u.indexOfProp(map_areas, "key", e.key);

                    if (index >= 0) {
                        $.extend(map_areas[index], e);
                    }
                    else {
                        map_areas.push(e);
                    }
                    ar = map_data.getDataForKey(e.key);
                    if (ar) {
                        $.extend(ar.options, e);
                    }
                });
            }
        }
        function merge_options(map_data, options) {
            var temp_opts = u.updateProps({}, options);
            delete temp_opts.areas;

            u.updateProps(map_data.options, temp_opts);

            merge_areas(map_data, options.areas);
            // refresh the area_option template
            u.updateProps(map_data.area_options, map_data.options);
        }

        // Most methods use the "Method" object which handles figuring out whether it's an image or area called and
        // parsing key parameters. The constructor wants:
        // this, the jQuery object
        // a function that is called when an image was passed (with a this context of the MapData)
        // a function that is called when an area was passed (with a this context of the AreaData)
        // options: first = true means only the first member of a jQuery object is handled
        //          key = the key parameters passed
        //          defaultReturn: a value to return other than the jQuery object (if its not chainable)
        //          args: the arguments
        // Returns a comma-separated list of user-selected areas. "staticState" areas are not considered selected for the purposes of this method.

        me.get = function (key) {
            var md = m.getMapData(this);
            if (!(md && md.complete)) {
                throw ("Can't access data until binding complete.");
            }

            return (new m.Method(this,
                function () {
                    // map_data return
                    return this.getSelected();
                },
                function () {
                    return this.isSelected();
                },
                {
                    name: 'get',
                    args: arguments,
                    key: key,
                    first: true,
                    allowAsync: true,
                    defaultReturn: ''
                }
            )).go();
        };
        me.data = function (key) {
            return (new m.Method(this,
                null,
                function () {
                    return this;
                },
                {
                    name: 'data',
                    args: arguments,
                    key: key
                }
            )).go();
        };


        // Set or return highlight state.
        //  $(img).mapster('highlight') -- return highlighted area key, or null if none
        //  $(area).mapster('highlight') -- highlight an area
        //  $(img).mapster('highlight','area_key') -- highlight an area
        //  $(img).mapster('highlight',false) -- remove highlight
        me.highlight = function (key) {
            return (new m.Method(this,
                function () {
                    if (key === false) {
                        this.ensureNoHighlight();
                    } else {
                        var id = this.highlightId;
                        return id >= 0 ? this.data[id].key : null;
                    }
                },
                function () {
                    this.highlight();
                },
                {
                    name: 'highlight',
                    args: arguments,
                    key: key,
                    first: true
                }
            )).go();
        };
        // Return the primary keys for an area or group key.
        // $(area).mapster('key')
        // includes all keys (not just primary keys)
        // $(area).mapster('key',true)
        // $(img).mapster('key','group-key')

        // $(img).mapster('key','group-key', true)
        me.keys = function (key, all) {
            var keyList = [],
                md = m.getMapData(this);

            if (!(md && md.complete)) {
                throw ("Can't access data until binding complete.");
            }


            function addUniqueKeys(ad) {
                var areas, keys = [];
                if (!all) {
                    keys.push(ad.key);
                } else {
                    areas = ad.areas();
                    $.each(areas, function (i, e) {
                        keys = keys.concat(e.keys);
                    });
                }
                $.each(keys, function (i, e) {
                    if ($.inArray(e, keyList) < 0) {
                        keyList.push(e);
                    }
                });
            }

            if (!(md && md.complete)) {
                return '';
            }
            if (typeof key === 'string') {
                if (all) {
                    addUniqueKeys(md.getDataForKey(key));
                } else {
                    keyList = [md.getKeysForGroup(key)];
                }
            } else {
                all = key;
                this.each(function (i, e) {
                    if (e.nodeName === 'AREA') {
                        addUniqueKeys(md.getDataForArea(e));
                    }
                });
            }
            return keyList.join(',');


        };
        me.select = function () {
            me.set.call(this, true);
        };
        me.deselect = function () {
            me.set.call(this, false);
        };

        /**
         * Select or unselect areas. Areas can be identified by a single string key, a comma-separated list of keys, 
         * or an array of strings.
         * 
         * 
         * @param {boolean} selected Determines whether areas are selected or deselected
         * @param {string|string[]} key A string, comma-separated string, or array of strings indicating 
         *                              the areas to select or deselect
         * @param {object} options Rendering options to apply when selecting an area
         */

        me.set = function (selected, key, options) {
            var lastMap, map_data, opts = options,
                key_list, area_list; // array of unique areas passed

            function setSelection(ar) {
                var newState = selected;
                if (ar) {
                    switch (selected) {
                        case true:
                            ar.select(opts); break;
                        case false:
                            ar.deselect(true); break;
                        default:
                            newState = ar.toggle(opts); break;
                    }
                    return newState;
                }
            }
            function addArea(ar) {
                if (ar && $.inArray(ar, area_list) < 0) {
                    area_list.push(ar);
                    key_list += (key_list === '' ? '' : ',') + ar.key;
                }
            }
            // Clean up after a group that applied to the same map
            function finishSetForMap(map_data) {
                $.each(area_list, function (i, el) {
                    var newState = setSelection(el);
                    if (map_data.options.boundList) {
                        m.setBoundListProperties(map_data.options, m.getBoundList(map_data.options, key_list), newState);
                    }
                });
                if (!selected) {
                    map_data.removeSelectionFinish();
                }

            }

            this.filter('img,area').each(function (i, e) {
                var keys;
                map_data = m.getMapData(e);

                if (map_data !== lastMap) {
                    if (lastMap) {
                        finishSetForMap(lastMap);
                    }

                    area_list = [];
                    key_list = '';
                }

                if (map_data) {

                    keys = '';
                    if (e.nodeName.toUpperCase() === 'IMG') {
                        if (!m.queueCommand(map_data, $(e), 'set', [selected, key, opts])) {
                            if (key instanceof Array) {
                                if (key.length) {
                                    keys = key.join(",");
                                }
                            }
                            else {
                                keys = key;
                            }

                            if (keys) {
                                $.each(u.split(keys), function (i, key) {
                                    addArea(map_data.getDataForKey(key.toString()));
                                    lastMap = map_data;
                                });
                            }
                        }
                    } else {
                        opts = key;
                        if (!m.queueCommand(map_data, $(e), 'set', [selected, opts])) {
                            addArea(map_data.getDataForArea(e));
                            lastMap = map_data;
                        }

                    }
                }
            });

            if (map_data) {
                finishSetForMap(map_data);
            }


            return this;
        };
        me.unbind = function (preserveState) {
            return (new m.Method(this,
                function () {
                    this.clearEvents();
                    this.clearMapData(preserveState);
                    removeMap(this);
                },
                null,
                {
                    name: 'unbind',
                    args: arguments
                }
            )).go();
        };


        // refresh options and update selection information.
        me.rebind = function (options) {
            return (new m.Method(this,
                function () {
                    var me = this;

                    me.complete = false;
                    me.configureOptions(options);
                    me.bindImages().then(function () {
                        me.buildDataset(true);
                        me.complete = true;
                    });
                    //this.redrawSelections();
                },
                null,
                {
                    name: 'rebind',
                    args: arguments
                }
            )).go();
        };
        // get options. nothing or false to get, or "true" to get effective options (versus passed options)
        me.get_options = function (key, effective) {
            var eff = u.isBool(key) ? key : effective; // allow 2nd parm as "effective" when no key
            return (new m.Method(this,
                function () {
                    var opts = $.extend({}, this.options);
                    if (eff) {
                        opts.render_select = u.updateProps(
                            {},
                            m.render_defaults,
                            opts,
                            opts.render_select);

                        opts.render_highlight = u.updateProps(
                            {},
                            m.render_defaults,
                            opts,
                            opts.render_highlight);
                    }
                    return opts;
                },
                function () {
                    return eff ? this.effectiveOptions() : this.options;
                },
                {
                    name: 'get_options',
                    args: arguments,
                    first: true,
                    allowAsync: true,
                    key: key
                }
            )).go();
        };

        // set options - pass an object with options to set,
        me.set_options = function (options) {
            return (new m.Method(this,
                function () {
                    merge_options(this, options);
                },
                null,
                {
                    name: 'set_options',
                    args: arguments
                }
            )).go();
        };
        me.unload = function () {
            var i;
            for (i = m.map_cache.length - 1; i >= 0; i--) {
                if (m.map_cache[i]) {
                    me.unbind.call($(m.map_cache[i].image));
                }
            }
            me.graphics = null;
        };

        me.snapshot = function () {
            return (new m.Method(this,
                function () {
                    $.each(this.data, function (i, e) {
                        e.selected = false;
                    });

                    this.base_canvas = this.graphics.createVisibleCanvas(this);
                    $(this.image).before(this.base_canvas);
                },
                null,
                { name: 'snapshot' }
            )).go();
        };

        // do not queue this function

        me.state = function () {
            var md, result = null;
            $(this).each(function (i, e) {
                if (e.nodeName === 'IMG') {
                    md = m.getMapData(e);
                    if (md) {
                        result = md.state();
                    }
                    return false;
                }
            });
            return result;
        };

        me.bind = function (options) {

            return this.each(function (i, e) {
                var img, map, usemap, md;

                // save ref to this image even if we can't access it yet. commands will be queued
                img = $(e);

                md = m.getMapData(e);

                // if already bound completely, do a total rebind

                if (md) {
                    me.unbind.apply(img);
                    if (!md.complete) {
                        // will be queued
                        img.bind();
                        return true;
                    }
                    md = null;
                }

                // ensure it's a valid image
                // jQuery bug with Opera, results in full-url#usemap being returned from jQuery's attr.
                // So use raw getAttribute instead.

                usemap = this.getAttribute('usemap');
                map = usemap && $('map[name="' + usemap.substr(1) + '"]');
                if (!(img.is('img') && usemap && map.size() > 0)) {
                    return true;
                }

                // sorry - your image must have border:0, things are too unpredictable otherwise.
                img.css('border', 0);

                if (!md) {
                    md = new m.MapData(this, options);

                    md.index = addMap(md);
                    md.map = map;
                    md.bindImages().then(function () {
                        md.initialize();
                    });
                }
            });
        };

        me.init = function (useCanvas) {
            var style, shapes;

            // for testing/debugging, use of canvas can be forced by initializing 
            // manually with "true" or "false". But generally we test for it.

            m.hasCanvas = function () {
                if (!u.isBool(m.hasCanvas.value)) {
                    m.hasCanvas.value = u.isBool(useCanvas) ?
                        useCanvas :
                        hasCanvas();
                }
                return m.hasCanvas.value;
            };

            m.hasVml = function () {
                if (!u.isBool(m.hasVml.value)) {
                    // initialize VML the first time we detect its presence.
                    var d = namespaces();

                    if (d && !d.v) {
                        d.add("v", "urn:schemas-microsoft-com:vml");
                        style = document.createStyleSheet();
                        shapes = ['shape', 'rect', 'oval', 'circ', 'fill', 'stroke', 'imagedata', 'group', 'textbox'];
                        $.each(shapes,
                        function (i, el) {
                            style.addRule('v\\:' + el, "behavior: url(#default#VML); antialias:true");
                        });
                    }
                    m.hasVml.value = hasVml();
                }

                return m.hasVml.value;
            };

            m.isTouch = !!document.documentElement.ontouchstart;

            u.indexOf = Array.prototype.indexOf || u.indexOf;

            $.extend(m.defaults, m.render_defaults, m.shared_defaults);
            $.extend(m.area_defaults, m.render_defaults, m.shared_defaults);

        };
        me.test = function (obj) {
            return eval(obj);
        };
        return me;
    }());

    $.mapster.impl.init();


}(jQuery));
/* graphics.js
   Graphics object handles all rendering.
*/
(function ($) {
    var p, m = $.mapster,
        u = m.utils,
        canvasMethods,
        vmlMethods;

    /**
     * Implemenation to add each area in an AreaData object to the canvas
     * @param {Graphics} graphics The target graphics object
     * @param {AreaData} areaData The AreaData object (a collection of area elements and metadata)
     * @param {object} options Rendering options to apply when rendering this group of areas
     */
    function addShapeGroupImpl(graphics, areaData, options) {
        var me = graphics,
            md = me.map_data,
            isMask = options.isMask;

        // first get area options. Then override fade for selecting, and finally merge in the 
        // "select" effect options.

        $.each(areaData.areas(), function (i, e) {
            options.isMask = isMask || (e.nohref && md.options.noHrefIsMask);
            me.addShape(e, options);
        });

        // it's faster just to manipulate the passed options isMask property and restore it, than to 
        // copy the object each time

        options.isMask = isMask;

    }

    /**
    * Convert a hex value to decimal
    * @param  {string} hex A hexadecimal toString
    * @return {int} Integer represenation of the hex string
    */

    function hex_to_decimal(hex) {
        return Math.max(0, Math.min(parseInt(hex, 16), 255));
    }
    function css3color(color, opacity) {
        return 'rgba(' + hex_to_decimal(color.substr(0, 2)) + ','
                + hex_to_decimal(color.substr(2, 2)) + ','
                + hex_to_decimal(color.substr(4, 2)) + ',' + opacity + ')';
    }
    /**
     * An object associated with a particular map_data instance to manage renderin.
     * @param {MapData} map_data The MapData object bound to this instance
     */

    m.Graphics = function (map_data) {
        //$(window).unload($.mapster.unload);
        // create graphics functions for canvas and vml browsers. usage:
        // 1) init with map_data, 2) call begin with canvas to be used (these are separate b/c may not require canvas to be specified
        // 3) call add_shape_to for each shape or mask, 4) call render() to finish

        var me = this;
        me.active = false;
        me.canvas = null;
        me.width = 0;
        me.height = 0;
        me.shapes = [];
        me.masks = [];
        me.map_data = map_data;
    };

    p = m.Graphics.prototype = {
        constructor: m.Graphics,

        /**
         * Initiate a graphics request for a canvas
         * @param  {Element} canvas The canvas element that is the target of this operation
         * @param  {string} [elementName] The name to assign to the element (VML only)
         */

        begin: function (canvas, elementName) {
            var c = $(canvas);

            this.elementName = elementName;
            this.canvas = canvas;

            this.width = c.width();
            this.height = c.height();
            this.shapes = [];
            this.masks = [];
            this.active = true;

        },

        /**
         * Add an area to be rendered to this canvas. 
         * @param {MapArea} mapArea The MapArea object to render
         * @param {object} options An object containing any rendering options that should override the
         *                         defaults for the area
         */

        addShape: function (mapArea, options) {
            var addto = options.isMask ? this.masks : this.shapes;
            addto.push({ mapArea: mapArea, options: options });
        },

        /**
         * Create a canvas that is sized and styled for the MapData object
         * @param  {MapData} mapData The MapData object that will receive this new canvas
         * @return {Element} A canvas element
         */

        createVisibleCanvas: function (mapData) {
            return $(this.createCanvasFor(mapData))
                .addClass('mapster_el')
                .css(m.canvas_style)[0];
        },

        /**
         * Add a group of shapes from an AreaData object to the canvas
         * 
         * @param {AreaData} areaData An AreaData object (a set of area elements)
         * @param {string} mode     The rendering mode, "select" or "highlight". This determines the target 
         *                          canvas and which default options to use.
         * @param {striong} options  Rendering options
         */

        addShapeGroup: function (areaData, mode, options) {
            // render includeKeys first - because they could be masks
            var me = this,
                list, name, canvas,
                map_data = this.map_data,
                opts = areaData.effectiveRenderOptions(mode);

            if (options) {
                $.extend(opts, options);
            }

            if (mode === 'select') {
                name = "static_" + areaData.areaId.toString();
                canvas = map_data.base_canvas;
            } else {
                canvas = map_data.overlay_canvas;
            }

            me.begin(canvas, name);

            if (opts.includeKeys) {
                list = u.split(opts.includeKeys);
                $.each(list, function (i, e) {
                    var areaData = map_data.getDataForKey(e.toString());
                    addShapeGroupImpl(me, areaData, areaData.effectiveRenderOptions(mode));
                });
            }

            addShapeGroupImpl(me, areaData, opts);
            me.render();
            if (opts.fade) {

                // fading requires special handling for IE. We must access the fill elements directly. The fader also has to deal with 
                // the "opacity" attribute (not css)

                u.fader(m.hasCanvas() ?
                    canvas :
                    $(canvas).find('._fill').not('.mapster_mask'),
                0,
                m.hasCanvas() ?
                    1 :
                    opts.fillOpacity,
                opts.fadeDuration);

            }

        }

        // These prototype methods are implementation dependent
    };

    function noop() { }


    // configure remaining prototype methods for ie or canvas-supporting browser

    canvasMethods = {
        renderShape: function (context, mapArea, offset) {
            var i,
                c = mapArea.coords(null, offset);

            switch (mapArea.shape) {
                case 'rect':
                    context.rect(c[0], c[1], c[2] - c[0], c[3] - c[1]);
                    break;
                case 'poly':
                    context.moveTo(c[0], c[1]);

                    for (i = 2; i < mapArea.length; i += 2) {
                        context.lineTo(c[i], c[i + 1]);
                    }
                    context.lineTo(c[0], c[1]);
                    break;
                case 'circ':
                case 'circle':
                    context.arc(c[0], c[1], c[2], 0, Math.PI * 2, false);
                    break;
            }
        },
        addAltImage: function (context, image, mapArea, options) {
            context.beginPath();

            this.renderShape(context, mapArea);
            context.closePath();
            context.clip();

            context.globalAlpha = options.altImageOpacity || options.fillOpacity;

            context.drawImage(image, 0, 0, mapArea.owner.scaleInfo.width, mapArea.owner.scaleInfo.height);
        },
        render: function () {
            // firefox 6.0 context.save() seems to be broken. to work around,  we have to draw the contents on one temp canvas,
            // the mask on another, and merge everything. ugh. fixed in 1.2.2. unfortunately this is a lot more code for masks,
            // but no other way around it that i can see.

            var maskCanvas, maskContext,
                        me = this,
                        md = me.map_data,
                        hasMasks = me.masks.length,
                        shapeCanvas = me.createCanvasFor(md),
                        shapeContext = shapeCanvas.getContext('2d'),
                        context = me.canvas.getContext('2d');

            if (hasMasks) {
                maskCanvas = me.createCanvasFor(md);
                maskContext = maskCanvas.getContext('2d');
                maskContext.clearRect(0, 0, maskCanvas.width, maskCanvas.height);

                $.each(me.masks, function (i, e) {
                    maskContext.save();
                    maskContext.beginPath();
                    me.renderShape(maskContext, e.mapArea);
                    maskContext.closePath();
                    maskContext.clip();
                    maskContext.lineWidth = 0;
                    maskContext.fillStyle = '#000';
                    maskContext.fill();
                    maskContext.restore();
                });

            }

            $.each(me.shapes, function (i, s) {
                shapeContext.save();
                if (s.options.fill) {
                    if (s.options.altImageId) {
                        me.addAltImage(shapeContext, md.images[s.options.altImageId], s.mapArea, s.options);
                    } else {
                        shapeContext.beginPath();
                        me.renderShape(shapeContext, s.mapArea);
                        shapeContext.closePath();
                        //shapeContext.clip();
                        shapeContext.fillStyle = css3color(s.options.fillColor, s.options.fillOpacity);
                        shapeContext.fill();
                    }
                }
                shapeContext.restore();
            });


            // render strokes at end since masks get stroked too

            $.each(me.shapes.concat(me.masks), function (i, s) {
                var offset = s.options.strokeWidth === 1 ? 0.5 : 0;
                // offset applies only when stroke width is 1 and stroke would render between pixels.

                if (s.options.stroke) {
                    shapeContext.save();
                    shapeContext.strokeStyle = css3color(s.options.strokeColor, s.options.strokeOpacity);
                    shapeContext.lineWidth = s.options.strokeWidth;

                    shapeContext.beginPath();

                    me.renderShape(shapeContext, s.mapArea, offset);
                    shapeContext.closePath();
                    shapeContext.stroke();
                    shapeContext.restore();
                }
            });

            if (hasMasks) {
                // render the new shapes against the mask

                maskContext.globalCompositeOperation = "source-out";
                maskContext.drawImage(shapeCanvas, 0, 0);

                // flatten into the main canvas
                context.drawImage(maskCanvas, 0, 0);
            } else {
                context.drawImage(shapeCanvas, 0, 0);
            }

            me.active = false;
            return me.canvas;
        },

        // create a canvas mimicing dimensions of an existing element
        createCanvasFor: function (md) {
            return $('<canvas width="' + md.scaleInfo.width + '" height="' + md.scaleInfo.height + '"></canvas>')[0];
        },
        clearHighlight: function () {
            var c = this.map_data.overlay_canvas;
            c.getContext('2d').clearRect(0, 0, c.width, c.height);
        },
        // Draw all items from selected_list to a new canvas, then swap with the old one. This is used to delete items when using canvases.
        refreshSelections: function () {
            var canvas_temp, map_data = this.map_data;
            // draw new base canvas, then swap with the old one to avoid flickering
            canvas_temp = map_data.base_canvas;

            map_data.base_canvas = this.createVisibleCanvas(map_data);
            $(map_data.base_canvas).hide();
            $(canvas_temp).before(map_data.base_canvas);

            map_data.redrawSelections();

            $(map_data.base_canvas).show();
            $(canvas_temp).remove();
        }
    };

    vmlMethods = {

        renderShape: function (mapArea, options, cssclass) {
            var me = this, fill, stroke, e, t_fill, el_name, el_class, template, c = mapArea.coords();
            el_name = me.elementName ? 'name="' + me.elementName + '" ' : '';
            el_class = cssclass ? 'class="' + cssclass + '" ' : '';

            t_fill = '<v:fill color="#' + options.fillColor + '" class="_fill" opacity="' +
                (options.fill ?
                    options.fillOpacity :
                    0) +
                '" /><v:stroke class="_fill" opacity="' +
                options.strokeOpacity + '"/>';


            stroke = options.stroke ?
                ' strokeweight=' + options.strokeWidth + ' stroked="t" strokecolor="#' +
                    options.strokeColor + '"' :
                ' stroked="f"';

            fill = options.fill ?
                ' filled="t"' :
                ' filled="f"';

            switch (mapArea.shape) {
                case 'rect':
                    template = '<v:rect ' + el_class + el_name + fill + stroke +
                        ' style="zoom:1;margin:0;padding:0;display:block;position:absolute;left:' +
                          c[0] + 'px;top:' + c[1] + 'px;width:' + (c[2] - c[0]) +
                          'px;height:' + (c[3] - c[1]) + 'px;">' + t_fill + '</v:rect>';
                    break;
                case 'poly':
                    template = '<v:shape ' + el_class + el_name + fill + stroke + ' coordorigin="0,0" coordsize="' + me.width + ',' + me.height
                                + '" path="m ' + c[0] + ',' + c[1] + ' l ' + c.slice(2).join(',')
                                + ' x e" style="zoom:1;margin:0;padding:0;display:block;position:absolute;top:0px;left:0px;width:' + me.width + 'px;height:' + me.height + 'px;">' + t_fill + '</v:shape>';
                    break;
                case 'circ':
                case 'circle':
                    template = '<v:oval ' + el_class + el_name + fill + stroke
                                + ' style="zoom:1;margin:0;padding:0;display:block;position:absolute;left:' + (c[0] - c[2]) + 'px;top:' + (c[1] - c[2])
                                + 'px;width:' + (c[2] * 2) + 'px;height:' + (c[2] * 2) + 'px;">' + t_fill + '</v:oval>';
                    break;
            }
            e = $(template);
            $(me.canvas).append(e);

            return e;
        },
        render: function () {
            var opts, me = this;

            $.each(this.shapes, function (i, e) {
                me.renderShape(e.mapArea, e.options);
            });

            if (this.masks.length) {
                $.each(this.masks, function (i, e) {
                    opts = u.updateProps({},
                        e.options, {
                            fillOpacity: 1,
                            fillColor: e.options.fillColorMask
                        });
                    me.renderShape(e.mapArea, opts, 'mapster_mask');
                });
            }

            this.active = false;
            return this.canvas;
        },

        createCanvasFor: function (md) {
            var w = md.scaleInfo.width,
                h = md.scaleInfo.height;
            return $('<var width="' + w + '" height="' + h
                + '" style="zoom:1;overflow:hidden;display:block;width:'
                + w + 'px;height:' + h + 'px;"></var>')[0];
        },

        clearHighlight: function () {
            $(this.map_data.overlay_canvas).children().remove();
        },
        // remove single or all selections
        removeSelections: function (area_id) {
            if (area_id >= 0) {
                $(this.map_data.base_canvas).find('[name="static_' + area_id.toString() + '"]').remove();
            }
            else {
                $(this.map_data.base_canvas).children().remove();
            }
        }

    };

    // for all methods with two implemenatations, add a function that will automatically replace itself with the correct
    // method on first invocation

    $.each(['renderShape',
           'addAltImage',
           'render',
           'createCanvasFor',
           'clearHighlight',
           'removeSelections',
           'refreshSelections'],
        function (i, e) {
            p[e] = (function (method) {
                return function () {
                    p[method] = (m.hasCanvas() ?
                        canvasMethods[method] :
                        vmlMethods[method]) || noop;

                    return p[method].apply(this, arguments);
                };
            }(e));
        });


}(jQuery));
/* mapimage.js
   the MapImage object, repesents an instance of a single bound imagemap
*/

(function ($) {

    var m = $.mapster,
        u = m.utils,
        ap = [];
    /**
     * An object encapsulating all the images used by a MapData.
     */

    m.MapImages = function (owner) {
        this.owner = owner;
        this.clear();
    };


    m.MapImages.prototype = {
        constructor: m.MapImages,

        /* interface to make this array-like */

        slice: function () {
            return ap.slice.apply(this, arguments);
        },
        splice: function () {
            ap.slice.apply(this.status, arguments);
            var result = ap.slice.apply(this, arguments);
            return result;
        },

        /** 
         * a boolean value indicates whether all images are done loading 
         * @return {bool} true when all are done
         */
        complete: function () {
            return $.inArray(false, this.status) < 0;
        },

        /**
         * Save an image in the images array and return its index 
         * @param  {Image} image An Image object
         * @return {int} the index of the image
         */

        _add: function (image) {
            var index = ap.push.call(this, image) - 1;
            this.status[index] = false;
            return index;
        },

        /**
         * Return the index of an Image within the images array
         * @param  {Image} img An Image
         * @return {int} the index within the array, or -1 if it was not found
         */

        indexOf: function (image) {
            return u.indexOf(this, image);
        },

        /**
         * Clear this object and reset it to its initial state after binding.
         */

        clear: function () {
            var me = this;

            if (me.ids && me.ids.length > 0) {
                $.each(me.ids, function (i, e) {
                    delete me[e];
                });
            }

            /**
             * A list of the cross-reference IDs bound to this object
             * @type {string[]}
             */

            me.ids = [];

            /**
             * Length property for array-like behavior, set to zero when initializing. Array prototype
             * methods will update it after that.
             * 
             * @type {int}
             */

            me.length = 0;

            /**
             * the loaded status of the corresponding image
             * @type {boolean[]}
             */

            me.status = [];


            // actually erase the images

            me.splice(0);

        },

        /**
         * Bind an image to the map and add it to the queue to be loaded; return an ID that
         * can be used to reference the
         * 
         * @param {Image|string} image An Image object or a URL to an image
         * @param {string} [id] An id to refer to this image
         * @returns {int} an ID referencing the index of the image object in 
         *                map_data.images
         */

        add: function (image, id) {
            var index, src, me = this;

            if (!image) { return; }

            if (typeof image === 'string') {
                src = image;
                image = me[src];
                if (typeof image === 'object') {
                    return me.indexOf(image);
                }

                image = $('<img />')
                    .addClass('mapster_el')
                    .hide();

                index = me._add(image[0]);

                image
                    .bind('load', function (e) {
                        me.imageLoaded.call(me, e);
                    })
                    .bind('error', function (e) {
                        me.imageLoadError.call(me, e);
                    });

                image.attr('src', src);
            } else {

                // use attr because we want the actual source, not the resolved path the browser will return directly calling image.src

                index = me._add($(image)[0]);
            }
            if (id) {
                if (this[id]) {
                    throw (id + " is already used or is not available as an altImage alias.");
                }
                me.ids.push(id);
                me[id] = me[index];
            }
            return index;
        },

        /**
         * Bind the images in this object, 
         * @param  {boolean} retry when true, indicates that the function is calling itself after failure 
         * @return {Promise} a promise that resolves when the images have finished loading
         */

        bind: function (retry) {
            var me = this,
                promise,
                triesLeft = me.owner.options.configTimeout / 200,

            /* A recursive function to continue checking that the images have been 
               loaded until a timeout has elapsed */

            check = function () {
                var i;

                // refresh status of images

                i = me.length;

                while (i-- > 0) {
                    if (!me.isLoaded(i)) {
                        break;
                    }
                }

                // check to see if every image has already been loaded

                if (me.complete()) {
                    me.resolve();
                } else {
                    // to account for failure of onLoad to fire in rare situations
                    if (triesLeft-- > 0) {
                        me.imgTimeout = window.setTimeout(function () {
                            check.call(me, true);
                        }, 50);
                    } else {
                        me.imageLoadError.call(me);
                    }
                }

            };

            promise = me.deferred = u.defer();

            check();
            return promise;
        },

        resolve: function () {
            var me = this,
                resolver = me.deferred;

            if (resolver) {
                // Make a copy of the resolver before calling & removing it to ensure
                // it is not called twice
                me.deferred = null;
                resolver.resolve();
            }
        },

        /**
         * Event handler for image onload
         * @param  {object} e jQuery event data
         */

        imageLoaded: function (e) {
            var me = this,
                index = me.indexOf(e.target);

            if (index >= 0) {

                me.status[index] = true;
                if ($.inArray(false, me.status) < 0) {
                    me.resolve();
                }
            }
        },

        /**
         * Event handler for onload error
         * @param  {object} e jQuery event data
         */

        imageLoadError: function (e) {
            clearTimeout(this.imgTimeout);
            this.triesLeft = 0;
            var err = e ? 'The image ' + e.target.src + ' failed to load.' :
                'The images never seemed to finish loading. You may just need to increase the configTimeout if images could take a long time to load.';
            throw err;
        },
        /**
         * Test if the image at specificed index has finished loading
         * @param  {int}  index The image index
         * @return {boolean} true if loaded, false if not
         */

        isLoaded: function (index) {
            var img,
                me = this,
                status = me.status;

            if (status[index]) { return true; }
            img = me[index];

            if (typeof img.complete !== 'undefined') {
                status[index] = img.complete;
            } else {
                status[index] = !!u.imgWidth(img);
            }
            // if complete passes, the image is loaded, but may STILL not be available because of stuff like adblock.
            // make sure it is.

            return status[index];
        }
    };
}(jQuery));
/* mapdata.js
   the MapData object, repesents an instance of a single bound imagemap
*/


(function ($) {

    var m = $.mapster,
        u = m.utils;

    /**
     * Set default values for MapData object properties
     * @param  {MapData} me The MapData object
     */

    function initializeDefaults(me) {
        $.extend(me, {
            complete: false,         // (bool)    when configuration is complete       
            map: null,                // ($)      the image map
            base_canvas: null,       // (canvas|var)  where selections are rendered
            overlay_canvas: null,    // (canvas|var)  where highlights are rendered
            commands: [],            // {}        commands that were run before configuration was completed (b/c images weren't loaded)
            data: [],                // MapData[] area groups
            mapAreas: [],            // MapArea[] list. AreaData entities contain refs to this array, so options are stored with each.
            _xref: {},               // (int)      xref of mapKeys to data[]
            highlightId: -1,        // (int)      the currently highlighted element.
            currentAreaId: -1,
            _tooltip_events: [],     // {}         info on events we bound to a tooltip container, so we can properly unbind them
            scaleInfo: null,         // {}         info about the image size, scaling, defaults
            index: -1,                 // index of this in map_cache - so we have an ID to use for wraper div
            activeAreaEvent: null
        });
    }

    /**
     * Return an array of all image-containing options from an options object; 
     * that is, containers that may have an "altImage" property
     * 
     * @param  {object} obj     An options object
     * @return {object[]}       An array of objects
     */
    function getOptionImages(obj) {
        return [obj, obj.render_highlight, obj.render_select];
    }

    /**
     * Parse all the altImage references, adding them to the library so they can be preloaded
     * and aliased.
     * 
     * @param  {MapData} me The MapData object on which to operate
     */
    function configureAltImages(me) {
        var opts = me.options,
            mi = me.images;

        // add alt images

        if (m.hasCanvas()) {
            // map altImage library first

            $.each(opts.altImages || {}, function (i, e) {
                mi.add(e, i);
            });

            // now find everything else

            $.each([opts].concat(opts.areas), function (i, e) {
                $.each(getOptionImages(e), function (i2, e2) {
                    if (e2 && e2.altImage) {
                        e2.altImageId = mi.add(e2.altImage);
                    }
                });
            });
        }

        // set area_options
        me.area_options = u.updateProps({}, // default options for any MapArea
            m.area_defaults,
            opts);
    }

    /**
     * Queue a mouse move action based on current delay settings 
     * (helper for mouseover/mouseout handlers)
     * 
     * @param  {MapData}    me       The MapData context
     * @param  {number}     delay    The number of milliseconds to delay the action
     * @param  {AreaData}   area     AreaData affected
     * @param  {Deferred}   deferred A deferred object to return (instead of a new one)
     * @return {Promise}    A promise that resolves when the action is completed
     */
    function queueMouseEvent(me, delay, area, deferred) {

        deferred = deferred || u.when.defer();

        function cbFinal(areaId) {
            if (me.currentAreaId !== areaId && me.highlightId >= 0) {
                deferred.resolve();
            }
        }
        if (me.activeAreaEvent) {
            window.clearTimeout(me.activeAreaEvent);
            me.activeAreaEvent = 0;
        }
        if (delay < 0) {
            deferred.reject();
        } else {
            if (area.owner.currentAction || delay) {
                me.activeAreaEvent = window.setTimeout((function () {
                    return function () {
                        queueMouseEvent(me, 0, area, deferred);
                    };
                }(area)),
                    delay || 100);
            } else {
                cbFinal(area.areaId);
            }
        }
        return deferred;
    }

    /**
    * Mousedown event. This is captured only to prevent browser from drawing an outline around an
    * area when it's clicked.
    *
    * @param  {EventData} e jQuery event data
    */

    function mousedown(e) {
        if (!m.hasCanvas()) {
            this.blur();
        }
        e.preventDefault();
    }

    /**
     * Mouseover event. Handle highlight rendering and client callback on mouseover
     * 
     * @param  {MapData} me The MapData context
     * @param  {EventData} e jQuery event data
     * @return {[type]}   [description]
     */

    function mouseover(me, e) {
        var arData = me.getAllDataForArea(this),
            ar = arData.length ? arData[0] : null;

        // mouseover events are ignored entirely while resizing, though we do care about mouseout events
        // and must queue the action to keep things clean.

        if (!ar || ar.isNotRendered() || ar.owner.currentAction) {
            return;
        }

        if (me.currentAreaId === ar.areaId) {
            return;
        }
        if (me.highlightId !== ar.areaId) {
            me.clearEffects();

            ar.highlight();

            if (me.options.showToolTip) {
                $.each(arData, function (i, e) {
                    if (e.effectiveOptions().toolTip) {
                        e.showToolTip();
                    }
                });
            }
        }

        me.currentAreaId = ar.areaId;

        if ($.isFunction(me.options.onMouseover)) {
            me.options.onMouseover.call(this,
            {
                e: e,
                options: ar.effectiveOptions(),
                key: ar.key,
                selected: ar.isSelected()
            });
        }
    }

    /**
     * Mouseout event.
     *
     * @param  {MapData} me The MapData context
     * @param  {EventData} e jQuery event data
     * @return {[type]}   [description]
     */

    function mouseout(me, e) {
        var newArea,
            ar = me.getDataForArea(this),
            opts = me.options;


        if (me.currentAreaId < 0 || !ar) {
            return;
        }

        newArea = me.getDataForArea(e.relatedTarget);

        if (newArea === ar) {
            return;
        }

        me.currentAreaId = -1;
        ar.area = null;

        queueMouseEvent(me, opts.mouseoutDelay, ar)
            .then(me.clearEffects);

        if ($.isFunction(opts.onMouseout)) {
            opts.onMouseout.call(this,
            {
                e: e,
                options: opts,
                key: ar.key,
                selected: ar.isSelected()
            });
        }

    }

    /**
     * Clear any active tooltip or highlight
     *
     * @param  {MapData} me The MapData context
     * @param  {EventData} e jQuery event data
     * @return {[type]}   [description]
     */

    function clearEffects(me) {
        var opts = me.options;

        me.ensureNoHighlight();

        if (opts.toolTipClose
            && $.inArray('area-mouseout', opts.toolTipClose) >= 0
            && me.activeToolTip) {
            me.clearToolTip();
        }
    }

    /**
     * Mouse click event handler
     *
     * @param  {MapData} me The MapData context
     * @param  {EventData} e jQuery event data
     * @return {[type]}   [description]
     */

    function click(me, e) {
        var selected, list, list_target, newSelectionState, canChangeState, cbResult,
            that = this,
            ar = me.getDataForArea(this),
            opts = me.options;

        function clickArea(ar) {
            var areaOpts, target;
            canChangeState = (ar.isSelectable() &&
                (ar.isDeselectable() || !ar.isSelected()));

            if (canChangeState) {
                newSelectionState = !ar.isSelected();
            } else {
                newSelectionState = ar.isSelected();
            }

            list_target = m.getBoundList(opts, ar.key);

            if ($.isFunction(opts.onClick)) {
                cbResult = opts.onClick.call(that,
                {
                    e: e,
                    listTarget: list_target,
                    key: ar.key,
                    selected: newSelectionState
                });

                if (u.isBool(cbResult)) {
                    if (!cbResult) {
                        return false;
                    }
                    target = $(ar.area).attr('href');
                    if (target !== '#') {
                        window.location.href = target;
                        return false;
                    }
                }
            }

            if (canChangeState) {
                selected = ar.toggle();
            }

            if (opts.boundList && opts.boundList.length > 0) {
                m.setBoundListProperties(opts, list_target, ar.isSelected());
            }

            areaOpts = ar.effectiveOptions();
            if (areaOpts.includeKeys) {
                list = u.split(areaOpts.includeKeys);
                $.each(list, function (i, e) {
                    var ar = me.getDataForKey(e.toString());
                    if (!ar.options.isMask) {
                        clickArea(ar);
                    }
                });
            }
        }

        mousedown.call(this, e);

        if (opts.clickNavigate && ar.href) {
            window.location.href = ar.href;
            return;
        }

        if (ar && !ar.owner.currentAction) {
            opts = me.options;
            clickArea(ar);
        }
    }

    /**
     * Prototype for a MapData object, representing an ImageMapster bound object
     * @param {Element} image   an IMG element
     * @param {object} options  ImageMapster binding options
     */
    m.MapData = function (image, options) {
        var me = this;

        // (Image)  main map image

        me.image = image;

        me.images = new m.MapImages(me);
        me.graphics = new m.Graphics(me);

        // save the initial style of the image for unbinding. This is problematic, chrome 
        // duplicates styles when assigning, and cssText is apparently not universally supported.
        // Need to do something more robust to make unbinding work universally.

        me.imgCssText = image.style.cssText || null;

        initializeDefaults(me);

        me.configureOptions(options);

        // create context-bound event handlers from our private functions

        me.mouseover = function (e) { mouseover.call(this, me, e); };
        me.mouseout = function (e) { mouseout.call(this, me, e); };
        me.click = function (e) { click.call(this, me, e); };
        me.clearEffects = function (e) { clearEffects.call(this, me, e); };
    };

    m.MapData.prototype = {
        constructor: m.MapData,

        /**
        * Set target.options from defaults + options
        * @param  {[type]} target      The target
        * @param  {[type]} options     The options to merge
        */

        configureOptions: function (options) {
            this.options = u.updateProps({}, m.defaults, options);
        },

        /**
         * Ensure all images are loaded
         * @return {Promise} A promise that resolves when the images have finished loading (or fail)
         */

        bindImages: function () {
            var me = this,
                mi = me.images;

            // reset the images if this is a rebind

            if (mi.length > 2) {
                mi.splice(2);
            } else if (mi.length === 0) {

                // add the actual main image
                mi.add(me.image);
                // will create a duplicate of the main image, we need this to get raw size info
                mi.add(me.image.src);
            }

            configureAltImages(me);

            return me.images.bind();
        },

        /**
         * Test whether an async action is currently in progress
         * @return {Boolean} true or false indicating state
         */

        isActive: function () {
            return !this.complete || this.currentAction;
        },

        /**
         * Return an object indicating the various states. This isn't really used by 
         * production code.
         * 
         * @return {object} An object with properties for various states
         */

        state: function () {
            return {
                complete: this.complete,
                resizing: this.currentAction === 'resizing',
                zoomed: this.zoomed,
                zoomedArea: this.zoomedArea,
                scaleInfo: this.scaleInfo
            };
        },

        /**
         * Get a unique ID for the wrapper of this imagemapster
         * @return {string} A string that is unique to this image
         */

        wrapId: function () {
            return 'mapster_wrap_' + this.index;
        },
        _idFromKey: function (key) {
            return typeof key === "string" && this._xref.hasOwnProperty(key) ?
                        this._xref[key] : -1;
        },

        /**
         * Return a comma-separated string of all selected keys
         * @return {string} CSV of all keys that are currently selected
         */

        getSelected: function () {
            var result = '';
            $.each(this.data, function (i, e) {
                if (e.isSelected()) {
                    result += (result ? ',' : '') + this.key;
                }
            });
            return result;
        },

        /**
         * Get an array of MapAreas associated with a specific AREA based on the keys for that area
         * @param  {Element} area   An HTML AREA
         * @param  {number} atMost  A number limiting the number of areas to be returned (typically 1 or 0 for no limit)
         * @return {MapArea[]}      Array of MapArea objects
         */

        getAllDataForArea: function (area, atMost) {
            var i, ar, result,
                me = this,
                key = $(area).filter('area').attr(me.options.mapKey);

            if (key) {
                result = [];
                key = u.split(key);

                for (i = 0; i < (atMost || key.length) ; i++) {
                    ar = me.data[me._idFromKey(key[i])];
                    ar.area = area.length ? area[0] : area;
                    // set the actual area moused over/selected
                    // TODO: this is a brittle model for capturing which specific area - if this method was not used,
                    // ar.area could have old data. fix this.
                    result.push(ar);
                }
            }

            return result;
        },
        getDataForArea: function (area) {
            var ar = this.getAllDataForArea(area, 1);
            return ar ? ar[0] || null : null;
        },
        getDataForKey: function (key) {
            return this.data[this._idFromKey(key)];
        },

        /**
         * Get the primary keys associated with an area group.
         * If this is a primary key, it will be returned.
         * 
         * @param  {string key An area key
         * @return {string} A CSV of area keys
         */

        getKeysForGroup: function (key) {
            var ar = this.getDataForKey(key);

            return !ar ? '' :
                ar.isPrimary ?
                    ar.key :
                    this.getPrimaryKeysForMapAreas(ar.areas()).join(',');
        },

        /**
         * given an array of MapArea object, return an array of its unique primary keys
         * @param  {MapArea[]} areas The areas to analyze
         * @return {string[]} An array of unique primary keys
         */

        getPrimaryKeysForMapAreas: function (areas) {
            var keys = [];
            $.each(areas, function (i, e) {
                if ($.inArray(e.keys[0], keys) < 0) {
                    keys.push(e.keys[0]);
                }
            });
            return keys;
        },
        getData: function (obj) {
            if (typeof obj === 'string') {
                return this.getDataForKey(obj);
            } else if (obj && obj.mapster || u.isElement(obj)) {
                return this.getDataForArea(obj);
            } else {
                return null;
            }
        },
        // remove highlight if present, raise event
        ensureNoHighlight: function () {
            var ar;
            if (this.highlightId >= 0) {
                this.graphics.clearHighlight();
                ar = this.data[this.highlightId];
                ar.changeState('highlight', false);
                this.setHighlightId(-1);
            }
        },
        setHighlightId: function (id) {
            this.highlightId = id;
        },

        /**
         * Clear all active selections on this map
         */

        clearSelections: function () {
            $.each(this.data, function (i, e) {
                if (e.selected) {
                    e.deselect(true);
                }
            });
            this.removeSelectionFinish();

        },

        /**
         * Set area options from an array of option data.
         * 
         * @param {object[]} areas An array of objects containing area-specific options
         */

        setAreaOptions: function (areas) {
            var i, area_options, ar;
            areas = areas || [];

            // refer by: map_data.options[map_data.data[x].area_option_id]

            for (i = areas.length - 1; i >= 0; i--) {
                area_options = areas[i];
                if (area_options) {
                    ar = this.getDataForKey(area_options.key);
                    if (ar) {
                        u.updateProps(ar.options, area_options);

                        // TODO: will not deselect areas that were previously selected, so this only works
                        // for an initial bind.

                        if (u.isBool(area_options.selected)) {
                            ar.selected = area_options.selected;
                        }
                    }
                }
            }
        },
        // keys: a comma-separated list
        drawSelections: function (keys) {
            var i, key_arr = u.asArray(keys);

            for (i = key_arr.length - 1; i >= 0; i--) {
                this.data[key_arr[i]].drawSelection();
            }
        },
        redrawSelections: function () {
            $.each(this.data, function (i, e) {
                if (e.isSelectedOrStatic()) {
                    e.drawSelection();
                }
            });

        },
        ///called when images are done loading
        initialize: function () {
            var imgCopy, base_canvas, overlay_canvas, wrap, parentId, css, i, size,
                img, sort_func, sorted_list, scale,
                        me = this,
                        opts = me.options;

            if (me.complete) {
                return;
            }

            img = $(me.image);

            parentId = img.parent().attr('id');

            // create a div wrapper only if there's not already a wrapper, otherwise, own it

            if (parentId && parentId.length >= 12 && parentId.substring(0, 12) === "mapster_wrap") {
                wrap = img.parent();
                wrap.attr('id', me.wrapId());
            } else {
                wrap = $('<div id="' + me.wrapId() + '"></div>');

                if (opts.wrapClass) {
                    if (opts.wrapClass === true) {
                        wrap.addClass(img[0].className);
                    }
                    else {
                        wrap.addClass(opts.wrapClass);
                    }
                }
            }
            me.wrapper = wrap;

            // me.images[1] is the copy of the original image. It should be loaded & at its native size now so we can obtain the true
            // width & height. This is needed to scale the imagemap if not being shown at its native size. It is also needed purely
            // to finish binding in case the original image was not visible. It can be impossible in some browsers to obtain the
            // native size of a hidden image.

            me.scaleInfo = scale = u.scaleMap(me.images[0], me.images[1], opts.scaleMap);

            me.base_canvas = base_canvas = me.graphics.createVisibleCanvas(me);
            me.overlay_canvas = overlay_canvas = me.graphics.createVisibleCanvas(me);

            // Now we got what we needed from the copy -clone from the original image again to make sure any other attributes are copied
            imgCopy = $(me.images[1])
                .addClass('mapster_el ' + me.images[0].className)
                .attr({ id: null, usemap: null });

            size = u.size(me.images[0]);

            if (size.complete) {
                imgCopy.css({
                    width: size.width,
                    height: size.height
                });
            }

            me.buildDataset();

            // now that we have processed all the areas, set css for wrapper, scale map if needed

            css = {
                display: 'block',
                position: 'relative',
                padding: 0,
                width: scale.width,
                height: scale.height
            };

            if (opts.wrapCss) {
                $.extend(css, opts.wrapCss);
            }
            // if we were rebinding with an existing wrapper, the image will aready be in it
            if (img.parent()[0] !== me.wrapper[0]) {

                img.before(me.wrapper);
            }

            wrap.css(css);

            // move all generated images into the wrapper for easy removal later

            $(me.images.slice(2)).hide();
            for (i = 1; i < me.images.length; i++) {
                wrap.append(me.images[i]);
            }

            //me.images[1].style.cssText = me.image.style.cssText;

            wrap.append(base_canvas)
                        .append(overlay_canvas)
                        .append(img.css(m.canvas_style));

            // images[0] is the original image with map, images[1] is the copy/background that is visible

            u.setOpacity(me.images[0], 0);
            $(me.images[1]).show();

            u.setOpacity(me.images[1], 1);

            if (opts.isSelectable && opts.onGetList) {
                sorted_list = me.data.slice(0);
                if (opts.sortList) {
                    if (opts.sortList === "desc") {
                        sort_func = function (a, b) {
                            return a === b ? 0 : (a > b ? -1 : 1);
                        };
                    }
                    else {
                        sort_func = function (a, b) {
                            return a === b ? 0 : (a < b ? -1 : 1);
                        };
                    }

                    sorted_list.sort(function (a, b) {
                        a = a.value;
                        b = b.value;
                        return sort_func(a, b);
                    });
                }

                me.options.boundList = opts.onGetList.call(me.image, sorted_list);
            }

            me.complete = true;
            me.processCommandQueue();

            if (opts.onConfigured && typeof opts.onConfigured === 'function') {
                opts.onConfigured.call(img, true);
            }
        },

        // when rebind is true, the MapArea data will not be rebuilt.
        buildDataset: function (rebind) {
            var sel, areas, j, area_id, $area, area, curKey, mapArea, key, keys, mapAreaId, group_value, dataItem, href,
                me = this,
                opts = me.options,
                default_group;

            function addAreaData(key, value) {
                var dataItem = new m.AreaData(me, key, value);
                dataItem.areaId = me._xref[key] = me.data.push(dataItem) - 1;
                return dataItem.areaId;
            }

            me._xref = {};
            me.data = [];
            if (!rebind) {
                me.mapAreas = [];
            }

            default_group = !opts.mapKey;
            if (default_group) {
                opts.mapKey = 'data-mapster-key';
            }

            // the [attribute] selector is broken on old IE with jQuery. hasVml() is a quick and dirty
            // way to test for that

            sel = m.hasVml() ? 'area' :
                        (default_group ?
                            'area[coords]' :
                            'area[' + opts.mapKey + ']');

            areas = $(me.map).find(sel).unbind('.mapster');

            for (mapAreaId = 0; mapAreaId < areas.length; mapAreaId++) {
                area_id = 0;
                area = areas[mapAreaId];
                $area = $(area);

                // skip areas with no coords - selector broken for older ie
                if (!area.coords) {
                    continue;
                }
                // Create a key if none was assigned by the user

                if (default_group) {
                    curKey = String(mapAreaId);
                    $area.attr('data-mapster-key', curKey);

                } else {
                    curKey = area.getAttribute(opts.mapKey);
                }

                // conditions for which the area will be bound to mouse events
                // only bind to areas that don't have nohref. ie 6&7 cannot detect the presence of nohref, so we have to also not bind if href is missing.

                if (rebind) {
                    mapArea = me.mapAreas[$area.data('mapster') - 1];
                    mapArea.configure(curKey);
                } else {
                    mapArea = new m.MapArea(me, area, curKey);
                    me.mapAreas.push(mapArea);
                }

                keys = mapArea.keys; // converted to an array by mapArea


                // Iterate through each mapKey assigned to this area
                for (j = keys.length - 1; j >= 0; j--) {
                    key = keys[j];

                    if (opts.mapValue) {
                        group_value = $area.attr(opts.mapValue);
                    }
                    if (default_group) {
                        // set an attribute so we can refer to the area by index from the DOM object if no key
                        area_id = addAreaData(me.data.length, group_value);
                        dataItem = me.data[area_id];
                        dataItem.key = key = area_id.toString();
                    }
                    else {
                        area_id = me._xref[key];
                        if (area_id >= 0) {
                            dataItem = me.data[area_id];
                            if (group_value && !me.data[area_id].value) {
                                dataItem.value = group_value;
                            }
                        }
                        else {
                            area_id = addAreaData(key, group_value);
                            dataItem = me.data[area_id];
                            dataItem.isPrimary = j === 0;
                        }
                    }
                    mapArea.areaDataXref.push(area_id);
                    dataItem.areasXref.push(mapAreaId);
                }

                href = $area.attr('href');
                if (href && href !== '#' && !dataItem.href) {
                    dataItem.href = href;
                }

                if (!mapArea.nohref) {
                    $area.bind('click.mapster', me.click)
                        .bind('mouseover.mapster', me.mouseover)
                        .bind('mouseout.mapster', me.mouseout)
                        .bind('mousedown.mapster', me.mousedown);



                }

                // store an ID with each area. 
                $area.data("mapster", mapAreaId + 1);
            }

            // TODO listenToList
            //            if (opts.listenToList && opts.nitG) {
            //                opts.nitG.bind('click.mapster', event_hooks[map_data.hooks_index].listclick_hook);
            //            }

            // populate areas from config options
            me.setAreaOptions(opts.areas);
            me.redrawSelections();

        },
        processCommandQueue: function () {

            var cur, me = this;
            while (!me.currentAction && me.commands.length) {
                cur = me.commands[0];
                me.commands.splice(0, 1);
                m.impl[cur.command].apply(cur.that, cur.args);
            }
        },
        clearEvents: function () {
            $(this.map).find('area')
                        .unbind('.mapster');
            $(this.images)
                        .unbind('.mapster');
        },
        _clearCanvases: function (preserveState) {
            // remove the canvas elements created
            if (!preserveState) {
                $(this.base_canvas).remove();
            }
            $(this.overlay_canvas).remove();
        },
        clearMapData: function (preserveState) {
            var me = this;
            this._clearCanvases(preserveState);

            // release refs to DOM elements
            $.each(this.data, function (i, e) {
                e.reset();
            });
            this.data = null;
            if (!preserveState) {
                // get rid of everything except the original image
                this.image.style.cssText = this.imgCssText;
                $(this.wrapper).before(this.image).remove();
            }

            me.images.clear();

            this.image = null;
            u.ifFunction(this.clearTooltip, this);
        },

        // Compelete cleanup process for deslecting items. Called after a batch operation, or by AreaData for single
        // operations not flagged as "partial"

        removeSelectionFinish: function () {
            var g = this.graphics;

            g.refreshSelections();
            // do not call ensure_no_highlight- we don't really want to unhilight it, just remove the effect
            g.clearHighlight();
        }
    };
}(jQuery));
/* areadata.js
   AreaData and MapArea protoypes
*/

(function ($) {
    var m = $.mapster, u = m.utils;

    /**
     * Select this area
     * 
     * @param {AreaData} me  AreaData context
     * @param {object} options Options for rendering the selection
     */
    function select(options) {
        // need to add the new one first so that the double-opacity effect leaves the current one highlighted for singleSelect

        var me = this, o = me.owner;
        if (o.options.singleSelect) {
            o.clearSelections();
        }

        // because areas can overlap - we can't depend on the selection state to tell us anything about the inner areas.
        // don't check if it's already selected
        if (!me.isSelected()) {
            if (options) {

                // cache the current options, and map the altImageId if an altimage 
                // was passed

                me.optsCache = $.extend(me.effectiveRenderOptions('select'),
                    options,
                    {
                        altImageId: o.images.add(options.altImage)
                    });
            }

            me.drawSelection();

            me.selected = true;
            me.changeState('select', true);
        }

        if (o.options.singleSelect) {
            o.graphics.refreshSelections();
        }
    }

    /**
     * Deselect this area, optionally deferring finalization so additional areas can be deselected
     * in a single operation
     * 
     * @param  {boolean} partial when true, the caller must invoke "finishRemoveSelection" to render 
     */

    function deselect(partial) {
        var me = this;
        me.selected = false;
        me.changeState('select', false);

        // release information about last area options when deselecting.

        me.optsCache = null;
        me.owner.graphics.removeSelections(me.areaId);

        // Complete selection removal process. This is separated because it's very inefficient to perform the whole
        // process for multiple removals, as the canvas must be totally redrawn at the end of the process.ar.remove

        if (!partial) {
            me.owner.removeSelectionFinish();
        }
    }

    /**
     * Toggle the selection state of this area
     * @param  {object} options Rendering options, if toggling on
     * @return {bool} The new selection state
     */
    function toggle(options) {
        var me = this;
        if (!me.isSelected()) {
            me.select(options);
        }
        else {
            me.deselect();
        }
        return me.isSelected();
    }

    /**
     * An AreaData object; represents a conceptual area that can be composed of 
     * one or more MapArea objects
     * 
     * @param {MapData} owner The MapData object to which this belongs
     * @param {string} key   The key for this area
     * @param {string} value The mapValue string for this area
     */

    m.AreaData = function (owner, key, value) {
        $.extend(this, {
            owner: owner,
            key: key || '',
            // means this represents the first key in a list of keys (it's the area group that gets highlighted on mouseover)
            isPrimary: true,
            areaId: -1,
            href: '',
            value: value || '',
            options: {},
            // "null" means unchanged. Use "isSelected" method to just test true/false 
            selected: null,
            // xref to MapArea objects
            areasXref: [],
            // (temporary storage) - the actual area moused over
            area: null,
            // the last options used to render this. Cache so when re-drawing after a remove, changes in options won't
            // break already selected things. 
            optsCache: null
        });
    };

    /**
     * The public API for AreaData object
     */

    m.AreaData.prototype = {
        constuctor: m.AreaData,
        select: select,
        deselect: deselect,
        toggle: toggle,
        areas: function () {
            var i, result = [];
            for (i = 0; i < this.areasXref.length; i++) {
                result.push(this.owner.mapAreas[this.areasXref[i]]);
            }
            return result;
        },
        // return all coordinates for all areas
        coords: function (offset) {
            var coords = [];
            $.each(this.areas(), function (i, el) {
                coords = coords.concat(el.coords(offset));
            });
            return coords;
        },
        reset: function () {
            $.each(this.areas(), function (i, e) {
                e.reset();
            });
            this.areasXref = [];
            this.options = null;
        },
        // Return the effective selected state of an area, incorporating staticState
        isSelectedOrStatic: function () {

            var o = this.effectiveOptions();
            return u.isBool(o.staticState) ? o.staticState :
                        this.isSelected();
        },
        isSelected: function () {
            return u.isBool(this.selected) ? this.selected :
                u.isBool(this.owner.area_options.selected) ? this.owner.area_options.selected : false;
        },
        isSelectable: function () {
            return u.isBool(this.effectiveOptions().staticState) ? false :
                        (u.isBool(this.owner.options.staticState) ? false : u.boolOrDefault(this.effectiveOptions().isSelectable, true));
        },
        isDeselectable: function () {
            return u.isBool(this.effectiveOptions().staticState) ? false :
                        (u.isBool(this.owner.options.staticState) ? false : u.boolOrDefault(this.effectiveOptions().isDeselectable, true));
        },
        isNotRendered: function () {
            var area = $(this.area);
            return area.attr('nohref') ||
                !area.attr('href') ||
                this.effectiveOptions().isMask;

        },


        /**
        * Return the overall options effective for this area. 
        * This should get the default options, and merge in area-specific options, finally
        * overlaying options passed by parameter
        * 
        * @param  {[type]} options  options which will supercede all other options for this area
        * @return {[type]}          the combined options
        */

        effectiveOptions: function (options) {

            var opts = u.updateProps({},
                    this.owner.area_options,
                    this.options,
                    options || {},
                    {
                        id: this.areaId
                    }
                );

            opts.selected = this.isSelected();

            return opts;
        },

        /**
         * Return the options effective for this area for a "render" or "highlight" mode. 
         * This should get the default options, merge in the areas-specific options, 
         * and then the mode-specific options.
         * @param  {string} mode    'render' or 'highlight'
         * @param  {[type]} options  options which will supercede all other options for this area
         * @return {[type]}          the combined options
         */

        effectiveRenderOptions: function (mode, options) {
            var allOpts, opts = this.optsCache;

            if (!opts || mode === 'highlight') {
                allOpts = this.effectiveOptions(options);
                opts = u.updateProps({},
                    allOpts,
                    allOpts["render_" + mode]
                );

                if (mode !== 'highlight') {
                    this.optsCache = opts;
                }
            }
            return $.extend({}, opts);
        },

        // Fire callback on area state change
        changeState: function (state_type, state) {
            if ($.isFunction(this.owner.options.onStateChange)) {
                this.owner.options.onStateChange.call(this.owner.image,
                    {
                        key: this.key,
                        state: state_type,
                        selected: state
                    }
                );
            }
        },

        // highlight this area

        highlight: function (options) {
            var o = this.owner;
            if (this.effectiveOptions().highlight) {
                o.graphics.addShapeGroup(this, "highlight", options);
            }
            o.setHighlightId(this.areaId);
            this.changeState('highlight', true);
        },

        // select this area. if "callEvent" is true then the state change event will be called. (This method can be used
        // during config operations, in which case no event is indicated)

        drawSelection: function () {


            this.owner.graphics.addShapeGroup(this, "select");

        }


    };
    // represents an HTML area
    m.MapArea = function (owner, areaEl, keys) {
        if (!owner) {
            return;
        }
        var me = this;
        me.owner = owner;   // a MapData object
        me.area = areaEl;
        me.areaDataXref = []; // a list of map_data.data[] id's for each areaData object containing this
        me.originalCoords = [];
        $.each(u.split(areaEl.coords), function (i, el) {
            me.originalCoords.push(parseFloat(el));
        });
        me.length = me.originalCoords.length;
        me.shape = areaEl.shape.toLowerCase();
        me.nohref = areaEl.nohref || !areaEl.href;
        me.configure(keys);
    };
    m.MapArea.prototype = {
        constructor: m.MapArea,
        configure: function (keys) {
            this.keys = u.split(keys);
        },
        reset: function () {
            this.area = null;
        },
        coords: function (offset) {
            return $.map(this.originalCoords, function (e) {
                return offset ? e : e + offset;
            });
        }
    };
}(jQuery));
/* areacorners.js
   determine the best place to put a box of dimensions (width,height) given a circle, rect or poly
*/

(function ($) {
    var u = $.mapster.utils;


    /**
     * Compute positions that will place a target with dimensions [width,height] outside 
     * but near the boundaries of the elements "elements". When an imagemap is passed, the 
     *
     * @param  {Element|Element[]} elements An element or an array of elements (such as a jQuery object)
     * @param  {Element} image The image to which area elements are bound, if this is an image map.
     * @param  {Element} container The contianer in which the target must be constrained (or document, if missing)
     * @param  {int} width The width of the target object
     * @return {object} a structure with the x and y positions
     */
    u.areaCorners = function (elements, image, container, width, height) {
        var pos, found, minX, minY, maxX, maxY, bestMinX, bestMaxX, bestMinY, bestMaxY, curX, curY, nest, j,
           offsetx = 0,
           offsety = 0,
           rootx,
           rooty,
           iCoords, radius, angle, el,
           coords = [];

        // if a single element was passed, map it to an array

        elements = elements.length ?
            elements :
            [elements];

        container = container ?
            $(container) :
            $(document.body);

        // get the relative root of calculation

        pos = container.offset();
        rootx = pos.left;
        rooty = pos.top;

        // with areas, all we know about is relative to the top-left corner of the image. We need to add an offset compared to
        // the actual container. After this calculation, offsetx/offsety can be added to either the area coords, or the target's
        // absolute position to get the correct top/left boundaries of the container.

        if (image) {
            pos = $(image).offset();
            offsetx = pos.left;
            offsety = pos.top;
        }

        // map the coordinates of any type of shape to a poly and use the logic. simpler than using three different
        // calculation methods. Circles use a 20 degree increment for this estimation.

        for (j = 0; j < elements.length; j++) {
            el = elements[j];
            if (el.nodeName === 'AREA') {
                iCoords = u.split(el.coords, parseInt);

                switch (el.shape) {
                    case 'circle':
                        curX = iCoords[0];
                        curY = iCoords[1];
                        radius = iCoords[2];
                        coords = [];
                        for (j = 0; j < 360; j += 20) {
                            angle = j * Math.PI / 180;
                            coords.push(curX + radius * Math.cos(angle), curY + radius * Math.sin(angle));
                        }
                        break;
                    case 'rect':
                        coords.push(iCoords[0], iCoords[1], iCoords[2], iCoords[1], iCoords[2], iCoords[3], iCoords[0], iCoords[3]);
                        break;
                    default:
                        coords = coords.concat(iCoords);
                        break;
                }

                // map area positions to it's real position in the container

                for (j = 0; j < coords.length; j += 2) {
                    coords[j] = parseInt(coords[j], 10) + offsetx;
                    coords[j + 1] = parseInt(coords[j + 1], 10) + offsety;
                }
            } else {
                el = $(el);
                pos = el.position();
                coords.push(pos.left, pos.top,
                            pos.left + el.width(), pos.top,
                            pos.left + el.width(), pos.top + el.height(),
                            pos.left, pos.top + el.height());

            }
        }

        minX = minY = bestMinX = bestMinY = 999999;
        maxX = maxY = bestMaxX = bestMaxY = -1;

        for (j = coords.length - 2; j >= 0; j -= 2) {
            curX = coords[j];
            curY = coords[j + 1];

            if (curX < minX) {
                minX = curX;
                bestMaxY = curY;
            }
            if (curX > maxX) {
                maxX = curX;
                bestMinY = curY;
            }
            if (curY < minY) {
                minY = curY;
                bestMaxX = curX;
            }
            if (curY > maxY) {
                maxY = curY;
                bestMinX = curX;
            }

        }

        // try to figure out the best place for the tooltip

        if (width && height) {
            found = false;
            $.each([[bestMaxX - width, minY - height], [bestMinX, minY - height],
                             [minX - width, bestMaxY - height], [minX - width, bestMinY],
                             [maxX, bestMaxY - height], [maxX, bestMinY],
                             [bestMaxX - width, maxY], [bestMinX, maxY]
            ], function (i, e) {
                if (!found && (e[0] > rootx && e[1] > rooty)) {
                    nest = e;
                    found = true;
                    return false;
                }
            });

            // default to lower-right corner if nothing fit inside the boundaries of the image

            if (!found) {
                nest = [maxX, maxY];
            }
        }
        return nest;
    };
}(jQuery));
/* scale.js: resize and zoom functionality
   requires areacorners.js, when.js
*/


(function ($) {
    var m = $.mapster, u = m.utils, p = m.MapArea.prototype;

    m.utils.getScaleInfo = function (eff, actual) {
        var pct;
        if (!actual) {
            pct = 1;
            actual = eff;
        } else {
            pct = eff.width / actual.width || eff.height / actual.height;
            // make sure a float error doesn't muck us up
            if (pct > 0.98 && pct < 1.02) { pct = 1; }
        }
        return {
            scale: (pct !== 1),
            scalePct: pct,
            realWidth: actual.width,
            realHeight: actual.height,
            width: eff.width,
            height: eff.height,
            ratio: eff.width / eff.height
        };
    };
    // Scale a set of AREAs, return old data as an array of objects
    m.utils.scaleMap = function (image, imageRaw, scale) {

        // stunningly, jQuery width can return zero even as width does not, seems to happen only
        // with adBlock or maybe other plugins. These must interfere with onload events somehow.


        var vis = u.size(image),
            raw = u.size(imageRaw, true);

        if (!raw.complete()) {
            throw ("Another script, such as an extension, appears to be interfering with image loading. Please let us know about this.");
        }
        if (!vis.complete()) {
            vis = raw;
        }
        return this.getScaleInfo(vis, scale ? raw : null);
    };

    /**
     * Resize the image map. Only one of newWidth and newHeight should be passed to preserve scale
     * 
     * @param  {int}   width       The new width OR an object containing named parameters matching this function sig
     * @param  {int}   height      The new height
     * @param  {int}   effectDuration Time in ms for the resize animation, or zero for no animation
     * @param  {function} callback    A function to invoke when the operation finishes
     * @return {promise}              NOT YET IMPLEMENTED
     */

    m.MapData.prototype.resize = function (width, height, duration, callback) {
        var p, promises, newsize, els, highlightId, ratio,
            me = this;

        // allow omitting duration
        callback = callback || duration;

        function sizeCanvas(canvas, w, h) {
            if (m.hasCanvas()) {
                canvas.width = w;
                canvas.height = h;
            } else {
                $(canvas).width(w);
                $(canvas).height(h);
            }
        }

        // Finalize resize action, do callback, pass control to command queue

        function cleanupAndNotify() {

            me.currentAction = '';

            if ($.isFunction(callback)) {
                callback();
            }

            me.processCommandQueue();
        }

        // handle cleanup after the inner elements are resized

        function finishResize() {
            sizeCanvas(me.overlay_canvas, width, height);

            // restore highlight state if it was highlighted before
            if (highlightId >= 0) {
                var areaData = me.data[highlightId];
                areaData.tempOptions = { fade: false };
                me.getDataForKey(areaData.key).highlight();
                areaData.tempOptions = null;
            }
            sizeCanvas(me.base_canvas, width, height);
            me.redrawSelections();
            cleanupAndNotify();
        }

        function resizeMapData() {
            $(me.image).css(newsize);
            // start calculation at the same time as effect
            me.scaleInfo = u.getScaleInfo({
                width: width,
                height: height
            },
                {
                    width: me.scaleInfo.realWidth,
                    height: me.scaleInfo.realHeight
                });
            $.each(me.data, function (i, e) {
                $.each(e.areas(), function (i, e) {
                    e.resize();
                });
            });
        }

        if (me.scaleInfo.width === width && me.scaleInfo.height === height) {
            return;
        }

        highlightId = me.highlightId;


        if (!width) {
            ratio = height / me.scaleInfo.realHeight;
            width = Math.round(me.scaleInfo.realWidth * ratio);
        }
        if (!height) {
            ratio = width / me.scaleInfo.realWidth;
            height = Math.round(me.scaleInfo.realHeight * ratio);
        }

        newsize = { 'width': String(width) + 'px', 'height': String(height) + 'px' };
        if (!m.hasCanvas()) {
            $(me.base_canvas).children().remove();
        }

        // resize all the elements that are part of the map except the image itself (which is not visible)
        // but including the div wrapper
        els = $(me.wrapper).find('.mapster_el').add(me.wrapper);

        if (duration) {
            promises = [];
            me.currentAction = 'resizing';
            els.each(function (i, e) {
                p = u.defer();
                promises.push(p);

                $(e).animate(newsize, {
                    duration: duration,
                    complete: p.resolve,
                    easing: "linear"
                });
            });

            p = u.defer();
            promises.push(p);

            // though resizeMapData is not async, it needs to be finished just the same as the animations,
            // so add it to the "to do" list.

            u.when.all(promises).then(finishResize);
            resizeMapData();
            p.resolve();
        } else {
            els.css(newsize);
            resizeMapData();
            finishResize();

        }
    };


    m.MapArea = u.subclass(m.MapArea, function () {
        //change the area tag data if needed
        this.base.init();
        if (this.owner.scaleInfo.scale) {
            this.resize();
        }
    });

    p.coords = function (percent, coordOffset) {
        var j, newCoords = [],
                    pct = percent || this.owner.scaleInfo.scalePct,
                    offset = coordOffset || 0;

        if (pct === 1 && coordOffset === 0) {
            return this.originalCoords;
        }

        for (j = 0; j < this.length; j++) {
            //amount = j % 2 === 0 ? xPct : yPct;
            newCoords.push(Math.round(this.originalCoords[j] * pct) + offset);
        }
        return newCoords;
    };
    p.resize = function () {
        this.area.coords = this.coords().join(',');
    };

    p.reset = function () {
        this.area.coords = this.coords(1).join(',');
    };

    m.impl.resize = function (width, height, duration, callback) {
        if (!width && !height) {
            return false;
        }
        var x = (new m.Method(this,
                function () {
                    this.resize(width, height, duration, callback);
                },
                null,
                {
                    name: 'resize',
                    args: arguments
                }
            )).go();
        return x;
    };

    /*
        m.impl.zoom = function (key, opts) {
            var options = opts || {};
    
            function zoom(areaData) {
                // this will be MapData object returned by Method
    
                var scroll, corners, height, width, ratio,
                        diffX, diffY, ratioX, ratioY, offsetX, offsetY, newWidth, newHeight, scrollLeft, scrollTop,
                        padding = options.padding || 0,
                        scrollBarSize = areaData ? 20 : 0,
                        me = this,
                        zoomOut = false;
    
                if (areaData) {
                    // save original state on first zoom operation
                    if (!me.zoomed) {
                        me.zoomed = true;
                        me.preZoomWidth = me.scaleInfo.width;
                        me.preZoomHeight = me.scaleInfo.height;
                        me.zoomedArea = areaData;
                        if (options.scroll) {
                            me.wrapper.css({ overflow: 'auto' });
                        }
                    }
                    corners = $.mapster.utils.areaCorners(areaData.coords(1, 0));
                    width = me.wrapper.innerWidth() - scrollBarSize - padding * 2;
                    height = me.wrapper.innerHeight() - scrollBarSize - padding * 2;
                    diffX = corners.maxX - corners.minX;
                    diffY = corners.maxY - corners.minY;
                    ratioX = width / diffX;
                    ratioY = height / diffY;
                    ratio = Math.min(ratioX, ratioY);
                    offsetX = (width - diffX * ratio) / 2;
                    offsetY = (height - diffY * ratio) / 2;
    
                    newWidth = me.scaleInfo.realWidth * ratio;
                    newHeight = me.scaleInfo.realHeight * ratio;
                    scrollLeft = (corners.minX) * ratio - padding - offsetX;
                    scrollTop = (corners.minY) * ratio - padding - offsetY;
                } else {
                    if (!me.zoomed) {
                        return;
                    }
                    zoomOut = true;
                    newWidth = me.preZoomWidth;
                    newHeight = me.preZoomHeight;
                    scrollLeft = null;
                    scrollTop = null;
                }
    
                this.resize({
                    width: newWidth,
                    height: newHeight,
                    duration: options.duration,
                    scroll: scroll,
                    scrollLeft: scrollLeft,
                    scrollTop: scrollTop,
                    // closure so we can be sure values are correct
                    callback: (function () {
                        var isZoomOut = zoomOut,
                                scroll = options.scroll,
                                areaD = areaData;
                        return function () {
                            if (isZoomOut) {
                                me.preZoomWidth = null;
                                me.preZoomHeight = null;
                                me.zoomed = false;
                                me.zoomedArea = false;
                                if (scroll) {
                                    me.wrapper.css({ overflow: 'inherit' });
                                }
                            } else {
                                // just to be sure it wasn't canceled & restarted
                                me.zoomedArea = areaD;
                            }
                        };
                    } ())
                });
            }
            return (new m.Method(this,
                    function (opts) {
                        zoom.call(this);
                    },
                    function () {
                        zoom.call(this.owner, this);
                    },
                    {
                        name: 'zoom',
                        args: arguments,
                        first: true,
                        key: key
                    }
                    )).go();
    
    
        };
        */
}(jQuery));
/* tooltip.js - tooltip functionality
   requires areacorners.js
*/

(function ($) {

    var m = $.mapster, u = m.utils;

    $.extend(m.defaults, {
        toolTipContainer: '<div style="border: 2px solid black; background: #EEEEEE; width:160px; padding:4px; margin: 4px; -moz-box-shadow: 3px 3px 5px #535353; ' +
        '-webkit-box-shadow: 3px 3px 5px #535353; box-shadow: 3px 3px 5px #535353; -moz-border-radius: 6px 6px 6px 6px; -webkit-border-radius: 6px; ' +
        'border-radius: 6px 6px 6px 6px; opacity: 0.9;"></div>',
        showToolTip: false,
        toolTipFade: true,
        toolTipClose: ['area-mouseout', 'image-mouseout'],
        onShowToolTip: null,
        onHideToolTip: null
    });

    $.extend(m.area_defaults, {
        toolTip: null,
        toolTipClose: null
    });


    /**
     * Show a tooltip positioned near this area.
     * 
     * @param {string|jquery} html A string of html or a jQuery object containing the tooltip content.
     * @param {string|jquery} [template] The html template in which to wrap the content
     * @param {string|object} [css] CSS to apply to the outermost element of the tooltip 
     * @return {jquery} The tooltip that was created
     */

    function createToolTip(html, template, css) {
        var tooltip;

        // wrap the template in a jQuery object, or clone the template if it's already one.
        // This assumes that anything other than a string is a jQuery object; if it's not jQuery will
        // probably throw an error.

        if (template) {
            tooltip = typeof template === 'string' ?
                $(template) :
                $(template).clone();

            tooltip.append(html);
        } else {
            tooltip = $(html);
        }

        // always set display to block, or the positioning css won't work if the end user happened to
        // use a non-block type element.

        tooltip.css($.extend((css || {}), {
            display: "block",
            position: "absolute"
        })).hide();

        $('body').append(tooltip);

        // we must actually add the tooltip to the DOM and "show" it in order to figure out how much space it
        // consumes, and then reposition it with that knowledge.
        // We also cache the actual opacity setting to restore finally.

        tooltip.attr("data-opacity", tooltip.css("opacity"))
            .css("opacity", 0);

        // doesn't really show it because opacity=0

        return tooltip.show();
    }


    /**
     * Show a tooltip positioned near this area.
     * 
     * @param {jquery} tooltip The tooltip
     * @param {object} [options] options for displaying the tooltip.
     *  @config {int} [left] The 0-based absolute x position for the tooltip
     *  @config {int} [top] The 0-based absolute y position for the tooltip
     *  @config {string|object} [css] CSS to apply to the outermost element of the tooltip 
     *  @config {bool} [fadeDuration] When non-zero, the duration in milliseconds of a fade-in effect for the tooltip.
     */

    function showToolTipImpl(tooltip, options) {
        var tooltipCss = {
            "left": options.left + "px",
            "top": options.top + "px"
        },
            actalOpacity = tooltip.attr("data-opacity") || 0,
            zindex = tooltip.css("z-index");

        if (parseInt(zindex, 10) === 0
            || zindex === "auto") {
            tooltipCss["z-index"] = 9999;
        }

        tooltip.css(tooltipCss)
            .addClass('mapster_tooltip');


        if (options.fadeDuration && options.fadeDuration > 0) {
            u.fader(tooltip[0], 0, actalOpacity, options.fadeDuration);
        } else {
            u.setOpacity(tooltip[0], actalOpacity);
        }
    }

    /**
     * Hide and remove active tooltips
     * 
     * @param  {MapData} this The mapdata object to which the tooltips belong
     */

    m.MapData.prototype.clearToolTip = function () {
        if (this.activeToolTip) {
            this.activeToolTip.stop().remove();
            this.activeToolTip = null;
            this.activeToolTipID = null;
            u.ifFunction(this.options.onHideToolTip, this);
        }
    };

    /**
     * Configure the binding between a named tooltip closing option, and a mouse event.
     *
     * If a callback is passed, it will be called when the activating event occurs, and the tooltip will
     * only closed if it returns true.
     *
     * @param  {MapData}  [this]     The MapData object to which this tooltip belongs.
     * @param  {String}   option     The name of the tooltip closing option
     * @param  {String}   event      UI event to bind to this option
     * @param  {Element}  target     The DOM element that is the target of the event
     * @param  {Function} [beforeClose] Callback when the tooltip is closed
     * @param  {Function} [onClose]  Callback when the tooltip is closed
     */
    function bindToolTipClose(options, bindOption, event, target, beforeClose, onClose) {
        var event_name = event + '.mapster-tooltip';

        if ($.inArray(bindOption, options) >= 0) {
            target.unbind(event_name)
                .bind(event_name, function (e) {
                    if (!beforeClose || beforeClose.call(this, e)) {
                        target.unbind('.mapster-tooltip');
                        if (onClose) {
                            onClose.call(this);
                        }
                    }
                });

            return {
                object: target,
                event: event_name
            };
        }
    }

    /**
     * Show a tooltip.
     *
     * @param {string|jquery}   [tooltip]       A string of html or a jQuery object containing the tooltip content.
     * 
     * @param {string|jquery}   [target]        The target of the tooltip, to be used to determine positioning. If null,
     *                                          absolute position values must be passed with left and top.
     *
     * @param {string|jquery}   [image]         If target is an [area] the image that owns it
     * 
     * @param {string|jquery}   [container]     An element within which the tooltip must be bounded
     *
     *
     * 
     * @param {object|string|jQuery} [options]  options to apply when creating this tooltip - OR -
     *                                          The markup, or a jquery object, containing the data for the tooltip 
     *                                         
     *  @config {string}        [closeEvents]   A string with one or more comma-separated values that determine when the tooltip
     *                                          closes: 'area-click','tooltip-click','image-mouseout' are valid values
     *                                          then no template will be used.
     *  @config {int}           [offsetx]       the horizontal amount to offset the tooltip 
     *  @config {int}           [offsety]       the vertical amount to offset the tooltip 
     *  @config {string|object} [css]           CSS to apply to the outermost element of the tooltip 
     */

    function showToolTip(tooltip, target, image, container, options) {
        var corners,
            ttopts = {};

        options = options || {};


        if (target) {

            var width = tooltip.outerWidth(true) || tooltip.children().outerWidth(true);
            var height = tooltip.outerHeight(true) || tooltip.children().outerHeight(true);

            corners = u.areaCorners(target, image, container, width, height);

            // Try to upper-left align it first, if that doesn't work, change the parameters

            ttopts.left = corners[0];
            ttopts.top = corners[1];

        } else {

            ttopts.left = options.left;
            ttopts.top = options.top;
        }

        ttopts.left += (options.offsetx || 0);
        ttopts.top += (options.offsety || 0);

        ttopts.css = options.css;
        ttopts.fadeDuration = options.fadeDuration;

        showToolTipImpl(tooltip, ttopts);

        return tooltip;
    }

    /**
     * Show a tooltip positioned near this area.
      *
     * @param {string|jquery}   [content]       A string of html or a jQuery object containing the tooltip content.
     
     * @param {object|string|jQuery} [options]  options to apply when creating this tooltip - OR -
     *                                          The markup, or a jquery object, containing the data for the tooltip 
     *  @config {string|jquery}   [container]     An element within which the tooltip must be bounded
     *  @config {bool}          [template]      a template to use instead of the default. If this property exists and is null,
     *                                          then no template will be used.
     *  @config {string}        [closeEvents]   A string with one or more comma-separated values that determine when the tooltip
     *                                          closes: 'area-click','tooltip-click','image-mouseout' are valid values
     *                                          then no template will be used.
     *  @config {int}           [offsetx]       the horizontal amount to offset the tooltip 
     *  @config {int}           [offsety]       the vertical amount to offset the tooltip 
     *  @config {string|object} [css]           CSS to apply to the outermost element of the tooltip 
     */
    m.AreaData.prototype.showToolTip = function (content, options) {
        var tooltip, closeOpts, target, tipClosed, template,
            ttopts = {},
            ad = this,
            md = ad.owner,
            areaOpts = ad.effectiveOptions();

        // copy the options object so we can update it
        options = options ? $.extend({}, options) : {};

        content = content || areaOpts.toolTip;
        closeOpts = options.closeEvents || areaOpts.toolTipClose || md.options.toolTipClose || 'tooltip-click';

        template = typeof options.template !== 'undefined' ?
                options.template :
                md.options.toolTipContainer;

        options.closeEvents = typeof closeOpts === 'string' ?
            closeOpts = u.split(closeOpts) :
            closeOpts;

        options.fadeDuration = options.fadeDuration ||
                 (md.options.toolTipFade ?
                    (md.options.fadeDuration || areaOpts.fadeDuration) : 0);

        target = ad.area ?
            ad.area :
            $.map(ad.areas(),
                function (e) {
                    return e.area;
                });

        if (md.activeToolTipID === ad.areaId) {
            return;
        }

        md.clearToolTip();

        md.activeToolTip = tooltip = createToolTip(content,
            template,
            options.css);

        md.activeToolTipID = ad.areaId;

        tipClosed = function () {
            md.clearToolTip();
        };

        bindToolTipClose(closeOpts, 'area-click', 'click', $(md.map), null, tipClosed);
        bindToolTipClose(closeOpts, 'tooltip-click', 'click', tooltip, null, tipClosed);
        bindToolTipClose(closeOpts, 'image-mouseout', 'mouseout', $(md.image), function (e) {
            return (e.relatedTarget && e.relatedTarget.nodeName !== 'AREA' && e.relatedTarget !== ad.area);
        }, tipClosed);


        showToolTip(tooltip,
                    target,
                    md.image,
                    options.container,
                    template,
                    options);

        u.ifFunction(md.options.onShowToolTip, ad.area,
        {
            toolTip: tooltip,
            options: ttopts,
            areaOptions: areaOpts,
            key: ad.key,
            selected: ad.isSelected()
        });

        return tooltip;
    };


    /**
     * Parse an object that could be a string, a jquery object, or an object with a "contents" property
     * containing html or a jQuery object.
     * 
     * @param  {object|string|jQuery} options The parameter to parse
     * @return {string|jquery} A string or jquery object
     */
    function getHtmlFromOptions(options) {

        // see if any html was passed as either the options object itself, or the content property

        return (options ?
            ((typeof options === 'string' || options.jquery) ?
            options :
                options.content) :
            null);
    }

    /**
     * Activate or remove a tooltip for an area. When this method is called on an area, the
     * key parameter doesn't apply and "options" is the first parameter.
     *
     * When called with no parameters, or "key" is a falsy value, any active tooltip is cleared.
     * 
     * When only a key is provided, the default tooltip for the area is used. 
     * 
     * When html is provided, this is used instead of the default tooltip.
     * 
     * When "noTemplate" is true, the default tooltip template will not be used either, meaning only
     * the actual html passed will be used.
     *  
     * @param  {string|AreaElement} key The area for which to activate a tooltip, or a DOM element.
     * 
     * @param {object|string|jquery} [options] options to apply when creating this tooltip - OR -
     *                                         The markup, or a jquery object, containing the data for the tooltip 
     *  @config {string|jQuery} [content]   the inner content of the tooltip; the tooltip text or HTML
     *  @config {Element|jQuery} [container]   the inner content of the tooltip; the tooltip text or HTML
     *  @config {bool}          [template]  a template to use instead of the default. If this property exists and is null,
     *                                      then no template will be used.
     *  @config {int}           [offsetx]   the horizontal amount to offset the tooltip.
     *  @config {int}           [offsety]   the vertical amount to offset the tooltip.
     *  @config {string|object} [css]       CSS to apply to the outermost element of the tooltip 
     *  @config {string|object} [css] CSS to apply to the outermost element of the tooltip 
     *  @config {bool}          [fadeDuration] When non-zero, the duration in milliseconds of a fade-in effect for the tooltip.
     * @return {jQuery} The jQuery object
     */

    m.impl.tooltip = function (key, options) {
        return (new m.Method(this,
        function mapData() {
            var tooltip, target, md = this;
            if (!key) {
                md.clearToolTip();
            } else {
                target = $(key);
                if (md.activeToolTipID === target[0]) {
                    return;
                }
                md.clearToolTip();

                md.activeToolTip = tooltip = createToolTip(getHtmlFromOptions(options),
                            options.template || md.options.toolTipContainer,
                            options.css);
                md.activeToolTipID = target[0];

                bindToolTipClose(['tooltip-click'], 'tooltip-click', 'click', tooltip, null, function () {
                    md.clearToolTip();
                });

                md.activeToolTip = tooltip = showToolTip(tooltip,
                    target,
                    md.image,
                    options.container,
                    options);
            }
        },
        function areaData() {
            if ($.isPlainObject(key) && !options) {
                options = key;
            }

            this.showToolTip(getHtmlFromOptions(options), options);
        },
        {
            name: 'tooltip',
            args: arguments,
            key: key
        }
    )).go();
    };
}(jQuery));;
// Domain Public by Eric Wendelin http://eriwen.com/ (2008)
//                  Luke Smith http://lucassmith.name/ (2008)
//                  Loic Dachary <loic@dachary.org> (2008)
//                  Johan Euphrosine <proppy@aminche.com> (2008)
//                  Oyvind Sean Kinsey http://kinsey.no/blog (2010)
//                  Victor Homyakov <victor-homyakov@users.sourceforge.net> (2010)
/*global module, exports, define, ActiveXObject*/
(function (global, factory) {
    if (typeof exports === 'object') {
        // Node
        module.exports = factory();
    } else if (typeof define === 'function' && define.amd) {
        // AMD
        define(factory);
    } else {
        // Browser globals
        global.printStackTrace = factory();
    }
}(this, function () {
    /**
     * Main function giving a function stack trace with a forced or passed in Error
     *
     * @cfg {Error} e The error to create a stacktrace from (optional)
     * @cfg {Boolean} guess If we should try to resolve the names of anonymous functions
     * @return {Array} of Strings with functions, lines, files, and arguments where possible
     */
    function printStackTrace(options) {
        options = options || { guess: true };
        var ex = options.e || null, guess = !!options.guess;
        var p = new printStackTrace.implementation(), result = p.run(ex);
        return (guess) ? p.guessAnonymousFunctions(result) : result;
    }

    printStackTrace.implementation = function () {
    };

    printStackTrace.implementation.prototype = {
        /**
         * @param {Error} [ex] The error to create a stacktrace from (optional)
         * @param {String} [mode] Forced mode (optional, mostly for unit tests)
         */
        run: function (ex, mode) {
            ex = ex || this.createException();
            mode = mode || this.mode(ex);
            if (mode === 'other') {
                return this.other(arguments.callee);
            } else {
                return this[mode](ex);
            }
        },

        createException: function () {
            try {
                this.undef();
            } catch (e) {
                return e;
            }
        },

        /**
         * Mode could differ for different exception, e.g.
         * exceptions in Chrome may or may not have arguments or stack.
         *
         * @return {String} mode of operation for the exception
         */
        mode: function (e) {
            if (e['arguments'] && e.stack) {
                return 'chrome';
            }

            if (e.stack && e.sourceURL) {
                return 'safari';
            }

            if (e.stack && e.number) {
                return 'ie';
            }

            if (e.stack && e.fileName) {
                return 'firefox';
            }

            if (e.message && e['opera#sourceloc']) {
                // e.message.indexOf("Backtrace:") > -1 -> opera9
                // 'opera#sourceloc' in e -> opera9, opera10a
                // !e.stacktrace -> opera9
                if (!e.stacktrace) {
                    return 'opera9'; // use e.message
                }
                if (e.message.indexOf('\n') > -1 && e.message.split('\n').length > e.stacktrace.split('\n').length) {
                    // e.message may have more stack entries than e.stacktrace
                    return 'opera9'; // use e.message
                }
                return 'opera10a'; // use e.stacktrace
            }

            if (e.message && e.stack && e.stacktrace) {
                // e.stacktrace && e.stack -> opera10b
                if (e.stacktrace.indexOf("called from line") < 0) {
                    return 'opera10b'; // use e.stacktrace, format differs from 'opera10a'
                }
                // e.stacktrace && e.stack -> opera11
                return 'opera11'; // use e.stacktrace, format differs from 'opera10a', 'opera10b'
            }

            if (e.stack && !e.fileName) {
                // Chrome 27 does not have e.arguments as earlier versions,
                // but still does not have e.fileName as Firefox
                return 'chrome';
            }

            return 'other';
        },

        /**
         * Given a context, function name, and callback function, overwrite it so that it calls
         * printStackTrace() first with a callback and then runs the rest of the body.
         *
         * @param {Object} context of execution (e.g. window)
         * @param {String} functionName to instrument
         * @param {Function} callback function to call with a stack trace on invocation
         */
        instrumentFunction: function (context, functionName, callback) {
            context = context || window;
            var original = context[functionName];
            context[functionName] = function instrumented() {
                callback.call(this, printStackTrace().slice(4));
                return context[functionName]._instrumented.apply(this, arguments);
            };
            context[functionName]._instrumented = original;
        },

        /**
         * Given a context and function name of a function that has been
         * instrumented, revert the function to it's original (non-instrumented)
         * state.
         *
         * @param {Object} context of execution (e.g. window)
         * @param {String} functionName to de-instrument
         */
        deinstrumentFunction: function (context, functionName) {
            if (context[functionName].constructor === Function &&
                context[functionName]._instrumented &&
                context[functionName]._instrumented.constructor === Function) {
                context[functionName] = context[functionName]._instrumented;
            }
        },

        /**
         * Given an Error object, return a formatted Array based on Chrome's stack string.
         *
         * @param e - Error object to inspect
         * @return Array<String> of function calls, files and line numbers
         */
        chrome: function (e) {
            return (e.stack + '\n')
                .replace(/^[\s\S]+?\s+at\s+/, ' at ') // remove message
                .replace(/^\s+(at eval )?at\s+/gm, '') // remove 'at' and indentation
                .replace(/^([^\(]+?)([\n$])/gm, '{anonymous}() ($1)$2')
                .replace(/^Object.<anonymous>\s*\(([^\)]+)\)/gm, '{anonymous}() ($1)')
                .replace(/^(.+) \((.+)\)$/gm, '$1@$2')
                .split('\n')
                .slice(0, -1);
        },

        /**
         * Given an Error object, return a formatted Array based on Safari's stack string.
         *
         * @param e - Error object to inspect
         * @return Array<String> of function calls, files and line numbers
         */
        safari: function (e) {
            return e.stack.replace(/\[native code\]\n/m, '')
                .replace(/^(?=\w+Error\:).*$\n/m, '')
                .replace(/^@/gm, '{anonymous}()@')
                .split('\n');
        },

        /**
         * Given an Error object, return a formatted Array based on IE's stack string.
         *
         * @param e - Error object to inspect
         * @return Array<String> of function calls, files and line numbers
         */
        ie: function (e) {
            return e.stack
                .replace(/^\s*at\s+(.*)$/gm, '$1')
                .replace(/^Anonymous function\s+/gm, '{anonymous}() ')
                .replace(/^(.+)\s+\((.+)\)$/gm, '$1@$2')
                .split('\n')
                .slice(1);
        },

        /**
         * Given an Error object, return a formatted Array based on Firefox's stack string.
         *
         * @param e - Error object to inspect
         * @return Array<String> of function calls, files and line numbers
         */
        firefox: function (e) {
            return e.stack.replace(/(?:\n@:0)?\s+$/m, '')
                .replace(/^(?:\((\S*)\))?@/gm, '{anonymous}($1)@')
                .split('\n');
        },

        opera11: function (e) {
            var ANON = '{anonymous}', lineRE = /^.*line (\d+), column (\d+)(?: in (.+))? in (\S+):$/;
            var lines = e.stacktrace.split('\n'), result = [];

            for (var i = 0, len = lines.length; i < len; i += 2) {
                var match = lineRE.exec(lines[i]);
                if (match) {
                    var location = match[4] + ':' + match[1] + ':' + match[2];
                    var fnName = match[3] || "global code";
                    fnName = fnName.replace(/<anonymous function: (\S+)>/, "$1").replace(/<anonymous function>/, ANON);
                    result.push(fnName + '@' + location + ' -- ' + lines[i + 1].replace(/^\s+/, ''));
                }
            }

            return result;
        },

        opera10b: function (e) {
            // "<anonymous function: run>([arguments not available])@file://localhost/G:/js/stacktrace.js:27\n" +
            // "printStackTrace([arguments not available])@file://localhost/G:/js/stacktrace.js:18\n" +
            // "@file://localhost/G:/js/test/functional/testcase1.html:15"
            var lineRE = /^(.*)@(.+):(\d+)$/;
            var lines = e.stacktrace.split('\n'), result = [];

            for (var i = 0, len = lines.length; i < len; i++) {
                var match = lineRE.exec(lines[i]);
                if (match) {
                    var fnName = match[1] ? (match[1] + '()') : "global code";
                    result.push(fnName + '@' + match[2] + ':' + match[3]);
                }
            }

            return result;
        },

        /**
         * Given an Error object, return a formatted Array based on Opera 10's stacktrace string.
         *
         * @param e - Error object to inspect
         * @return Array<String> of function calls, files and line numbers
         */
        opera10a: function (e) {
            // "  Line 27 of linked script file://localhost/G:/js/stacktrace.js\n"
            // "  Line 11 of inline#1 script in file://localhost/G:/js/test/functional/testcase1.html: In function foo\n"
            var ANON = '{anonymous}', lineRE = /Line (\d+).*script (?:in )?(\S+)(?:: In function (\S+))?$/i;
            var lines = e.stacktrace.split('\n'), result = [];

            for (var i = 0, len = lines.length; i < len; i += 2) {
                var match = lineRE.exec(lines[i]);
                if (match) {
                    var fnName = match[3] || ANON;
                    result.push(fnName + '()@' + match[2] + ':' + match[1] + ' -- ' + lines[i + 1].replace(/^\s+/, ''));
                }
            }

            return result;
        },

        // Opera 7.x-9.2x only!
        opera9: function (e) {
            // "  Line 43 of linked script file://localhost/G:/js/stacktrace.js\n"
            // "  Line 7 of inline#1 script in file://localhost/G:/js/test/functional/testcase1.html\n"
            var ANON = '{anonymous}', lineRE = /Line (\d+).*script (?:in )?(\S+)/i;
            var lines = e.message.split('\n'), result = [];

            for (var i = 2, len = lines.length; i < len; i += 2) {
                var match = lineRE.exec(lines[i]);
                if (match) {
                    result.push(ANON + '()@' + match[2] + ':' + match[1] + ' -- ' + lines[i + 1].replace(/^\s+/, ''));
                }
            }

            return result;
        },

        // Safari 5-, IE 9-, and others
        other: function (curr) {
            var ANON = '{anonymous}', fnRE = /function(?:\s+([\w$]+))?\s*\(/, stack = [], fn, args, maxStackSize = 10;
            var slice = Array.prototype.slice;
            while (curr && stack.length < maxStackSize) {
                fn = fnRE.test(curr.toString()) ? RegExp.$1 || ANON : ANON;
                try {
                    args = slice.call(curr['arguments'] || []);
                } catch (e) {
                    args = ['Cannot access arguments: ' + e];
                }
                stack[stack.length] = fn + '(' + this.stringifyArguments(args) + ')';
                try {
                    curr = curr.caller;
                } catch (e) {
                    stack[stack.length] = 'Cannot access caller: ' + e;
                    break;
                }
            }
            return stack;
        },

        /**
         * Given arguments array as a String, substituting type names for non-string types.
         *
         * @param {Arguments,Array} args
         * @return {String} stringified arguments
         */
        stringifyArguments: function (args) {
            var result = [];
            var slice = Array.prototype.slice;
            for (var i = 0; i < args.length; ++i) {
                var arg = args[i];
                if (arg === undefined) {
                    result[i] = 'undefined';
                } else if (arg === null) {
                    result[i] = 'null';
                } else if (arg.constructor) {
                    // TODO constructor comparison does not work for iframes
                    if (arg.constructor === Array) {
                        if (arg.length < 3) {
                            result[i] = '[' + this.stringifyArguments(arg) + ']';
                        } else {
                            result[i] = '[' + this.stringifyArguments(slice.call(arg, 0, 1)) + '...' + this.stringifyArguments(slice.call(arg, -1)) + ']';
                        }
                    } else if (arg.constructor === Object) {
                        result[i] = '#object';
                    } else if (arg.constructor === Function) {
                        result[i] = '#function';
                    } else if (arg.constructor === String) {
                        result[i] = '"' + arg + '"';
                    } else if (arg.constructor === Number) {
                        result[i] = arg;
                    } else {
                        result[i] = '?';
                    }
                }
            }
            return result.join(',');
        },

        sourceCache: {},

        /**
         * @return {String} the text from a given URL
         */
        ajax: function (url) {
            var req = this.createXMLHTTPObject();
            if (req) {
                try {
                    req.open('GET', url, false);
                    //req.overrideMimeType('text/plain');
                    //req.overrideMimeType('text/javascript');
                    req.send(null);
                    //return req.status == 200 ? req.responseText : '';
                    return req.responseText;
                } catch (e) {
                }
            }
            return '';
        },

        /**
         * Try XHR methods in order and store XHR factory.
         *
         * @return {XMLHttpRequest} XHR function or equivalent
         */
        createXMLHTTPObject: function () {
            var xmlhttp, XMLHttpFactories = [
                function () {
                    return new XMLHttpRequest();
                }, function () {
                    return new ActiveXObject('Msxml2.XMLHTTP');
                }, function () {
                    return new ActiveXObject('Msxml3.XMLHTTP');
                }, function () {
                    return new ActiveXObject('Microsoft.XMLHTTP');
                }
            ];
            for (var i = 0; i < XMLHttpFactories.length; i++) {
                try {
                    xmlhttp = XMLHttpFactories[i]();
                    // Use memoization to cache the factory
                    this.createXMLHTTPObject = XMLHttpFactories[i];
                    return xmlhttp;
                } catch (e) {
                }
            }
        },

        /**
         * Given a URL, check if it is in the same domain (so we can get the source
         * via Ajax).
         *
         * @param url {String} source url
         * @return {Boolean} False if we need a cross-domain request
         */
        isSameDomain: function (url) {
            return typeof location !== "undefined" && url.indexOf(location.hostname) !== -1; // location may not be defined, e.g. when running from nodejs.
        },

        /**
         * Get source code from given URL if in the same domain.
         *
         * @param url {String} JS source URL
         * @return {Array} Array of source code lines
         */
        getSource: function (url) {
            // TODO reuse source from script tags?
            if (!(url in this.sourceCache)) {
                this.sourceCache[url] = this.ajax(url).split('\n');
            }
            return this.sourceCache[url];
        },

        guessAnonymousFunctions: function (stack) {
            for (var i = 0; i < stack.length; ++i) {
                var reStack = /\{anonymous\}\(.*\)@(.*)/,
                    reRef = /^(.*?)(?::(\d+))(?::(\d+))?(?: -- .+)?$/,
                    frame = stack[i], ref = reStack.exec(frame);

                if (ref) {
                    var m = reRef.exec(ref[1]);
                    if (m) { // If falsey, we did not get any file/line information
                        var file = m[1], lineno = m[2], charno = m[3] || 0;
                        if (file && this.isSameDomain(file) && lineno) {
                            var functionName = this.guessAnonymousFunction(file, lineno, charno);
                            stack[i] = frame.replace('{anonymous}', functionName);
                        }
                    }
                }
            }
            return stack;
        },

        guessAnonymousFunction: function (url, lineNo, charNo) {
            var ret;
            try {
                ret = this.findFunctionName(this.getSource(url), lineNo);
            } catch (e) {
                ret = 'getSource failed with url: ' + url + ', exception: ' + e.toString();
            }
            return ret;
        },

        findFunctionName: function (source, lineNo) {
            // FIXME findFunctionName fails for compressed source
            // (more than one function on the same line)
            // function {name}({args}) m[1]=name m[2]=args
            var reFunctionDeclaration = /function\s+([^(]*?)\s*\(([^)]*)\)/;
            // {name} = function ({args}) TODO args capture
            // /['"]?([0-9A-Za-z_]+)['"]?\s*[:=]\s*function(?:[^(]*)/
            var reFunctionExpression = /['"]?([$_A-Za-z][$_A-Za-z0-9]*)['"]?\s*[:=]\s*function\b/;
            // {name} = eval()
            var reFunctionEvaluation = /['"]?([$_A-Za-z][$_A-Za-z0-9]*)['"]?\s*[:=]\s*(?:eval|new Function)\b/;
            // Walk backwards in the source lines until we find
            // the line which matches one of the patterns above
            var code = "", line, maxLines = Math.min(lineNo, 20), m, commentPos;
            for (var i = 0; i < maxLines; ++i) {
                // lineNo is 1-based, source[] is 0-based
                line = source[lineNo - i - 1];
                commentPos = line.indexOf('//');
                if (commentPos >= 0) {
                    line = line.substr(0, commentPos);
                }
                // TODO check other types of comments? Commented code may lead to false positive
                if (line) {
                    code = line + code;
                    m = reFunctionExpression.exec(code);
                    if (m && m[1]) {
                        return m[1];
                    }
                    m = reFunctionDeclaration.exec(code);
                    if (m && m[1]) {
                        //return m[1] + "(" + (m[2] || "") + ")";
                        return m[1];
                    }
                    m = reFunctionEvaluation.exec(code);
                    if (m && m[1]) {
                        return m[1];
                    }
                }
            }
            return '(?)';
        }
    };

    return printStackTrace;
}));;
'use strict';

/*
 * AngularJS Toaster
 * Version: 0.4.7
 *
 * Copyright 2013 Jiri Kavulak.  
 * All Rights Reserved.  
 * Use, reproduction, distribution, and modification of this code is subject to the terms and 
 * conditions of the MIT license, available at http://www.opensource.org/licenses/mit-license.php
 *
 * Author: Jiri Kavulak
 * Related to project of John Papa and Hans Fjällemark
 */

angular.module('toaster', ['ngAnimate'])
.service('toaster', ['$rootScope', function ($rootScope) {
    this.pop = function (type, title, body, timeout, bodyOutputType, clickHandler) {
        this.toast = {
            type: type,
            title: title,
            body: body,
            timeout: timeout,
            bodyOutputType: bodyOutputType,
            clickHandler: clickHandler
        };
        $rootScope.$broadcast('toaster-newToast');
    };

    this.clear = function () {
        $rootScope.$broadcast('toaster-clearToasts');
    };
}])
.constant('toasterConfig', {
    'limit': 0,                   // limits max number of toasts 
    'tap-to-dismiss': true,
    'close-button': false,
    'newest-on-top': true,
    //'fade-in': 1000,            // done in css
    //'on-fade-in': undefined,    // not implemented
    //'fade-out': 1000,           // done in css
    // 'on-fade-out': undefined,  // not implemented
    //'extended-time-out': 1000,    // not implemented
    'time-out': 5000, // Set timeOut and extendedTimeout to 0 to make it sticky
    'icon-classes': {
        error: 'toast-error',
        info: 'toast-info',
        wait: 'toast-wait',
        success: 'toast-success',
        warning: 'toast-warning'
    },
    'body-output-type': '', // Options: '', 'trustedHtml', 'template'
    'body-template': 'toasterBodyTmpl.html',
    'icon-class': 'toast-info',
    'position-class': 'toast-top-right',
    'title-class': 'toast-title',
    'message-class': 'toast-message'
})
.directive('toasterContainer', ['$compile', '$timeout', '$sce', 'toasterConfig', 'toaster',
function ($compile, $timeout, $sce, toasterConfig, toaster) {
    return {
        replace: true,
        restrict: 'EA',
        scope: true, // creates an internal scope for this directive
        link: function (scope, elm, attrs) {

            var id = 0,
                mergedConfig;

            mergedConfig = angular.extend({}, toasterConfig, scope.$eval(attrs.toasterOptions));

            scope.config = {
                position: mergedConfig['position-class'],
                title: mergedConfig['title-class'],
                message: mergedConfig['message-class'],
                tap: mergedConfig['tap-to-dismiss'],
                closeButton: mergedConfig['close-button']
            };

            scope.configureTimer = function configureTimer(toast) {
                var timeout = typeof (toast.timeout) == "number" ? toast.timeout : mergedConfig['time-out'];
                if (timeout > 0)
                    setTimeout(toast, timeout);
            };

            function addToast(toast) {
                toast.type = mergedConfig['icon-classes'][toast.type];
                if (!toast.type)
                    toast.type = mergedConfig['icon-class'];

                id++;
                angular.extend(toast, { id: id });

                // Set the toast.bodyOutputType to the default if it isn't set
                toast.bodyOutputType = toast.bodyOutputType || mergedConfig['body-output-type'];
                switch (toast.bodyOutputType) {
                    case 'trustedHtml':
                        toast.html = $sce.trustAsHtml(toast.body);
                        break;
                    case 'template':
                        toast.bodyTemplate = toast.body || mergedConfig['body-template'];
                        break;
                }

                scope.configureTimer(toast);

                if (mergedConfig['newest-on-top'] === true) {
                    scope.toasters.unshift(toast);
                    if (mergedConfig['limit'] > 0 && scope.toasters.length > mergedConfig['limit']) {
                        scope.toasters.pop();
                    }
                } else {
                    scope.toasters.push(toast);
                    if (mergedConfig['limit'] > 0 && scope.toasters.length > mergedConfig['limit']) {
                        scope.toasters.shift();
                    }
                }
            }

            function setTimeout(toast, time) {
                toast.timeout = $timeout(function () {
                    scope.removeToast(toast.id);
                }, time);
            }

            scope.toasters = [];
            scope.$on('toaster-newToast', function () {
                addToast(toaster.toast);
            });

            scope.$on('toaster-clearToasts', function () {
                scope.toasters.splice(0, scope.toasters.length);
            });
        },
        controller: ['$scope', '$element', '$attrs', function ($scope, $element, $attrs) {

            $scope.stopTimer = function (toast) {
                if (toast.timeout) {
                    $timeout.cancel(toast.timeout);
                    toast.timeout = null;
                }
            };

            $scope.restartTimer = function (toast) {
                if (!toast.timeout)
                    $scope.configureTimer(toast);
            };

            $scope.removeToast = function (id) {
                var i = 0;
                for (i; i < $scope.toasters.length; i++) {
                    if ($scope.toasters[i].id === id)
                        break;
                }
                $scope.toasters.splice(i, 1);
            };

            $scope.click = function (toaster) {
                if ($scope.config.tap === true) {
                    if (toaster.clickHandler && angular.isFunction($scope.$parent.$eval(toaster.clickHandler))) {
                        var result = $scope.$parent.$eval(toaster.clickHandler)(toaster);
                        if (result === true)
                            $scope.removeToast(toaster.id);
                    } else {
                        if (angular.isString(toaster.clickHandler))
                            console.log("TOAST-NOTE: Your click handler is not inside a parent scope of toaster-container.");
                        $scope.removeToast(toaster.id);
                    }
                }
            };
        }],
        template:
        '<div  id="toast-container" ng-class="config.position">' +
            '<div ng-repeat="toaster in toasters" class="toast" ng-class="toaster.type" ng-click="click(toaster)" ng-mouseover="stopTimer(toaster)"  ng-mouseout="restartTimer(toaster)">' +
              '<button class="toast-close-button" ng-show="config.closeButton">&times;</button>' +
              '<div ng-class="config.title">{{toaster.title}}</div>' +
              '<div ng-class="config.message" ng-switch on="toaster.bodyOutputType">' +
                '<div ng-switch-when="trustedHtml" ng-bind-html="toaster.html"></div>' +
                '<div ng-switch-when="template"><div ng-include="toaster.bodyTemplate"></div></div>' +
                '<div ng-switch-default >{{toaster.body}}</div>' +
              '</div>' +
            '</div>' +
        '</div>'
    };
}]);;
$(document).on('click', '.btn.btn-inverse', function (e) {
    e.preventDefault();
    var $this = $(this);
    var $group = $this.closest('.collapse-group');
    var $collapse = $group.find('.collapse');
    $collapse.collapse({ toggle: false });
    $collapse.collapse('toggle');
    $group.find('.btn.btn-inverse').toggle();
});;
/* Modernizr 2.7.1 (Custom Build) | MIT & BSD  
 * Build: http://modernizr.com/download/#-teststyles-testprop-testallprops-hasevent-prefixes-domprefixes-cors
 */
; window.Modernizr = function (a, b, c) { function z(a) { i.cssText = a } function A(a, b) { return z(l.join(a + ";") + (b || "")) } function B(a, b) { return typeof a === b } function C(a, b) { return !!~("" + a).indexOf(b) } function D(a, b) { for (var d in a) { var e = a[d]; if (!C(e, "-") && i[e] !== c) return b == "pfx" ? e : !0 } return !1 } function E(a, b, d) { for (var e in a) { var f = b[a[e]]; if (f !== c) return d === !1 ? a[e] : B(f, "function") ? f.bind(d || b) : f } return !1 } function F(a, b, c) { var d = a.charAt(0).toUpperCase() + a.slice(1), e = (a + " " + n.join(d + " ") + d).split(" "); return B(b, "string") || B(b, "undefined") ? D(e, b) : (e = (a + " " + o.join(d + " ") + d).split(" "), E(e, b, c)) } var d = "2.7.1", e = {}, f = b.documentElement, g = "modernizr", h = b.createElement(g), i = h.style, j, k = {}.toString, l = " -webkit- -moz- -o- -ms- ".split(" "), m = "Webkit Moz O ms", n = m.split(" "), o = m.toLowerCase().split(" "), p = {}, q = {}, r = {}, s = [], t = s.slice, u, v = function (a, c, d, e) { var h, i, j, k, l = b.createElement("div"), m = b.body, n = m || b.createElement("body"); if (parseInt(d, 10)) while (d--) j = b.createElement("div"), j.id = e ? e[d] : g + (d + 1), l.appendChild(j); return h = ["&#173;", '<style id="s', g, '">', a, "</style>"].join(""), l.id = g, (m ? l : n).innerHTML += h, n.appendChild(l), m || (n.style.background = "", n.style.overflow = "hidden", k = f.style.overflow, f.style.overflow = "hidden", f.appendChild(n)), i = c(l, a), m ? l.parentNode.removeChild(l) : (n.parentNode.removeChild(n), f.style.overflow = k), !!i }, w = function () { function d(d, e) { e = e || b.createElement(a[d] || "div"), d = "on" + d; var f = d in e; return f || (e.setAttribute || (e = b.createElement("div")), e.setAttribute && e.removeAttribute && (e.setAttribute(d, ""), f = B(e[d], "function"), B(e[d], "undefined") || (e[d] = c), e.removeAttribute(d))), e = null, f } var a = { select: "input", change: "input", submit: "form", reset: "form", error: "img", load: "img", abort: "img" }; return d }(), x = {}.hasOwnProperty, y; !B(x, "undefined") && !B(x.call, "undefined") ? y = function (a, b) { return x.call(a, b) } : y = function (a, b) { return b in a && B(a.constructor.prototype[b], "undefined") }, Function.prototype.bind || (Function.prototype.bind = function (b) { var c = this; if (typeof c != "function") throw new TypeError; var d = t.call(arguments, 1), e = function () { if (this instanceof e) { var a = function () { }; a.prototype = c.prototype; var f = new a, g = c.apply(f, d.concat(t.call(arguments))); return Object(g) === g ? g : f } return c.apply(b, d.concat(t.call(arguments))) }; return e }); for (var G in p) y(p, G) && (u = G.toLowerCase(), e[u] = p[G](), s.push((e[u] ? "" : "no-") + u)); return e.addTest = function (a, b) { if (typeof a == "object") for (var d in a) y(a, d) && e.addTest(d, a[d]); else { a = a.toLowerCase(); if (e[a] !== c) return e; b = typeof b == "function" ? b() : b, typeof enableClasses != "undefined" && enableClasses && (f.className += " " + (b ? "" : "no-") + a), e[a] = b } return e }, z(""), h = j = null, e._version = d, e._prefixes = l, e._domPrefixes = o, e._cssomPrefixes = n, e.hasEvent = w, e.testProp = function (a) { return D([a]) }, e.testAllProps = F, e.testStyles = v, e }(this, this.document), Modernizr.addTest("cors", !!(window.XMLHttpRequest && "withCredentials" in new XMLHttpRequest));
(function () {
    'use strict';

    var app = angular.module('common', []);

    app.value('webShop2AppSettings', { folderName: __appSettingsInitialValue.folderName, useSSL: __appSettingsInitialValue.useSSL, requiresSSL: __appSettingsInitialValue.requiresSSL, loadBasket: __appSettingsInitialValue.loadBasket == undefined ? true : __appSettingsInitialValue.loadBasket, homePageIsBooking: (__appSettingsInitialValue.homePageIsBooking != undefined && __appSettingsInitialValue.homePageIsBooking), disableTracking: (__appSettingsInitialValue.disableTracking != undefined) ? __appSettingsInitialValue.disableTracking : false, customization: (__appSettingsInitialValue.customization != undefined) ? __appSettingsInitialValue.customization : {} });
    app.value('pageModel', { currentStep: '', isLoading: false, startStep: (__appSettingsInitialValue.homePageIsBooking == undefined || !__appSettingsInitialValue.homePageIsBooking) ? '' : 'Shop', hasCrossSelling: false, lastAlertTime: '', requiresPatronLogin: false, hasPendingBasketValidation: false, editAddress: false });

    app.filter('range', function () {
        return function (input, start, end) {
            start = parseInt(start);
            end = parseInt(end);
            var direction = (start <= end) ? 1 : -1;
            while (start != end) {
                input.push(start);
                start += direction;
            }
            input.push(end);                        
            return input;
        };
    });
    app.filter('numberFixedLen', function () {
        return function (number, len) {
            var num = parseInt(number, 10);
            len = parseInt(len, 10);
            if (isNaN(num) || isNaN(len)) {
                return number;
            }
            num = '' + num;
            while (num.length < len) {
                num = '0' + num;
            }
            return num;
        };
    });

    app.filter("sanitize", ['$sce', function ($sce) {
        return function (htmlContent) {
            return $sce.trustAsHtml(htmlContent);
        }
    }]);
})();

String.prototype.format = function () {
    var theString = this;

    for (var i = 0; i < arguments.length; i++) {
        var regEx = new RegExp("\\{" + i + "\\}", "gm");
        var theString = theString.replace(regEx, arguments[i]);
    }
    return theString;
};

function getQueryStringValue(paramName, url) {
    if (url === undefined)
        url = location.search;
    paramName = paramName.replace(/[\[]/, "\\\[").replace(/[\]]/, "\\\]");
    var regex = new RegExp("[\\?&]" + paramName + "=([^&#]*)"),
            results = regex.exec(url);

    return results == null ? "" : decodeURIComponent(results[1].replace(/\+/g, " "));
};

alertModalInstanceController = function ($scope, $modalInstance, header, message) {
    $scope.header = header;
    $scope.message = message;

    $scope.ok = function () {
        $modalInstance.close();
    }

    $scope.cancel = cancelModalInstanceController;
}
alertModalInstanceController.$inject = ['$scope', '$modalInstance', 'header', 'message'];

cancelModalInstanceController = function ($scope, $modalInstance) {
    $scope.cancel = function () {
        $modalInstance.dismiss('cancel');
    }
}
cancelModalInstanceController.$inject = ['$scope', '$modalInstance'];

validateDate = function (date, pastDateOnly) {
    var flag = true;
    if (date.day || date.month || date.year) {
        var day = parseInt(date.day);
        var month = parseInt(date.month);
        var year = parseInt(date.year);
        if (!isNaN(day) || !isNaN(month) || !isNaN(year)) {
            if (day < 1 || day > 31)
                flag = false;
            if (month < 1 || month > 12)
                flag = false;
            if (year < 1800 || year > 2200)
                flag = false;

            if ((month == 4 || month == 6 || month == 9 || month == 11) && (day >= 31))
                flag = false;

            if (month == 2) {
                if (year % 4 != 0) {
                    if (day > 28)
                        flag = false;
                }
                if (year % 4 == 0) {
                    if (day > 29)
                        flag = false;
                }
            }

            var dob = new Date();
            dob.setFullYear(year, month - 1, day);
            if (isNaN(dob.getTime()))
                flag = false;

            if (pastDateOnly) {
                var today = new Date();
                if (dob > today)
                    flag = false;
            }
        }
    }

    return flag;
}

findStringInArray = function (search, array) {
    var results = [];
    for (var i = 0; i < array.length; i++) {
        if (array[i].indexOf(search) > -1) {
            results.push(array[i]);
        }
    }

    return results.length > 0 ? search + $.map(results, function (obj) { return obj.split('=')[1] }).join('<br>') : '';
}

checkCompareType = function (sourceValue, compareType, compareValue) {
    if (typeof sourceValue === "boolean")
        compareValue = compareValue ? compareValue.toString().toLowerCase() : '';

    sourceValue = sourceValue ? sourceValue.toString() : '';
    compareValue = compareValue ? compareValue.toString() : '';
    switch (compareType) {
        case 'Equals':
            return sourceValue == compareValue;
        case 'NotEquals':
            return sourceValue != compareValue;
        case 'Greater':
            return sourceValue > compareValue;
        case 'GreaterOrEqual':
            return sourceValue >= compareValue;
        case 'Lesser':
            return sourceValue < compareValue;
        case 'LesserOrEqual':
            return sourceValue <= compareValue;
        case 'In':
            return $.inArray(sourceValue, compareValue.split(',')) !== -1;
        case 'NotIn':
            return $.inArray(sourceValue, compareValue.split(',')) == -1;
        case 'Contains':
            return sourceValue.indexOf(compareValue) !== -1;
        case 'StartsWith':
            return sourceValue.indexOf(compareValue) == 0;
        case 'EndsWith':
            return sourceValue.indexOf(compareValue, sourceValue.length - compareValue.length) !== -1;;
    }
}

var ticketportal = window.ticketportal || {};
ticketportal.webshop = ticketportal.webshop || {};
ticketportal.webshop.fixUrl = function (folderName, url) {
    if (folderName && folderName.length > 0)
        url = "/" + folderName + url;
    return url;
};

ticketportal.webshop.handleOldBrowser = function (folderName) {
    if (document.cookie.indexOf("browserNotSupportedMessage") >= 0)
        return;

    if (folderName && folderName.length > 0)
        window.location = ticketportal.webshop.fixUrl(folderName, '/browsernotsupported');
    else
        window.location = '/browsernotsupported';
};

ticketportal.webshop.ensurePageIsSecure = function () {
    if (window.location.protocol != "https:") window.location.href = "https:" + window.location.href.substring(window.location.protocol.length);
};

ticketportal.webshop.inIframe = function () {
    try {
        return window.self !== window.top;
    } catch (e) {
        return true;
    }
};

/*
* Generate the tooltip text color depend on the background
*/
textColorGenerator = function (hexcolor) {

    // If a leading # is provided, remove it
    if (hexcolor.slice(0, 1) === '#') {
        hexcolor = hexcolor.slice(1);
    }

    // If a three-character hexcode, make six-character
    if (hexcolor.length === 3) {
        hexcolor = hexcolor.split('').map(function (hex) {
            return hex + hex;
        }).join('');
    }

    // Convert to RGB value
    var r = parseInt(hexcolor.substr(0, 2), 16);
    var g = parseInt(hexcolor.substr(2, 2), 16);
    var b = parseInt(hexcolor.substr(4, 2), 16);

    // Get YIQ ratio
    var yiq = ((r * 299) + (g * 587) + (b * 114)) / 1000;

    // Check contrast
    return (yiq > 128) ? 'black' : 'white';
};
;
(function () {
    'use strict';

    var app = angular.module('webShop2App', [
        // Angular modules
        'ngCookies',
        'ngAnimate',
        'ngSanitize', // added for tooltips with html content

        // Custom modules
        'common',
        'common.directives',
        'exceptionHandler',

        // 3rd Party Modules
        'ui.bootstrap',
        'toaster'
    ]);


    app.config(['$provide', function ($provide) {
        if (enableEnhancedEcommerce == false) {                         //enableEnhancedEcommerce is a global variable
            $provide.factory("ecommerceService", function service() {
                return {
                    pushImpressions: function () { },
                    pushAdmissionTicketToCart: function () { },
                    pushCrossSellingPerformanceToCart: function () { },
                    pushRemoveItemsFromCart: function () { },
                    pushStartCheckout: function () { },
                    pushCheckout: function () { },
                    pushAddToCartPerformance: function () { },
                    parseIdFromPerformanceTag: function () { },
                    pushAddToCartGiftCertificates: function () {},
                    pushAddToCartProduct: function () { },
                    addSubscriptionToCart: function () { },
                    pushAddPackageTicketToCart: function () { },
                    pushCrossSellingProductsToCart: function () { }
                }
            }
            );
        }
    }]);   


    // Execute bootstrapping code and any dependencies.
    app.run(['$rootScope',
        function ($rootScope) {
        }
    ]);

})();;
(function () {
    'use strict';

    var app = angular.module('exceptionHandler', []);

    var serviceId = 'stacktraceService';
    app.factory(serviceId, [stacktraceService]);

    function stacktraceService() {
        return ({
            print: printStackTrace
        })
    }

    var serviceId = 'errorLogService';
    app.factory(serviceId, ['$log', '$window', '$injector', 'stacktraceService', 'webShop2AppSettings', errorLogService]);

    function errorLogService($log, $window, $injector, stacktraceService, webShop2AppSettings) {
        function log(exception, cause) {
            // Pass off the error to the default error handler
            // on the AngualrJS logger. This will output the
            // error to the console (and let the application
            // keep running normally for the user).
            $log.error.apply($log, arguments);

            try {
                var errorMessage = exception.toString();
                var stackTrace = stacktraceService.print({ e: exception });
                var $cookies = $injector.get('$cookies');

                var formData = {
                    errorUrl: $window.location.href,
                    errorMessage: errorMessage,
                    stackTrace: angular.toJson(stackTrace),
                    cause: (cause || ""),
                    browserVersion: $window.navigator.userAgent,
                    cookies:  angular.toJson($cookies)
                };
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ErrorLog');

                $.ajax({
                    type: "POST",
                    url: postUrl,
                    data: formData,
                    dataType: "json"
                })
                .fail(function (response) {
                    if (response && response.status != 200) {
                        $log.warn("Error logging failed");
                        $log.log("Status " + response.status);
                    }
                });
            } catch (loggingError) {
                $log.warn("Error logging failed");
                $log.log(loggingError);
            } finally {
                var $rootScope = $injector.get('$rootScope');
                $rootScope.$broadcast('event:ErrorHandled');
            }
        }

        // Return the logging function.
        return (log);
    }

    app.provider("$exceptionHandler", function () {
        this.$get = ['errorLogService', function (errorLogService) {
            return (errorLogService);
        }]
    })

    var serviceId = 'errorHttpInterceptor';
    app.factory(serviceId, ['$q', '$log', '$window', '$injector', '$location', 'webShop2AppSettings', errorHttpInterceptorService]);
    
    function errorHttpInterceptorService($q, $log, $window, $injector, $location, webShop2AppSettings) {
        return {
            responseError: function responseError(rejection) {
                // Log error
                try {
                    var $cookies = $injector.get('$cookies');

                    var formData = {
                        errorUrl: rejection.config.url,
                        errorMessage: 'Status=' + rejection.status,
                        stackTrace: angular.toJson(rejection.config),
                        browserVersion: $window.navigator.userAgent,
                        cookies: angular.toJson($cookies)
                    };
                    var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ErrorLog');

                    $.ajax({
                        type: "POST",
                        url: postUrl,
                        data: formData,
                        dataType: "json"
                    })
                    .fail(function (response) {
                        if (response && response.status != 200) {
                            $log.warn("Error logging failed");
                            $log.log("Status " + response.status);
                        }
                    });
                } catch (loggingError) {
                    $log.warn("Error logging failed");
                    $log.log(loggingError);
                } finally {
                    var $rootScope = $injector.get('$rootScope');
                    $rootScope.$broadcast('event:ErrorHandled');
                }

                return $q.reject(rejection);
            },

            response: function response(result) {
                if (result.data.redirectUrl) {
                    $window.location.href = result.data.redirectUrl;
                    return $q.reject(result);
                }
                return result;
            }
        }
    }

    app.config(['$httpProvider', function ($httpProvider) {
        $httpProvider.interceptors.push('errorHttpInterceptor');
    }])

})();
;
(function () {
    'use strict';

    var module = angular.module('common.directives', [])
        .directive('btnLoading', [btnLoading])
        .directive('tpCustomvalidator', [tpCustomvalidator])
        .directive('enabledValidation', [enabledValidation])
        .directive('showErrorMessage', [showErrorMessage])
        .directive('showConstraintError', [showConstraintError])
        .directive('ngMatch', [ngMatch])
        .directive('enableConstraint', [enableConstraint])
        .directive('visibleConstraint', [visibleConstraint])
        .directive('requiredConstraint', [requiredConstraint])
        .directive('validationConstraint', [validationConstraint])
        .directive('contentConstraint', [contentConstraint])
        .directive('tpConvertToNumber', [tpConvertToNumber])
        .directive('tpPlusMinus', [tpPlusMinus]);

    function tpConvertToNumber() {
        return {
            require: 'ngModel',
            link: function (scope, element, attrs, ngModel) {
                ngModel.$parsers.push(function (val) {
                    return parseInt(val, 10);
                });
                ngModel.$formatters.push(function (val) {
                    return '' + val;
                });
            }
        };
    }

    function tpCustomvalidator() {
        return {
            restrict: 'A',
            require: 'ngModel',
            link: function (scope, elm, attr, ngModelCtrl) {
                var validateFunctionNames = attr["validateFunctions"].split(",");
                var validatorNames = attr["tpCustomvalidator"].split(",");
                ngModelCtrl.$parsers.push(function (value) {
                    var hasErrors = false;
                    angular.forEach(validateFunctionNames, function (functionName, index) {
                        if (!scope[functionName]) {
                            console.log('There is no function with name ' + functionName + ' available on the scope. Please make sure the function exists on current scope or its parent.');
                        } else {
                            var result = scope[functionName](value);
                            if (result && result != false) {
                                ngModelCtrl.$setValidity(validatorNames[index], true);
                            } else {
                                ngModelCtrl.$setValidity(validatorNames[index], false);
                                hasErrors = true;
                            }

                        }
                    });
                    //return hasErrors ? undefined : value;
                    return value; // we set the value even though it might be invalid
                });
            }
        }
    };

    function btnLoading() {
        return function (scope, element, attrs) {
            scope.$watch(function () {
                return scope.$eval(attrs.ngDisabled);
            },
                function (newVal) {
                    if (newVal) {
                        return;
                    } else {
                        return scope.$watch(function () {
                            return scope.$eval(attrs.btnLoading);
                        },
                            function (loading) {
                                if (typeof element.button === 'function') {
                                    if (loading)
                                        return element.button('loading');

                                    element.button('reset');
                                }
                            });
                    }
                });
        }
    }

    function enabledValidation() {
        return {
            require: '^form',
            restrict: 'A',
            link: function (scope, element, attrs, form) {
                var control;

                scope.$watchGroup([attrs.ngValidate1, attrs.ngValidate2], function (newValue) {
                    if (!control) {
                        control = form[element.attr("name")];
                    }
                    if (newValue[0] != false && newValue[1] != false) {
                        form.$addControl(control);
                        angular.forEach(control.$error, function (validity, validationToken) {
                            form.$setValidity(validationToken, !validity, control);
                        });
                    } else {
                        form.$removeControl(control);
                    }
                });
            }
        };
    }

    function showErrorMessage() {
        return {
            require: '^form',
            restrict: 'A',
            link: function (scope, element, attrs) {
                scope.$watch(attrs.ngMessageType, function (newValue) {
                    var errorMessages = element.attr('errorMessages').split('|');
                    if (newValue) {
                        var errorMessage = '';
                        if (newValue.required)
                            errorMessage = findStringInArray('Required=', errorMessages).split('=');
                        else if (newValue.number)
                            errorMessage = findStringInArray('Number=', errorMessages).split('=');
                        else if (newValue.minlength) {
                            if (element.attr('minLength')) {
                                var minLength = element.attr('minLength').length > 0 ? element.attr('minLength') : '8';
                                errorMessage = findStringInArray('MinLength=', errorMessages).replace('{0}', minLength).split('=');
                            }
                            else
                                errorMessage = findStringInArray('MinLength=', errorMessages).split('=');
                        }
                        else if (newValue.maxlength) {
                            if (element.attr('maxLength')) {
                                var maxLength = element.attr('maxLength').length > 0 ? element.attr('maxLength') : '8';
                                errorMessage = findStringInArray('MaxLength=', errorMessages).replace('{0}', maxLength).split('=');
                            }
                            else
                                errorMessage = findStringInArray('MaxLength=', errorMessages).split('=');
                        }
                        else if (newValue.min || newValue.max)
                            errorMessage = findStringInArray('Range=', errorMessages).split('=');
                        else if (newValue.pattern)
                            errorMessage = findStringInArray('Pattern=', errorMessages).split('=');
                        else if (newValue.date)
                            errorMessage = findStringInArray('Date=', errorMessages).split('=');
                        else if (newValue.datePast)
                            errorMessage = findStringInArray('DatePast=', errorMessages).split('=');
                        else if (newValue.customRequired)
                            errorMessage = findStringInArray('CustomRequired=', errorMessages).split('=');
                        else if (newValue.validation)
                            errorMessage = findStringInArray('Validation=', errorMessages).split('=');

                        if (errorMessage.length == 2)
                            element.html(errorMessage[1]);
                    }
                }, true);
            }
        };
    }

    function showConstraintError() {
        return {
            require: '^form',
            scope: true,
            restrict: 'A',
            link: function (scope, element, attrs) {
                scope.$watch(attrs.showConstraintError, function (newValue) {
                    if (newValue && newValue.length > 0) {
                        var constraintErrors = scope.$eval(attrs.showConstraintError);
                        var htmlString = '';
                        angular.forEach(constraintErrors, function (constraintError, index) {
                            if (htmlString.length > 0)
                                htmlString += '<br/>';

                            htmlString += constraintError;
                        });
                        element.html(htmlString);
                    }
                }, true);
            }
        };
    }

    function ngMatch() {
        return {
            restrict: 'A',
            scope: true,
            require: 'ngModel',
            link: function (scope, elm, attr, ngModelCtrl) {
                function validate() {
                    var e1 = scope.$eval(attr.ngModel);
                    var e2 = scope.$eval(attr.ngMatch);
                    return e1 == e2;
                };

                scope.$watch(validate, function (n) {
                    ngModelCtrl.$setValidity("match", n);
                });
            }
        };
    }

    function enableConstraint() {
        return {
            require: '^form',
            restrict: 'A',
            link: function (scope, element, attrs) {
                var targets = element.attr('targetEnable').match(/[^\|]+/g);
                if (!(element[0].name in scope))
                    scope[element[0].name] = {};

                scope[element[0].name]['enable'] = Array.apply(null, targets).map(function (x, i) { return { name: x.match(/[^\;]+/g)[0], value: true }; });
                $.each(targets, function (index, value) {
                    var targetParams = value.match(/[^\;]+/g);
                    var controlModel = $('#' + targetParams[0])[0].attributes['ng-model'].value;
                    var comparerEnable = targetParams[1];
                    var valueEnable = targetParams[2];
                    scope.$watch(controlModel, function (newValue) {
                        if (checkCompareType(newValue, comparerEnable, valueEnable)) {
                            $.grep(scope[element[0].name].enable.slice(), function (el) { return el.name === targetParams[0]; })[0].value = true;
                        }
                        else {
                            $.grep(scope[element[0].name].enable.slice(), function (el) { return el.name === targetParams[0]; })[0].value = false;
                        }

                        var isEnabled = scope[element[0].name].enable.every(function (el, index, array) { return el.value === true; });
                        element.prop("disabled", !isEnabled);
                    }, true);
                });
            }
        };
    }

    function visibleConstraint() {
        return {
            require: '^form',
            restrict: 'A',
            link: function (scope, element, attrs) {
                var targets = element.attr('targetVisible').match(/[^\|]+/g);
                if (!(element[0].name in scope))
                    scope[element[0].name] = {};

                scope[element[0].name]['visible'] = Array.apply(null, targets).map(function (x, i) { return { name: x.match(/[^\;]+/g)[0], value: true }; });
                $.each(targets, function (index, value) {
                    var targetParams = value.match(/[^\;]+/g);
                    var controlModel = $('#' + targetParams[0])[0].attributes['ng-model'].value;
                    var comparerVisible = targetParams[1];
                    var valueVisible = targetParams[2];
                    scope.$watch(controlModel, function (newValue) {
                        if (checkCompareType(newValue, comparerVisible, valueVisible)) {
                            $.grep(scope[element[0].name].visible.slice(), function (el) { return el.name === targetParams[0]; })[0].value = true;
                        }
                        else {
                            $.grep(scope[element[0].name].visible.slice(), function (el) { return el.name === targetParams[0]; })[0].value = false;
                        }

                        if (scope[element[0].name].visible.every(function (el, index, array) { return el.value === true; })) {
                            element.closest('.parentfield').show();
                            var path = element.attr('ng-model');
                            var attributes = path.split('.');
                            var property = getPathParentProperty(scope, attributes);
                            property.unregister();
                            delete property.unregister;

                        }
                        else {
                            element.closest('.parentfield').hide();
                            var path = element.attr('ng-model');
                            var attributes = path.split('.');
                            var property = getPathParentProperty(scope, attributes);
                            var unregister = scope.$watch(path, function (newValue) {
                                if (newValue) {
                                    var attributes = path.split('.');
                                    var property = getPathParentProperty(scope, attributes);
                                    property[attributes[0]] = '';
                                }
                            });
                            property.unregister = unregister;
                        }
                    }, true);
                });
            }
        };
    }

    function requiredConstraint() {
        return {
            require: '^form',
            restrict: 'A',
            link: function (scope, element, attrs, form) {
                var targets = element.attr('targetRequired').match(/[^\|]+/g);
                if (!(element[0].name in scope))
                    scope[element[0].name] = {};

                scope[element[0].name]['required'] = Array.apply(null, targets).map(function (x, i) { return { name: x.match(/[^\;]+/g)[0], value: true }; });
                $.each(targets, function (index, value) {
                    var targetParams = value.match(/[^\;]+/g);
                    var targetControl = $('#' + targetParams[0]);
                    var controlModel = targetControl[0].attributes['ng-model'].value;
                    var comparerRequired = targetParams[1];
                    var valueRequired = targetParams[2];
                    scope.$watch(controlModel, function (newValue) {
                        var modelValue = form[element.attr('name')].$modelValue;
                        modelValue = $(element).is('select') && modelValue == '0' ? '' : modelValue;
                        var isCompareValid = checkCompareType(newValue, comparerRequired, valueRequired) && modelValue == '';

                        // Set source constraint
                        $.grep(scope[element[0].name].required.slice(), function (el) { return el.name === targetParams[0]; })[0].value = isCompareValid;
                        var constraintErrorModel = $(element).nextAll(".form-field-validator:first").attr('show-constraint-error').split('.');
                        var constraintErrorProperty = scope;
                        $.each(constraintErrorModel, function (i, p) {
                            if (constraintErrorProperty)
                                constraintErrorProperty = constraintErrorProperty[p];
                            else
                                return false;
                        });
                        if (constraintErrorProperty) {
                            var validationControl = $(element).nextAll(".form-field-validator:first");
                            var errorMessage = $.grep(validationControl.attr('errormessages').split('|').slice(), function (el) { return el.startsWith(targetParams[0] + '_CustomRequired') })[0].split('=')[1];
                            var errorIndex = $.inArray(errorMessage, constraintErrorProperty);
                            if (isCompareValid && errorIndex == -1)
                                constraintErrorProperty.push(errorMessage);
                            else if (!isCompareValid && index > -1)
                                constraintErrorProperty.splice(errorIndex, 1);

                            var isValid = scope[element[0].name].required.every(function (el, index, array) { return el.value === false; });
                            form[element.attr('name')].$setValidity("customRequired", isValid);
                        }

                        // Set target constraint
                        var hasTargetConstraint = scope[targetParams[0]] !== undefined && scope[targetParams[0]].required !== undefined;
                        if (hasTargetConstraint) {
                            $.grep(scope[targetParams[0]].required.slice(), function (el) { return el.name === element[0].name; })[0].value = isCompareValid;
                            var constraintErrorModel = targetControl.nextAll(".form-field-validator:first").attr('show-constraint-error').split('.');
                            var constraintErrorProperty = scope;
                            $.each(constraintErrorModel, function (i, p) {
                                if (constraintErrorProperty)
                                    constraintErrorProperty = constraintErrorProperty[p];
                                else
                                    return false;
                            });
                            if (constraintErrorProperty) {
                                var validationControl = targetControl.nextAll(".form-field-validator:first");
                                var errorMessage = $.grep(validationControl.attr('errormessages').split('|').slice(), function (el) { return el.startsWith(element[0].name + '_CustomRequired') })[0].split('=')[1];
                                var errorIndex = $.inArray(errorMessage, constraintErrorProperty);
                                if (isCompareValid && errorIndex == -1)
                                    constraintErrorProperty.push(errorMessage);
                                else if (!isCompareValid && index > -1)
                                    constraintErrorProperty.splice(errorIndex, 1);

                                var isValid = scope[targetControl[0].name].required.every(function (el, index, array) { return el.value === false; });
                                form[targetParams[0]].$setValidity("customRequired", isValid);
                            }
                        }
                    }, true);
                });
            }
        };
    }

    function validationConstraint() {
        return {
            require: '^form',
            restrict: 'A',
            link: function (scope, element, attrs, form) {
                scope.$watch(attrs.ngModel, function (newValue) {
                    var targets = element.attr('targetValidation').match(/[^\|]+/g);
                    if (!(element[0].name in scope))
                        scope[element[0].name] = {};

                    scope[element[0].name]['validation'] = Array.apply(null, targets).map(function (x, i) { return { name: x.match(/[^\;]+/g)[0], value: true }; });
                    $.each(targets, function (index, value) {
                        var targetParams = value.match(/[^\;]+/g);
                        var controlValidation = form[targetParams[0]];
                        var comparerValidation = targetParams[1];
                        var valueValidation = controlValidation ? controlValidation.$modelValue : targetParams[2];
                        var isValid = checkCompareType(newValue, comparerValidation, valueValidation);
                        if (controlValidation) {
                            var targetControlValue = controlValidation.$modelValue;
                            isValid = checkCompareType(newValue, comparerValidation, targetControlValue);
                        }
                        else {
                            isValid = checkCompareType(newValue, comparerValidation, valueValidation);
                        }
                        $.grep(scope[element[0].name].validation.slice(), function (el) { return el.name === targetParams[0]; })[0].value = isValid;
                        var constraintErrorModel = $(element).nextAll(".form-field-validator:first").attr('show-constraint-error').split('.');
                        var constraintErrorProperty = scope;
                        $.each(constraintErrorModel, function (i, p) {
                            if (constraintErrorProperty)
                                constraintErrorProperty = constraintErrorProperty[p];
                            else
                                return false;
                        });
                        if (constraintErrorProperty) {
                            var validationControl = $(element).nextAll(".form-field-validator:first");
                            var errorMessage = $.grep(validationControl.attr('errormessages').split('|').slice(), function (el) { return el.startsWith(targetParams[0] + '_Validation') })[0].split('=')[1];
                            var errorIndex = $.inArray(errorMessage, constraintErrorProperty);
                            if (!isValid && errorIndex == -1)
                                constraintErrorProperty.push(errorMessage);
                            else if (isValid && index > -1)
                                constraintErrorProperty.splice(errorIndex, 1);

                            isValid = scope[element[0].name].validation.every(function (el, index, array) { return el.value === true; });
                            form[element.attr('name')].$setValidity("validation", isValid);
                        }
                    });
                }, true);
            }
        };
    }

    function contentConstraint() {
        return {
            require: '^form',
            restrict: 'A',
            link: function (scope, element, attrs) {
                var target = element.attr('targetContent').match(/[^\|]+/g)[0];
                var targetParams = target.match(/[^\;]+/g);
                var targetControl = $('#' + targetParams[0]);
                var controlModel = targetControl[0].attributes['ng-model'].value;
                scope.$parent.$watch(controlModel, function (newValue) {
                    if (newValue && newValue.length > 0 && scope.$parent !== null) {
                        scope.$parent.pageService.pageModel.isLoading = true;
                        if (targetControl.is("select")) {
                            var constraintModel = element[0].attributes['content-constraint'].value.split('.');
                            var url = ticketportal.webshop.fixUrl(scope.$parent.webShop2AppSettings.folderName, '/GetFormFieldContent');
                            url += "?targetControlId=" + element.attr('name');
                            url += "&targetControlCode=" + constraintModel[3];
                            url += "&selectedValue=" + newValue;
                            var constraintProperty = scope;
                            $.each(constraintModel, function (i, p) {
                                if (constraintProperty)
                                    constraintProperty = constraintProperty[p];
                                else
                                    return false;
                            });
                            if (constraintProperty) {
                                constraintProperty.items.length = 0;

                                $.ajax({
                                    type: "GET",
                                    url: url,
                                    dataType: "json",
                                    success: function (data) {
                                        if (data.succeeded) {
                                            constraintProperty.items.length = 0;
                                            var found = false;
                                            $.each(data.fieldContent, function (index, item) {
                                                if (item.id == constraintProperty.value)
                                                    found = true;
                                                constraintProperty.items.push({ value: item.id > 0 ? item.id : "", name: item.name });
                                            });

                                            if (!found)
                                                constraintProperty.value = "";
                                        }
                                    },
                                    complete: function () {
                                        scope.$parent.pageService.pageModel.isLoading = false;
                                        scope.$parent.$apply();
                                    }
                                });
                            }
                        }
                        else
                            scope.$parent.pageService.pageModel.isLoading = false;
                    }
                }, true);
            }
        };
    }

    function tpPlusMinus() {
        return {
            restrict: 'E', // elements only
            scope: {
                min: '=min',
                max: '=max',
                value: '=value'
            },
            controller: ['$scope', function ($scope) {
                $scope.updateValue = function (amount) {
                    var currentValueSafe = ($scope.value === undefined) ? 0 : +$scope.value;
                    var updatedValue = currentValueSafe + amount;
                    if (updatedValue >= $scope.min && ($scope.max === undefined || updatedValue <= +$scope.max)) {
                        $scope.value = updatedValue;
                    }
                }
            }],
            template: '<button ng-click="updateValue(+1)" type="button" class="btn"><i class="glyphicon glyphicon-plus"></i></button><button type="button" class="btn" ng-click="updateValue(-1)"><i class="glyphicon glyphicon-minus"></i></button>'
        };
    };

    function getPathParentProperty(scope, attributes) {
        var parent = scope;
        var current;
        var attribute;

        while (attributes.length > 1 && (attribute = attributes.shift()) && (current = parent[attribute] || (parent[attribute] = {}))) {
            parent = current;
        }

        return current;
    }

})();;
// Modified:		09.03.2017 Starticket AG, St. Gallen fbe  : #2481 Festi'Neuch: Discount selection is not aligned with price category

/*  
    Class: performanceModel
    Handles the reservation of a performance (single ticket / package)    
*/
function performanceModel(source) {
    this.isDisplayingReservationInfoTab = false;
    this.isDisplayingPriceInfoTab = false;
    this.isDisplayingPromotionInfoTab = false;
    this.showPriceCategoryDiscounts = false;
    this.flattenPriceCategory = null;        
    this.venueAndLocation = '';
    if (source) {
        if (source.venue && source.venue.length > 0)
            this.venueAndLocation = source.venue;

        if (source.location && source.location.length > 0 && source.location.localeCompare(this.venueAndLocation, undefined, { sensitivity: 'accent' }) !== 0) {
            if (this.venueAndLocation.length > 0)
                this.venueAndLocation += ', ';
            this.venueAndLocation += source.location;
        }
    }
        
    this.allowedCurrencies = new Array();

    if (source)
        angular.extend(this, source);
    
    this.init = function (options) {
        if (this.reservationInformation && this.reservationInformation.length > 0)
            this.isDisplayingReservationInfoTab = true;
        else
            this.isDisplayingPriceInfoTab = true;

        // find available currencies
        this.allowedCurrencies.length = 0;
        if (options && options.allowedCurrencies && options.allowedCurrencies.length > 0) {
            for (var i = options.allowedCurrencies.length - 1; i >= 0; --i) {
                var basketAllowedCurrency = options.allowedCurrencies[i];
                if (this.hasPriceForCurrency(basketAllowedCurrency.symbol)) {
                    this.allowedCurrencies.push(basketAllowedCurrency);
                }
            }
        }        
    };

    this.hasPriceForCurrency = function (currencySymbol) {
        for (var i = this.priceCategories.length - 1; i >= 0; --i) {
            var priceCategory = this.priceCategories[i];
            for (var priceIndex = priceCategory.prices.length - 1; priceIndex >= 0; --priceIndex) {
                if (priceCategory.prices[priceIndex].currencySymbol == currencySymbol)
                    return true;
            }
        }

        return false;
    };

    this.getPrice = function (priceCategoryID, currencySymbol) {
        var price = null;
        for (var i = this.priceCategories.length - 1; !price && i >= 0; --i) {
            var priceCategory = this.priceCategories[i];
            if (priceCategory.uniqueId == priceCategoryID)
                for (var priceIndex = priceCategory.prices.length - 1; !price && priceIndex >= 0; --priceIndex) {
                    if (priceCategory.prices[priceIndex].currencySymbol == currencySymbol)
                        price = priceCategory.prices[priceIndex];
                }
        }
        return price;
    }

    this.findSectionByCode = function (sectionCode) {
        for (var i = this.sections.length - 1; i >= 0; --i) {
            var section = this.sections[i];
            if (section.code == sectionCode)
                return section;
        }

        return null;
    }
};

/*
    Class: bestPlaceSelectionModel
    Collection of best place options
*/
function bestPlaceSelectionModel(sectionId, sectionName, sectionCode) {
    this.sectionId = sectionId == undefined ? '' : sectionId;
    this.sectionName = sectionName == undefined ? '' : sectionName;
    this.sectionCode = sectionCode == undefined ? '' : sectionCode;
    this.priceCategories = new Array();
    this.setPriceCategories = function (values) {
        values.sort(function (x, y) {
            var result = x.priority - y.priority;
            if (result == 0) {
                if (x.name < y.name)
                    result = -1
                else if (x.name > y.name)
                    result = 1;
            }
            return result;
        });

        for (var i = 0; i < values.length; ++i)
            this.priceCategories.push(values[i]);
    }
    this.isValid = false;
    this.reset = function () {
        this.sectionId = '';
        this.sectionName = '';
        this.sectionCode = '';
        this.priceCategories.length = 0;
        this.isValid = false;
    };

    this.isSoldOut = function () {
        return this.priceCategories.every(function (priceCategory) { return priceCategory.isSoldOut; });
    }
};

/*
    Class: bestPlaceSelectionCategoryModel
    Price categories of a bestPlaceSelectionModel
*/
function bestPlaceSelectionCategoryModel(id, name, fullPriceName, colorCode, priority, isSoldOut, prices, currencySymbol, itemOptions, ticketAvailabilityText, showDiscountBestPlace) {
    this.id = id == undefined ? '' : id;
    this.priority = priority == undefined ? 0 : priority;
    this.name = name == undefined ? '' : name;
    this.fullPriceName = fullPriceName == undefined ? '' : fullPriceName;
    this.colorCode = colorCode == undefined ? '' : colorCode;
    this.isSoldOut = isSoldOut == undefined ? false : isSoldOut;
    this.prices = prices;
    this.currencySymbol = currencySymbol;
    this.ticketAmount = '';
    this.itemOptions = itemOptions == undefined ? [] : itemOptions;
    this.price = '';
    this.ticketAvailabilityText = ticketAvailabilityText;
    this.discounts = [];
    // initialization
    for (var i = prices.length - 1; i >= 0; --i) {
        var currentPrice = prices[i];
        if (currentPrice.currencySymbol == currencySymbol) {
            this.price = currentPrice.formattedPrice;
            if (showDiscountBestPlace && currentPrice.discounts) {
                this.discounts.length = 0;
                for (var discountIndex = 0; discountIndex < currentPrice.discounts.length; ++discountIndex) {
                    var discountWithCategory = angular.copy(prices[i].discounts[discountIndex], {});
                    discountWithCategory.ticketAvailabilityText = this.ticketAvailabilityText;
                    discountWithCategory.categoryId = this.id;
                    discountWithCategory.isSoldOut = this.isSoldOut;
                    discountWithCategory.itemOptions = this.itemOptions;
                    this.discounts.push(discountWithCategory);
                }             
            }
            break; // found the item with the expected currency
        }
    }

    this.updateCurrency = function (currencySymbol, showDiscountBestPlace) {
        for (var i = this.prices.length - 1; i >= 0; --i) {
            var currentPrice = prices[i];
            if (currentPrice.currencySymbol == currencySymbol) {
                this.price = currentPrice.formattedPrice;
                if (showDiscountBestPlace && currentPrice.discounts) {
                    this.discounts.length = 0;
                    for (var discountIndex = 0; discountIndex < currentPrice.discounts.length; ++discountIndex) {
                        var discountWithCategory = angular.copy(prices[i].discounts[discountIndex], {});
                        discountWithCategory.ticketAvailabilityText = this.ticketAvailabilityText;
                        discountWithCategory.categoryId = this.id;
                        discountWithCategory.isSoldOut = this.isSoldOut;
                        discountWithCategory.itemOptions = this.itemOptions;
                        this.discounts.push(discountWithCategory);
                    }
                }
                break; // found the item with the expected currency
            }
        }
    };
}
;
(function () {
    'use strict';

    var commonModule = angular.module('common');

    var serviceId = 'commonService';
    commonModule.factory(serviceId, ['$location', 'webShop2AppSettings', service]);

    function service($location, webShop2AppSettings) {
        var service = {
            setSessionStorageItem: setSessionStorageItem,
            getSessionStorageItem: getSessionStorageItem,
            removeSessionStorageItem: removeSessionStorageItem,
            clearSessionStorage: clearSessionStorage,
            extendWithoutFunctions: extendWithoutFunctions,
            absoluteUrl: absoluteUrl
        };

        return service;

        function $broadcast() {
            return $rootScope.$broadcast.apply($rootScope, arguments);
        }

        function setSessionStorageItem(key, value) {
            if (storage)
                sessionStorage.setItem(key, value);
        }

        function getSessionStorageItem(key) {
            if (storage)
                return sessionStorage.getItem(key);
            return null;
        }

        function removeSessionStorageItem(key) {
            if (storage)
                return sessionStorage.removeItem(key);
        }

        function clearSessionStorage() {
            if (storage)
                return sessionStorage.clear();
        }

        function extendWithoutFunctions(destination, source) {
            var clone = $.extend(true, [], source);
            for (var property in clone) {
                if (clone.hasOwnProperty(property) && typeof (clone[property]) === 'function')
                    delete clone[property];
            }
            return angular.extend(destination, clone);
        }

        function absoluteUrl(url) {
            if (url.indexOf("http") > -1)
                return url;
            else {
                // Create base url
                var hostName = $location.host();
                var hostPort = $location.port();
                var protocol = webShop2AppSettings.useSSL ? 'https' : $location.protocol();
                var baseUrl = protocol + "://" + hostName;

                if (hostPort !== 80 && hostPort !== 443) {
                    baseUrl = baseUrl + ":" + hostPort;
                }

                return baseUrl + url;
            }
        }
    }

    var storage;
    try {
        var uid = new Date;
        (storage = sessionStorage).setItem(uid, uid);
        var fail = storage.getItem(uid) != uid;
        storage.removeItem(uid);
        fail && (storage = false);
    } catch (exception) {
        storage = null;
    }

})();;
var CURRENT_STEP_NONE = "";
var CURRENT_STEP_SHOP = "Shop";
var CURRENT_STEP_CROSSSELLING = "CrossSelling";
var CURRENT_STEP_BASKET = "Basket";
var CURRENT_STEP_PATRON = "Patron";
var CURRENT_STEP_CHECKOUT = "Checkout";

var ERROR_NONE = "None";
var ERROR_GENERALBASKETERROR = "GeneralBasketError";
var ERROR_BASKETEXPIRED = "BasketExpired";
var ERROR_BASKETRULEERROR = "BasketRuleError";
var ERROR_BASKETDOESNOTEXIST = "BasketDoesNotExist";
var ERROR_BASKETEMPTY = "BasketEmpty";
var ERROR_REQUIRESLATEBASKETVALIDATION = "RequiresLateBasketValidation";
var ERROR_GENERALPATRONERROR = "GeneralPatronError";
var ERROR_PATRONMISSING = "PatronMissing";
var ERROR_REQUIRESPATRONLOGIN = "RequiresPatronLogin";
var ERROR_ADDRESSMISSING = "AddressMissing";
var ERROR_REQUIRESLOGIN = "RequiresLogin";
var ERROR_PERMISSIONERROR = "PermissionError";
var ERROR_RESERVATIONFAILED = "ReservationFailed";
var ERROR_NOTENOUGHTICKETS = "NotEnoughTickets";
var ERROR_NOMORETICKETS = "NoMoreTickets";
var ERROR_NOTICKETPRICEDEFINED = "NoTicketPriceDefined";
var ERROR_ORDERDATACHECKFAILED = "OrderDataCheckFailed";
var ERROR_SEATRESERVED = "SeatReserved";
var ERROR_CONCURRENCYERROR = "ConcurrencyError";

(function () {
    'use strict';

    var serviceId = 'pageService';
    angular.module('common').service(serviceId, ['commonService', 'pageModel', 'toaster', 'webShop2AppSettings', '$modal', service]);

    function service(commonService, pageModel, toaster, webShop2AppSettings, $modal) {
        return {
            pageModel: pageModel,
            trackEnabled: !webShop2AppSettings || !webShop2AppSettings.disableTracking,

            // Gets the start step
            getStartStep: function () {
                return this.pageModel.startStep;
            },

            // Sets the start step. '' for tickets and 'Shop' for other items
            setStartStep: function (value) {
                this.pageModel.startStep = value;
            },

            setStep: function (value) {
                if (value !== this.pageModel.currentStep)
                    window.scrollTo(0, 0);

                this.pageModel.currentStep = value;
                commonService.setSessionStorageItem('currentStep', value);

                if (value && value.length > 0 && value != CURRENT_STEP_SHOP)
                    this.trackNavigation(value.toLowerCase());

                // in iFrame post a message saying we changed the current step
                this.postMessageToIFrameParent("changedStep," + this.getStep());
            },

            getStep: function () {
                return commonService.getSessionStorageItem('currentStep') ? commonService.getSessionStorageItem('currentStep') : this.pageModel.currentStep;
            },

            nextStep: function () {
                switch (this.getStep()) {
                    case CURRENT_STEP_NONE:
                        this.setStep(CURRENT_STEP_SHOP);
                        break;
                    case CURRENT_STEP_SHOP:
                        if (this.pageModel.hasCrossSelling)
                            this.setStep(CURRENT_STEP_CROSSSELLING);
                        else
                            this.setStep(CURRENT_STEP_BASKET);
                        break;
                    case CURRENT_STEP_CROSSSELLING:
                        this.setStep(CURRENT_STEP_BASKET);
                        break;
                    case CURRENT_STEP_BASKET:
                        this.setStep(CURRENT_STEP_PATRON);
                        break;
                    case CURRENT_STEP_PATRON:
                        this.setStep(CURRENT_STEP_CHECKOUT);
                        break;
                    case CURRENT_STEP_CHECKOUT:
                        // No action on next step
                        break;
                }
            },

            previousStep: function () {
                switch (this.getStep()) {
                    case CURRENT_STEP_NONE:
                    case CURRENT_STEP_SHOP:
                        // No action on back step
                        break;
                    case CURRENT_STEP_CROSSSELLING:
                        this.setStep(CURRENT_STEP_SHOP);
                        break;
                    case CURRENT_STEP_BASKET:
                        if (this.pageModel.hasCrossSelling)
                            this.setStep(CURRENT_STEP_CROSSSELLING);
                        else
                            this.setStep(CURRENT_STEP_SHOP);
                        break;
                    case CURRENT_STEP_CHECKOUT:
                    case CURRENT_STEP_PATRON:
                        this.setStep(CURRENT_STEP_BASKET);
                        break;
                }
            },

            clearStep: function () {
                this.pageModel.currentStep = '';
                commonService.setSessionStorageItem('currentStep', '');
            },

            clearSessionStorage: function () {
                commonService.clearSessionStorage();
            },

            getBasketItemCount: function () {
                return commonService.getSessionStorageItem('basketItemCount');
            },

            setBasketItemCount: function (value) {
                commonService.setSessionStorageItem('basketItemCount', value);
            },

            getLastAlertTime: function () {
                return commonService.getSessionStorageItem('lastAlertTime') ? commonService.getSessionStorageItem('lastAlertTime') : this.pageModel.lastAlertTime;
            },

            setLastAlertTime: function (value) {
                this.pageModel.lastAlertTime = value;
                commonService.setSessionStorageItem('lastAlertTime', value);
            },

            isMobile: function () {
                return (/Android|webOS|iPhone|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent));
            },

            isTablet: function () {
                return (/iPad/i.test(navigator.userAgent));
            },

            pop: function (type, title, body, timeout, bodyOutputType) {
                if (!bodyOutputType)
                    bodyOutputType = '';
                toaster.pop(type, title, body, timeout, bodyOutputType);
            },

            showChangePasswordForm: function () {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/changePassword');
                var modalInstance = $modal.open({
                    animation: false,
                    templateUrl: url,
                    controller: 'patronChangePasswordController'
                });
            },

            showAliasRegistrationForm: function (isMyAccount) {
                var baseUrl = '/payment/dataTrans/aliasRegistration';
                if (isMyAccount !== undefined && isMyAccount)
                    baseUrl += "?isMyAccount=true"
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, baseUrl);
                var modalInstance = $modal.open({
                    animation: false,
                    size: 'lg',
                    templateUrl: url,
                    controller: aliasRegistrationModalInstanceController
                });
            },

            showSeatingMapHelp: function () {
                if (commonService.getSessionStorageItem('seatingMapHelpDisplayed'))
                    return !commonService.getSessionStorageItem('seatingMapHelpDisplayed');
                else {
                    commonService.setSessionStorageItem('seatingMapHelpDisplayed', true);
                    return true;
                }
            },

            // Generic way to display a modal dialog message
            // $pageService.showModal({ 
            //      message: 'the text message',
            //      title: 'The title'
            //      onClose: function() { alert(' it was closed'); } 
            // });
            showModal: function (options) {
                var modalInstance = $modal.open({
                    animation: false,
                    templateUrl: 'WebShop2AlertModalTemplate.html',
                    controller: alertModalInstanceController,
                    resolve: {
                        message: function () { return options.message; }
                        , header: function () {
                            if (options.title)
                                return options.title;
                            return '';
                        }
                    }
                });

                if (options && options.onClose) {
                    modalInstance.result.then(function () {
                        options.onClose();
                    });
                }
            },

            trackNavigation: function (path) {
                if (this.trackEnabled) {
                    if (window.ga && window.ga !== undefined && path) {
                        if (webShop2AppSettings.folderName && webShop2AppSettings.folderName.length > 0)
                            path = webShop2AppSettings.folderName + '/' + path;
                        window.ga('send', 'pageview', '/' + path);
                    }
                }
            },

            // sends a message to parent window (only iframe)
            postMessageToIFrameParent: function (message) {
                // in iFrame post a message notifying which tab is active
                if (ticketportal.webshop.inIframe()) {
                    window.parent.parent.postMessage("ticketportal::" + message, "*");
                }
            },

            tryUpdateParentWindowSize: function () {
                if (window.ticketportalWebshop && ticketportalWebshop.iframeExt) {
                    ticketportalWebshop.iframeExt.update();
                }
            },

            updateHasPendingBasketValidation: function (value) {
                if (value != undefined && value != null) {
                    this.pageModel.hasPendingBasketValidation = value;
                    commonService.setSessionStorageItem('hasPendingBasketValidation', value);
                }
            },

            hasPendingBasketValidation: function () {
                var sessionValue = commonService.getSessionStorageItem('hasPendingBasketValidation');
                if (sessionValue == undefined || sessionValue == null)
                    return this.pageModel.hasPendingBasketValidation;

                return sessionValue;
            },

            // sets the loading status of the page
            loading: function (isLoading) {
                if (isLoading === undefined)
                    isLoading = true;
                this.pageModel.isLoading = isLoading;
            },

            // display starting page
            goHome: function () {
                var expectedPathName = '/' + webShop2AppSettings.folderName;
                if (expectedPathName.length > 1)
                    expectedPathName += '/';
                if (window.location.pathname !== expectedPathName || window.location.search.length > 0)
                    window.location = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/');
                else
                    location.reload();
            },

            setNavigationTitleIfNotExists: function (title) {
                if (!this.pageModel.navigationTitle && title) {
                    this.pageModel.navigationTitle = title;
                };
            },

            setNavigationTitle: function (title) {
                this.pageModel.navigationTitle = title;
            },

            getNavigationTitle: function () {
                return this.pageModel.navigationTitle;
            },

            hasNavigationTitle: function () {
                return this.pageModel.navigationTitle && this.pageModel.navigationTitle.trim().length > 0;
            },

            // Shows personalization modal
            showPersonalization: function (itemGroupsModel) {
                var modalInstance = $modal.open({
                    animation: false,
                    size: 'lg',
                    templateUrl: 'Personalization.html',
                    controller: personalizationModalInstanceController,
                    resolve: {
                        itemGroupsModel: function () { return itemGroupsModel; },
                    }
                });
            },

            /**
             * Function that validates a password
             * @param {string} password - The password to be validated
             * @returns {Array} - An array of error messages
             */
            validatePassword: password => {
                let errors = [];
                password = password || '';
                if (password.length < 8) {
                    errors.push(ticketportal.webshop.localization.errorPasswordTooShort);
                }
                if (!/[a-zA-Z]/.test(password)) {
                    errors.push(ticketportal.webshop.localization.errorPasswordMustHaveLetter);
                }
                if (!/\d/.test(password)) {
                    errors.push(ticketportal.webshop.localization.errorPasswordMustHaveNumber);
                }
                if (!/[^a-zA-Z\d]/.test(password)) {
                    errors.push(ticketportal.webshop.localization.errorPasswordMustHaveSpecialChar);
                }
                return errors;
            },

            /**
             * Function that validates an email
             * @param {string} email - The email to be validated
             * @returns {string} - Error message
             */
            validateEmail: email => {
                if (!email) {
                    return ticketportal.webshop.localization.errorEmailMissing;
                }

                let pattern = /(([^<>()\[\]\\.,;:\s@\x22]+(\.[^<>()\[\]\\.,;:\s@\x22]+)*)|(\x22.+\x22))@((\[[0-9]{1, 3}\.[0-9]{1, 3}\.[0-9]{1, 3}\.[0-9]{1, 3}])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/;
                if (!pattern.test(email)) {
                    return ticketportal.webshop.localization.errorEmailInvalidFormat;
                }

                return '';
            }
        }
    }
})();
aliasRegistrationModalInstanceController = function ($scope, $modalInstance, $location, $sce) {
    $scope.model = [];
    $scope.model.paymentMethod = '';
    $scope.model.inlineFrameSrcTemplate = '';
    $scope.model.inlineFrameSrc = '';
    $scope.model.resultUrl = $location.absUrl();
    $scope.cancel = function () {
        $modalInstance.dismiss('cancel');
    }

    $scope.onPaymentMethodChanged = function () {
        if ($scope.model.paymentMethod) {
            $scope.model.inlineFrameSrc = $sce.trustAsResourceUrl($scope.model.inlineFrameSrcTemplate + "&paymentmethod=" + $scope.model.paymentMethod);
        } else {
            $scope.model.inlineFrameSrc = '';
        }
    }

    $scope.needsCreditCardInfo = function () {
        switch ($scope.model.paymentMethod) {
            case "PFC":
            case "PST":
            case "PEF":
                return false;
        }
        return true;
    }
}
aliasRegistrationModalInstanceController.$inject = ['$scope', '$modalInstance', '$location', '$sce'];

personalizationModalInstanceController = function ($scope, $modalInstance, itemGroupsModel, basketService, pageService, commonService, $rootScope) {
    $scope.itemGroupsModel = filterItemGroupsModel(itemGroupsModel);
    $scope.itemGroups = {};
    commonService.extendWithoutFunctions($scope.itemGroups, $scope.itemGroupsModel);
    $scope.basketService = basketService;
    $scope.pageService = pageService;

    $scope.cancel = function () {
        $modalInstance.dismiss('cancel');
    }

    $scope.save = function (isValid) {
        if (isValid) {
            $scope.pageService.pageModel.isLoading = true;
            let items = [];
            $.each($scope.itemGroups, function (i, itemGroup) {
                $.each(itemGroup.items, function (i, item) {
                    items.push({ id: item.id, personalization: item.personalization });
                });
            });

            // Save personalization values
            $scope.basketService.setPersonalization(items).then(function (data) {
                $scope.pageService.pageModel.isLoading = false;

                // Copy saved values to model
                if (data.succeeded) {
                    $.each($scope.itemGroupsModel, function (i, groupModel) {
                        $.each(groupModel.items, function (i, itemModel) {
                            let group = $.grep(data.basket.itemGroups, function (g) {
                                return $.grep(g.items, function (i) { return i.id === itemModel.id; })[0];
                            })[0];
                            let item = $.grep(group.items, function (i) {
                                return i.id === itemModel.id;
                            })[0];
                            $.each(itemModel.personalization.fields, function (i, fieldModel) {
                                let field = $.grep(item.personalization.fields, function (f) {
                                    return f.id === fieldModel.id;
                                })[0];
                                fieldModel.value = field.value;
                            })
                        });
                    });
                } else {
                    $rootScope.$broadcast('event:ShowAlert', { title: ticketportal.webshop.localization.errorTitle, message: data.errorMessage });
                }
                $modalInstance.close();
            });
        }
    }

    // Filter item groups to return only groups with items that need personalization
    function filterItemGroupsModel(itemGroupsModel) {
        let itemGroups = [];
        for (let i = 0; i < itemGroupsModel.length; i++) {
            let basketItemGroup = itemGroupsModel[i];
            if (basketItemGroup.hasPersonalization) {
                let itemGroup = {
                    name: basketItemGroup.name,
                    venueName: basketItemGroup.venueName,
                    locationName: basketItemGroup.locationName,
                    venueCity: basketItemGroup.venueCity,
                    items: []
                };
                for (let j = 0; j < basketItemGroup.items.length; j++) {
                    let basketItem = basketItemGroup.items[j];
                    if (basketItem.personalization !== null) {
                        let item = {
                            id: basketItem.id,
                            isAdmissionTicket: basketItem.isAdmissionTicket,
                            priceCategoryName: basketItem.priceCategoryName,
                            sectionName: basketItem.sectionName,
                            description: basketItem.description,
                            discountName: basketItem.appliedDiscount ? basketItem.appliedDiscount.name : basketItem.discountName,
                            personalization: basketItem.personalization
                        };
                        itemGroup.items.push(item);
                    }
                }
                itemGroups.push(itemGroup);
            }
        }
        return itemGroups;
    }
};
personalizationModalInstanceController.$inject = ['$scope', '$modalInstance', 'itemGroupsModel', 'basketService', 'pageService', 'commonService', '$rootScope'];
;
(function () {
    'use strict';

    var serviceId = 'formGeneratorService';
    angular.module('common').value('formGeneratorModel', {});
    angular.module('common').service(serviceId, ['formGeneratorModel', service]);

    function service(formGeneratorModel) {
        return {
            formGeneratorModel: formGeneratorModel,

            validateDate: function (form, elementBaseName, date, propertyName) {
                var isDateValid = validateDate(date, false);
                formGeneratorModel[propertyName] = validateDate(date, true);
                angular.forEach(form, function (control) {
                    if (control && control.$name && control.$name.indexOf(elementBaseName) === 0) {
                        control.$setValidity("date", isDateValid);
                        if (isDateValid)
                            control.$setValidity("datePast", formGeneratorModel[propertyName]);
                    }
                });                
            },

            setPropertyNameValue: function (element, dynamicFormModel, propertyGroup, propertyCode, propertyName, propertyValue) {
                var controlElement = $('#' + element);
                var currentPropertyValues = $.grep(dynamicFormModel[propertyGroup][propertyCode][propertyName].split(','), function (n) { return (n); });
                if (controlElement.is('input')) {
                    if (controlElement.prop('checked')) {
                        if (controlElement.is(':radio'))
                            currentPropertyValues.length = 0;

                        if ($.inArray(propertyValue, currentPropertyValues) === -1)
                            currentPropertyValues.push(propertyValue);
                    }
                    else
                        currentPropertyValues = $.grep(currentPropertyValues, function (value) { return value != propertyValue; });
                }
                else {
                    var selectedOption = controlElement.find(":selected");
                    currentPropertyValues.length = 0;
                    currentPropertyValues.push(selectedOption.text());
                    var selectedOptionValue = parseInt((selectedOption.val() ? selectedOption.val().match(/\d+/) : "0") || "0");
                    if (selectedOptionValue === 0)
                        controlElement.css("font-weight", "lighter");
                    else
                        controlElement.css("font-weight", "");
                }

                dynamicFormModel[propertyGroup][propertyCode][propertyName] = currentPropertyValues.join();
            },

            findPropertyValueRecursive: function (object, propertyName) {
                var value;
                for (var property in object) {
                    if (object.hasOwnProperty(property)) {
                        if (property !== propertyName) {
                            if (typeof object[property] === "object") {
                                value = this.findPropertyValueRecursive(object[property], propertyName);
                                if (value !== undefined)
                                    break;
                            }
                        } else {
                            value = object[property].value;
                            break;
                        }
                    }
                }

                return value;
            },

            convertStringToDate: function (object) {
                for (var property in object) {
                    if (object.hasOwnProperty(property)) {
                        if (typeof object[property] === "object") {
                            this.convertStringToDate(object[property]);
                        } else {
                            if (property === "value") {
                                if (moment(object[property], "YYYY-MM-DDTHH:mm:ss", true).isValid()) {
                                    object[property] = new Date(object[property]);
                                }
                            }
                        }
                    }
                }
            },

            convertDateToString: function (object) {
                for (var property in object) {
                    if (object.hasOwnProperty(property)) {
                        if (typeof object[property] === "object") {
                            if ((object[property] instanceof Date)) {
                                object[property] = moment(object[property]).format("YYYY-MM-DDTHH:mm:ss");
                            } else {
                                this.convertDateToString(object[property]);
                            }
                        }
                    }
                }
            }
        };
    }

})();
;
// Modified:		01.12.2016 Starticket AG, St. Gallen rha    : (Task 598) Add full price name to price category
// Modified:        17.03.2016 ticketportal AG, St. Gallen FBE  : (SCR 8951) Tattoo SG: wrong information on webshop --> price doesn't include fees
// Modified:		12.01.2016 ticketportal AG, St. Gallen fbe  : (SCR 8298) Admission Ticket
(function () {
    'use strict';

    var commonModule = angular.module('common');

    var serviceId = 'customizationService';
    commonModule.factory(serviceId, ['webShop2AppSettings', '$sce', customizationService]);

    function customizationService(webShop2AppSettings, $sce) {
        var service = {
            updateHomeModel: updateHomeModel,
            onPerformanceSelected: onPerformanceSelected,
            buildFlattenPriceCategory: buildFlattenPriceCategory,            
            updateBasketModel: updateBasketModel,
            updateCheckoutModel: updateCheckoutModel,
            onBeforePlaceOrder: onBeforePlaceOrder
        };

        return service;

        function updateHomeModel(data, homeModel) {
            homeModel.updateModel(data);
        }

        function onPerformanceSelected(performanceModel) {

        }

        function buildFlattenPriceCategory(parentObject) {
            if (parentObject.flattenPriceCategory === undefined || parentObject.flattenPriceCategory == null)
                parentObject.flattenPriceCategory = new Array();

            for (var i = 0; i < parentObject.priceCategories.length; i++) {
                var currentPriceCategory = parentObject.priceCategories[i];
                for (var priceIndex = 0; priceIndex < currentPriceCategory.prices.length; ++priceIndex) {
                    var currentPrice = currentPriceCategory.prices[priceIndex];
                    parentObject.flattenPriceCategory.push({
                        priceCategoryUniqueId: currentPriceCategory.uniqueId,
                        discountId: 0,
                        isDiscount: false,
                        currencySymbol: currentPrice.currencySymbol,
                        name: currentPriceCategory.name,
                        fullPriceName: currentPriceCategory.fullPriceName,
                        formattedPrice: currentPrice.formattedPrice,
                        formattedPriceWithFees: $sce.trustAsHtml(currentPrice.formattedPriceWithFees),
                        colorCode: currentPriceCategory.colorCode,
                        attributes: currentPriceCategory.attributes
                    });

                    for (var discountIndex = 0; discountIndex < currentPrice.discounts.length; ++discountIndex) {
                        var currentDiscount = currentPrice.discounts[discountIndex];
                        parentObject.flattenPriceCategory.push({
                            isDiscount: true,
                            priceCategoryUniqueId: currentPriceCategory.uniqueId,
                            discountId: currentDiscount.id,                            
                            currencySymbol: currentPrice.currencySymbol,
                            name: currentPriceCategory.name,
                            fullPriceName: currentPriceCategory.fullPriceName,
                            discountName: currentDiscount.name,
                            discountNameAndValue: currentDiscount.nameAndValue,
                            formattedPrice: currentDiscount.formattedPrice,
                            formattedPriceWithFees: $sce.trustAsHtml(currentDiscount.formattedPriceWithFees),
                            colorCode: currentPriceCategory.colorCode,
                            attributes: null    // only the full price element gets the attributes for now
                        });
                        parentObject.hasPriceCategoryDiscount = true;
                    }
                }
            }

            return parentObject.flattenPriceCategory;
        }

        function updateBasketModel(data, basketModel) {
            basketModel.updateFromService(data);
        }

        function onBeforePlaceOrder(options) {

        }

        function updateCheckoutModel($scope, data, checkoutModel) {
            checkoutModel.update(data);
        }

        
    }

})();;
(function () {
    'use strict';    

    var controllerId = 'homeController';
    angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', '$sce', '$modal', '$cookies','$window', 'homeService', 'homeModel', 'crossSellingModel', 'pageService', 'webShop2AppSettings', 'customizationService', 'commonService','ecommerceService', 'basketService', controller]);
    
    function controller($rootScope, $scope, $sce, $modal, $cookies, $window, homeService, homeModel, crossSellingModel, pageService, webShop2AppSettings, customizationService, commonService, ecommerceService, basketService) {
        $scope.model = homeModel;
        $scope.model.cartData = null;
        $scope.crossSellingModel = crossSellingModel;
        $scope.homeService = homeService;
        $scope.pageService = pageService;
        $scope.webShop2AppSettings = webShop2AppSettings;
        $scope.modal = $modal;
        $scope.customizationService = customizationService;
        $scope.ecommerceService = ecommerceService;

        $scope.updateModelFromRaw = function (data) {
            $scope.customizationService.updateHomeModel(data, $scope.model);         
            $scope.crossSellingModel.updateFromService(data);
            $scope.pageService.pageModel.hasCrossSelling = $scope.crossSellingModel.hasCrossSelling;
            $scope.homeService.setLoginName($scope.model.loggedUserLogin);
        };

        $scope.init = function (bookingObjectType, bookingObjectTag, loadBasket) {

            // check if the browser supports 'cors'
            if ($scope.webShop2AppSettings.requiresSSL) {
                ticketportal.webshop.ensurePageIsSecure();
            }
            else if ($scope.webShop2AppSettings.useSSL) {
                if (!Modernizr.cors) {
                    ticketportal.webshop.ensurePageIsSecure();
                }
            }

            // Verify cookie support
            $scope.cookieSupport();

            $scope.model.bookingObjectType = bookingObjectType.toLowerCase();
            $scope.model.bookingObjectTag = bookingObjectTag;

            if ($scope.webShop2AppSettings.loadBasket && (loadBasket == undefined || loadBasket)) {
                $scope.homeService.getBasketContents($scope.model.bookingObjectType, $scope.model.bookingObjectTag, ticketportal.webshop.inIframe()).then(function(data) {

                    // only update if it really changed
                    if ($scope.pageService.pageModel.requiresPatronLogin != data.requiresPatronLogin)
                        $scope.pageService.pageModel.requiresPatronLogin = data.requiresPatronLogin;

                    $scope.updateModelFromRaw(data);

                    var currentStep = $scope.pageService.getStep();
                    if (data.hasSpecifiedObjectInBasket) {
                        // check if we have parameter to go to 'basket' or 'checkout'
                        var urlStep = getQueryStringValue('step');
                        if (urlStep == 'basket') {
                            $scope.pageService.setStep(CURRENT_STEP_BASKET);
                        } else if (urlStep == 'checkout' && $scope.canGoToCheckoutStep()) {
                            $scope.pageService.setStep(CURRENT_STEP_CHECKOUT);
                        } else if (urlStep == 'shop' || currentStep == '') {
                            $scope.pageService.setStep(CURRENT_STEP_SHOP);
                        } else if (currentStep == CURRENT_STEP_CROSSSELLING) {
                            $scope.pageService.setStep(CURRENT_STEP_BASKET);
                        }
                    }
                    else {
                        $scope.pageService.setStep($scope.pageService.getStartStep());
                    }
                });
            } else {
                $scope.pageService.setStep($scope.pageService.getStartStep());                
            }

            // load event categories
            if (window.__eventCategories) {
                var baseEventUrl = ticketportal.webshop.fixUrl($scope.webShop2AppSettings.folderName, '/event');
                $scope.model.updateEventCategories(baseEventUrl, window.__eventCategories, window.__defaultEventCategory, $scope.sortEventByPriorityAndName);
                if ($scope.model.eventCategories.length) {
                    $scope.selectEventCategory($scope.model.eventCategories[0]);
                }
            }
        }

        // Pushing impressions if a page is loaded. (check if page is a product page happens in pushImpressions method.
        $(window).on("load",
            function() {
                ecommerceService.pushImpressions($window); // Push Impression - dataLayer
            });

        // display starting page
        $scope.goHome = function () {
            $scope.pageService.goHome();    // refactory: moved to pageService
        }

        $scope.toggleToShopStep = function () {
            if ($scope.pageService.getStep() != CURRENT_STEP_SHOP) {
                $scope.pageService.setStep(CURRENT_STEP_SHOP);
            }
        }

        $scope.toggleToBasketStep = function () {
            if ($scope.pageService.getStep() != CURRENT_STEP_BASKET && $scope.model.basketItemCount > 0) {
                $scope.pageService.setStep(CURRENT_STEP_BASKET);
            }
        }

        $scope.toggleToCrossSellingStep = function () {
            if ($scope.pageService.getStep() != CURRENT_STEP_CROSSSELLING && $scope.model.basketItemCount > 0) {
                $scope.pageService.setStep(CURRENT_STEP_CROSSSELLING);
            }
        }

        $scope.toggleToPatronStep = function () {
            if ($scope.pageService.getStep() != CURRENT_STEP_PATRON && $scope.model.hasPatron) {
                $scope.ensureBasketIsValidatedAndGoToStep(CURRENT_STEP_PATRON);
            }
        }

        $scope.ensureBasketIsValidatedAndGoToStep = function(step) {
            
            if ($scope.pageService.getStep() == CURRENT_STEP_BASKET && $scope.pageService.hasPendingBasketValidation()) {
                // currently on the basket page and it has pending basket validation
                // THEN do the validation and after continue
                $rootScope.$broadcast('basket:validateAndMoveToStep', { step: step });
            } else {
                $scope.pageService.setStep(step);
            }
        }

        $scope.canGoToCheckoutStep = function () {
            return $scope.model.basketItemCount > 0 && $scope.model.hasPatron;
        }

        $scope.toggleToCheckoutStep = function () {
            if ($scope.pageService.getStep() != CURRENT_STEP_CHECKOUT && $scope.canGoToCheckoutStep()) {
                $scope.ensureBasketIsValidatedAndGoToStep(CURRENT_STEP_CHECKOUT);
            }
        }

        $scope.changeLanguage = function (language) {
            $scope.homeService.changeLanguage(language).then(function () {
                window.location.reload(true);
            });
        }

        // called by the 'basket' icon in the top of the page
        $scope.goToBasket = function () {
            if ($scope.model.goToBasketURL && $scope.model.goToBasketURL.length > 0)
                window.location = $scope.model.goToBasketURL;
            else
                $scope.toggleToBasketStep();
        }

        // logout
        $scope.logout = function () {
            window.location = ticketportal.webshop.fixUrl($scope.webShop2AppSettings.folderName, '/logout');
        }

        // goes to my account page
        $scope.goToMyAccount = function (isRegistration) {
            let path = '/myaccount';
            if (isRegistration) {
                path += '?isRegistration=true';
            }
                
            window.location = ticketportal.webshop.fixUrl($scope.webShop2AppSettings.folderName, path);
        }

        /*
        * Ensures that the current page is SSL
        */
        $scope.ensureSSLPage = function () {
            if (window.location.protocol != "https:") {
                window.location.href = "https:" + window.location.href.substring(window.location.protocol.length);
            }
        }

        $scope.toggleIsRunning = function () {
            $scope.model.isRunning = true;
            setTimeout(function () {
                $scope.$apply(function () {
                    $scope.model.isRunning = false;
                })
            }
                , 1000);
        }

        $scope.$on('event:BasketChanged', function (event, args) {
            $scope.updateModelFromRaw(args);
            var itemCountDelta = (args.amountOfItemsAdded ? args.amountOfItemsAdded : 0) - (args.amountOfItemsRemoved ? args.amountOfItemsRemoved : 0);
            var basketItemCount = $scope.pageService.getBasketItemCount();
            if (basketItemCount && basketItemCount != $scope.model.basketItemCount - itemCountDelta) {
                if ($scope.model.basketItemCount == 0) {                    
                    $scope.$broadcast('event:ShowAlert', { title: '', message: ticketportal.webshop.localization.basketExpired });
                }
                else {
                    $scope.$broadcast('event:ShowAlert', { title: '', message: ticketportal.webshop.localization.itemsExpired });
                }
            }
            else if (args.hasConcurrencyError) {
                $scope.$broadcast('event:ShowAlert', { title: '', message: args.errorMessage });
            }

            if ($scope.pageService.pageModel.hasCrossSelling && args.amountOfItemsAdded && args.amountOfItemsAdded > 0) {
                var message = args.isCrossSelling ? ticketportal.webshop.localization.specialsAdded : ticketportal.webshop.localization.itemsAdded;
                $scope.pageService.pop('info', '', message.replace('({0})', args.amountOfItemsAdded), 4000);
            }

            $scope.pageService.setBasketItemCount($scope.model.basketItemCount);
        });

        // Shows alert modal message
        $scope.$on('event:ShowAlert', function (event, args) {
            var checkTime = new Date();
            checkTime.setSeconds(checkTime.getSeconds() - 1);
            var lastAlertTime = pageService.getLastAlertTime().length > 0 ? new Date(pageService.getLastAlertTime()) : '';
            if (lastAlertTime < checkTime) {
                pageService.setLastAlertTime(new Date());
                var message = '';
                var title = '';
                if (args && args != undefined) {
                    if (args.title)
                        title = args.title;
                    if (args.message)
                        message = args.message;

                }
                var modalInstance = $scope.modal.open({
                    templateUrl: 'WebShop2AlertModalTemplate.html',
                    controller: alertModalInstanceController,
                    animation: false,
                    resolve: {
                        header: function () { return title; },
                        message: function () { return message; }

                    }
                });
            }
        });

        $scope.$on('event:ErrorHandled', function (event, args) {
            try {
                $scope.pageService.pageModel.isLoading = false;
                $scope.$broadcast('event:ShowAlert', { title: '', message: ticketportal.webshop.localization.generalError });
            }
            catch (err) {
            }    
        });

        $rootScope.$on('event:LoggedOn', function (event, args) {
            $scope.model.isLoggedOn = true;
        });

        $scope.clearCache = function () {
            $scope.homeService.clearCache();
        }

        $scope.showTab = function (tabKey) {
            $scope.model.displayTab = tabKey;
        }       

        $scope.selectEventCategory = function (eventCategory) {

            if (eventCategory !== undefined)
                $scope.model.selectedEventCategory = eventCategory;

            $scope.model.isDisplayingAllEventsByCategory = (!$scope.model.selectedEventCategory || $scope.model.selectedEventCategory.id == 0);
            $scope.updateDisplayedCategoryEvents();            
        }

        $scope.updateDisplayedCategoryEvents = function () {
            var eventsToDisplay = null;
            if ($scope.model.selectedEventCategory.id == 0) {
                // get all events from all categories
                eventsToDisplay = new Array();
                var allEventsKey = {};
                for (var i = 0; i < $scope.model.eventCategories.length; i++) {
                    var categoryModel = $scope.model.eventCategories[i];

                    for (var j = 0; j < categoryModel.events.length; j++) {
                        var eventModel = categoryModel.events[j];
                        if (!allEventsKey.hasOwnProperty(eventModel.id.toString())) {
                            eventsToDisplay.push(eventModel);
                            allEventsKey[eventModel.id.toString()] = true;
                        }
                    }
                }

                eventsToDisplay.sort($scope.sortEventByPriorityAndName);

            } else {
                eventsToDisplay = $scope.model.selectedEventCategory.events;
            }


            $scope.model.eventsByCategory.length = 0;

            var amountOfEventsToDisplay = eventsToDisplay.length;
            if (!$scope.model.isDisplayingAllEventsByCategory) {
                amountOfEventsToDisplay = Math.min(eventsToDisplay.length, $scope.model.maxEventsByCategoryToDisplay);
                $scope.model.hasMoreEventsByCategoryToDisplay = eventsToDisplay.length > $scope.model.maxEventsByCategoryToDisplay;
            } else {
                $scope.model.hasMoreEventsByCategoryToDisplay = false;
            }

            for (var i = 0; i < amountOfEventsToDisplay; i++) {
                $scope.model.eventsByCategory.push(eventsToDisplay[i]);
            }

            // in iFrame post a message saying we changed the events being displayed
            $scope.pageService.postMessageToIFrameParent("eventsByCategoryChanged");
        }

        $scope.displayAllEvents = function () {
            $scope.model.isDisplayingAllEventsByCategory = true;
            $scope.updateDisplayedCategoryEvents();
        }        

        $scope.sortEventByPriorityAndName = function (a, b) {

            if (a.priority == 0 && b.priority > 0)
                return 1; // b first
            else if (b.priority == 0 && a.priority > 0)
                return -1; // a first

            var result = a.priority - b.priority;
            if (result == 0) {
                if (a.name > b.name)
                    result = 1;
                else if (a.name < b.name)
                    result = -1;
                else
                    result = 0;
            }

            return result;
        };

        $scope.cookieSupport = function () {
            $scope.homeService.cookieSupport().then(function (response) {
                if (response.data.succeeded) {
                    $scope.model.isCookieDisabled = false;
                } else {
                    $scope.model.isCookieDisabled = true;
                    $scope.pageService.showModal({ title: '', message: response.data.errorMessage, onClose: $scope.pageService.goHome });
                }
            });
        };

        $scope.showInfo = function (message) {
            $scope.pageService.showModal({
                message: message
            });
        };
        
        // Get Cart
        $scope.getCart = function () {
            basketService.get().then(res => {
                $scope.model.cartData = res;
            });
        };
    }
})();;
// Modified:		14.09.2016 ticketportal AG, St. Gallen fbe  : (SCR 9395) Coop ticket shop 2016: add css changes from Nexum Agentur
(function () {
    'use strict';

    var modelId = 'homeModel';
    angular.module('webShop2App').factory(modelId, [model]);

    function eventCategoryModel(source) {
        this.name = source.name;
        this.id = source.id;
        this.code = (source.code != undefined) ? source.code : '';
            
        this.isSelected = false;
        this.events = new Array();
        if (source.events) {
            for (var i = 0; i < source.events.length; i++) {
                var sourceEvent = source.events[i];
                this.events.push(new eventModel(sourceEvent));
            }
        }
        
    };

    function eventModel(source) {
        this.id = source.id;
        this.name = source.name;
        this.tag = source.tag;
        this.priceInformation = source.priceInformation;
        this.priceConditionInformation = source.priceConditionInformation;
        this.bigImage = source.bigImage;
        this.smallImage = source.smallImage;
        this.mediumImage = source.mediumImage;
        this.isExternal = source.isExternal;
        this.date = source.date;
        this.locations = source.locations;
        this.priority = source.priority;
        this.categoryPriority = source.categoryPriority;        
        this.location = source.city ? source.city : source.location;
    };

    function model() {
        var model = {
            basketItemCount: 0,
            basketTotal: '',
            basketTotalToPay: '',
            hasPatron: false,
            isLoggedOn: false,
            loggedUserLogin: false,
            isRunning: false,
            bookingObjectType: '',
            bookingObjectTag: '',
            goToBasketURL: '',
            allowedCurrencies: new Array(),
            selectedEventCategory: null,
            eventsByCategory: new Array(),
            eventCategories: new Array(),           
            maxEventsByCategoryToDisplay: 2000,
            isDisplayingAllEventsByCategory: false,
            hasMoreEventsByCategoryToDisplay: false,

            // functions
            selectCurrencyBySymbol: selectCurrencyBySymbol,
            updateModel: updateModel,
            updateEventCategories: updateEventCategories
        };
        return model;

        // sets the current currency by symbol
        function selectCurrencyBySymbol(currencySymbol) {
            for (var i = this.allowedCurrencies.length - 1; i >= 0; --i) {
                if (currencySymbol == this.allowedCurrencies[i].symbol) {
                    this.selectedCurrency = this.allowedCurrencies[i];
                    this.selectedCurrency.isSelected = true;
                } else {
                    this.allowedCurrencies[i].isSelected = false;
                }
            }
        }

        function updateModel(data) {
            if (data.hasPatron != undefined) this.hasPatron = data.hasPatron;
            if (data.loggedUserLogin != undefined) this.loggedUserLogin = data.loggedUserLogin;
            if (data.isLoggedOn != undefined) {
                this.isLoggedOn = data.isLoggedOn;
                if (!this.isLoggedOn)
                    this.loggedUserLogin = null;
            }
            if (data.formattedTotal != undefined) this.basketTotal = data.formattedTotal;
            if (data.formattedTotalToPay != undefined) this.basketTotalToPay = data.formattedTotalToPay;
            if (data.totalItems != undefined) this.basketItemCount = data.totalItems;
            if (data.basketTooltipFormat != undefined) this.basketTooltipFormat = data.basketTooltipFormat;
            if (this.basketTooltipFormat)
                this.basketTooltip = this.basketTooltipFormat.replace('$items', this.basketItemCount).replace('$total', this.basketTotal);

            this.goToBasketURL = '';
            if (data.basketUrl != undefined)
                this.goToBasketURL = data.basketUrl;

            if (data.allowedCurrencies != undefined) {
                this.allowedCurrencies.length = 0;
                this.selectedCurrency = null;
                for (var i = 0; i < data.allowedCurrencies.length; i++) {
                    var currency = {
                        id: data.allowedCurrencies[i].value,
                        symbol: data.allowedCurrencies[i].text,
                        isSelected: data.allowedCurrencies[i].selected
                    };
                    this.allowedCurrencies.push(currency);
                    if (currency.isSelected)
                        this.selectedCurrency = currency;
                }
            }
        }
      

        function updateEventCategories(baseEventUrl, source, defaultEventCategory, eventSortFunction) {
            this.eventCategories.length = 0;
            var allEvents = new Array();
            var allEventsKey = {};

            if (defaultEventCategory) {
                var defaultCategoryModel = new eventCategoryModel(defaultEventCategory);
                this.eventCategories.push(defaultCategoryModel);                
            }

            for (var i = 0; i < source.length; i++) {
                var categoryModel = new eventCategoryModel(source[i]);
                this.eventCategories.push(categoryModel);
                
                for (var j = 0; j < categoryModel.events.length; j++) {
                    var eventModel = categoryModel.events[j];
                    eventModel.url = baseEventUrl + '/' + eventModel.tag;         
                }
            }

            // sort the events
            if (eventSortFunction !== undefined && eventSortFunction) {
                for (var categoryIndex = 0; categoryIndex < this.eventCategories.length; ++categoryIndex) {
                    this.eventCategories[categoryIndex].events.sort(eventSortFunction);
                }
            }
        }        
        
    }

})();;


var userLoginData = { "loginName" : ""};

(function () {
    'use strict';

    var serviceId = 'homeService';
    angular.module('webShop2App').factory(serviceId, ['$http', 'webShop2AppSettings', service]);

    function service($http, webShop2AppSettings) {
        return {
            getBasketContents: function (bookingObjectType, bookingObjectTag, isIFrame) {
                var getBasketStatusURL = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/GetBasketStatus');

                if (bookingObjectType) {
                    getBasketStatusURL += '?bookingobjecttype=' + bookingObjectType;

                    if (bookingObjectTag) {
                        getBasketStatusURL += '&bookingobjecttag=' + bookingObjectTag;
                    }
                }

                var externalSystemTokenValue = getQueryStringValue('token');
                if (!externalSystemTokenValue)
                    externalSystemTokenValue = getQueryStringValue('uid');
                var data = {
                    externalSystemToken: externalSystemTokenValue,
                    isIFrame: ticketportal.webshop.inIframe()
                };

                return $http({
                    method: 'POST',
                    url: getBasketStatusURL,
                    data: $.param(data),
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });


            },

            // changes the language of the current user / session
            changeLanguage: function (language) {
                var switchLanguageUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/switchlanguage?language=') + language;
                return $http({ method: 'POST', url: switchLanguageUrl }).then(function (response) {
                    return response.data;
                });
            },

            clearCache: function () {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/clearcache');
                return $http({ method: 'POST', url: url }).then(function (response) {
                    return response.data;
                });
            },

            setLoginName: function (userName) {
                userLoginData = { 'loginName': userName };
            },

            getLoginName: function () {
                return userLoginData.loginName;
            },

            cookieSupport: function () {
                var cookieSupportUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cookieSupport');
                return $http({ method: 'GET', url: cookieSupportUrl + "?cookieAction=setCookie", withCredentials: true }).then(function (response) {
                    var responseData = response.data;
                    if (responseData && responseData.succeeded && responseData.value) {
                        return $http({ method: 'GET', url: cookieSupportUrl + "?cookieAction=verifyCookie&cookieValue=" + responseData.value, withCredentials: true }).then(function (response) {
                            return response;
                        });
                    }

                    return response;
                });
            }
        };
    }
})();
;
(function () {
    'use strict';

    var controllerId = 'ticketController';
    angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', '$timeout','ecommerceService', 'ticketService', 'ticketModel', 'pageService', 'commonService', 'homeModel', '$uibModal', 'selectedPerformancePlacesFinderService', 'customizationService', 'webShop2AppSettings', '$window', controller]);

    var PERFORMANCE_RESERVATIONINOF_VIEW = "PerformanceReservationInfoTab.html";
    var PERFORMANCE_PROMOTIONSINFO_VIEW = "PerformancePromotionsTab.html";
    var PERFORMANCE_DESCRIPTIONINFO_VIEW = "PerformanceDescriptionTab.html";
    var PERFORMANCE_PRICEINFO_VIEW = "PerformancePriceInfoTab.html";

    function controller($rootScope, $scope, $timeout, ecommerceService, ticketService, ticketModel, pageService, commonService, homeModel, $uibModal, selectedPerformancePlacesFinderService, customizationService, webShop2AppSettings, $window) {
        $scope.ticketService = ticketService;
        $scope.model = ticketModel;
        $scope.pageService = pageService;
        $scope.modal = $uibModal;
        $scope.model.isMobile = pageService.isMobile();
        $scope.mainModel = homeModel;
        $scope.selectedPlacesFinderService = selectedPerformancePlacesFinderService;
        $scope.customizationService = customizationService;
        $scope.model.selectedSeats = [];
        $scope.model.selectedSeats.expanded = false;
        $scope.webShop2AppSettings = webShop2AppSettings;
        $scope.ecommerceService = ecommerceService;
        $scope.ticketportal = $window.ticketportal;
        $scope.model.performanceReservationInfoView = PERFORMANCE_RESERVATIONINOF_VIEW;
        $scope.model.performancePromotionsInfoView = PERFORMANCE_PROMOTIONSINFO_VIEW;
        $scope.model.performanceDescriptionInfoView = PERFORMANCE_DESCRIPTIONINFO_VIEW;
        $scope.model.performancePriceInfoView = PERFORMANCE_PRICEINFO_VIEW;

        // Starts the booking process
        $scope.startBookingProcess = function () {
            if (!$scope.pageService.getStep())
                $scope.pageService.setStep(CURRENT_STEP_SHOP);
        }

        // Starts the booking process with a promotion
        $scope.startPromotionBookingProcess = function (promotionId) {
            $scope.model.promotionId = promotionId;
            $scope.updateCalendarMonth();
            if (!$scope.pageService.getStep())
                $scope.pageService.setStep(CURRENT_STEP_SHOP);
        }

        $scope.createSectionTooltip = function (sectionCode) {
            var sectionData = $.grep($scope.model.selectedPerformance.sections, function (e) { return e.code == sectionCode; })[0];
            if (!sectionData) return null;
            var content = '<h1 class="section-tooltip">' + sectionData.name + '</h1>';

            if (sectionData.isSoldOut) {
                content += '<p class="section-tooltip no-tickets-available section-"' + sectionData.code + ' >' + ticketportal.webshop.localization.noTicketsAvailableInThisSector + "</p>";
                return content;
            }
            
            for (var i = 0; i < sectionData.priceCategories.length; i++) {
                var categoryAvailability = sectionData.priceCategories[i];
                var categoryColorHtml = '';
                if (categoryAvailability.colorCode !== undefined) {
                    categoryColorHtml = "<span class='price-category-color-bar in-section-tooltip' style='background-color: " + categoryAvailability.colorCode + ";'></span>";
                }
                content += '<p>' + categoryColorHtml + '<strong>' + categoryAvailability.name + '</strong>';
                if (categoryAvailability.ticketAvailabilityText.length > 0)
                    content += ': ' + categoryAvailability.ticketAvailabilityText;
                content += '</p>';
            }

            return content;
        }

        // Change the current currency
        $scope.changeCurrency = function (currency) {
            $scope.pageService.pageModel.isLoading = true;
            $scope.ticketService.setBasketCurrency(currency.symbol).then(function (data) {
                $rootScope.$broadcast('event:BasketChanged', data);

                // update selected performance model with new currency
                if (data.allowedCurrencies)
                    $scope.model.updateAvailableCurrencies(data.allowedCurrencies);

                $scope.pageService.pageModel.isLoading = false;
            });
        }

        $scope.bindImageMap = function () {
            // TODO: create a directive out of it
            var mapSeatingMapName = $scope.model.isSeatingMapOpen ? '#mapSeatingMapModal' : '#mapSeatingMap';
            var imgSeatingMapName = $scope.model.isSeatingMapOpen ? '#imgSeatingMapModal' : '#imgSeatingMap';

            $(mapSeatingMapName).children().remove();
            if ($scope.model.selectedSeatingImage) {
                for (var i = 0; i < $scope.model.selectedSeatingImage.areas.length; ++i) {
                    var area = $scope.model.selectedSeatingImage.areas[i];
                    var name = area.isImageUrl ? 'perspective-' + i : area.key;
                    var imageUrl = area.isImageUrl ? area.key : '';
                    var imageName = area.imageName ? area.imageName : '';
                    $(mapSeatingMapName).append('<area shape="' + area.shape + '" coords="' + area.coordinates + '" name="' + name + '" imageUrl="' + imageUrl + '" imageName="' + imageName + '" href="#" />');
                }

                var areas = new Array();
                var htmlAreas = $(mapSeatingMapName + ' > area');
                for (var i = htmlAreas.length - 1; i >= 0; i--) {
                    var name = $(htmlAreas[i]).attr('name');
                    var toolTipContent = '';
                    if (name.indexOf('perspective-') >= 0) {
                        var imageUrl = $(htmlAreas[i]).attr('imageUrl');
                        toolTipContent = '<img class="image" src="' + imageUrl + '"></img>';
                    }
                    else {
                        toolTipContent = '<div class="seating-map-section-tooltip">' + $scope.createSectionTooltip(name) + '</div>';
                    }

                    if (toolTipContent && toolTipContent.length > 0)
                        areas.push({ key: name, toolTip: toolTipContent });
                }

                var $img = $(imgSeatingMapName);

                $img.attr('src', $scope.model.selectedSeatingImage.imageUrl);

                // customize the seating map based on the webShop2AppSettings.customization
                var fillOpacity = 0.4;
                var fillColor = 'd42e16';
                var strokeColor = '3320FF';
                var strokeOpacity = 0.8;
                var strokeWidth = 4;
                var displayStroke = true;

                if ($scope.webShop2AppSettings.customization !== undefined && $scope.webShop2AppSettings.customization.seatingMap !== undefined) {
                    var seatingMapCustomization = $scope.webShop2AppSettings.customization.seatingMap;

                    if (seatingMapCustomization.imageMapFillColor != undefined)
                        fillColor = seatingMapCustomization.imageMapFillColor;
                    if (seatingMapCustomization.imageMapFillOpacity != undefined)
                        fillOpacity = seatingMapCustomization.imageMapFillOpacity;
                    if (seatingMapCustomization.imageMapStrokeColor != undefined)
                        strokeColor = seatingMapCustomization.imageMapStrokeColor;
                    if (seatingMapCustomization.imageMapStrokeOpacity != undefined)
                        strokeOpacity = seatingMapCustomization.imageMapStrokeOpacity;
                    if (seatingMapCustomization.imageMapStrokeWidth != undefined)
                        strokeWidth = seatingMapCustomization.imageMapStrokeWidth;
                    if (seatingMapCustomization.imageMapDisplayStroke != undefined)
                        displayStroke = seatingMapCustomization.imageMapDisplayStroke;
                }

                if (typeof $img.mapster === 'function') {
                    $img.mapster({
                        fillOpacity: fillOpacity,
                        fillColor: fillColor,
                        stroke: displayStroke,
                        strokeColor: strokeColor,
                        strokeOpacity: strokeOpacity,
                        strokeWidth: strokeWidth,
                        singleSelect: true,
                        mapKey: 'name',
                        listKey: 'name',
                        onClick: $scope.selectImageArea,
                        toolTipContainer: '<div class="seating-map-tooltip"></div>',
                        toolTipClose: ["tooltip-click", "area-click", "img-mouseout", "area-mouseout"],
                        areas: areas,
                        showToolTip: !$scope.pageService.isMobile() && !$scope.pageService.isTablet(),
                        onShowToolTip: $scope.pageService.isMobile() ? null : function (e) {
                            var imagePos = $(imgSeatingMapName).offset();
                            var left = ((imagePos.left - (e.toolTip.width() || e.toolTip.children().width())) - 40);
                            left = Math.max(0, left) + 'px';
                            var top = (imagePos.top + ($(imgSeatingMapName).height() / 2)) - ((e.toolTip.height() || e.toolTip.children().height()) / 2);
                            top = Math.max(0, top) + 'px';
                            e.toolTip.css({ left: left, top: top });
                        }
                    });

                    if ($scope.pageService.isMobile()) {
                        $img.mapster('resize', 320, 0, 0);
                    }
                }

                if ($scope.model.isSeatingMapOpen)
                    $scope.setSeatingMap(false);
            }

        }

        /**
         * 
         * Hack to expand the promotion tab in mobile view if the performance has no description
         * 
         * Please kill me
         * 
         * */
        $scope.hackExpandPromotionTab = function () {
            // get the performance accordion div for mobile view
            let e = document.getElementById('performance-accordion');

            // check if it's visible
            let visible = window.getComputedStyle(e).display !== 'none';

            // if it's visible and the description tab is not shown, expand the promotion tab
            if (visible && !$scope.model.selectedPerformance.isDisplayingReservationInfoTab)
                $scope.showTab('promotionInfo', false);
        }

        $scope.showTab = function (tab, selfCollapse) {
            if (tab == 'reservationinfo') {
                $scope.model.selectedPerformance.isDisplayingReservationInfoTab = selfCollapse ? !$scope.model.selectedPerformance.isDisplayingReservationInfoTab : true;                
                $scope.model.selectedPerformance.isDisplayingPromotionInfoTab = false;
                $scope.model.selectedPerformance.isDisplayingPriceInfoTab = false;
            } else if (tab == 'priceinfo') {                  
                $scope.model.selectedPerformance.isDisplayingReservationInfoTab = false;
                $scope.model.selectedPerformance.isDisplayingPromotionInfoTab = false;
                $scope.model.selectedPerformance.isDisplayingPriceInfoTab = selfCollapse ? !$scope.model.selectedPerformance.isDisplayingPriceInfoTab : true;

            } else if (tab == 'promotionInfo') {
                $scope.model.selectedPerformance.isDisplayingPromotionInfoTab = selfCollapse ? !$scope.model.selectedPerformance.isDisplayingPromotionInfoTab : true;
                $scope.model.selectedPerformance.isDisplayingReservationInfoTab = false;
                $scope.model.selectedPerformance.isDisplayingPriceInfoTab = false;
            }

            // in iFrame post a message notifying which tab is active
            $scope.pageService.postMessageToIFrameParent("viewPerformanceTab-" + tab);
        }

        // Goes back to the calendar/performance list view
        $scope.backToPerformanceList = function () {
            $scope.resetSelectedPerformance();
        }

        // resets the current selected performance, in other words, makes that no performance is currently selected
        $scope.resetSelectedPerformance = function () {
            $scope.model.selectedPerformance = null;
            $scope.model.selectedDate = null;
            $scope.model.displayBackToMainViewButton = false;
            $scope.model.showPriceCategoryDiscounts = false;
            $scope.model.selectedSection = null;
            $scope.model.selectedSeatingImage = null;
            $scope.model.bestPlaceSelection.reset();
            $scope.model.hasPerformanceDeepLink = false;

            // in iFrame post a message notifying that the calendar is visible again
            $scope.pageService.postMessageToIFrameParent("event/performancelist");

            // reset breadcrumb navigation
            $scope.pageService.setNavigationTitle('');
        }

        // Reloads the current performance
        $scope.reloadCurrentPerformance = function () {
            if ($scope.model.selectedPerformance)
                $scope.selectPerformance($scope.model.selectedPerformance.tag, $scope.model.selectedPerformance.startDate);
        }

        // sets the current performance
        $scope.selectPerformance = function (tag, date, options) {
            $scope.model.icalURL = $scope.ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + encodeURIComponent(tag.replace('|', '-')) + '/ical');            
            $scope.pageService.pageModel.isLoading = true;
            $scope.seatMapController = null;
            $scope.ticketService.getPerformance(tag, date).then(function (data) {
                if (data.succeeded != false) {
                    if (options && options.changeToShopStep) {
                        $scope.pageService.setStep(CURRENT_STEP_SHOP);
                    }

                    $scope.pageService.setNavigationTitle(data.name);

                    $scope.pageService.trackNavigation('ticket/' + data.tag);

                    var selectedPerformanceData = new $scope.model.performanceModel(data);
                    $scope.model.canvasModel = data.canvasModel;
                    if ($scope.model.canvasModel == undefined) {
                        $scope.model.canvasModel = { useCanvas: false }
                    }
                    $scope.model.selectedPerformance = selectedPerformanceData;
                    $scope.model.selectedDate = date || data.startDate;
                    $scope.model.displayBackToMainViewButton = false;
                    $scope.model.showPriceCategoryDiscounts = false;
                    $scope.model.selectedSection = null;
                    $scope.model.selectedSeatingImage = null;
                    $scope.model.seatingPlanCode = data.seatingPlanCode;
                    $scope.model.bestPlaceSelection.reset();
                    $scope.model.selectedSeats = [];
                    $scope.model.performanceName2 = data.name2;
                    $scope.model.performanceName3 = data.name3;
                    $scope.model.performanceImageUrl = data.performanceImageUrl;
                    $scope.model.performanceDescription = data.performanceDescription;

                    $scope.model.selectedPerformance.init({ allowedCurrencies: $scope.mainModel.allowedCurrencies });

                    $scope.customizationService.onPerformanceSelected($scope.model.selectedPerformance);

                    if ($scope.model.selectedPerformance.isReservationAllowed) {
                        $scope.customizationService.buildFlattenPriceCategory($scope.model.selectedPerformance);

                        var isSilentSelection = options && options.silent;

                        // for mobiles disable seating map reservation
                        if ($scope.model.selectedPerformance.isSeatingMapReservationAllowed && $scope.model.selectedPerformance.isBestPlaceReservationAllowed) {
                            if ($scope.pageService.isMobile())
                                $scope.model.selectedPerformance.isSeatingMapReservationAllowed = false;
                        }

                        if ($scope.model.canvasModel && $scope.model.canvasModel !== undefined && $scope.model.canvasModel.useCanvas) {
                            var preSelectedSection = options && options.section;
                            $scope.safeBindCanvasSeatingMap(data.canvasModel, preSelectedSection);
                        }
                        else {
                            // set the current seating map
                            if ($scope.model.selectedPerformance.seatingMapImage && !$scope.model.seatingPlanCode) {
                                $scope.model.selectedPerformance.seatingMapImage.isTopLevel = true;
                                $scope.model.selectedSeatingImage = $scope.model.selectedPerformance.seatingMapImage;
                                $scope.bindImageMap();
                            }

                            // display seating map help
                            if (!$scope.model.seatingMapReservationHelpAlreadyDisplayed && $scope.model.selectedPerformance.isSeatingMapReservationAllowed) {
                                if (!isSilentSelection)
                                    $scope.showSeatingMapBookingHelp();
                                $scope.model.seatingMapReservationHelpAlreadyDisplayed = true;
                            }

                            // select section
                            if (!$scope.model.selectedPerformance.showSectionDirectly) {

                                // if there is a section to pre select do it (scenario: click on basket item)
                                var preSelectedSection = options && options.section;
                                if (preSelectedSection && preSelectedSection.length > 0) {
                                    $scope.selectSection(preSelectedSection);
                                } else {
                                    // for a only best place performance display the categories
                                    if ($scope.model.selectedPerformance.isBestPlaceReservationAllowed && !$scope.model.selectedPerformance.isSeatingMapReservationAllowed) {
                                        $scope.buildBestPlaceSelectionModel();
                                    }
                                }
                            }
                            else {
                                // show section directly
                                // 1. select the section
                                if ($scope.model.selectedPerformance.sections.length > 0) {
                                    $scope.selectSection($scope.model.selectedPerformance.sections[0].code, false);
                                }
                            }
                        }
                    }

                    if ($scope.model.seatingPlanCode) {
                        $scope.ticketService.getSeatingPlan(tag, $scope.model.selectedPerformance.seatingPlanCode).then(function (data) {
                            var seatingPlan = JSON.parse(data.seatingPlan.jsonString);
                            var seatPicker = new SeatPicker({
                                "canvas": document.querySelector("canvas.canvas-seatpicker"),
                                "tooltip": {
                                    "container": document.querySelector(".tooltip-seatpicker"),
                                    "overrides": {
                                        "content": true,
                                        "display": false,
                                        "hide": false,
                                        "position": true
                                    }
                                },
                                "categories": data.categories,
                                "layout": seatingPlan.SeatingPlan.Layout,
                                "colourPalette": {
                                    "selectedSeat": data.selectedSeatColor
                                },
                                "canvasControls": {
                                    "zoomInButton": document.querySelector(".seatingPlan-zoomIn"),
                                    "zoomOutButton": document.querySelector(".seatingPlan-zoomOut"),
                                    "zoomResetButton": document.querySelector(".seatingPlan-reset")
                                },
                                "seatStyles": {
                                    "selected": {
                                        "icon": "StrokeCircleWithTick"
                                    }
                                }
                            });

                            // Set reserved/sold/not available seats
                            seatPicker.categories.forEach(function (category) {
                                if (category.status && category.status !== 1) {
                                    // Reserved
                                    if (category.status == 2) {
                                        category.seatStyle.icon = seatPicker.enums.SeatIcon.StrokeCircleWithSlash;
                                    }
                                    // Sold or not available
                                    else if (category.status == 3 || category.status == 4) {
                                        category.seatStyle.icon = seatPicker.enums.SeatIcon.StrokeCircleWithCross;
                                    }
                                    seatPicker.addUpdateCategory(category, false);
                                }
                            });

                            // Add sectionMappings to seatmap controller
                            seatPicker.sectionMappings = data.sections;

                            // Add seat mappings to seatpicker
                            seatPicker.seatMappings = data.seats;

                            // Remove seat function
                            seatPicker.removeSeat = function (seat) {
                                seatPicker.unselectSeat(seat);
                                var index = $scope.model.selectedSeats.indexOf(seat);
                                if (index !== -1) {
                                    $scope.model.selectedSeats.splice(index, 1);
                                }
                            }

                            // Zoom to section function
                            seatPicker.goToSection = function (sectionCode, seat) {
                                var blocks = seatPicker.getBlocks(sectionCode);
                                var block = !!seat ? blocks.find(function (b) {
                                    return b.rows.find(function (r) {
                                        return r.seats.find(function (s) {
                                            return s.identifiers.includes(seat.identifiers[0])
                                        })
                                    })
                                }) : blocks[0];
                                seatPicker.zoomToBlock(block)
                            }

                            // Set selected seats
                            seatPicker.setSelectedSeats = function (selectedSeats) {
                                selectedSeats.forEach(function (selectedSeat) {
                                    seatPicker.setAdditionalSeatProperties(selectedSeat);
                                    $scope.model.selectedSeats.push(selectedSeat);
                                });
                            }

                            // Set additional seat properties
                            seatPicker.setAdditionalSeatProperties = function (seat) {
                                var seatMapping = $scope.seatMapController.seatMappings.find(function (m) { return m.seatMappingId === seat.identifiers[0] });
                                if (!!seatMapping) {
                                    var section = $scope.seatMapController.sectionMappings.find(function (s) { return s.code === seatMapping.sectionCode });
                                    var priceCategory = $scope.seatMapController.categories.find(function (c) { return c.seatIdentifiers && c.seatIdentifiers.includes(seat.identifiers[0]) });
                                    var ticketPrice = $scope.model.selectedPerformance.priceCategories.find(function (c) { return c.uniqueId === seatMapping.priceCategoryId });
                                    seat.id = seatMapping.id;
                                    seat.sectionCode = seatMapping.sectionCode;
                                    seat.sectionName = !!section ? section.name : '';
                                    seat.priceCategoryName = !!priceCategory ? priceCategory.name : '';
                                    seat.rowIdentifier = seatMapping.rowIdentifier;
                                    seat.seatIdentifier = seatMapping.seatIdentifier;
                                    seat.price = !!ticketPrice ? ticketPrice.prices[0].formattedPriceWithFees : '';
                                    seat.status = seatMapping.status;
                                    seat.statusName = seatMapping.statusName;
                                    seat.color = !!priceCategory ? priceCategory.seatStyle.colour : '';;
                                } else {
                                    var section = $scope.seatMapController.sectionMappings.find(function (s) { return s.code === seat.blockLabel });
                                    var priceCategory = $scope.seatMapController.categories.find(function (c) { return c.seatIdentifiers && c.seatIdentifiers.includes(seat.identifiers[0]) });
                                    seat.sectionCode = seat.blockLabel;
                                    seat.sectionName = !!section ? section.name : '';
                                    seat.priceCategoryName = !!priceCategory ? priceCategory.name : '';
                                    seat.rowIdentifier = seat.rowLabel;
                                    seat.seatIdentifier = seat.label;
                                    seat.price = !!priceCategory ? priceCategory.price : '';
                                    seat.color = !!priceCategory ? priceCategory.seatStyle.colour : '';;
                                }
                            }

                            // Set and display tooltip
                            seatPicker.setTooltip = function (seat) {
                                var html = '';
                                if (!!seat) {
                                    seatPicker.setAdditionalSeatProperties(seat);
                                    if (seat.color) {
                                        seatPicker.tooltip.container.style.background = seat.color;
                                        seatPicker.tooltip.container.style.color = textColorGenerator(seat.color);
                                    } else {
                                        seatPicker.tooltip.container.style.background = '';
                                        seatPicker.tooltip.container.style.color = '';
                                    }

                                    html += '<div class="tooltip-seat-location">';
                                    if (seat.sectionName) {
                                        html += '<span class="tooltip-seat-section">' + seat.sectionName + '</span>';
                                    }
                                    html += '<span class="tooltip-seat-identifier">' + ticketportal.webshop.localization.seatRow + ": " + seat.rowIdentifier + '</span>';
                                    html += '<span class="tooltip-seat-identifier">' + ticketportal.webshop.localization.seatIdentifier + ': ' + seat.seatIdentifier + '</span>';
                                    html += '</div>';
                                    html += '<div class="tooltip-seat-information">';
                                    if (seat.priceCategoryName) {
                                        html += '<span class="tooltip-seat-category">' + seat.priceCategoryName + '</span>';
                                    }
                                    if (seat.price) {
                                        html += '<span class="tooltip-seat-price">' + seat.price + '</span>';
                                    }
                                    if (seat.status) {
                                        html += '<span class="tooltip-seat-status">' + seat.statusName + '</span>';
                                    }
                                    html += '</div>';

                                    var seatCoords = seat.layoutProperties.centre;
                                    var canvasCoords = seatPicker.translateToCanvasCoordinates(seatCoords);
                                    seatPicker.tooltip.container.innerHTML = html;
                                    seatPicker.tooltip.x = canvasCoords.x + seatPicker.canvas.offsetLeft + 10;
                                    seatPicker.tooltip.y = canvasCoords.y + seatPicker.canvas.offsetTop + 10;
                                    seatPicker.displayTooltip();
                                }
                            }

                            // Add events
                            document.addEventListener(seatPicker.events.selectionChanged, function (event) {
                                $scope.model.selectedSeats = [];
                                seatPicker.setSelectedSeats(event.selectedSeats);
                                $scope.$apply(); 
                            });

                            document.addEventListener(seatPicker.events.click, function (event) {
                                seatPicker.zoomToBlock(event.block);
                            });

                            document.addEventListener(seatPicker.events.mousemove, function (event) {
                                seatPicker.setTooltip(event.seat);
                            });

                            document.addEventListener(seatPicker.events.tap, function (event) {
                                seatPicker.setTooltip(event.seat);
                            });

                            $scope.seatMapController = seatPicker;

                            // Set checked seats
                            $scope.seatMapController.seatMappings.forEach(function (seatMapping) {
                                if (seatMapping.isSeatChecked) {
                                    var seat = seatPicker.getSeat(seatMapping.seatMappingId);
                                    seatPicker.selectSeat(seat);
                                    if (!$scope.model.selectedSeats.some(function (s) { return s.identifiers[0] === seatMapping.seatMappingId })) {
                                        seatPicker.setSelectedSeats([seat]);
                                    }
                                }
                            });
                        });
                    }

                    $scope.pageService.pageModel.isLoading = false;

                    // in iFrame post a message saying a performance was selected
                    $scope.pageService.postMessageToIFrameParent("performanceSelected");
                }
                else {
                    $scope.pageService.pageModel.isLoading = false;
                    $rootScope.$broadcast('event:ShowAlert', { title: '', message: data.error.message });
                }
            });
        }

        // opens the external performance
        $scope.selectExternalPerformance = function (tag, date, url) {
            $window.open(url);
            $scope.pageService.trackNavigation('ticket/external/' + tag);
        }

        $scope.onSeatSelected = function (seat, selected) {
            if (selected)
                $scope.model.selectedSeats.push(seat);
            else {
                for (var i = 0; i < $scope.model.selectedSeats.length; i++)
                    if ($scope.model.selectedSeats[i].id == seat.id) {
                        $scope.model.selectedSeats.splice(i, 1);
                        break;
                    }
            }
        }

        $scope.onSectionSelected = function (section) {
            var sectionModel = $scope.model.selectedPerformance.findSectionByCode(section.code);
            if (sectionModel && sectionModel.isBestPlace) {
                var modalInstance = $uibModal.open({
                    controller: 'sectionBookingController',
                    templateUrl: '/Templates/SectionBooking.aspx',
                    resolve: {
                        section: function () {
                            return sectionModel;
                        },
                        currencyCode: function () {
                            return $scope.model.selectedPerformance.currencySymbol;
                        }
                    }
                });
                modalInstance.result.then($scope.addBestPlacesToBasket);
            }
        }

        $scope.safeBindCanvasSeatingMap = function (canvasModel, initialSectionCode) {
            if ($("#canvas")[0].clientWidth > 0) {

                var seatMapControllerOptions = {
                    host: "canvas",
                    scale: "scaleSlider",
                    reset: "reset",
                    moveUp: 'moveUp',
                    moveDown: 'moveDown',
                    moveLeft: 'moveLeft',
                    moveRight: 'moveRight',
                    refresh: 'refresh',
                    performanceModel: $scope.model.selectedPerformance,
                    onSeatSelected: function (seat, selected) {
                        $timeout(function () {
                            $scope.onSeatSelected(seat, selected);
                        }, 0);
                    },
                    onSectionSelected: function (section) {
                        $timeout(function () {
                            $scope.onSectionSelected(section);
                        }, 0);
                    }
                };

                var seatMapController = new seatingMapController(canvasModel, ticketService, seatMapControllerOptions, function () {
                    $scope.seatMapController = seatMapController;
                    if ($scope.seatMapController != null) {
                        if (initialSectionCode)
                            $scope.seatMapController.goToSection(initialSectionCode);

                        $scope.model.selectedSeats.length = 0;
                        for (var i = 0; i < canvasModel.selectedSeats.length; i++)
                            $scope.model.selectedSeats.push($scope.seatMapController.getTicketInfoFromSeat(canvasModel.selectedSeats[i]));
                    }
                });
                if ($scope.seatMapController == null)
                    $scope.seatMapController = seatMapController;
            } else {
                $timeout(function () { $scope.safeBindCanvasSeatingMap(canvasModel, initialSectionCode); }, 0);
            }
        }

        // Called by the image map
        $scope.selectImageArea = function (area) {
            if (area.key) {
                if (area.key.indexOf('perspective-') >= 0) {
                    var imageUrl = area.e.currentTarget.getAttribute('imageUrl');
                    var imageName = area.e.currentTarget.getAttribute('imageName');
                    var img = $('<img />', { src: imageUrl, 'class': 'image' });
                    $('#showImageModal .modal-title').html(imageName);
                    $('#showImageModal .modal-body').html(img);
                    $('#showImageModal').modal('show');
                }
                else
                    $scope.selectSection(area.key, true);
            }
        }

        // Shows membership login modal
        $scope.showMembershipLogin = function () {
            $rootScope.$broadcast('patron:displayMembershipLogin');
        }

        $rootScope.$on('event:ReloadCurrentData', function (event, args) {
            $scope.reloadCurrentPerformance();
        });

        $('#seatingMapModal').on('hidden.bs.modal', function () {
            $scope.backToMainSeatingMapView(true);
            $scope.model.isSeatingMapOpen = false;
        })

        $('.modal').on('hidden.bs.modal', function () {
            var area = $('area');
            if (area.length > 0)
                $('area').mapster('deselect');
        })
        
        $(function () {
            $('[data-toggle="popover"]').popover({ trigger: 'hover' });
        })

        // Show seating map booking help
        $scope.showSeatingMapBookingHelp = function () {
            if ($scope.pageService.showSeatingMapHelp())
                $('#seating-map-help').modal('show');
        }

        // Creates an array with ticket amount options
        $scope.createTicketAmountOptions = function (maxItems) {
            var itemOptions = new Array();
            for (var itemAmount = 0; itemAmount <= maxItems; ++itemAmount) {
                itemOptions.push(itemAmount);
            }

            return itemOptions;
        }

        $scope.buildBestPlaceSelectionModel = function (isAsync) {
            var defaultMaxItemOptions = $scope.createTicketAmountOptions($scope.model.selectedPerformance.basketMaximumItemCount);

            $scope.model.bestPlaceSelection.priceCategories.length = 0;

            if ($scope.model.selectedSection) {
                $scope.model.bestPlaceSelection.sectionId = $scope.model.selectedSection.id;
                $scope.model.bestPlaceSelection.sectionName = $scope.model.selectedSection.name;
                $scope.model.bestPlaceSelection.sectionCode = $scope.model.selectedSection.code;

               
                var priceCategoryList = new Array();
                for (var priceCategoryIndex = 0; priceCategoryIndex < $scope.model.selectedSection.priceCategories.length; priceCategoryIndex++) {
                    var priceCategory = $scope.model.selectedSection.priceCategories[priceCategoryIndex];
                    var itemOptions = defaultMaxItemOptions;
                    if (priceCategory.maximumTicketReservation)
                        itemOptions = $scope.createTicketAmountOptions(priceCategory.maximumTicketReservation);
                    priceCategoryList.push(new $scope.model.bestPlaceSelectionCategoryModel(priceCategory.uniqueId, priceCategory.name, priceCategory.fullPriceName, priceCategory.colorCode, priceCategory.priority, priceCategory.isSoldOut, priceCategory.prices, $scope.model.selectedPerformance.currencySymbol, itemOptions, priceCategory.ticketAvailabilityText, $scope.model.selectedPerformance.showDiscountBestPlace));
                }
                $scope.model.bestPlaceSelection.setPriceCategories(priceCategoryList);

            } else {
                $scope.model.bestPlaceSelection.sectionId = ''
                $scope.model.bestPlaceSelection.sectionName = '';
                $scope.model.bestPlaceSelection.sectionCode = '';

                // best place among all sections
                var priceCategoryList = new Array();
                for (var priceCategoryIndex = 0; priceCategoryIndex < $scope.model.selectedPerformance.priceCategories.length; priceCategoryIndex++) {
                    var priceCategory = $scope.model.selectedPerformance.priceCategories[priceCategoryIndex];
                    var itemOptions = defaultMaxItemOptions;
                    if (priceCategory.maximumTicketReservation)
                        itemOptions = $scope.createTicketAmountOptions(priceCategory.maximumTicketReservation);
                    priceCategoryList.push(new $scope.model.bestPlaceSelectionCategoryModel(priceCategory.uniqueId, priceCategory.name, priceCategory.fullPriceName, priceCategory.colorCode, priceCategory.priority, priceCategory.isSoldOut, priceCategory.prices, $scope.model.selectedPerformance.currencySymbol, itemOptions, priceCategory.ticketAvailabilityText, $scope.model.selectedPerformance.showDiscountBestPlace));
                }
                $scope.model.bestPlaceSelection.setPriceCategories(priceCategoryList);
            }

            if (isAsync) {
                $scope.$apply(function () {
                    if (($scope.model.selectedPerformance.isBestPlaceReservationAllowed && $scope.model.selectedPerformance.sections.length == 1) ||
                        $scope.model.selectedSection)
                        $scope.model.bestPlaceSelection.isValid = true;
                });
            } else {
                if (($scope.model.selectedPerformance.isBestPlaceReservationAllowed && $scope.model.selectedPerformance.sections.length == 1) ||
                    $scope.model.selectedSection)
                    $scope.model.bestPlaceSelection.isValid = true;
            }
        }

        $scope.backToMainSeatingMapView = function (isModalClose) {
            // 1. reset any best place being displayed
            $scope.model.bestPlaceSelection.reset();

            // 2. reset current section
            $scope.model.selectedSection = null;
            $scope.model.displayBackToMainViewButton = false;

            // 3. update the current displayed seating map
            if (!isModalClose) {
                $("#raster-seatingmap").remove();
                $scope.model.selectedSeatingImage = $scope.model.selectedPerformance.seatingMapImage;
                $scope.bindImageMap();
            }

            if ($scope.model.isSeatingMapOpen) {
                $scope.setSeatingMap(true);
                $scope.model.sectionNavigationItems.length = 0;
                $('#imgSeatingMapRenderModal').show();
            }
        }

        $scope.mobileDisplaySeatingMap = function () {
            if ($scope.model.selectedSection) {
                $scope.selectSection($scope.model.selectedSection.code, false, true);
            }
        }

        // Display the current section
        $scope.selectSection = function (sectionCode, isAsync, allowSeatingMapInMobile) {
            var selectedSection = $.grep($scope.model.selectedPerformance.sections, function (e) { return e.code == sectionCode; })[0];
            if (selectedSection) {
                $scope.model.selectedSection = selectedSection;

                var showAsBestPlace = selectedSection.isBestPlace;
                if ($scope.pageService.isMobile() && !allowSeatingMapInMobile)
                    showAsBestPlace = true;

                // check if it has no place and it is seating map selection
                if (!selectedSection.isGroupingSection && selectedSection.isSoldOut && !showAsBestPlace) {
                    alert(ticketportal.webshop.localization.noTicketsAvailableInThisSector);
                    return;
                }

                $scope.model.selectedSeatingImage = selectedSection.seatingMapImage;

                if (selectedSection.isGroupingSection) {
                    $scope.model.bestPlaceSelection.reset();
                    if (!$scope.pageService.isMobile()) {
                        $scope.model.isSeatingMapOpen = true;
                        $('#seatingMapModal').modal({ keyboard: true });
                        $('#imgSeatingMapRenderModal').show();
                    }
                    $scope.bindImageMap();
                    $scope.setSeatingMap(true);

                    $scope.pageService.postMessageToIFrameParent("performance-showing-section-" + sectionCode);

                } else if (showAsBestPlace) {
                    $scope.buildBestPlaceSelectionModel(isAsync);
                    if ($scope.model.isSeatingMapOpen) {
                        $('#seatingMapRenderModal').html("");
                        $('#seatingMapLegendModal').html("");
                        $('#imgSeatingMapRenderModal').hide();
                    }

                    $scope.pageService.postMessageToIFrameParent("performance-showing-section-" + sectionCode);

                } else {
                    $scope.model.bestPlaceSelection.reset();
                    $scope.pageService.pageModel.isLoading = true;
                    // display seating map booking process
                    $scope.ticketService.getPerformanceSeatingMapHtml($scope.model.selectedPerformance.tag, $scope.model.selectedSection.code).then(function (data) {
                        if (data.succeeded != false) {
                            if ($scope.model.selectedPerformance.showSectionDirectly) {
                                var elements = $(data);
                                var seatMap = $('#seatingMapRenderModal', elements);
                                var legend = jQuery.grep(elements, function (element) { return element.id == 'seatingMapLegendModal'; });
                                $('#seatingMapRender').html(seatMap);
                                $('#seatingMapLegend').html(legend);
                                $scope.pageService.pageModel.isLoading = false;
                                $scope.bindImageMap();

                                // allow iframe to recalculate it's size
                                $scope.pageService.postMessageToIFrameParent("performance-showing-section-" + sectionCode);
                            }
                            else {
                                $scope.model.isSeatingMapOpen = true;
                                // TODO: use directives to display the seating map modal
                                $('#seatingMapModal .modal-body-seatingMapRender').html(data);
                                $scope.pageService.pageModel.isLoading = false;
                                $('#seatingMapModal').modal({ keyboard: true });
                                $('#imgSeatingMapRenderModal').hide();
                                $scope.toggleAddPlaceToBasketButton()
                                $scope.buildSectionNavigation();
                            }
                        }
                        else {
                            $scope.pageService.pageModel.isLoading = false;
                            $rootScope.$broadcast('event:ShowAlert', { title: '', message: data.error.message });
                        }
                    });
                }

                // Display back button: if is not 'show section directly' and current section has a parent sections or is a grouping section
                var selectSectionHasParent = selectedSection.parentSectionCode != null && selectedSection.parentSectionCode.length > 0;
                var newDisplayBackToMainViewButtonValue = !$scope.model.selectedPerformance.showSectionDirectly && (selectedSection.isGroupingSection || selectSectionHasParent);

                if (isAsync) {
                    $scope.$apply(function () {
                        if ($scope.model.isSeatingMapOpen) {
                            $scope.buildSectionNavigation();
                            $scope.toggleAddPlaceToBasketButton();
                        }
                        else
                            $scope.model.displayBackToMainViewButton = newDisplayBackToMainViewButtonValue;
                    });
                } else {
                    if ($scope.model.isSeatingMapOpen) {
                        $scope.buildSectionNavigation();
                        $scope.toggleAddPlaceToBasketButton();
                    }
                    else
                        $scope.model.displayBackToMainViewButton = newDisplayBackToMainViewButtonValue;
                }
            }
        }

        $scope.buildSectionNavigation = function () {
            $scope.model.sectionNavigationItems.length = 0;
            if ($scope.model.selectedSection.parentSectionCode != null && $scope.model.selectedSection.parentSectionCode.length > 0) {
                var sectionCode = $scope.model.selectedSection.parentSectionCode;
                var sectionNavigation = null;
                while (sectionCode != null && sectionCode.length > 0) {
                    var selectedSection = $.grep($scope.model.selectedPerformance.sections, function (e) { return e.code == sectionCode; })[0];
                    $scope.model.sectionNavigationItems.push({
                        code: selectedSection.code,
                        name: selectedSection.name,
                        imageUrl: selectedSection.isGroupingSection ? selectedSection.seatingMapImage.imageUrl : '',
                        active: false
                    });
                    sectionCode = selectedSection.parentSectionCode;
                }
            }
            $scope.model.sectionNavigationItems.push({
                code: $scope.model.selectedSection.code,
                name: $scope.model.selectedSection.name,
                imageUrl: $scope.model.selectedSection.isGroupingSection ? $scope.model.selectedSeatingImage.imageUrl : '',
                active: true
            });
        }

        $scope.closeSeatingMap = function () {
            $('#seatingMapModal').modal('hide');
        }

        $scope.toggleAddPlaceToBasketButton = function () {
            if ($scope.model.selectedSection.isBestPlace) {
                $('#seatingMapModal .add-places-to-basket').hide();
                $('#seatingMapModal .add-bestplaces-to-basket').show();
            }
            else {
                $('#seatingMapModal .add-places-to-basket').show();
                $('#seatingMapModal .add-bestplaces-to-basket').hide();
            }
        }

        $scope.setSeatingMap = function (reset) {
            if (reset) {
                $('#seatingMapRenderModal').html("");
                $('#seatingMapLegendModal').html("");
            }
        }

        $scope.addModalBestPlacesToBasket = function () {
            $scope.addBestPlacesToBasket();
        }

        $scope.addBestPlacesToBasket = function (section) {
            var ticketsToReserve = new Array();
            var sectionCode;

            if (section === undefined) {
                sectionCode = $scope.model.bestPlaceSelection.sectionCode;
                if ($scope.model.bestPlaceSelection.isValid) {
                    for (var categoryIndex = 0; categoryIndex < $scope.model.bestPlaceSelection.priceCategories.length; categoryIndex++) {
                        var priceCategory = $scope.model.bestPlaceSelection.priceCategories[categoryIndex];
                        if (priceCategory.ticketAmount && priceCategory.ticketAmount != '') {
                            ticketsToReserve.push({ ticketId: 'bsp_sec-' + priceCategory.id + '-' + $scope.model.bestPlaceSelection.sectionCode, amount: priceCategory.ticketAmount });
                        }
                        for (var discountIndex = 0; discountIndex < priceCategory.discounts.length; discountIndex++) {
                            var discount = priceCategory.discounts[discountIndex];
                            if (discount.ticketAmount && discount.ticketAmount != '') {
                                ticketsToReserve.push({ ticketId: 'bspd_sec-' + priceCategory.id + '-' + discount.id + '-' + $scope.model.bestPlaceSelection.sectionCode, amount: discount.ticketAmount });
                            }
                        }
                    }
                }
            }
            else {  // handle best place for seat map 2 (canvas)
                for (var categoryIndex = 0; categoryIndex < section.priceCategories.length; categoryIndex++) {
                    var priceCategory = section.priceCategories[categoryIndex];
                    if (priceCategory.selectedAmount > 0) {
                        ticketsToReserve.push({ ticketId: 'bsp_sec-' + priceCategory.uniqueId + '-' + section.code, amount: priceCategory.selectedAmount });
                    }
                }
            }

            if (ticketsToReserve.length > 0) {
                $scope.pageService.pageModel.isLoading = true;

                $scope.ticketService.addBestPlacesToBasket($scope.model.selectedPerformance.tag, ticketsToReserve, sectionCode, $scope.model.selectedDate).then(function (data) {
                    $scope.handleReservationResult(data);
                    if (data.succeeded) {
                        $scope.ecommerceService.pushAddToCartPerformance($scope.mainModel, $scope.model);
                    }
                });
            }
        }

        $scope.handleReservationResult = function (data) {
            if (data.succeeded) {
                $rootScope.$broadcast('event:BasketChanged', data);
                $scope.pageService.pageModel.isLoading = false;

                if ($scope.model.isSeatingMapOpen) {
                    $scope.closeSeatingMap();
                }

                if ($scope.seatMapController != null && $scope.seatMapController.ensureNotInFullScreenMode)
                    $scope.seatMapController.ensureNotInFullScreenMode();

                if ($scope.pageService.pageModel.hasCrossSelling)
                    $scope.pageService.setStep(CURRENT_STEP_CROSSSELLING);
                else
                    $scope.pageService.setStep(CURRENT_STEP_BASKET);
            } else {
                $scope.pageService.pageModel.isLoading = false;
                var title = null;
                if ($scope.model.selectedSection)
                    title = $scope.model.selectedSection.name;
                $rootScope.$broadcast('event:ShowAlert', { title: title, message: data.error.message });
            }
        }

        $scope.addPlacesToBasket = function () {
            var selectedPlaces = $scope.selectedPlacesFinderService.find($scope.seatMapController, $scope.model.canvasModel.useCanvas);

            if (selectedPlaces.length > 0) {
                if ($scope.amountOfPlacesToSelect) {
                    if (selectedPlaces.length != $scope.amountOfPlacesToSelect) {
                        $rootScope.$broadcast('event:ShowAlert', { title: $scope.selectedSection.name, message: ticketportal.webshop.localization.seatingMapExactAmountOfPlaceAlertFormat.format(ticketportal.webshop.ticket.amountOfPlacesToSelect) });
                        return;
                    }
                }

                $scope.pageService.pageModel.isLoading = true;
                var section = null;
                if ($scope.model.selectedSection && $scope.model.selectedSection.code)
                    section = $scope.model.selectedSection.code;
                $scope.ticketService.addPlacesToBasket($scope.model.selectedPerformance.tag, section, selectedPlaces).then(function (data) {
                    if (data.succeeded) {
                        $scope.ecommerceService.pushAddToCartPerformance($scope.mainModel, $scope.model, selectedPlaces);
                    }
                    $scope.handleReservationResult(data);
                });
            }
        }

        $scope.setPerformancePromotion = function (isValid, promotion, exclusivePromotionCode) {
            $scope.model.performancePromotionFormSubmitted = true;
            if (isValid) {
                $scope.pageService.pageModel.isLoading = true;
                var promotionTitle = '';
                if (promotion != null) {
                    promotionTitle = promotion.title;
                    $scope.pageService.trackNavigation('ticket/' + $scope.model.selectedPerformance.tag + '/promotion/' + promotion.id);
                }
                $scope.ticketService.setPerformancePromotion($scope.model.selectedPerformance.tag, promotion, exclusivePromotionCode).then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;
                    if (data.succeeded) {
                        $scope.reloadCurrentPerformance();
                    } else {
                        $rootScope.$broadcast('event:ShowAlert', { title: promotionTitle, message: data.errorMessage });
                    }
                });
            }
        }

        $scope.updateCalendarMonth = function () {
            $scope.model.calendarUrl = $scope.getCalendarUrl();
        }

        $scope.performanceListPageChange = function (pageIndex) {
            $scope.model.calendarUrl = $scope.getCalendarUrl(pageIndex);
        }

        $scope.getCalendarUrl = function (pageIndex) {
            var parameterSeparator = $scope.model.baseCalendarUrl.indexOf('?') == -1 ? '?' : '&';
            var newCalendarUrl = $scope.model.baseCalendarUrl + parameterSeparator + 'date=' + $scope.model.selectedCalendarMonth;
            if ($scope.model.brokerPromotion && $scope.model.brokerPromotion.length > 0)
                newCalendarUrl += '&brokerPromotion=' + encodeURIComponent($scope.model.brokerPromotion);

            if ($scope.model.promotionId > 0)
                newCalendarUrl += '&promotion=' + $scope.model.promotionId;

            if (pageIndex)
                newCalendarUrl += '&pageIndex=' + pageIndex;

            return newCalendarUrl;
        }

        // handle 'display performance is item was clicked in basket'
        $scope.$watch('pageService.pageModel.currentStep', function () {
            if ($scope.pageService.getStep() == CURRENT_STEP_SHOP) {
                if (getQueryStringValue('step') == 'shop') {
                    if (!$scope.model.handledPerformanceDeepLink) {
                        $scope.model.handledPerformanceDeepLink = true;
                        if ($scope.mainModel.basketItemCount > 0) {
                            var performance = getQueryStringValue('performance');
                            if (performance && performance.length > 0) {
                                var selectPerformanceOptions = { silent: true, section: getQueryStringValue('section') };
                                $scope.selectPerformance(performance, getQueryStringValue('performanceDate'), selectPerformanceOptions);
                                return;
                            }
                        }
                    }
                }
                if ($scope.model.selectedPerformance && $scope.model.canvasModel && $scope.model.canvasModel.useCanvas)
                    $scope.reloadCurrentPerformance();
            }
        });

        $scope.handlePerformanceDeepLink = function () {
            if (!getQueryStringValue('step')) {
                if (getQueryStringValue('performance')) {
                    var performance = getQueryStringValue('performance');
                    if (performance && performance.length > 0) {
                        $scope.model.handledPerformanceDeepLink = true;
                        $scope.pageService.setStartStep(CURRENT_STEP_SHOP);
                        var selectPerformanceOptions = { silent: true, changeToShopStep: true };
                        $scope.selectPerformance(performance, null /* performanceDate */, selectPerformanceOptions);
                    }
                } else if (getQueryStringValue('performanceType')) {
                    var performanceType = getQueryStringValue('performanceType');
                    if (performanceType && performanceType.length > 0) {

                        $scope.model.handledPerformanceDeepLink = true;
                        $scope.pageService.setStartStep(CURRENT_STEP_SHOP);

                        // calendar is loaded async, update it after a delay
                        $timeout(function () {
                            $('#panel-type-' + performanceType + ' a.trigger').removeClass('collapsed');
                            $("#panel-collapse-type-" + performanceType).collapse('show');
                        }
                        , 500 // delay in ms
                        , false // invokeApply
                        );
                    }
                }
            }
        }

        $scope.showSelectedSeats = function () {
            if ($scope.model.selectedSeats.length > 0)
                $scope.model.selectedSeats.expanded = true;
        }

        $scope.hideSelectedSeats = function () {
            $scope.model.selectedSeats.expanded = false;
        }


        $scope.toggleSelectedSeatsVisibility = function () {
            if ($scope.model.selectedSeats.length > 0)
                $scope.model.selectedSeats.expanded = !$scope.model.selectedSeats.expanded;
            else
                $scope.model.selectedSeats.expanded = false;
        }

        $scope.handlePerformanceDeepLink();

        $scope.showInfo = function (message) {
            $scope.pageService.showModal({
                message: message
            });
        }

        $scope.$on('event:BasketChanged', function (event, args) {
            if (args.basket && (args.amountOfItemsRemoved || 0) > 0) {
                var seatIds = args.basket.itemGroups.map(function (g) {
                    return g.items.map(function (i) {
                        return parseInt(i.ticketID)
                    })
                }).flat();
                if (seatIds.length > 0) {
                    var seatIdsToRemove = $.grep($scope.model.selectedSeats.map(function (s) {
                        return s.id
                    }), function (seatId) {
                        return $.inArray(seatId, seatIds) == -1
                    });
                    seatIdsToRemove.forEach(function (seatId) {
                        if ($scope.model.canvasModel.useCanvas) {
                            $scope.seatMapController.removeSeat(seatId);
                        } else {
                            var seat = $scope.model.selectedSeats.find(function (s) { return s.id === seatId });
                            $scope.seatMapController.removeSeat(seat);
                        }
                    });
                } else {
                    $scope.model.selectedSeats.forEach(function (seat) {
                        if ($scope.model.canvasModel.useCanvas) {
                            $scope.seatMapController.removeSeat(seat.id);
                        } else {
                            $scope.seatMapController.unselectSeat(seat);
                        }
                    });

                    $scope.model.selectedSeats = [];
                }
            }
        });

        $scope.canvasWidthInit = function () {
            var canvasSeatPicker = document.querySelector(".canvas-seatpicker");
            var canvasWidth = canvasSeatPicker.parentElement.offsetWidth;
            canvasSeatPicker.width = canvasWidth;
        };


        $scope.alternatePriceInfoRow = function (priceInfoRows, discountVisible, index) {
            var items = discountVisible ? priceInfoRows : $.grep(priceInfoRows, function (e) { return !e.isDiscount });
            var itemIndex = items.indexOf(priceInfoRows[index]);

            return !(itemIndex % 2);
        }
    }
})();;
(function () {
    'use strict';

    var modelId = 'ticketModel';
    angular.module('webShop2App').factory(modelId, [model]);

    function model() {
        var performance = getQueryStringValue('performance');
        var model = {
            selectedPerformance: null,
            selectedSection: null,
            selectedSeatingImage: null,
            amountOfPlacesToSelect: 0,
            isSeatingMapOpen: false,
            showPriceCategoryDiscounts: false,
            seatingMapReservationHelpAlreadyDisplayed: false,
            selectedCalendarMonth: '',
            calendarUrl: '',
            baseCalendarUrl: '',
            promotionId: 0,
            brokerPromotion: '',
            icalURL: '',
            displayBackToMainViewButton: false,
            isMobile: false,
            bestPlaceSelection: new bestPlaceSelectionModel(),
            performancePromotionFormSubmitted: false,
            handledPerformanceDeepLink: false, // indicates if the handling of the basket item/performance deep linked already happened.
            hasPerformanceDeepLink: performance && performance.length > 0,
            sectionNavigationItems: [],

            // functions
            updateAvailableCurrencies: updateAvailableCurrencies,

            // child models
            performanceModel: performanceModel,
            bestPlaceSelectionModel: bestPlaceSelectionModel,
            bestPlaceSelectionCategoryModel: bestPlaceSelectionCategoryModel
        };

        return model;

        function updateAvailableCurrencies(allowedCurrencies) {
            if (this.selectedPerformance) {
                // 1. update the currency code and the array of allowed currencies
                for (var currencyIndex = 0; currencyIndex < allowedCurrencies.length; ++currencyIndex) {
                    if (allowedCurrencies[currencyIndex].selected) {

                        // update the current selected currency symbol
                        this.selectedPerformance.currencySymbol = allowedCurrencies[currencyIndex].text;

                        // update the list of available currencies in the performance model
                        for (var j = this.selectedPerformance.allowedCurrencies.length - 1; j >= 0; --j)
                            this.selectedPerformance.allowedCurrencies[j].isSelected = this.selectedPerformance.allowedCurrencies[j].symbol == allowedCurrencies[currencyIndex].text;

                        break;
                    }
                }

                // 2. if there is best place selection active, update with the new currency
                if (this.bestPlaceSelection && this.bestPlaceSelection.isValid) {
                    for (var i = 0; i < this.bestPlaceSelection.priceCategories.length; i++) {
                        this.bestPlaceSelection.priceCategories[i].updateCurrency(this.selectedPerformance.currencySymbol, this.selectedPerformance.showDiscountBestPlace);
                    }
                }
            }
        }
    }

})();;
// Modified:		07.02.2017 Starticket AG, St. Gallen lbo : #1495: Erweiterung Exklusive Promotionen
// Modified:		19.02.2016 ticketporal AG, St Gallen fbe	: (SCR 8833) Web shop 2: Cannot display seating map section if performance name has a pipe "|"
(function () {
    'use strict';

    var serviceId = 'ticketService';
    angular.module('webShop2App').factory(serviceId, ['$http', '$sce', 'webShop2AppSettings', service]);

    function service($http, $sce, webShop2AppSettings) {
        return {
            getPerformance: function (tag, selectedDate) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + tag + '?json=1&selectedDate=' + selectedDate);
                return $http.post(postUrl).then(function (response) {

                    // transform the reservationInformation html
                    // http://stackoverflow.com/questions/19415394/with-ng-bind-html-unsafe-removed-how-do-i-inject-html
                    response.data.reservationInformationSafe = $sce.trustAsHtml(response.data.reservationInformation);
                    response.data.reservationNotAllowedMessage = $sce.trustAsHtml(response.data.reservationNotAllowedMessage);
                    response.data.ticketPriceInformation = $sce.trustAsHtml(response.data.ticketPriceInformation);

                    if (response.data.promotions) {
                        for (var i = 0; i < response.data.promotions.length; i++) {
                            var promotion = response.data.promotions[i];                            
                            if (promotion.description) {
                                promotion.descriptionSafe = $sce.trustAsHtml(promotion.description);
                            }
                        }
                    }

                    return response.data;
                });               
            },

            // gets the html with the seating map
            getPerformanceSeatingMapHtml: function (tag, sectionCode) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + encodeURIComponent(tag.replace('|', '-')) + '/seatingmap?section=' + encodeURIComponent(sectionCode));
                return $http.post(postUrl).then(function (response) {
                    return response.data;
                });
            },

            // gets the seating map drawing objects
            getSeatMap: function (tag) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + tag + '/seatmap');
                return $http.post(postUrl).then(function (response) {
                    return response.data;
                });
            },

            // gets the seats for the region
            getSeats: function (performanceID, areas) {
                var data = {
                    performanceID: performanceID,
                    areas: []
                };
                
                for (var i = 0; i < areas.length; i++)
                    data.areas.push({
                        X1: areas[i].x1,
                        Y1: areas[i].y1,
                        X2: areas[i].x2,
                        Y2: areas[i].y2
                    });
                    
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + performanceID + '/seats');
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: data,
                }).then(function (response) {
                    return { data: response.data, areas: areas };
                });
            },

            // adds seating places to the basket
            addPlacesToBasket: function (tag, sectionCode, places) {            
                var data = {
                    json: 1,
                    placeList: places,
                    section: sectionCode,
                    tag: tag
                };
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + tag + '/reservation?section=' + encodeURIComponent(sectionCode));
                var postResult = $http.post(postUrl, data);
                return postResult.then(function (result) {
                    return result.data;
                });
            },

            // adds best places to the basket
            addBestPlacesToBasket: function (tag, tickets, sectionCode, selectedDate) {
                var data = {
                    json: 1,
                    tag: tag,
                    selectedDate: selectedDate
                };
                for (var i = 0; i < tickets.length; i++) {
                    data[tickets[i].ticketId] = tickets[i].amount;
                }

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + tag + '/reservation?section=' + encodeURIComponent(sectionCode));
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // Sets the basket currency
            setBasketCurrency: function (currencySymbol) {
                var data = { 
                    json: 1,
                    basketAction: 'changeCurrency',
                    basketActionParameter: currencySymbol
                };

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/');
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // set performance promotion
            setPerformancePromotion: function (tag, promotion, exclusivePromotionCode) {
                var data = {
                    json: 1,
                    action: 'promotion',
                    promotionId: promotion != null ? promotion.id : 0,
                    promotionCode: promotion != null ? promotion.userCode : exclusivePromotionCode
                };
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + tag + '/performanceaction');
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            getSeatingPlan: function (tag, seatingPlanId) {
                var getSeatMapUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + tag + '/seatingPlan');
                return $http({ method: 'GET', url: getSeatMapUrl + "?seatingPlanId=" + seatingPlanId, withCredentials: true }).then(function (response) {
                    return response.data;
               });
            }
        }
    }

    var selectedPlacesServiceId = 'selectedPerformancePlacesFinderService';
    angular.module('webShop2App').factory(selectedPlacesServiceId, selectedPerformancePlacesFinderService);
    function selectedPerformancePlacesFinderService() {
        return {
            find: function (seatMapController, useCanvas) {
                var selectedPlaces = [];
                var placeList = $('#placeList');
                var selectedPlacesId = [];
                if (placeList.length == 1) {
                    selectedPlacesId = placeList.val().split(',');
                }
                else if (typeof seatMapController !== "undefined" && seatMapController !== null) {
                    if (useCanvas) {
                        selectedPlacesId = seatMapController.getSelectedSeatsId();
                    } else {
                        selectedPlaces = seatMapController.getSelectedSeats().map(function (s) { return { SeatID: s.id, SectionCode: s.sectionCode } });
                    }
                }
                else if (typeof rasterSeatingMapRender !== "undefined" && rasterSeatingMapRender !== null) {
                    // raster
                    selectedPlacesId = rasterSeatingMapRender.getSelectedPlacesId();
                }

                if (selectedPlacesId.length > 0) {
                    for (var i = selectedPlacesId.length - 1; i >= 0; --i) {
                        if (selectedPlacesId[i]) {
                            selectedPlaces.push({
                                SeatID: selectedPlacesId[i]
                            });
                        }
                    }
                }

                return selectedPlaces;
            }
        };
    }
})();;
(function () {
    'use strict';

    var controllerId = 'basketController';
    angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', 'ecommerceService', 'basketService', 'basketModel', 'pageService', 'customizationService', '$window', '$modal', '$timeout', controller]);

    function controller($rootScope, $scope, ecommerceService, basketService, basketModel, pageService, customizationService, $window, $modal, $timeout) {
        $scope.basketService = basketService;
        $scope.model = basketModel;
        $scope.pageService = pageService;
        $scope.modal = $modal;
        $scope.customizationService = customizationService;
        $scope.ecommerceService = ecommerceService;
        $scope.enableEnhancedEcommerce = enableEnhancedEcommerce;
        $scope.model.promotionRedemptionText = ticketportal.webshop.localization.promotionRedemptionTextYes;

        $scope.$watch('pageService.pageModel.currentStep', function () {
            if ($scope.pageService.getStep() === CURRENT_STEP_BASKET) {
                if (!$scope.model.loaded) {
                    $scope.pageService.pageModel.isLoading = true;
                    $scope.basketService.get().then(function (data) {
                        $scope.customizationService.updateBasketModel(data, $scope.model);
                        $scope.pageService.pageModel.isLoading = false;

                        // in iFrame post a message saying that the basket is ready
                        // give it time until the basket html is rendered
                        var currScope = $scope;
                        $timeout(function () {
                            currScope.pageService.postMessageToIFrameParent("basketIsReady");
                        }, 250);
                    });
                } else {
                    // in iFrame post a message saying that the basket is ready
                    // give it time until the basket html is rendered
                    var currScope = $scope;
                    $timeout(function () {
                        currScope.pageService.postMessageToIFrameParent("basketIsReady");
                    }, 250);  
                }

                // ensure the promotion redemption is not open
                $scope.model.isPromotionRedemptionActivated = false;
            }
        });

        $rootScope.$on('event:BasketChanged', function (event, args) {
            // indicate the the basket modal must be reloaded
            $scope.model.loaded = false;
        });

        $scope.applyDiscount = function (item, discount) {
            $scope.pageService.pageModel.isLoading = true;
            var discountId = discount ? discount.id : 0;
            var value = discount ? discount.value : 0;
            $scope.basketService.applyDiscount(item.id, discountId, value).then(function (data) {
                if (data.succeeded) {
                    $rootScope.$broadcast('event:BasketChanged', data);
                    $scope.customizationService.updateBasketModel(data, $scope.model);
                    $scope.pageService.updateHasPendingBasketValidation(data.error.code === ERROR_REQUIRESLATEBASKETVALIDATION);
                }
                else
                    $rootScope.$broadcast('event:ShowAlert', { title: ticketportal.webshop.localization.errorTitle, message: data.errorMessage });

                $scope.pageService.pageModel.isLoading = false;
            });
        };

        $scope.applyItemPromotion = function (item) {
            $scope.pageService.pageModel.isLoading = true;
            $scope.model.resetItemErrors();
            $scope.basketService.applyItemPromotion(item.id, item.promotionCodeToApply, $scope.model.ticketInsurance.isActivated).then(function (data) {
                if (data.succeeded) {
                    $rootScope.$broadcast('event:BasketChanged', data);
                    $scope.customizationService.updateBasketModel(data, $scope.model);
                    $scope.pageService.updateHasPendingBasketValidation(data.error.code === ERROR_REQUIRESLATEBASKETVALIDATION);

                    // notify hosting app that basket item promotion has changed
                    $scope.pageService.postMessageToIFrameParent("basketItemPromotionChanged");
                } else {
                    if (data.error !== undefined && data.error.code === ERROR_CONCURRENCYERROR) {
                        $rootScope.$broadcast('event:BasketChanged', data);
                        $scope.customizationService.updateBasketModel(data, $scope.model);
                    }
                    else
                        item.promotionRedemptionError = data.errorMessage;
                }

                $scope.pageService.pageModel.isLoading = false;
            });
        };

        $scope.removeItem = function (item) {
            $scope.pageService.pageModel.isLoading = true;

            // Execute RemoveFromCart Tag
            if (enableEnhancedEcommerce === true) {
                if ($scope.model.itemGroups.length > 0) {
                    //parsing id from model.tag e.g.:(test-erlebnisrundgang-90)
                    let performanceTagId = $scope.ecommerceService.parseIdFromPerformanceTag(item.performanceTag); 
                    let itemGroup = $scope.model.itemGroups;
                    let filteredItemGroup = '';
                    if (performanceTagId !== null) {
                        //get Correct itemGroup for id
                        filteredItemGroup = itemGroup.filter(function (d) { return d.id === performanceTagId; })[0]; 
                    }
                    let data = {};
                    data.id = performanceTagId;
                    data.name = filteredItemGroup.name;
                    data.price = item.finalPrice;
                    data.quantity = item.quantity;
                    $scope.ecommerceService.pushRemoveItemsFromCart(data);
                }
            }
            $scope.basketService.removeItem(item.id).then(function (data) {
                if (data.succeeded) {
                    $rootScope.$broadcast('event:BasketChanged', data);
                    $scope.customizationService.updateBasketModel(data, $scope.model);
                    $scope.pageService.updateHasPendingBasketValidation(data.error.code === ERROR_REQUIRESLATEBASKETVALIDATION);
                    $scope.pageService.pageModel.isLoading = false;

                    // notify hosting app that basket item was removed
                    $scope.pageService.postMessageToIFrameParent("basketItemRemoved");
                } else {
                    if (data.error !== undefined && data.error.code === ERROR_CONCURRENCYERROR) {
                        $rootScope.$broadcast('event:BasketChanged', data);
                        $scope.customizationService.updateBasketModel(data, $scope.model);
                    }
                    else {
                        $rootScope.$broadcast('event:ShowAlert', { title: ticketportal.webshop.localization.errorTitle, message: data.errorMessage });
                    }
                    $scope.pageService.pageModel.isLoading = false;
                }
            });
        };

        // remove all items of the basket
        $scope.emptyBasket = function () {
            $scope.pageService.pageModel.isLoading = true;
            // Execute RemoveFromCart Tag
            if (enableEnhancedEcommerce === true) {
                if ($scope.model.itemGroups.length > 0) {
                    let itemGroup = $scope.model.itemGroups;
                    for (let i = 0; i < itemGroup.length; i++) {
                        for (let j = 0; j < itemGroup[i].items.length; j++) {
                            $scope.ecommerceService.pushRemoveItemsFromCart(itemGroup[i].items[j], itemGroup);
                        }
                    }
                }
            }

            $scope.basketService.emptyBasket().then(function (data) {
                if (data.succeeded) {
                    $rootScope.$broadcast('event:BasketChanged', data);
                    $scope.customizationService.updateBasketModel(data, $scope.model);
                    $scope.pageService.updateHasPendingBasketValidation(data.error.code === ERROR_REQUIRESLATEBASKETVALIDATION);
                    $scope.pageService.pageModel.isLoading = false;

                    // notify hosting app that basket has been cleared
                    $scope.pageService.postMessageToIFrameParent("basketCleared");


                } else {
                    if (data.error !== undefined && data.error.code === ERROR_CONCURRENCYERROR) {
                        $rootScope.$broadcast('event:BasketChanged', data);
                        $scope.customizationService.updateBasketModel(data, $scope.model);
                    }
                    else {
                        $rootScope.$broadcast('event:ShowAlert', { title: ticketportal.webshop.localization.errorTitle, message: data.errorMessage });
                    }
                    $scope.pageService.pageModel.isLoading = false;
                }
            });
        };

        // updates the ticket insurance activation
        $scope.updateTicketInsurance = function (newValue) {

            if (newValue !== undefined) {
                $scope.model.ticketInsurance.isActivated = newValue;
            }

            $scope.pageService.pageModel.isLoading = true;
            $scope.basketService.updateTicketInsurance($scope.model.ticketInsurance.isActivated).then(function (data) {
                if (data.succeeded) {
                    $rootScope.$broadcast('event:BasketChanged', data);
                    $scope.customizationService.updateBasketModel(data, $scope.model);
                    $scope.pageService.pageModel.isLoading = false;
                } else {
                    if (data.error !== undefined && data.error.code === ERROR_CONCURRENCYERROR) {
                        $rootScope.$broadcast('event:BasketChanged', data);
                        $scope.customizationService.updateBasketModel(data, $scope.model);
                    }
                    else {
                        $rootScope.$broadcast('event:ShowAlert', { title: ticketportal.webshop.localization.errorTitle, message: data.errorMessage });
                    }
                    $scope.pageService.pageModel.isLoading = false;
                }
            });
        };

        $scope.togglePromotionRedemption = function () {
            $scope.model.isPromotionRedemptionActivated = !$scope.model.isPromotionRedemptionActivated;
            $scope.model.promotionRedemptionText = $scope.model.promotionRedemptionText === ticketportal.webshop.localization.promotionRedemptionTextYes ? ticketportal.webshop.localization.promotionRedemptionTextNo : ticketportal.webshop.localization.promotionRedemptionTextYes;
            
            // notify hosting app that basket promotion redemption has been toggled
            $scope.pageService.postMessageToIFrameParent("basketPromotionRedemptionToggled");
        };

        $scope.changeCurrency = function (currency) {
            $scope.pageService.pageModel.isLoading = true;
            $scope.basketService.setBasketCurrency(currency.text).then(function (data) {
                $rootScope.$broadcast('event:BasketChanged', data);
                $scope.customizationService.updateBasketModel(data, $scope.model);
                $scope.pageService.pageModel.isLoading = false;
            });
        };

        $scope.ensureBasketIsValidatedAndMoveToStep = function (step) {
            $scope.pageService.pageModel.isLoading = true;
            $scope.basketService.validateBasket().then(function (data) {
                $scope.pageService.pageModel.isLoading = false;
                if (data.succeeded) {
                    $scope.pageService.updateHasPendingBasketValidation(false);
                    if (step)
                        $scope.pageService.setStep(step);
                    else {
                        $scope.ecommerceService.pushStartCheckout($scope.model);
                        if (data.isLoggedOn) {
                            $scope.pageService.setStep(CURRENT_STEP_CHECKOUT);
                        }
                        else {
                            $scope.pageService.setStep(CURRENT_STEP_PATRON);
                        }
                    }
                } else {
                    $rootScope.$broadcast('event:ShowAlert', { title: ticketportal.webshop.localization.errorTitle, message: data.errorMessage });
                }
            });
        };

        $rootScope.$on('basket:validateAndMoveToStep', function (event, args) {
            $scope.ensureBasketIsValidatedAndMoveToStep(args.step);
        });


        $scope.moveToNextStep = function () {
            $scope.ensureBasketIsValidatedAndMoveToStep();
        };

        $scope.backToPreviousStep = function () {
            $scope.pageService.previousStep();
        };

        // display the item
        $scope.showItem = function (item) {
            var newUrl;
            if (item.typeName == 'Ticket') {
                newUrl = item.url + "?step=shop&performance=" + encodeURIComponent(item.performanceTag);
                if (item.isAdmissionTicket) {
                    newUrl += "&date=" + encodeURIComponent(item.validFrom);
                    newUrl += "&dayDefinitionId=" + item.dayDefinitionID;
                    newUrl += "&timeDefinitionId=" + item.timeDefinitionID;
                }
                else
                    newUrl += "&section=" + encodeURIComponent(item.sectionCode);
                $window.location = newUrl;
            } else if (item.typeName == 'Subscription') {
                newUrl = item.url + "?step=shop&section=" + encodeURIComponent(item.sectionCode);
                $window.location = newUrl;
            }
        };

        // goes back to the start shopping step
        $scope.continueShopping = function () {
            var homeUrl = window.location.protocol + "//" + window.location.host;
            if ($scope.webShop2AppSettings.folderName && $scope.webShop2AppSettings.folderName.length > 0) {
                homeUrl += "/" + $scope.webShop2AppSettings.folderName;
            }

            if (window.location.href != homeUrl)
                window.location = homeUrl;
            else
                location.reload();
        };

        $scope.removeItemPromotion = function (item) {
            $scope.pageService.pageModel.isLoading = true;
            $scope.basketService.removeItemPromotion(item.id).then(function (data) {
                if (data.succeeded) {
                    $rootScope.$broadcast('event:BasketChanged', data);
                    $scope.customizationService.updateBasketModel(data, $scope.model);
                    $scope.pageService.updateHasPendingBasketValidation(data.error.code === ERROR_REQUIRESLATEBASKETVALIDATION);

                    // notify hosting app that basket item promotion has changed
                    $scope.pageService.postMessageToIFrameParent("basketItemPromotionChanged");
                } else {
                    if (data.error !== undefined && data.error.code === ERROR_CONCURRENCYERROR) {
                        $rootScope.$broadcast('event:BasketChanged', data);
                        $scope.customizationService.updateBasketModel(data, $scope.model);
                    }
                    else
                        item.promotionRedemptionError = data.errorMessage;
                }

                $scope.pageService.pageModel.isLoading = false;
            });
        };
    }
})();;
(function () {
    'use strict';

    var modelId = 'basketModel';
    angular.module('webShop2App').factory(modelId, ['$sce', model]);

    function model($sce) {
        var model = {
            basketId: '',
            itemGroups: [],
            formattedTotal: '',
            formattedTotalToPay: '',
            promotions: [],
            hasPromotions: false,
            isPromotionRedemptionActivated: false,
            itemBasePriceTotal: 0,
            itemFeeTotal: 0,
            itemTotal: 0,
            reservationDateTime: null,
            ticketInsurance: { isEnabled: false, name: '', description: '', isActivated: false, termAndConditionsUrl: null, total: 0, formattedItemValueText: '' },
            loaded: false,
            allowedCurrencies: new Array(),
            hasPersonalization: false,

            // functions
            updateFromService: updateFromService,
            resetItemErrors: resetItemErrors
        }
        return model;

        // Updates the model based on the contents from the server
        function updateFromService(data) {
            this.hasPersonalization = false;
            var basketData = data.basket ? data.basket : data;
            this.basketId = basketData.basketId != undefined ? basketData.basketId : this.basketId;
            if (basketData.formattedTotal != undefined) this.formattedTotal = $sce.trustAsHtml(basketData.formattedTotal);
            if (basketData.formattedTotalToPay != undefined) this.formattedTotalToPay = basketData.formattedTotalToPay;
            if (basketData.itemGroups != undefined) {
                var promotionDictionary = {};
                this.itemGroups = basketData.itemGroups;
                for (var groupIndex = 0; groupIndex < this.itemGroups.length; groupIndex++) {
                    var group = this.itemGroups[groupIndex];
                    if (group.hasPersonalization) {
                        this.hasPersonalization = true;
                    }
                    for (var itemIndex = 0; itemIndex < group.items.length; itemIndex++) {
                        var item = group.items[itemIndex];
                        if (item.promotionOptions != null && item.promotionOptions.length > 0) {
                            for (var promotionIndex = item.promotionOptions.length - 1; promotionIndex >= 0; --promotionIndex) {
                                var promotion = item.promotionOptions[promotionIndex];
                                promotionDictionary[promotion.id] = promotion;
                            }
                        }
                    }
                }

                this.promotions.length = 0;
                for (var promotionId in promotionDictionary)
                    this.promotions.push(promotionDictionary[promotionId]);
                this.hasPromotions = this.promotions.length > 0;
            }

            this.allowedCurrencies.length = 0;
            if (basketData.allowedCurrencies) {
                for (var currencyIndex = 0; currencyIndex < basketData.allowedCurrencies.length; ++currencyIndex) {
                    this.allowedCurrencies.push(basketData.allowedCurrencies[currencyIndex]);
                }
            }

            var ticketInsuranceData = data.ticketInsurance || basketData.ticketInsurance;
            if (ticketInsuranceData) {
                this.ticketInsurance.isEnabled = ticketInsuranceData.isEnabled;
                this.ticketInsurance.isActivated = ticketInsuranceData.isActivated;
                this.ticketInsurance.termAndConditionsUrl = ticketInsuranceData.termAndConditionsUrl;
                this.ticketInsurance.total = ticketInsuranceData.total;
                this.ticketInsurance.formattedItemValueText = ticketInsuranceData.formattedItemValueText;
                this.ticketInsurance.name = ticketInsuranceData.name;
                this.ticketInsurance.description = $sce.trustAsHtml(ticketInsuranceData.description);
            }

            this.loaded = true;
        }

        // Resets the promotion item errors
        function resetItemErrors() {
            if (this.itemGroups) {
                for (var i = 0; i < this.itemGroups.length; i++) {
                    var group = this.itemGroups[i];
                    for (var k = 0; k < group.items.length; k++) {
                        var item = group.items[k];
                        item.promotionRedemptionError = '';
                    }
                }
            }
        }
    }

})();;
(function () {
    'use strict';

    var serviceId = 'basketService';
    angular.module('webShop2App').factory(serviceId, ['$http', '$sce', 'webShop2AppSettings', service]);

    function service($http, $sce, webShop2AppSettings) {
        return {
            get: function () {
                var formData = {
                    json: 1
                };

                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/?json=1');

                return $http({
                    method: 'POST',
                    url: url,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // removes item from the basket and returns the basket
            removeItem: function (itemId) {
                var formData = {
                    json: 1,
                    includeCrossSelling: 1,
                    includeItemOptions: 1,
                    includeBasket: 1,
                    basketAction: 'itemRemove',
                    basketActionParameter: itemId
                };

                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/?json=1');

                return $http({
                    method: 'POST',
                    url: url,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // applies discount to an item
            applyDiscount: function (itemId, discountId, value) {
                var formData = {
                    json: 1,
                    includeItemOptions: 1,
                    includeBasket: 1
                };

                formData['discount-' + itemId] = itemId + '_' + discountId;
                formData['discountValue-' + itemId] = value;

                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/?json=1');

                return $http({
                    method: 'POST',
                    url: url,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // applies promotion to an item
            applyItemPromotion: function (itemId, code, ticketInsuranceActivated) {
                var formData = {
                    json: 1,
                    includeItemOptions: 1,
                    includeBasket: 1,
                    singlePage: 1,
                    basketAction: 'promotionApply',
                    basketActionParameter: itemId,
                    insurance: ticketInsuranceActivated ? 'on' : ''
                };

                formData['promotion-' + itemId] = code;

                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/?json=1');

                return $http({
                    method: 'POST',
                    url: url,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // Sets the basket currency
            setBasketCurrency: function (currencySymbol) {
                var data = {
                    basketAction: 'changeCurrency',
                    json: 1,
                    basketActionParameter: currencySymbol
                };

                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/');

                return $http({
                    method: 'POST',
                    url: url,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // Updates the ticket insurance usage
            updateTicketInsurance: function (ticketInsuranceActivated) {
                var data = {
                    json: 1,
                    includeItemOptions: 1,
                    includeBasket: 1,
                    singlePage: 1,
                    basketAction: 'insurance',
                    insurance: ticketInsuranceActivated ? 'on' : ''
                };

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/');

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            validateBasket: function () {
                var data = {
                    json: 1,
                    singlePage: 1,
                    basketAction: 'validate'
                };

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/');

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            emptyBasket: function () {
                var data = {
                    json: 1,
                    singlePage: 1,
                    basketAction: 'emptyBasket'
                };

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/');

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // removes promotion from an item
            removeItemPromotion: function (itemId) {
                var formData = {
                    json: 1,
                    includeItemOptions: 1,
                    includeBasket: 1,
                    singlePage: 1,
                    basketAction: 'promotionRemove',
                    basketActionParameter: itemId
                };

                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/?json=1');

                return $http({
                    method: 'POST',
                    url: url,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            setPersonalization: function (items) {
                var formData = {
                    json: 1,
                    basketAction: 'personalizationApply',
                    basketActionParameter: JSON.stringify(items)
                };

                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/');

                return $http({
                    method: 'POST',
                    url: url,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            }
        };
    }

})();;
(function () {
    'use strict';

    var MyAccountReplaceCurrentBasketModalInstanceController = function($scope, $modalInstance) {
        $scope.model = [];
        $scope.no = function () {
            $modalInstance.dismiss('cancel');
        }
        $scope.yes = function () {
            $modalInstance.close(true);
        }        
    }
    MyAccountReplaceCurrentBasketModalInstanceController.$inject = ['$scope', '$modalInstance'];

    var MENU_VIEW = "MyAccountMenu.html";
    var ORDER_HISTORY_VIEW = "MyAccountOrderHistory.html";
    var ORDER_DETAIL_VIEW = "MyAccountOrderDetail.html";
    var MEMBERSHIP_PRIVILEGES_VIEW = "MyAccountMembershipPrivileges.html";
    var RESERVATIONS_VIEW = "MyAccountReservations.html";
    var RESERVATIONS_DETAIL_VIEW = "MyAccountReservationDetail.html";

    var controllerId = 'patronAccountController';
    angular.module('webShop2App').controller(controllerId, ['$scope', 'patronAccountService', 'patronAccountModel', 'pageService', 'patronService', 'webShop2AppSettings', '$modal', '$sce', controller]);

    function controller($scope, patronAccountService, patronAccountModel, pageService, patronService, webShop2AppSettings, $modal, $sce) {
        $scope.patronAccountService = patronAccountService;
        $scope.model = patronAccountModel;
        $scope.webShop2AppSettings = webShop2AppSettings;
        $scope.pageService = pageService;
        $scope.patronService = patronService;
        $scope.model.dynamicForm = $scope.model.dynamicFormModel(__formGenerator);
        $scope.model.registrationDynamicFormKey = __registrationFormGeneratorKey;
        $scope.model.mainAddressInfo = $scope.model.addressInfoModel(__formGenerator, $scope.model.registrationDynamicFormKey);
        $scope.model.deliveryAddressDynamicFormKey = __deliveryAddressFormGeneratorKey;
        $scope.model.deliveryAddressInfo = $scope.model.addressInfoModel(__formGenerator, $scope.model.deliveryAddressDynamicFormKey);

        $scope.model.creditCardAliases = __creditCardAliases;
        $scope.model.membershipPrivileges = __membershipPrivileges;
        $scope.model.billingAddressId = __billingAddressId;
        $scope.model.deliveryAddressId = __deliveryAddressId;
        $scope.model.isDeliveryAddressSameAsBillingAddress = __billingAddressId === __deliveryAddressId;
        $scope.model.isRegistration = __isRegistration;
        $scope.model.hasAccountInformation = __hasAccountInformation;
        $scope.model.socialLogins = __socialLogins;
        
        $scope.model.messages = {};
        $scope.model.messages.confirmCreditCardRemoval = __messageConfirmCreditCardRemoval;
        $scope.model.messages.confirmDisconnectSocialLogin = __messageConfirmDisconnectSocialLogin;
        $scope.model.messages.disconnectManuallySocialLogin = __messageDisconnectManuallySocialLogin;

        if (typeof __popMessage != 'undefined')
            $scope.pageService.pop('success', '', __popMessage, 4000, 'trustedHtml');

        if (typeof __errorMessageFromTempData == 'string' && __errorMessageFromTempData.length > 0)
            $scope.pageService.pop('error', '', __errorMessageFromTempData, 4000, 'trustedHtml');

        $scope.priceFormatter = function (value) {
            let newValue = value;
            let converterdValue = JSON.stringify(value);
            var split = converterdValue.split(".");
            if (!split[1]) {
                newValue = converterdValue+".00"
            } else {
                split[1].length === 1 && (newValue = converterdValue+"0"); 
            };            
            return newValue
        };

        $scope.displayOrderHistory = function () {
            $scope.pageService.trackNavigation('myaccount/orderhistory');
            $scope.pageService.postMessageToIFrameParent('myaccount/orderhistory');

            $scope.model.currentView = ORDER_HISTORY_VIEW;
            if (!$scope.model.isOrderHistoryCollectionLoaded) {
                $scope.pageService.pageModel.isLoading = true;

                // load orders
                $scope.patronAccountService.getOrderHistory().then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;
                    if (data.succeeded) {
                        $scope.model.isOrderHistoryCollectionLoaded = true;
                        $scope.model.updateOrderHistory(data.orders);
                    }
                });
            }
        }

        $scope.displayReservations = function () {
            $scope.pageService.trackNavigation('myaccount/reservations');
            $scope.pageService.postMessageToIFrameParent('myaccount/reservations');

            $scope.model.currentView = RESERVATIONS_VIEW;
            if (!$scope.model.isReservationsCollectionLoaded) {
                $scope.pageService.pageModel.isLoading = true;

                // load reservations
                $scope.patronAccountService.getReservations().then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;
                    if (data.succeeded) {
                        $scope.model.isReservationsCollectionLoaded = true;
                        $scope.model.updateReservations(data.reservations);
                    }
                });
            }
        }

        $scope.displayReservationDetail = function (id) {
            $scope.pageService.trackNavigation('myaccount/reservationdetail');
            $scope.pageService.postMessageToIFrameParent('myaccount/reservationdetail');
            $scope.pageService.pageModel.isLoading = true;

            // load orders
            $scope.patronAccountService.getReservationDetail(id).then(function (data) {
                $scope.pageService.pageModel.isLoading = false;
                if (data.succeeded) {
                    $scope.model.updateCurrentReservationDetail(data);
                    $scope.model.currentView = RESERVATIONS_DETAIL_VIEW;
                } else {
                }
            });
        }

        $scope.activateReservation = function () {

            if ($scope.model.reservationDetail && $scope.model.reservationDetail.canActivate) {

                // if there is a reservation already open ask a confirmation
                if ($scope.model.reservationDetail.hasOpenReservationInSession) {

                    var modalInstance = $modal.open({
                        animation: false,
                        size: 'lg',
                        templateUrl: 'MyAccountReplaceCurrentBasketModal.html',
                        controller: MyAccountReplaceCurrentBasketModalInstanceController
                    });

                    modalInstance.result.then(function (confirmResult) {
                        if (confirmResult)
                            $scope.doActivateReservation();
                    });

                }
                else {
                    $scope.doActivateReservation();
                }
            }
        }

        $scope.doActivateReservation = function() {
            
            $scope.pageService.pageModel.isLoading = true;

            // load orders
            $scope.patronAccountService.activateReservation($scope.model.reservationDetail.id).then(function (data) {
                $scope.pageService.pageModel.isLoading = false;
                if (data.succeeded) {

                    $scope.pageService.showModal({
                        message: ticketportal.webshop.localization.reservationActivatedMessage,
                        onClose: function () { window.location = data.nextUrl; }
                    });

                } else {
                }
            });
        }
        
        $scope.displayMembershipPrivileges = function () {
            $scope.pageService.trackNavigation('myaccount/membership-privileges');
            $scope.pageService.postMessageToIFrameParent('myaccount/membership-privileges');
            $scope.model.currentView = MEMBERSHIP_PRIVILEGES_VIEW;
        }

        $scope.displayMenu = function () {
            if ($scope.model.isStepRegisterUser) {
                $scope.model.isStepRegisterUser = false;

                // restore address
                $scope.model.mainAddressInfo = $scope.model.originalMainAddress;
                $scope.model.deliveryAddressInfo = $scope.model.originalDeliveryAddress;
            }
            else {
                // Set current registration adress info
                $scope.setCurrentAddressInfo($scope.model.mainAddressInfo, $scope.model.dynamicForm[$scope.model.registrationDynamicFormKey]);

                // Set current delivery adress info
                $scope.setCurrentAddressInfo($scope.model.deliveryAddressInfo, $scope.model.dynamicForm[$scope.model.deliveryAddressDynamicFormKey]);
            }

            $scope.model.isDeliveryAddressSameAsBillingAddress = $scope.model.billingAddressId === $scope.model.deliveryAddressId;
            $scope.model.currentView = MENU_VIEW;
            $scope.pageService.postMessageToIFrameParent('myaccount');
        }

        $scope.setCurrentAddressInfo = function (addressInfo, currentAddressInfo) {
            for (var name in currentAddressInfo) {
                switch (true) {
                    case name.indexOf('salutation') === 0:
                        addressInfo.salutationName = currentAddressInfo[name].name;
                        break;
                    case name.indexOf('companyName') === 0:
                        addressInfo.companyName = currentAddressInfo[name].value;
                        break;
                    case name.indexOf('firstName') === 0:
                        addressInfo.firstName = currentAddressInfo[name].value;
                        break;
                    case name.indexOf('surname') === 0:
                        addressInfo.surname = currentAddressInfo[name].value;
                        break;
                    case name.indexOf('address1') === 0:
                        addressInfo.address1 = currentAddressInfo[name].value;
                        break;
                    case name.indexOf('address2') === 0:
                        addressInfo.address2 = currentAddressInfo[name].value;
                        break;
                    case name.indexOf('city') === 0:
                        addressInfo.city = currentAddressInfo[name].value;
                        break;
                    case name.indexOf('postalCode') === 0:
                        addressInfo.postalCode = currentAddressInfo[name].value;
                        break;
                    case name.indexOf('state') === 0:
                        addressInfo.stateName = currentAddressInfo[name].name;
                        break;
                    case name.indexOf('country') === 0:
                        addressInfo.countryName = currentAddressInfo[name].name;
                        break;
                    case name.indexOf('email') === 0:
                        addressInfo.email = currentAddressInfo[name].value;
                        break;
                    case name.indexOf('homePhone') === 0:
                        addressInfo.homePhone = currentAddressInfo[name].value;
                        break;
                    case name.indexOf('mobilePhone') === 0:
                        addressInfo.mobilePhone = currentAddressInfo[name].value;
                        break;
                    case name.indexOf('companyPhone') === 0:
                        addressInfo.companyPhone = currentAddressInfo[name].value;
                        break;
                    case name.indexOf('fax') === 0:
                        addressInfo.fax = currentAddressInfo[name].value;
                        break;
                }
            }
        }

        $scope.displayOrderDetail = function (orderId) {
            function setData (orderId) {                
                $scope.pageService.trackNavigation('myaccount/orderdetail');
                $scope.pageService.postMessageToIFrameParent('myaccount/orderdetail');
                $scope.pageService.pageModel.isLoading = true;            
                
                // load orders
                $scope.patronAccountService.getOrderDetail(orderId).then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;
                    if (data.succeeded) {
                        $scope.model.updateCurrentOrderDetail(data.order);
                        $scope.model.currentOrderDetailsView = ORDER_DETAIL_VIEW;
                    }
                });
            };
            
            // For expanding order details 
            if ($scope.model.orderDetail === null) {
                setData(orderId);
            } else {
                if ($scope.model.orderDetail.id == orderId) {
                    $scope.model.orderDetail = null;
                    $scope.model.currentOrderDetailsView = undefined;
                } else {
                    setData(orderId);
                }
            };
        }

        $scope.goToChangePassword = function () {
            $scope.pageService.trackNavigation('myaccount/changepassword');
            $scope.pageService.showChangePasswordForm();
        }

        $scope.goToAliasRegistration = function () {
            $scope.pageService.trackNavigation('myaccount/aliasregistration');
            $scope.pageService.showAliasRegistrationForm(true);
        }

        $scope.removeCreditCardAlias = function (id) {
            if (confirm($scope.model.messages.confirmCreditCardRemoval)) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.patronService.removeCreditCardAlias(id).then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;
                    if (data.succeeded) {
                        for (var i = $scope.model.creditCardAliases.length - 1; i >= 0; i--)
                            if ($scope.model.creditCardAliases[i].id == id) {
                                $scope.model.creditCardAliases.splice(i, 1);
                                break;
                            }
                    } else {
                        if (data.errorMessage.length > 0)
                            $scope.model.errorMessage = data.errorMessage;
                    }
                });
            }
        }

        /**
         * Disconnect patron's social account.
         * 
         * Note: Patron must be loggedin via the social account to be able to disconnect automatically using the respective sdk/api.
         * 
         * @param {int} id
         * @param {string} socialAccountTyp
         */
        $scope.disconnectSocialLogin = function (id, socialAccountType) {
            if (confirm($scope.model.messages.confirmDisconnectSocialLogin)) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.patronAccountService.disconnectSocialLogin(id).then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;
                    if (data.succeeded) {

                        // use api to disconnect, only works is connected through the social account
                        try {                            
                            if (socialAccountType == 'Google') {
                                gapi.auth2.getAuthInstance().disconnect();
                            }
                            if (socialAccountType == 'Facebook') {
                                FB.api('/me/permissions', 'delete', function (response) {
                                    if (response == true || response.success == true) {
                                    } else {
                                        // must disconnect manually
                                        throw 'Could not revoke facebook permissions: ' + response;
                                    }
                                });
                            }
                            if (socialAccountType == 'Apple') {
                                // nothing to do, gently ask patron to revoke access in his apple id account https://support.apple.com/en-us/HT210426
                                throw 'Revoke apple association manually';
                            }
                        } catch (e) {
                            console.log(e)
                            $scope.pageService.showModal({
                                message: $scope.model.messages.disconnectManuallySocialLogin
                            });
                        }                        

                        // remove the social login from the list in my account page
                        for (var i = $scope.model.socialLogins.length - 1; i >= 0; i--) {
                            if ($scope.model.socialLogins[i].id == id) {
                                $scope.model.socialLogins.splice(i, 1);
                                break;
                            }
                        }
                        
                    } else {
                        if (data.errorMessage.length > 0)
                            $scope.model.errorMessage = data.errorMessage;
                    }
                });
            }
        }

        $scope.register = function (isValid) {
            $scope.model.dynamicWebFormSubmitted = true;

            var isRegisterStep = ($scope.model.hasAccountInformation && !$scope.model.isRegisterStepAccount) || (!$scope.model.hasAccountInformation && !$scope.model.isRegisterStepBilling);
            if (isRegisterStep) {
                $scope.registerNextStep(isValid);
                return;
            }

            if (isValid && ($scope.model.validDate === undefined || $scope.model.validDate === true)) {
                $scope.pageService.pageModel.isLoading = true;

                $scope.model.loginErrorMessage = '';
                // register patron
                $scope.patronService.register(false, $scope.model.dynamicForm, false, false, $scope.model.patronId, $scope.model.billingAddressId, $scope.model.deliveryAddressId, $scope.model.isDeliveryAddressSameAsBillingAddress).then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;

                    if (data.loginSucceeded) {
                        $scope.onLoggedOn(data);
                    } else if (data.succeeded) {
                        $scope.model.userDataSaved = true;
                        $scope.model.needsConfirmation = data.needsConfirmation;
                        $scope.model.isStepRegisterUser = false;
                        $scope.model.patronId = data.patronId;
                        $scope.model.billingAddressId = data.billingAddressId;
                        $scope.model.deliveryAddressId = data.deliveryAddressId;
                        $scope.model.isDeliveryAddressSameAsBillingAddress = data.billingAddressId === data.deliveryAddressId;
                        $scope.model.patronCategoryMissing = false;
                        if (!$scope.model.isLoggedOn) {
                            $scope.model.isStepLogin = true;
                        }

                        if ($scope.model.messages && $scope.model.messages.patronCategoryStatusMessages)
                            $scope.model.messages.patronCategoryStatusMessages.length = 0;
                        if (data.patronCategoryStatusMessages) {
                            $scope.model.messages = $scope.model.messages || {};
                            $scope.model.messages.patronCategoryStatusMessages = $scope.model.messages.patronCategoryStatusMessages || [];
                            data.patronCategoryStatusMessages.forEach(function (statusMessage) {
                                $scope.model.messages.patronCategoryStatusMessages.push($sce.trustAsHtml(statusMessage));
                            });
                        }

                        $scope.model.mainAddressInfo = $scope.model.addressInfoModel(__formGenerator, $scope.model.registrationDynamicFormKey);
                        var isDeliveryAddressSameAsBillingAddress = $scope.model.billingAddressId === $scope.model.deliveryAddressId;
                        if (isDeliveryAddressSameAsBillingAddress) {
                            $scope.model.deliveryAddressInfo = $scope.model.addressInfoModel(__formGenerator, $scope.model.registrationDynamicFormKey);
                        } else {
                            $scope.model.deliveryAddressInfo = $scope.model.addressInfoModel(__formGenerator, $scope.model.deliveryAddressDynamicFormKey);
                        }

                        $scope.displayMenu();
                    }
                    else {
                        if (data.hasConstraintError)
                            $scope.model.dynamicForm = new $scope.model.dynamicFormModel(data.formGenerator);
                        else {
                            $scope.model.loginErrorMessage = data.errorMessage;
                            $("html, body").animate({ scrollTop: 0 }, "slow");
                        }
                    }
                });
            }
        }

        $scope.goToAccountRegistrationStep = function () {            
            $scope.model.isStepLogin = false;
            $scope.model.isStepRegisterUser = true;
            $scope.model.isStepRegisterAsGuest = false;
            $scope.model.isRegisterStepPersonal = true;
            $scope.model.isRegisterStepBilling = false;
            $scope.model.isRegisterStepAccount = false;
            $scope.model.displayRegistrationFields = true;
            $scope.model.userDataSaved = false;
            $scope.model.dynamicWebFormSubmitted = false;
            $scope.model.originalMainAddress = angular.copy($scope.model.mainAddressInfo);
            $scope.model.originalDeliveryAddress = angular.copy($scope.model.deliveryAddressInfo);
            $scope.pageService.trackNavigation('myaccount/addressedit');
            $scope.pageService.postMessageToIFrameParent('myaccount/addressedit');
            $scope.model.currentView = '';            
        }

        $scope.registerPreviousStep = function () {
            if ($scope.model.isRegisterStepAccount) {
                $scope.model.isRegisterStepPersonal = false;
                $scope.model.isRegisterStepBilling = true;
                $scope.model.isRegisterStepAccount = false;
            } else if ($scope.model.isRegisterStepBilling) {
                $scope.model.isRegisterStepPersonal = true;
                $scope.model.isRegisterStepBilling = false;
                $scope.model.isRegisterStepAccount = false;
            }
        }

        $scope.registerNextStep = function (isValid) {
            if (isValid) {
                $scope.model.dynamicWebFormSubmitted = false;

                if ($scope.model.isRegisterStepPersonal) {
                    $scope.model.isRegisterStepPersonal = false;
                    $scope.model.isRegisterStepBilling = true;
                    $scope.model.isRegisterStepAccount = false;
                } else if ($scope.model.isRegisterStepBilling) {
                    $scope.model.isRegisterStepPersonal = false;
                    $scope.model.isRegisterStepBilling = false;
                    $scope.model.isRegisterStepAccount = true;
                }
                $scope.copyDynamicFormGroupValues($scope.model.dynamicForm[$scope.model.registrationDynamicFormKey]);
            }
        }

        $scope.copyDynamicFormGroupValues = function (dynamicForm) {
            var groups = [];
            $.each(dynamicForm, function () {
                groups.push(this);
            });

            $.each(groups, function (index, source) {
                for (let key in source) {
                    let destination = groups[index + 1];
                    if (destination && key in destination) {
                        if (!destination[key]['value']) {
                            destination[key]['value'] = source[key]['value'];
                        }
                    }
                }
            });
        }
        if ($scope.model.isRegistration) {
            $scope.model.isRegisterStepPersonal = true;
            $scope.model.isRegisterStepBilling = false;
            $scope.model.isRegisterStepAccount = false;
            $scope.goToAccountRegistrationStep();
        }
    }
})();
;
(function () {
    'use strict';


    function patronOrderSummaryModel(src) {
        this.id = (src != undefined) ? src.id : 0;
        this.currencySymbol = (src != undefined) ? src.currencySymbol : '';
        this.date = (src != undefined) ? src.date : null;
        this.deliveryMethod = (src != undefined) ? src.deliveryMethod : '';
        this.paymentMethod = (src != undefined) ? src.paymentMethod : '';
        this.statusName = (src != undefined) ? src.statusName : '';
        this.total = (src != undefined) ? src.total : 0;
    }


    var modelId = 'patronAccountModel';
    angular.module('webShop2App').factory(modelId, ['formGeneratorService', model]);

    function model(formGeneratorService) {
        var model = {
            currentView: 'MyAccountMenu.html',
            isOrderHistoryCollectionLoaded: false,
            isReservationsCollectionLoaded: false,
            orderHistoryCollection: new Array(),
            reservationsCollection: new Array(),
            isSeminarRegistrationCollectionLoaded: false,
            seminarRegistrationCollection: new Array(),
            orderDetail: null,
            reservationDetail: null,

            updateSeminarRegistrations: updateSeminarRegistrations,
            updateOrderHistory: updateOrderHistory,
            updateCurrentOrderDetail: updateCurrentOrderDetail,
            updateReservations: updateReservations,
            updateCurrentReservationDetail: updateCurrentReservationDetail,

            // dynamic form
            dynamicForm: new dynamicFormModel(),

            // address information
            mainAddressInfo: new addressInfoModel(),
            deliveryAddressInfo: new addressInfoModel(),

            // child models
            dynamicFormModel: dynamicFormModel,
            addressInfoModel: addressInfoModel,

            // social logins
            socialLogins: new Array(),
        };
        return model;

        function updateSeminarRegistrations(registrations) {
            this.seminarRegistrationCollection.length = 0;
            for (var i = 0; i < registrations.length; i++) {
                this.seminarRegistrationCollection.push(registrations[i]);
            }
        }

        function updateOrderHistory(orders) {
            this.orderHistoryCollection.length = 0;
            for (var i = 0; i < orders.length; i++) {
                this.orderHistoryCollection.push(orders[i]);
            }
        }

        function updateReservations(reservations) {
            this.reservationsCollection.length = 0;
            for (var i = 0; i < reservations.length; i++) {
                this.reservationsCollection.push(reservations[i]);
            }
        }

        function updateCurrentReservationDetail(data) {
            this.reservationDetail = data.reservation;
            this.reservationDetail.canActivate = data.canActivateReservation;
            this.reservationDetail.hasOpenReservationInSession = data.hasOpenReservationInSession;
            this.reservationDetail.hasPersonalization = false;
            for (var groupIndex = 0; groupIndex < this.reservationDetail.itemGroups.length; groupIndex++) {
                var group = this.reservationDetail.itemGroups[groupIndex];
                if (group.hasPersonalization) {
                    this.reservationDetail.hasPersonalization = true;
                    for (var itemIndex = 0; itemIndex < group.items.length; itemIndex++) {
                        var item = group.items[itemIndex];
                        item.personalization.text = $.map(item.personalization.fields, function (field) { return field.value }).join(' ');
                    }
                }
            }
        }

        function updateCurrentOrderDetail(value) {
            this.orderDetail = value;
            this.orderDetail.hasPersonalization = false;
            for (var groupIndex = 0; groupIndex < this.orderDetail.itemGroups.length; groupIndex++) {
                var group = this.orderDetail.itemGroups[groupIndex];
                if (group.hasPersonalization) {
                    this.orderDetail.hasPersonalization = true;
                    for (var itemIndex = 0; itemIndex < group.items.length; itemIndex++) {
                        var item = group.items[itemIndex];
                        item.personalization.text = $.map(item.personalization.fields, function (field) { return field.value }).join(' ');
                    }
                }
            }
        }

        function dynamicFormModel(source) {
            if (source && source.dynamicForm) {
                formGeneratorService.convertStringToDate(source.dynamicForm);
            }

            var model = source ? source.dynamicForm : {};

            // functions
            model.addressExists = addressExists;
            return model;

            function addressExists(formKey) {
                var hasAddress1 = false;
                var hasLastName = false;
                var hasCompanyName = false;
                if (this[formKey]) {
                    var address1 = formGeneratorService.findPropertyValueRecursive(this[formKey], 'address1');
                    var lastName = formGeneratorService.findPropertyValueRecursive(this[formKey], 'surname');
                    var companyName = formGeneratorService.findPropertyValueRecursive(this[formKey], 'companyName');
                    hasAddress1 = address1 != null && address1.length > 0;
                    hasLastName = lastName != null && lastName.length > 0;
                    hasCompanyName = companyName != null && companyName.length > 0;
                }

                return hasAddress1 && (hasLastName || hasCompanyName);
            }
        }

        function addressInfoModel(source, formKey) {
            var model = {
                salutationName: '',
                companyName: '',
                firstName: '',
                surname: '',
                address1: '',
                address2: '',
                city: '',
                postalCode: '',
                stateName: '',
                countryName: '',
                email: '',
                homePhone: '',
                mobilePhone: '',
                companyPhone: '',
                fax: '',
            };

            if (source) {
                model.salutationName = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'salutationName');
                model.companyName = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'companyName');
                model.firstName = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'firstName');
                model.surname = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'surname');
                model.address1 = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'address1');
                model.address2 = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'address2');
                model.city = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'city');
                model.postalCode = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'postalCode');
                model.stateName = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'stateName');
                model.countryName = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'countryName');
                model.email = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'email');
                model.homePhone = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'homePhone');
                model.mobilePhone = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'mobilePhone');
                model.companyPhone = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'companyPhone');
                model.fax = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'fax');
            }
            return model;
        }
    }
})();;
// Modified:		15.08.2016 ticketportal AG, St. Gallen fbe  : (SCR 9328) WebShop2: Activate a Reservation
(function () {
    'use strict';

    var serviceId = 'patronAccountService';
    angular.module('webShop2App').factory(serviceId, ['$http', '$sce', 'webShop2AppSettings', 'commonService', service]);

    function service($http, $sce, webShop2AppSettings, commonService) {
        return {

            cancelOrder: function (order) {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/cancelorder');
               
                return $http({
                    method: 'POST',
                    url: commonService.absoluteUrl(url),
                    data: order
                }).then(function (response) {
                    return response.data;
                });
            },

            getSeminarRegistrations: function () {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/registrations');
               
                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),                    
                }).then(function (response) {
                    return response.data;
                });
            },

            getOrderHistory: function () {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/orders');
               
                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),                    
                }).then(function (response) {
                    return response.data;
                });
            },

            getOrderDetail: function (id) {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/orders?id=' + encodeURIComponent(id));
               
                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),                    
                }).then(function (response) {
                    return response.data;
                });
            },

            getReservations: function () {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/reservations');

                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),
                }).then(function (response) {
                    return response.data;
                });
            },

            getReservationDetail: function (id) {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/reservations?id=' + encodeURIComponent(id));

                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),
                }).then(function (response) {
                    return response.data;
                });
            },

            activateReservation: function (id) {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/activatereservation?id=' + encodeURIComponent(id));

                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),
                }).then(function (response) {
                    return response.data;
                });
            },

            disconnectSocialLogin: function (id) {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/disconnectsocialaccount/' + encodeURIComponent(id));

                return $http({
                    method: 'POST',
                    url: commonService.absoluteUrl(url),
                    data: id
                }).then(function (response) {
                    return response.data;
                });
            }


        }
    }

})();;
(function () {
    'use strict';

    var controllerId = 'patronChangePasswordController';
    angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', 'patronService', 'patronChangePasswordModel', 'pageService', 'webShop2AppSettings', '$modal', '$modalInstance', controller]);

    function controller($rootScope, $scope, patronService, patronChangePasswordModel, pageService, webShop2AppSettings, $modal, $modalInstance) {
        $scope.model = patronChangePasswordModel;
        $scope.patronService = patronService;
        $scope.pageService = pageService;
        $scope.model.reset();

        $scope.changePassword = function (isValid) {
            $scope.model.formSubmitted = true;
            if (isValid) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.patronService.changePassword($scope.model.oldPassword, $scope.model.newPassword, $scope.model.newPasswordConfirm).then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;
                    if (data.succeeded) {
                        $scope.pageService.pop('success', '', __messagePasswordChanged, 4000, 'trustedHtml');
                        $modalInstance.close();
                    }
                    else
                        $scope.model.errorMessage = data.errorMessage;
                });
            }
        }
        $scope.cancel = function () {
            $modalInstance.dismiss('cancel');
        }

    }
})();;
(function () {
    'use strict';

    var modelId = 'patronChangePasswordModel';
    angular.module('webShop2App').factory(modelId, [model]);

    function model() {
        var model = {
            oldPassword: '',
            newPassword: '',
            newPasswordConfirm: '',
            formSubmitted: false,
            passwordChanged: false,
            errorMessage: '',

            onPasswordChanged: onPasswordChanged,

            reset: reset
        }
        return model;

        function onPasswordChanged() {
            this.reset();
            this.passwordChanged = true;
        }

        function reset() {
            this.oldPassword = '';
            this.newPassword = '';
            this.newPasswordConfirm = '';
            this.errorMessage = '';
            this.passwordChanged = false;
            this.formSubmitted = false;
        }
    }
})();;
(function () {
    'use strict';

    var controllerId = 'patronController';
    angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', '$window', 'patronService', 'patronModel', 'formGeneratorService', 'pageService', 'webShop2AppSettings', '$modal', '$timeout', controller]);

    function controller($rootScope, $scope, $window, patronService, patronModel, formGeneratorService, pageService, webShop2AppSettings, $modal, $timeout) {
        $scope.patronService = patronService;
        $scope.model = patronModel;
        $scope.webShop2AppSettings = webShop2AppSettings;
        var patronContentUrl = (webShop2AppSettings.folderName) ? '/' + webShop2AppSettings.folderName + '/patron/register' : '/patron/register';
        $scope.model.patronContentUrl = patronContentUrl;
        $scope.formGeneratorService = formGeneratorService;
        $scope.pageService = pageService;

        $scope.$watch('pageService.pageModel.currentStep', function () {
            if ($scope.pageService.getStep() == CURRENT_STEP_PATRON) {
                $scope.model.isRegisterStepPersonal = !$scope.pageService.pageModel.editAddress;
                $scope.model.isRegisterStepBilling = $scope.pageService.pageModel.editAddress;
                $scope.model.isRegisterStepAccount = false;
                $scope.pageService.pageModel.editAddress = $scope.model.isLoggedOn && !$scope.model.dynamicForm.addressExists($scope.model.registrationDynamicFormKey);
                if ($scope.pageService.pageModel.requiresPatronLogin || (!$scope.model.isGuestBookingEnabled && !$scope.model.isLoggedOn)) {
                    $scope.model.isStepLogin = true;
                    $scope.model.isStepRegisterUser = false;
                    $scope.model.isStepRegisterAsGuest = false;
                    $scope.patronService.getConfigurationAndAddress().then(function (data) {
                        $scope.model.update(data);
                    });
                } else {
                    if (!$scope.model.dynamicForm.addressExists($scope.model.registrationDynamicFormKey)) {
                        $scope.pageService.pageModel.isLoading = true;
                        $scope.patronService.getConfigurationAndAddress().then(function (data) {
                            $scope.model.update(data);

                            // handle patron when patron already exists
                            if (($scope.model.dynamicForm.addressExists($scope.model.registrationDynamicFormKey) || $scope.pageService.pageModel.editAddress) && ($scope.model.isLoggedOn || !$scope.model.isPatronRegistered)) {
                                $scope.model.isStepRegisterUser = !$scope.pageService.pageModel.requiresPatronLogin && $scope.model.isLoggedOn;
                                $scope.model.isStepRegisterAsGuest = !$scope.pageService.pageModel.requiresPatronLogin && !$scope.model.isLoggedOn;
                                $scope.model.isStepLogin = $scope.pageService.pageModel.requiresPatronLogin;
                                if ($scope.model.isLoggedOn && !$scope.model.isLoginByMemberProviderSystem)
                                    $scope.model.displayRegistrationFields = true;
                            }

                            $scope.pageService.pageModel.isLoading = false;
                        });
                    } else {
                        if ($scope.model.isLoggedOn || !$scope.model.isPatronRegistered) {
                            $scope.model.isStepRegisterUser = $scope.model.isStepRegisterUser || (!$scope.pageService.pageModel.requiresPatronLogin && $scope.model.isLoggedOn);
                            $scope.model.isStepRegisterAsGuest = !$scope.model.isStepRegisterUser && !$scope.pageService.pageModel.requiresPatronLogin && !$scope.model.isLoggedOn;
                            $scope.model.isStepLogin = $scope.pageService.pageModel.requiresPatronLogin;
                        } else {
                            $scope.model.isStepRegisterUser = false;
                            $scope.model.isStepRegisterAsGuest = false;
                            $scope.model.isStepLogin = true;
                        }

                        if ($scope.model.isLoggedOn && !$scope.model.isLoginByMemberProviderSystem)
                            $scope.model.displayRegistrationFields = true;
                    }
                }
            }
        });

        $scope.$watch('pageService.pageModel.requiresPatronLogin', function () {
            if ($scope.pageService.pageModel.requiresPatronLogin) {
                $scope.model.isStepRegisterUser = false;
                $scope.model.isStepRegisterAsGuest = false;
                $scope.model.isStepLogin = true;

                $timeout(function () {
                    $scope.pageService.setStep(CURRENT_STEP_PATRON);
                    $scope.pageService.showModal({ title: '', message: ticketportal.webshop.localization.loginExpired });
                }, 50);
                
            }
        });

        $scope.displayCurrentPatronAddress = function () {
            if ($scope.pageService.pageModel.requiresPatronLogin)
                $scope.model.isStepLogin = true;
            else {
                // reload the partial view
                var patronContentUrl = (webShop2AppSettings.folderName) ? '/' + webShop2AppSettings.folderName + '/patron/register' : '/patron/register';
                    patronContentUrl += "?r=" + Math.random();
                $scope.model.patronContentUrl = patronContentUrl;
                $scope.model.isStepRegisterUser = !$scope.model.isLoginByMemberProviderSystem;
                $scope.model.isStepRegisterAsGuest = $scope.model.isLoginByMemberProviderSystem;
                $scope.model.isStepLogin = false;
            }
        }

        $scope.login = function (isValid, bookingProcessLogin) {
            $scope.model.loginFormSubmitted = true;
            if (isValid) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.model.loginErrorMessage = '';
                var captchaResponse = $("#g-recaptcha-response").length ? $("#g-recaptcha-response").val() : '';
                $scope.patronService.login($scope.model.loginEmail, $scope.model.loginPassword, captchaResponse, bookingProcessLogin).then(function (data) {
                    $scope.handleLoginResponse(data);
                });
            } else {
                $scope.pageService.postMessageToIFrameParent("patronLoginInvalidForm");
            }
        }

        $rootScope.$on('patron:displayMembershipLogin', function (event, args) {
            if (!$scope.model.isDisplayingMembershipLogin) {
                $scope.model.isDisplayingMembershipLogin = true;
                $scope.model.inlineLoginErrorMessage = '';
                var modalInstance = $modal.open({
                    templateUrl: 'MembershipLogin.html',
                    controller: membershipLoginModalInstanceController
                });
            }
        });

        // Membership login
        $scope.membershipLogin = function (isValid) {
            $scope.model.inlineLoginFormSubmitted = true;
            if (isValid) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.model.loginErrorMessage = '';
                patronService.login($scope.model.inlineLoginEmail, $scope.model.inlineLoginPassword).then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;
                    if (data.loginSucceeded) {
                        $rootScope.$broadcast('event:ReloadCurrentData');
                        $scope.cancel();
                    } else {
                        $scope.model.inlineLoginErrorMessage = data.errorMessages[0];
                    }
                });
                
                $("#btn-inline-login").button('reset');
                $scope.pageService.pageModel.isLoading = false;
            }
        }

        // called by the registration page
        $scope.bootstrapRegistrationPage = function () {
            $scope.model.isStepRegisterUser = true;
            $scope.model.isStepRegisterAsGuest = false;
            $scope.model.displayRegistrationFields = true;
            $scope.model.registrationPage = {
                isRegistrationFinished: false,
                isActive: true
            };            
        }

        $scope.goToGuestRegistrationStep = function () {
            $scope.model.isStepLogin = false;
            $scope.model.isStepRegisterUser = false;
            $scope.model.isStepRegisterAsGuest = true;
            $scope.model.displayRegistrationFields = false;

            $scope.updatePageTracking();
        }

        $scope.goToAccountRegistrationStep = function () {
            $scope.model.isStepLogin = false;
            $scope.model.isStepRegisterUser = true;
            $scope.model.isStepRegisterAsGuest = false;
            $scope.model.displayRegistrationFields = true;
            $scope.model.isRegisterStepPersonal = true;
            $scope.model.isRegisterStepBilling = false;
            $scope.model.isRegisterStepAccount = false;

            $scope.updatePageTracking();
        }

        $scope.backToPreviousStep = function () {
            $scope.pageService.previousStep();
        }

        $scope.back = function () {
            if (($scope.model.isLoggedOn || $scope.model.isLoginByMemberProviderSystem) && ($scope.model.dynamicForm.addressExists($scope.model.registrationDynamicFormKey) || $scope.pageService.pageModel.editAddress)) {
                $scope.pageService.previousStep();
                return;
            }

            if ($scope.model.isStepRegisterUser || $scope.model.isStepRegisterAsGuest) {
                $scope.model.isStepLogin = true;
                if ($scope.model.isGuestBookingEnabled) {
                    // guest registraton is possible, move back to the registration type selection
                    $scope.model.isStepRegisterUser = false;
                    $scope.model.isStepRegisterAsGuest = false;
                    $scope.model.isLoginActive = false;
                    $scope.model.isRegistrationActive = true;
                    $scope.updatePageTracking();
                } else {
                    // no guest registration is possible, switch back to the login screen
                    $scope.model.isStepRegisterUser = false;
                    $scope.model.isStepRegisterAsGuest = false;
                    $scope.model.isLoginActive = true;
                    $scope.model.isRegistrationActive = false;
                    $scope.updatePageTracking();
                }
            }          
        }

        $scope.updatePageTracking = function () {
            var iFrameMessage = '';
            if ($scope.model.isStepRegisterAsGuest) {
                $scope.pageService.trackNavigation('patron/register-as-guest');
                iFrameMessage = 'patronRegisterAsGuest';
            }
            else if ($scope.model.isStepRegisterUser) {
                $scope.pageService.trackNavigation('patron/register-with-account');
                iFrameMessage = 'patronRegisterWithAccount';
            }

            // in iFrame post a message notifying the actual action
            $scope.pageService.postMessageToIFrameParent(iFrameMessage);
        }

        $scope.proceedWithoutSaving = function () {

            $scope.pageService.pageModel.isLoading = true;

            $scope.patronService.proceedWithoutSaving().then(function (data) {
                $scope.pageService.pageModel.isLoading = false;
                if (data.succeeded) {
                    $scope.pageService.setStep(CURRENT_STEP_CHECKOUT);
                } else {
                    $rootScope.$broadcast('event:ShowAlert', { title: '', message: data.errorMessage });
                }
            })
        }

        $scope.register = function (isValid) {
            $scope.model.dynamicWebFormSubmitted = true;

            var isRegisterStep = ($scope.model.isStepRegisterAsGuest && !$scope.model.isRegisterStepBilling) || ($scope.model.isStepRegisterUser && ($scope.model.hasAccountInformation ? !$scope.model.isRegisterStepAccount : !$scope.model.isRegisterStepBilling));
            if (isRegisterStep) {
                $scope.registerNextStep(isValid);
                return;
            }

            if (isValid && $scope.model.validDate) {
                $scope.pageService.pageModel.isLoading = true;

                // check if registration needs to be handled as guest patron
                var isGuest = $scope.model.isStepRegisterAsGuest || $scope.model.isLoginByMemberProviderSystem || (!$scope.model.isLoggedOn && !$scope.model.registerPassword && $scope.model.dynamicForm.addressExists($scope.model.registrationDynamicFormKey));

                let dynamicForm = {};
                if ($scope.model.registrationDynamicFormKey.length > 0) {                   
                    for (var formKey in $scope.model.dynamicForm) {
                        // Copy form in case the visitor changes his mind and wants to register as a user after having already filled the guest form
                        if (formKey !== $scope.model.registrationDynamicFormKey && formKey !== $scope.model.deliveryAddressDynamicFormKey && typeof $scope.model.dynamicForm[formKey] === "object") {
                            $scope.copyDynamicFormValues($scope.model.dynamicForm[formKey], $scope.model.dynamicForm[$scope.model.registrationDynamicFormKey]);
                        }
                    }

                    // clone the current dynamic form to give to $scope.patronService.register
                    dynamicForm[$scope.model.registrationDynamicFormKey] = {};
                    angular.extend(dynamicForm[$scope.model.registrationDynamicFormKey], $scope.model.dynamicForm[$scope.model.registrationDynamicFormKey]);         
                    dynamicForm[$scope.model.deliveryAddressDynamicFormKey] = {};
                    angular.extend(dynamicForm[$scope.model.deliveryAddressDynamicFormKey], $scope.model.dynamicForm[$scope.model.deliveryAddressDynamicFormKey]);         
                }

                // register patron
                var isRegistrationPage = $scope.model.registrationPage && $scope.model.registrationPage.isActive;
                $scope.patronService.register(isGuest, dynamicForm, false, isRegistrationPage, $scope.model.patronId, $scope.model.billingAddressId, $scope.model.deliveryAddressId, $scope.model.isDeliveryAddressSameAsBillingAddress, $scope.model.socialAccount).then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;

                    if (data.succeeded) {
                        $scope.model.patronId = data.patronId;
                        if (!isGuest) {
                            $rootScope.$broadcast('event:LoggedOn');
                        }

                        if ($scope.model.registrationPage) {
                            $scope.model.registrationPage.isRegistrationFinished = true;
                        } else {
                            $scope.pageService.setStep(CURRENT_STEP_CHECKOUT);
                        }                        
                    }
                    else {
                        $scope.model.isRegisterStepPersonal = false;
                        $scope.model.isRegisterStepBilling = false;
                        $scope.model.isRegisterStepAccount = false;

                        if (data.isBasketExpiredError) {

                            // reset the amount of items known to be in the basket
                            $scope.pageService.setBasketItemCount(0);
                            $scope.pageService.showModal({
                                message: data.errorMessage,
                                onClose: function () { window.location.reload(true); }
                            });
                        } else if (data.requiresPatronLogin) {
                            $scope.pageService.pageModel.requiresPatronLogin = data.requiresPatronLogin;
                        } else if (data.hasConstraintError) {
                            $scope.model.dynamicForm = new $scope.model.dynamicFormModel(data.formGenerator);
                            var propertyFound = false;
                            var groupName;
                            function findConstraintErrorGroup(object) {
                                for (var property in object) {
                                    if (object.hasOwnProperty(property)) {
                                        if (property.match(/_[0-9]+$/)) {
                                            groupName = property;
                                        }
                                        if (typeof object[property] == "object") {
                                            findConstraintErrorGroup(object[property]);
                                            if (propertyFound) {
                                                break;
                                            }
                                        } else {
                                            if ((object.constraintErrors?.length) > 0) {
                                                propertyFound = true;
                                                break;
                                            }
                                        }
                                    }
                                }
                            }
                            findConstraintErrorGroup($scope.model.dynamicForm);
                            if (groupName.startsWith('registrationPersonalInformation')) {
                                $scope.model.isRegisterStepPersonal = true;
                            } else if (groupName.startsWith('registrationBillingInformation')) {
                                $scope.model.isRegisterStepBilling = true;
                            } else if (groupName.startsWith('registrationAccountInformation')) {
                                $scope.model.isRegisterStepAccount = true;
                            }
                        } else if (data.isBasketRuleError) {
                            $scope.pageService.updateHasPendingBasketValidation(true);
                            $scope.pageService.showModal({
                                message: data.errorMessage,
                                onClose: function () { $scope.pageService.setStep(CURRENT_STEP_BASKET); }
                            });
                        } else {
                            $scope.model.isRegisterStepPersonal = true;
                            $rootScope.$broadcast('event:ShowAlert', { title: '', message: data.errorMessage });
                        }
                    }
                });
            } else {
                $scope.pageService.postMessageToIFrameParent('patronRegistrationInvalidForm');
            }
        }

        /**
         * Copy the values of the source form into the destination form. Only existing keys in both forms are used.
         * @param {any} destination - The destination form will have its value overwritten by the source form
         * @param {any} source - We use the values of the source form to overwrite the destination form values
         */
        $scope.copyDynamicFormValues = function (destination, source) {
            for (let key in source)
                if (key in destination)
                    destination[key]['value'] = source[key]['value'];
        };

        // opens the 'forgot password' modal dialog
        $scope.forgotPassword = function () {

            // reset: password not sent yet
            $scope.model.passwordRetrievalSucceeded = undefined; // not yet known            
            $scope.model.passwordRetrievalEmail = '';

            var modalInstance = $modal.open({
                animation: false,
                templateUrl: 'ForgotPassword.html',
                controller: forgotPasswordModalInstanceController
            });
            $scope.pageService.trackNavigation('forgot-password-dialog');
        }

        // Retrieve password
        $scope.retrievePassword = function (isValid) {            
            $scope.model.passwordRetrievalFormSubmitted = true;
            if (isValid) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.model.passwordRetrievalErrorMessage = '';
                patronService.retrievePassword($scope.model.passwordRetrievalEmail).then(function (data) {
                    $scope.model.passwordRetrievalSucceeded = data.succeeded;
                });

                $scope.pageService.pageModel.isLoading = false;
            }
        }

        // Reset password
        $scope.resetPassword = function (isValid) {
            $scope.model.resetPasswordFormSubmitted = true;
            if (isValid) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.model.resetPasswordErrorMessage = '';
                $scope.patronService.resetPassword($scope.model.resetPassword).then(function (data) {
                    $scope.model.succeeded = data.succeeded;
                    $scope.model.resetPasswordErrorMessage = data.errorMessage;
                });

                $scope.pageService.pageModel.isLoading = false;
            }
        }

        $scope.moveToNextStep = function () {
        }

        $scope.validateDOB = function (birthDay) {
            $scope.model.validDate = validateDate(birthDay, true);
        }

        $scope.loadDynamicForm = function () {
            if (typeof __formGenerator !== 'undefined' && __formGenerator !== null) {
                $scope.model.dynamicForm = $scope.model.dynamicFormModel(__formGenerator);
                $scope.model.billingAddressId = __billingAddressId;
                $scope.model.deliveryAddressId = __deliveryAddressId;
                $scope.model.isDeliveryAddressSameAsBillingAddress = __billingAddressId === __deliveryAddressId;
                $scope.model.socialAccount = $scope.model.socialAccountModel(__socialAccount);;
            }
        }

        $scope.socialAccountSignIn = function (socialAccountType, bookingProcessLogin) {
            if (socialAccountType == 'Google') {
                gapi.auth2.getAuthInstance().signIn().then(function (googleUser) {
                    $scope.socialLogin(socialAccountType, googleUser.getAuthResponse().id_token, bookingProcessLogin);
                }, function (error) {
                    console.log(error.error + (error.type ? ': ' + error.type : ''));
                });
            }
            if (socialAccountType == 'Facebook') {
                FB.login(function (response) {
                    if (response.authResponse) {
                        $scope.socialLogin(socialAccountType, response.authResponse.accessToken, bookingProcessLogin);
                    }
                    else {
                        console.log(response);
                    }
                }, {scope:'email'});
            }
            if (socialAccountType == 'Apple') {
                const result = AppleID.auth.signIn();
                result.then(function (response) {
                    // Apple only returns the user object the first time the user authorizes the app.
                    // To use the first name and last name to register the user, we have to collect them here.
                    let firstName = '';
                    let lastName = '';
                    if (typeof response.user !== 'undefined' && typeof response.user.name !== 'undefined') {
                        firstName = response.user.name.firstName;
                        lastName = response.user.name.lastName;
                    }
                    $scope.socialLogin(socialAccountType, response.authorization.id_token, bookingProcessLogin, firstName, lastName);
                }).catch(function (error) {
                    console.log(error);
                });
            }
        }

        $scope.socialLogin = function (socialAccountType, socialToken, bookingProcessLogin, firstName = '', lastName = '') {
            if (socialToken) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.model.loginErrorMessage = '';
                var captchaResponse = $("#g-recaptcha-response").length ? $("#g-recaptcha-response").val() : '';
                $scope.patronService.socialAccountLogin(socialAccountType, socialToken, captchaResponse, bookingProcessLogin, firstName, lastName).then(function (data) {
                    if (data.startRegistrationProcess) {
                        if (data.bookingProcessLogin) {
                            $scope.model.dynamicForm = new $scope.model.dynamicFormModel(data.formGenerator);
                            $scope.model.socialAccount = $scope.model.socialAccountModel(data.socialAccount);
                            $scope.model.isGuestBookingEnabled = false;
                            $scope.pageService.pageModel.requiresPatronLogin = false;
                            $scope.goToAccountRegistrationStep();
                            $scope.pageService.pageModel.isLoading = false;
                        } else {
                            $window.location = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/Registration?createNewAccount=True&socialAccountType=' + socialAccountType + '&socialToken=' + encodeURIComponent(socialToken) + '&firstName=' + firstName + '&lastName=' + lastName);
                        }
                    }
                    else {
                        $scope.handleLoginResponse(data);
                    }
                });
            } else {
                $scope.pageService.postMessageToIFrameParent("patronLoginInvalidForm");
            }
        }

        $scope.handleLoginResponse = function (data) {
            if (data.loginSucceeded) {
                $scope.model.patronId = data.patronId;
                $scope.model.billingAddressId = data.billingAddressId;
                $scope.model.deliveryAddressId = data.deliveryAddressId;

                // redirect the user if configured to do so
                if ($scope.model.redirectAfterLogon) {
                    if ($scope.model.redirectAfterLogonURL)
                        $window.location = unescape($scope.model.redirectAfterLogonURL);
                    else if (($scope.webShop2AppSettings.folderName))
                        $window.location = '/' + $scope.webShop2AppSettings.folderName;
                    else
                        $window.location = '/';
                    return;
                }

                $scope.pageService.pageModel.isLoading = false;
                $scope.pageService.pageModel.requiresPatronLogin = false;

                // notify basket that we have a patron
                $rootScope.$broadcast('event:BasketChanged', { hasPatron: true, loggedUserLogin: data.email, isLoggedOn: true });

                // update the model
                if (data.formGenerator) {
                    $scope.model.dynamicForm = new $scope.model.dynamicFormModel(data.formGenerator);
                    $scope.model.isStepRegisterUser = !$scope.model.isLoginByMemberProviderSystem;
                    $scope.model.isStepRegisterAsGuest = $scope.model.isLoginByMemberProviderSystem;
                    $scope.model.isStepLogin = false;
                    $scope.model.displayRegistrationFields = !$scope.model.isLoginByMemberProviderSystem;
                    $scope.model.isLoggedOn = true;

                    $scope.displayCurrentPatronAddress();

                    // in iFrame post a message saying that the patron has logged in
                    $scope.pageService.postMessageToIFrameParent("patronLoggedIn");

                    if (data.basketRuleWarning) {
                        $scope.pageService.showModal({
                            message: data.basketRuleWarning,
                            onClose: function () {
                                $scope.pageService.setStep(CURRENT_STEP_BASKET);
                            }
                        });
                    }

                    // Next step is checkout after successful login
                    $scope.pageService.setStep(CURRENT_STEP_CHECKOUT);
                }
            } else {
                $scope.pageService.pageModel.isLoading = false;
                $scope.pageService.pageModel.requiresPatronLogin = false;
                $scope.model.isCaptchaActive = data.isCaptchaActive;
                try { grecaptcha.reset(); }
                catch (e) { }

                if (data.errorMessages && data.errorMessages.length > 0)
                    $scope.model.loginErrorMessage = data.errorMessages[0];
            }
        }

        $scope.registerPreviousStep = function () {
            if ($scope.model.isRegisterStepAccount) {
                $scope.model.isRegisterStepPersonal = false;
                $scope.model.isRegisterStepBilling = true;
                $scope.model.isRegisterStepAccount = false;
            } else if ($scope.model.isRegisterStepBilling) {
                $scope.model.isRegisterStepPersonal = true;
                $scope.model.isRegisterStepBilling = false;
                $scope.model.isRegisterStepAccount = false;
            }
        }

        $scope.registerNextStep = function (isValid) {
            if (isValid) {
                $scope.model.dynamicWebFormSubmitted = false;

                if ($scope.model.isRegisterStepPersonal) {
                    $scope.model.isRegisterStepPersonal = false;
                    $scope.model.isRegisterStepBilling = true;
                    $scope.model.isRegisterStepAccount = false;
                } else if ($scope.model.isRegisterStepBilling) {
                    $scope.model.isRegisterStepPersonal = false;
                    $scope.model.isRegisterStepBilling = false;
                    $scope.model.isRegisterStepAccount = true;
                }
                $scope.copyDynamicFormGroupValues($scope.model.dynamicForm[$scope.model.registrationDynamicFormKey]);
            }
        }

        $scope.copyDynamicFormGroupValues = function (dynamicForm) {
            var groups = [];
            $.each(dynamicForm, function () {
                groups.push(this);
            });

            $.each(groups, function (index, source) {
                for (let key in source) {
                    let destination = groups[index + 1];
                    if (destination && key in destination) {
                        if (!destination[key]['value']) {
                            destination[key]['value'] = source[key]['value'];
                        }
                    }
                }
            });
        }

        $scope.toggleLoginTab = function (loginTabActive) {
            $('[data-toggle="popover"]').popover({ trigger: 'hover' });
            $scope.model.isLoginActive = loginTabActive;
            $scope.model.isRegistrationActive = !loginTabActive;
        }

        $scope.toggleShowPassword = function () {
            $scope.model.showPassword = !$scope.model.showPassword;
        }

        $scope.startAccountRegistration = function (isValid, bookingProcessLogin) {
            $scope.model.registerFormSubmitted = true;
            $scope.model.emailValidationErrorMessage = $scope.pageService.validateEmail($scope.model.registerEmail);
            isValid = isValid && $scope.model.emailValidationErrorMessage.length == 0;
            let passwordValidationErrorMessages = $scope.pageService.validatePassword($scope.model.registerPassword);
            isValid = isValid && passwordValidationErrorMessages.length == 0;
            $scope.model.passwordValidationErrorMessages.length = 0;
            if (!isValid) {
                $scope.model.passwordValidationErrorMessages.push(...passwordValidationErrorMessages);
                $scope.pageService.postMessageToIFrameParent("patronLoginInvalidForm");
                return;
            }

            $scope.model.loginErrorMessage = '';
            let registrationValues = { firstName: $scope.model.registerFirstName, surname: $scope.model.registerLastName, email: $scope.model.registerEmail, password: $scope.model.registerPassword, passwordConfirmation: $scope.model.registerPassword };
            $scope.patronService.registration(registrationValues).then(function (data) {
                $scope.pageService.pageModel.isLoading = false;

                if (!data.succeeded) {
                    $rootScope.$broadcast('event:ShowAlert', { title: '', message: data.errorMessage });
                    return;
                }

                if (!bookingProcessLogin) {
                    $window.location = ticketportal.webshop.fixUrl($scope.webShop2AppSettings.folderName, '/myaccount?isRegistration=true');
                    return;
                }

                $scope.model.patronId = data.patron.id;
                $scope.model.billingAddressId = $scope.model.deliveryAddressId = data.patron.mainAddress.id;
                $scope.model.isDeliveryAddressSameAsBillingAddress = true;
                $scope.copyStartRegistrationValuesToDynamicForm(registrationValues);
                $scope.goToAccountRegistrationStep();
            });
        }

        $scope.registerPasswordOnChangeHandler = () => {
            if ($scope.model.registerFormSubmitted) {
                let passwordValidationErrorMessages = $scope.pageService.validatePassword($scope.model.registerPassword);
                $scope.model.passwordValidationErrorMessages.length = 0;
                $scope.model.passwordValidationErrorMessages.push(...passwordValidationErrorMessages);
            }
        }

        $(function () {
            $('[data-toggle="popover"]').popover({ trigger: 'hover' });
        });

        $scope.startGuestRegistration = function (isValid) {
            $scope.model.guestFormSubmitted = true;
            if (isValid) {
                $scope.model.loginErrorMessage = '';
                let registrationValues = { firstName: '', surname: '', email: $scope.model.guestEmail, password: '' };
                $scope.copyStartRegistrationValuesToDynamicForm(registrationValues);
                $scope.goToGuestRegistrationStep();
            } else {
                $scope.pageService.postMessageToIFrameParent("patronLoginInvalidForm");
            }
        }

        $scope.copyStartRegistrationValuesToDynamicForm = function (registrationValues) {
            $.each($scope.model.dynamicForm, function (index, form) {
                $.each(form, function (index, group) {
                    for (let key in group) {
                        if (key in registrationValues) {
                            group[key]['value'] = registrationValues[key];
                        }
                    }
                });
            });
        }

        $scope.registerEmailOnChangeHandler = () => {
            if ($scope.model.registerFormSubmitted) {
                $scope.model.emailValidationErrorMessage = $scope.pageService.validateEmail($scope.model.registerEmail);
            }
        }
    }
})();

membershipLoginModalInstanceController = function ($scope, $modalInstance, patronModel) {
    $scope.cancel = function () {
        patronModel.isDisplayingMembershipLogin = false;
        $modalInstance.dismiss('cancel');
    }
}
membershipLoginModalInstanceController.$inject = ['$scope', '$modalInstance', 'patronModel'];

forgotPasswordModalInstanceController = function ($scope, $modalInstance, patronModel) {
    $scope.cancel = function () {
        patronModel.passwordRetrievalFormSubmitted = false;
        $modalInstance.dismiss('cancel');
    }

    $scope.reset = function () {
        patronModel.passwordRetrievalFormSubmitted = false;
        delete patronModel.succeeded;
    }
}
forgotPasswordModalInstanceController.$inject = ['$scope', '$modalInstance', 'patronModel'];
;
(function () {
    'use strict';

    var modelId = 'patronModel';
    angular.module('webShop2App').factory(modelId, ['formGeneratorService', model]);

    function model(formGeneratorService) {
        var model = {
            isStepLogin: true,
            isStepRegisterUser: false,
            isStepRegisterAsGuest: false,
            loginErrorMessage: "",
            isPatronLoginEnabled: true,
            isGuestBookingEnabled: true,
            enforceSecurePage: false,
            loginEmail: "",
            loginPassword: "",
            patronContentUrl: "",
            displayRegistrationFields: false,
            isLoggedOn: false,
            loginFormSubmitted: false,
            dynamicWebFormSubmitted: false,
            inlineLoginFormSubmitted: false,
            inlineLoginEmail: '',
            inlineLoginPassword: '',
            inlineLoginErrorMessage: '',
            isDisplayingMembershipLogin: false,
            validDate: true,
            passwordRetrievalFormSubmitted: false,
            passwordRetrievalEmail: '',
            userDataSaved: false,
            isCaptchaActive: false,
            resetPasswordFormSubmitted: false,
            resetPassword: '',
            resetPasswordErrorMessage: '',
            needsConfirmation: false,
            patronId: 0,
            registrationDynamicFormKey: '',
            deliveryAddressDynamicFormKey: '',
            registrationPage: null, // flags when the page /patron/registration is being displayed    
            isPatronRegistered: false,
            isLoginActive: true,
            isRegistrationActive: false,
            registerFirstName: '',
            registerLastName: '',
            registerEmail: '',
            registerPassword: '',
            passwordValidationErrorMessages: [],

            // dynamic form
            dynamicForm: new dynamicFormModel(),

            // address information
            mainAddressInfo: new addressInfoModel(),
            deliveryAddressInfo: new addressInfoModel(),

            // social account
            socialAccount: new socialAccountModel(),

            // child models
            dynamicFormModel: dynamicFormModel,
            addressInfoModel: addressInfoModel,
            socialAccountModel: socialAccountModel,

            // functions
            update: update
        };
        return model;

        // update the model based on the 
        function update(source) {
            this.enforceSecurePage = source.configuration.enforceSecurePage;
            this.isGuestBookingEnabled = source.configuration.isGuestBookingEnabled;
            this.isPatronLoginEnabled = source.configuration.isPatronLoginEnabled;
            this.dynamicForm = new dynamicFormModel(source.formGenerator);
            this.registrationDynamicFormKey = source.formGenerator ? Object.keys(source.formGenerator.dynamicForm)[0] : '';
            this.mainAddressInfo = new addressInfoModel(source.formGenerator, this.registrationDynamicFormKey);
            this.deliveryAddressInfo = new addressInfoModel(source.formGenerator, this.deliveryAddressDynamicFormKey);
            this.isPatronRegistered = source.configuration.isPatronRegistered;
            this.billingAddressId = source.configuration.billingAddressId;
            this.deliveryAddressId = source.configuration.deliveryAddressId;
            this.isDeliveryAddressSameAsBillingAddress = this.billingAddressId === this.deliveryAddressId;
            this.isLoginActive = !source.configuration.isRegistrationActive;
            this.isRegistrationActive = source.configuration.isRegistrationActive;
        }

        function dynamicFormModel(source) {
            if (source && source.dynamicForm) {
                formGeneratorService.convertStringToDate(source.dynamicForm);
            }

            var model = source ? source.dynamicForm : {};

            // functions
            model.addressExists = addressExists;
            return model;

            function addressExists(formKey) {
                var hasAddress1 = false;
                var hasLastName = false;
                var hasCompanyName = false;
                if (this[formKey]) {
                    var address1 = formGeneratorService.findPropertyValueRecursive(this[formKey], 'address1');
                    var lastName = formGeneratorService.findPropertyValueRecursive(this[formKey], 'surname');
                    var companyName = formGeneratorService.findPropertyValueRecursive(this[formKey], 'companyName');
                    hasAddress1 = address1 != null && address1.length > 0;
                    hasLastName = lastName != null && lastName.length > 0;
                    hasCompanyName = companyName != null && companyName.length > 0;
                }

                return hasAddress1 && (hasLastName || hasCompanyName);
            }
        }

        function addressInfoModel(source, formKey) {
            var model = {
                salutationName: '',
                companyName: '',
                firstName: '',
                surname: '',
                address1: '',
                address2: '',
                // type: '',
                city: '',
                postalCode: '',
                stateName: '',
                countryName: '',
                email: '',
                homePhone: '',
                mobilePhone: '',
                companyPhone: '',
                fax: '',
            };

            if (source) {
                model.salutationName = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'salutationName');
                model.companyName = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'companyName');
                model.firstName = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'firstName');
                model.surname = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'surname');
                model.address1 = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'address1');
                model.address2 = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'address2');
                model.city = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'city');
                model.postalCode = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'postalCode');
                model.stateName = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'stateName');
                model.countryName = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'countryName');
                model.email = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'email');
                model.homePhone = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'homePhone');
                model.mobilePhone = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'mobilePhone');
                model.companyPhone = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'companyPhone');
                model.fax = formGeneratorService.findPropertyValueRecursive(source.dynamicForm[formKey], 'fax');
            }
            return model;
        }

        function socialAccountModel(source) {
            var model = {
                hasSocialAccount: source && source.uniqueId ? true : false,
                socialAccountUniqueId: source ? source.uniqueId : '',
                socialAccountName: source ? source.name : '',
                socialAccountEmail: source ? source.email : '',
                firstName: '',
                lastName: '',
            }

            return model;
        }
    }

})();;
// Modified:		14.01.2016 ticketportal AG, St. Gallen fbe  : (SCR 8739) Patron category specific schedules don't work on performance/event level
// Modified:		25.11.2015 ticketporal AG, St Gallen fbe	: (SCR 8604) Messen und Märkte: Loginseite ergänzen mit Neuregistrierung
(function () {
    'use strict';

    var serviceId = 'patronService';
    angular.module('webShop2App').factory(serviceId, ['$http', '$sce', '$location', 'webShop2AppSettings', 'commonService', 'formGeneratorService', service]);

    function service($http, $sce, $location, webShop2AppSettings, commonService, formGeneratorService) {
        return {
            login: function (email, password, captchaResponse, preSalesRegistration, bookingProcessLogin) {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/login');
                var data = { Email: email, Password: password, CaptchaResponse: captchaResponse ? captchaResponse : '', PreSalesRegistration: preSalesRegistration ? true : false, BookingProcessLogin: bookingProcessLogin ? true : false };

                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    if (response.data && response.data.errorMessages) {
                        if (response.data.errorMessages.length > 0) {
                            for (var errorMessageIndex = 0; errorMessageIndex < response.data.errorMessages.length; errorMessageIndex++) {
                                response.data.errorMessages[errorMessageIndex] = $sce.trustAsHtml(response.data.errorMessages[errorMessageIndex]);
                            }
                        }
                    }
                    return response.data;
                });
            },

            logout: function () {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/logout');
                return $http.post(url).then(function (response) {
                    return response.data;
                });
            },

            changePassword: function (oldPassword, newPassword, newPasswordConfirm) {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/changePassword');
                var data = { oldPassword: oldPassword, newPassword: newPassword, newPasswordConfirm: newPasswordConfirm };

                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            register: function (isGuest, dynamicForm, preSalesRegistration, isRegistrationPage, patronId, billingAddressId, deliveryAddressId, isDeliveryAddressSameAsBillingAddress, socialAccount) {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/register');
                var data = { json: '1', singlePage: 1, isPatronRegistration: isGuest ? 0 : 1, isPreSalesRegistration: preSalesRegistration ? 1 : 0, isRegistrationPage: isRegistrationPage ? 1 : 0, patronId: patronId, billingAddressId: billingAddressId, deliveryAddressId: deliveryAddressId, isDeliveryAddressSameAsBillingAddress: isDeliveryAddressSameAsBillingAddress ? 1 : 0 };
                commonService.extendWithoutFunctions(data, dynamicForm);
                commonService.extendWithoutFunctions(data, socialAccount);
                formGeneratorService.convertDateToString(data);

                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            proceedWithoutSaving: function () {

                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/register');
                var data = { json: '1', singlePage: 1, proceedWithoutSaving: 1 };

                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            getConfigurationAndAddress: function () {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/configuration');
                return $http.post(url).then(function (response) {
                    return response.data;
                });
            },

            removeCreditCardAlias: function (id) {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/removeCreditCardAlias');
                var data = { creditCardID: id };

                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            retrievePassword: function (email) {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/retrievePassword?emailAddress=' + email);

                return $http({
                    method: 'POST',
                    url: url,
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            resetPassword: function (password) {
                var data = { newPassword: password };

                return $http({
                    method: 'POST',
                    url: $location.absUrl(),
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            socialAccountLogin: function (socialAccountType, socialToken, captchaResponse, bookingProcessLogin, firstName = '', lastName = '') {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/login');
                var data = { SocialAccountType: socialAccountType, SocialToken: socialToken, CaptchaResponse: captchaResponse ? captchaResponse : '', BookingProcessLogin: bookingProcessLogin ? true : false, FirstName: firstName, LastName: lastName };

                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),
                    data: $.param(data),
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }
                }).then(function (response) {
                    if (response.data && response.data.errorMessages) {
                        if (response.data.errorMessages.length > 0) {
                            for (var errorMessageIndex = 0; errorMessageIndex < response.data.errorMessages.length; errorMessageIndex++) {
                                response.data.errorMessages[errorMessageIndex] = $sce.trustAsHtml(response.data.errorMessages[errorMessageIndex]);
                            }
                        }
                    }
                    return response.data;
                });
            },

            registration: function (registrationValues) {
                var url = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/patron/registration');

                return $http({
                    method: 'POST',
                    withCredentials: true,
                    url: commonService.absoluteUrl(url),
                    data: $.param(registrationValues),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

        }
    }

})();;
(function () {
    'use strict';

    var controllerId = 'preSalesRegistrationController';
    var controller = angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', 'patronService', 'patronModel', 'formGeneratorService', 'pageService', 'webShop2AppSettings', '$modal', '$sce', controller]);

    function controller($rootScope, $scope, patronService, patronModel, formGeneratorService, pageService, webShop2AppSettings, $modal, $sce) {
        $scope.patronService = patronService;
        $scope.model = patronModel;
        $scope.webShop2AppSettings = webShop2AppSettings;
        var patronContentUrl = (webShop2AppSettings.folderName) ? '/' + webShop2AppSettings.folderName + '/patron/register' : '/patron/register';
        $scope.model.patronContentUrl = patronContentUrl;
        $scope.formGeneratorService = formGeneratorService;
        $scope.pageService = pageService;
        $scope.model.displayRegistrationFields = false;
        $scope.model.isStepLogin = true;
        $scope.model.errorMessage = '';
        if (typeof __errorMessageFromTempData != 'undefined')
            $scope.model.errorMessage = __errorMessageFromTempData;
        $scope.model.creditCardAliases = [];
        if (typeof __creditCardAliases != 'undefined' && __creditCardAliases != null)
            $scope.model.creditCardAliases = __creditCardAliases;

        $scope.model.messages = {};

        $scope.model.messages.patronCategoryStatusMessages = [];
        if (typeof __patronCategoryStatusMessages != 'undefined' && __patronCategoryStatusMessages != null)
            __patronCategoryStatusMessages.forEach(function (message) {
                $scope.model.messages.patronCategoryStatusMessages.push($sce.trustAsHtml(message));
            });

        $scope.model.messages.preSalesRegistrationSuccess = __messagePreSalesRegistrationSuccess;
        $scope.model.messages.preSalesRegistrationUserInfoSaved = __messagePreSalesRegistrationUserInfoSaved;
        $scope.model.messages.confirmCreditCardRemoval = __messageConfirmCreditCardRemoval;
        $scope.model.messages.patronCategoryAcceptanceMessage = null;
        $scope.model.messages.patronCategoryConfirmationMessage = null;
        $scope.model.patronCategoryMissing = false;

        if (typeof __popMessage != 'undefined' && __popMessage.length > 0)
            $scope.pageService.pop('success', '', __popMessage, 4000, 'trustedHtml');

        $scope.login = function (isValid) {
            $scope.model.loginFormSubmitted = true;
            if (isValid) {
                $scope.model.userDataSaved = false;
                $scope.pageService.pageModel.isLoading = true;
                $scope.model.loginErrorMessage = '';
                var captchaResponse = $("#g-recaptcha-response").length ? $("#g-recaptcha-response").val() : '';
                $scope.patronService.login($scope.model.loginEmail, $scope.model.loginPassword, captchaResponse, true).then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;
                    if (data.loginSucceeded) {
                        $scope.onLoggedOn(data);
                    } else {
                        $scope.model.isCaptchaActive = data.isCaptchaActive;
                        try { grecaptcha.reset(); }
                        catch (e) { }

                        if (data.errorMessages && data.errorMessages.length > 0)
                            $scope.model.loginErrorMessage = data.errorMessages[0];
                    }
                });
            }
        }

        $scope.onLoggedOn = function (data) {
            $scope.model.patronId = data.patronId;
            $scope.model.patronCategoryMissing = data.patronCategoryMissing;
            if (data.patronCategoryAcceptanceMessage)
                $scope.model.messages.patronCategoryAcceptanceMessage = $sce.trustAsHtml(data.patronCategoryAcceptanceMessage);
            if (data.patronCategoryConfirmationMessage)
                $scope.model.messages.patronCategoryConfirmationMessage = data.patronCategoryConfirmationMessage;
            // update the model
            if (data.formGenerator) {
                $scope.model.dynamicForm = new $scope.model.dynamicFormModel(data.formGenerator);
                var formKey = data.formGenerator ? Object.keys(data.formGenerator.dynamicForm)[0] : '';
                $scope.model.mainAddressInfo = new $scope.model.addressInfoModel(data.formGenerator, formKey);
                $scope.model.isStepRegisterUser = false;
                $scope.model.isStepRegisterAsGuest = false;
                $scope.model.isStepLogin = false;
                $scope.model.displayRegistrationFields = false;
                $scope.model.isLoggedOn = true;
            }
            $scope.model.creditCardAliases.length = 0;
            if (data.creditCardAliases) {
                data.creditCardAliases.forEach(function (creditCard) {
                    $scope.model.creditCardAliases.push(creditCard);
                });
            }
            $scope.model.messages.patronCategoryStatusMessages.length = 0;
            if (data.patronCategoryStatusMessages) {
                data.patronCategoryStatusMessages.forEach(function (statusMessage) {
                    $scope.model.messages.patronCategoryStatusMessages.push($sce.trustAsHtml(statusMessage));
                });
            }
            if ($scope.model.patronCategoryMissing)
                $scope.goToAccountRegistrationStep();
        }

        $scope.logout = function () {
            $scope.patronService.logout().then(function (data) {
                $scope.model.isStepRegisterUser = false;
                $scope.model.mainAddressInfo = new $scope.model.addressInfoModel();
                $scope.model.isLoggedOn = false;
                $scope.model.isStepLogin = true;
                $scope.model.userDataSaved = false;
                $scope.model.messages.patronCategoryAcceptanceMessage = null;
                $scope.model.messages.patronCategoryConfirmationMessage = null;
                $scope.model.patronCategoryMissing = false;
            });
        }

        $scope.goToAccountRegistrationStep = function () {
            $scope.model.isStepLogin = false;
            $scope.model.isStepRegisterUser = true;
            $scope.model.isStepRegisterAsGuest = false;
            $scope.model.displayRegistrationFields = true;
            $scope.model.userDataSaved = false;
            $scope.model.dynamicWebFormSubmitted = false;
            $scope.model.originalMainAddress = [];
            if ($scope.model.isLoggedOn) {
                $scope.model.originalMainAddress = angular.copy($scope.model.mainAddressInfo);
            }
        }

        $scope.back = function () {
            if ($scope.model.isStepRegisterUser) {
                if (!$scope.model.isLoggedOn) {
                    $scope.model.isStepLogin = true;
                }
                else
                    angular.copy($scope.model.originalMainAddress, $scope.model.mainAddressInfo);
                $scope.model.isStepRegisterUser = false;
            }
        }

        $scope.register = function (isValid) {
            $scope.model.dynamicWebFormSubmitted = true;

            if (isValid && $scope.model.validDate) {
                $scope.pageService.pageModel.isLoading = true;

                $scope.model.loginErrorMessage = '';
                // register patron
                $scope.patronService.register(false, null, true, false, $scope.model.patronId).then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;

                    if (data.loginSucceeded) {
                        $scope.onLoggedOn(data);
                    } else if (data.succeeded) {
                        $scope.model.userDataSaved = true;
                        $scope.model.needsConfirmation = data.needsConfirmation;
                        $scope.model.isStepRegisterUser = false;
                        $scope.model.patronId = data.patronId;
                        $scope.model.patronCategoryMissing = false;
                        if (!$scope.model.isLoggedOn) {
                            $scope.model.isStepLogin = true;
                        }
                        $scope.model.messages.patronCategoryStatusMessages.length = 0;
                        if (data.patronCategoryStatusMessages) {
                            data.patronCategoryStatusMessages.forEach(function (statusMessage) {
                                $scope.model.messages.patronCategoryStatusMessages.push($sce.trustAsHtml(statusMessage));
                            });
                        }
                    }
                    else {
                        if (data.hasConstraintError)
                            $scope.model.dynamicForm = new $scope.model.dynamicFormModel(data.formGenerator);
                        else {
                            $scope.model.loginErrorMessage = data.errorMessage;
                            $("html, body").animate({ scrollTop: 0 }, "slow");
                        }
                    }
                });
            }
        }

        $scope.forgotPassword = function () {

            // reset: password not sent yet
            $scope.model.passwordRetrievalSucceeded = undefined; // not yet known
            $scope.model.passwordRetrievalEmail = '';

            var modalInstance = $modal.open({
                animation: false,
                templateUrl: 'ForgotPassword.html',
                controller: forgotPasswordModalInstanceController
            });
        }

        // Retrieve password
        $scope.retrievePassword = function (isValid) {
            $scope.model.passwordRetrievalFormSubmitted = true;
            if (isValid) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.model.passwordRetrievalErrorMessage = '';
                patronService.retrievePassword($scope.model.passwordRetrievalEmail).then(function (data) {
                    $scope.model.passwordRetrievalSucceeded = data.succeeded;
                });

                $scope.pageService.pageModel.isLoading = false;
            }
        }

        $scope.goToChangePassword = function () {
            $scope.pageService.showChangePasswordForm();
        }

        $scope.goToAliasRegistration = function () {
            $scope.pageService.showAliasRegistrationForm();
        }

        $scope.removeCreditCardAlias = function (id) {
            if (confirm($scope.model.messages.confirmCreditCardRemoval)) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.patronService.removeCreditCardAlias(id).then(function (data) {
                    $scope.pageService.pageModel.isLoading = false;
                    if (data.succeeded) {
                        for (var i = $scope.model.creditCardAliases.length - 1; i >= 0; i--)
                            if ($scope.model.creditCardAliases[i].id == id) {
                                $scope.model.creditCardAliases.splice(i, 1);
                                break;
                            }
                    } else {
                        if (data.errorMessage.length > 0)
                            $scope.model.errorMessage = data.errorMessage;
                    }
                });
            }
        }

        $scope.validateDOB = function (birthDay) {
            $scope.model.validDate = validateDate(birthDay, true);
        }

        $scope.$watch("model.userDataSaved", function (newValue, oldValue) {
            if (newValue) {
                var message;

                if ($scope.model.isLoggedOn) {
                    if ($scope.model.messages.patronCategoryConfirmationMessage)
                        message = $scope.model.messages.patronCategoryConfirmationMessage;
                    else
                        message = $scope.model.messages.preSalesRegistrationUserInfoSaved;
                }
                else
                    message = $scope.model.messages.preSalesRegistrationSuccess;

                $scope.pageService.pop('success', '', message, 4000, 'trustedHtml');
            }
        });
    }

})();
;
/**
 * Copyright 2014 Google Inc. All rights reserved.
 *
 * Use of this source code is governed by a BSD-style
 * license that can be found in the LICENSE file.
 *
 * @fileoverview Description of this file.
 *
 * A polyfill for HTML Canvas features, including
 * Path2D support.
 */

(function (CanvasRenderingContext2D, nodeRequire) {

if (CanvasRenderingContext2D == undefined) {
  CanvasRenderingContext2D = nodeRequire('canvas').Context2d;
}

if (CanvasRenderingContext2D.prototype.ellipse == undefined) {
  CanvasRenderingContext2D.prototype.ellipse = function(x, y, radiusX, radiusY, rotation, startAngle, endAngle, antiClockwise) {
    this.save();
    this.translate(x, y);
    this.rotate(rotation);
    this.scale(radiusX, radiusY);
    this.arc(0, 0, 1, startAngle, endAngle, antiClockwise);
    this.restore();
  }
}

if (typeof Path2D !== 'function' || 
    typeof new Path2D().addPath !== 'function') {
  (function() {

    // Include the SVG path parser.
    parser = (function() {
      /*
       * Generated by PEG.js 0.8.0.
       *
       * http://pegjs.majda.cz/
       */
    
      function peg$subclass(child, parent) {
        function ctor() { this.constructor = child; }
        ctor.prototype = parent.prototype;
        child.prototype = new ctor();
      }
    
      function SyntaxError(message, expected, found, offset, line, column) {
        this.message  = message;
        this.expected = expected;
        this.found    = found;
        this.offset   = offset;
        this.line     = line;
        this.column   = column;
    
        this.name     = "SyntaxError";
      }
    
      peg$subclass(SyntaxError, Error);
    
      function parse(input) {
        var options = arguments.length > 1 ? arguments[1] : {},
    
            peg$FAILED = {},
    
            peg$startRuleFunctions = { svg_path: peg$parsesvg_path },
            peg$startRuleFunction  = peg$parsesvg_path,
    
            peg$c0 = peg$FAILED,
            peg$c1 = [],
            peg$c2 = null,
            peg$c3 = function(d) { return ops; },
            peg$c4 = /^[Mm]/,
            peg$c5 = { type: "class", value: "[Mm]", description: "[Mm]" },
            peg$c6 = function(ch, args) {
                  var moveCh = ch
                  // If this is the first move cmd then force it to be absolute.
                  if (firstSubPath) {
                    moveCh = 'M';
                    firstSubPath = false;
                  }
                  ops.push({type: 'moveTo', args: makeAbsolute(moveCh, args[0])});
                  for (var i=1; i < args.length; i++) {
                    // The lineTo args are either abs or relative, depending on the
                    // original moveto command.
                    ops.push({type: 'lineTo', args: makeAbsolute(ch, args[i])});
                  }
                },
            peg$c7 = function(one, rest) { return concatSequence(one, rest); },
            peg$c8 = /^[Zz]/,
            peg$c9 = { type: "class", value: "[Zz]", description: "[Zz]" },
            peg$c10 = function() { ops.push({type: 'closePath', args: []}); },
            peg$c11 = /^[Ll]/,
            peg$c12 = { type: "class", value: "[Ll]", description: "[Ll]" },
            peg$c13 = function(ch, args) {
                  for (var i=0; i < args.length; i++) {
                    ops.push({type: 'lineTo', args: makeAbsolute(ch, args[i])});
                  }
                },
            peg$c14 = /^[Hh]/,
            peg$c15 = { type: "class", value: "[Hh]", description: "[Hh]" },
            peg$c16 = function(ch, args) {
                for (var i=0; i < args.length; i++) {
                  ops.push({type: 'lineTo', args: makeAbsoluteFromX(ch, args[i])});
                }
              },
            peg$c17 = /^[Vv]/,
            peg$c18 = { type: "class", value: "[Vv]", description: "[Vv]" },
            peg$c19 = function(ch, args) {
                for (var i=0; i < args.length; i++) {
                  ops.push({type: 'lineTo', args: makeAbsoluteFromY(ch, args[i])});
                }
              },
            peg$c20 = /^[Cc]/,
            peg$c21 = { type: "class", value: "[Cc]", description: "[Cc]" },
            peg$c22 = function(ch, args) {
                for (var i=0; i < args.length; i++) {
                  ops.push({type: 'bezierCurveTo', args: makeAbsoluteMultiple(ch, args[i])});
                }
              },
            peg$c23 = function(cp1, cp2, last) { return cp1.concat(cp2, last); },
            peg$c24 = /^[Ss]/,
            peg$c25 = { type: "class", value: "[Ss]", description: "[Ss]" },
            peg$c26 = function(ch, args) {
                for (var i=0; i < args.length; i++) {
                  ops.push({type: 'bezierCurveTo', args: makeReflected().concat(makeAbsoluteMultiple(ch, args[i]))});
                }
              },
            peg$c27 = function(cp1, last) { return cp1.concat(last); },
            peg$c28 = /^[Qq]/,
            peg$c29 = { type: "class", value: "[Qq]", description: "[Qq]" },
            peg$c30 = function(ch, args) {
                for (var i=0; i < args.length; i++) {
                  ops.push({type: 'quadraticCurveTo', args: makeAbsoluteMultiple(ch, args[i])});
                }
              },
            peg$c31 = /^[Tt]/,
            peg$c32 = { type: "class", value: "[Tt]", description: "[Tt]" },
            peg$c33 = function(ch, args) {
                for (var i=0; i < args.length; i++) {
                  var reflected = makeReflected();
                  ops.push({type: 'quadraticCurveTo', args: reflected.concat(makeAbsoluteMultiple(ch, args[i]))});
                  lastControl = reflected.slice(0);
                }
              },
            peg$c34 = /^[Aa]/,
            peg$c35 = { type: "class", value: "[Aa]", description: "[Aa]" },
            peg$c36 = function(ch, args) {
                for (var i=0; i < args.length; i++) {
                  var x1 = [lastCoord.slice()];
                  var x2 = [makeAbsolute(ch, args[i].slice(-2))];
                  absArgs = x1.concat(args[i].slice(0, -2), x2);
                  ellipseFromEllipticalArc.apply(this, absArgs);
                }
              },
            peg$c37 = function(rx, ry, xrot, large, sweep, last) { return [parseFloat(rx), parseFloat(ry), parseFloat(flatten(xrot).join('')), parseInt(large), parseInt(sweep), last[0], last[1]]; },
            peg$c38 = function(x, y) { return [x, y] },
            peg$c39 = function(number) { return parseFloat(flatten(number).join('')) },
            peg$c40 = "0",
            peg$c41 = { type: "literal", value: "0", description: "\"0\"" },
            peg$c42 = "1",
            peg$c43 = { type: "literal", value: "1", description: "\"1\"" },
            peg$c44 = ",",
            peg$c45 = { type: "literal", value: ",", description: "\",\"" },
            peg$c46 = ".",
            peg$c47 = { type: "literal", value: ".", description: "\".\"" },
            peg$c48 = /^[eE]/,
            peg$c49 = { type: "class", value: "[eE]", description: "[eE]" },
            peg$c50 = "+",
            peg$c51 = { type: "literal", value: "+", description: "\"+\"" },
            peg$c52 = "-",
            peg$c53 = { type: "literal", value: "-", description: "\"-\"" },
            peg$c54 = /^[0-9]/,
            peg$c55 = { type: "class", value: "[0-9]", description: "[0-9]" },
            peg$c56 = function(digits) { return digits.join('') },
            peg$c57 = /^[ \t\n\r]/,
            peg$c58 = { type: "class", value: "[ \\t\\n\\r]", description: "[ \\t\\n\\r]" },
    
            peg$currPos          = 0,
            peg$reportedPos      = 0,
            peg$cachedPos        = 0,
            peg$cachedPosDetails = { line: 1, column: 1, seenCR: false },
            peg$maxFailPos       = 0,
            peg$maxFailExpected  = [],
            peg$silentFails      = 0,
    
            peg$result;
    
        if ("startRule" in options) {
          if (!(options.startRule in peg$startRuleFunctions)) {
            throw new Error("Can't start parsing from rule \"" + options.startRule + "\".");
          }
    
          peg$startRuleFunction = peg$startRuleFunctions[options.startRule];
        }
    
        function text() {
          return input.substring(peg$reportedPos, peg$currPos);
        }
    
        function offset() {
          return peg$reportedPos;
        }
    
        function line() {
          return peg$computePosDetails(peg$reportedPos).line;
        }
    
        function column() {
          return peg$computePosDetails(peg$reportedPos).column;
        }
    
        function expected(description) {
          throw peg$buildException(
            null,
            [{ type: "other", description: description }],
            peg$reportedPos
          );
        }
    
        function error(message) {
          throw peg$buildException(message, null, peg$reportedPos);
        }
    
        function peg$computePosDetails(pos) {
          function advance(details, startPos, endPos) {
            var p, ch;
    
            for (p = startPos; p < endPos; p++) {
              ch = input.charAt(p);
              if (ch === "\n") {
                if (!details.seenCR) { details.line++; }
                details.column = 1;
                details.seenCR = false;
              } else if (ch === "\r" || ch === "\u2028" || ch === "\u2029") {
                details.line++;
                details.column = 1;
                details.seenCR = true;
              } else {
                details.column++;
                details.seenCR = false;
              }
            }
          }
    
          if (peg$cachedPos !== pos) {
            if (peg$cachedPos > pos) {
              peg$cachedPos = 0;
              peg$cachedPosDetails = { line: 1, column: 1, seenCR: false };
            }
            advance(peg$cachedPosDetails, peg$cachedPos, pos);
            peg$cachedPos = pos;
          }
    
          return peg$cachedPosDetails;
        }
    
        function peg$fail(expected) {
          if (peg$currPos < peg$maxFailPos) { return; }
    
          if (peg$currPos > peg$maxFailPos) {
            peg$maxFailPos = peg$currPos;
            peg$maxFailExpected = [];
          }
    
          peg$maxFailExpected.push(expected);
        }
    
        function peg$buildException(message, expected, pos) {
          function cleanupExpected(expected) {
            var i = 1;
    
            expected.sort(function(a, b) {
              if (a.description < b.description) {
                return -1;
              } else if (a.description > b.description) {
                return 1;
              } else {
                return 0;
              }
            });
    
            while (i < expected.length) {
              if (expected[i - 1] === expected[i]) {
                expected.splice(i, 1);
              } else {
                i++;
              }
            }
          }
    
          function buildMessage(expected, found) {
            function stringEscape(s) {
              function hex(ch) { return ch.charCodeAt(0).toString(16).toUpperCase(); }
    
              return s
                .replace(/\\/g,   '\\\\')
                .replace(/"/g,    '\\"')
                .replace(/\x08/g, '\\b')
                .replace(/\t/g,   '\\t')
                .replace(/\n/g,   '\\n')
                .replace(/\f/g,   '\\f')
                .replace(/\r/g,   '\\r')
                .replace(/[\x00-\x07\x0B\x0E\x0F]/g, function(ch) { return '\\x0' + hex(ch); })
                .replace(/[\x10-\x1F\x80-\xFF]/g,    function(ch) { return '\\x'  + hex(ch); })
                .replace(/[\u0180-\u0FFF]/g,         function(ch) { return '\\u0' + hex(ch); })
                .replace(/[\u1080-\uFFFF]/g,         function(ch) { return '\\u'  + hex(ch); });
            }
    
            var expectedDescs = new Array(expected.length),
                expectedDesc, foundDesc, i;
    
            for (i = 0; i < expected.length; i++) {
              expectedDescs[i] = expected[i].description;
            }
    
            expectedDesc = expected.length > 1
              ? expectedDescs.slice(0, -1).join(", ")
                  + " or "
                  + expectedDescs[expected.length - 1]
              : expectedDescs[0];
    
            foundDesc = found ? "\"" + stringEscape(found) + "\"" : "end of input";
    
            return "Expected " + expectedDesc + " but " + foundDesc + " found.";
          }
    
          var posDetails = peg$computePosDetails(pos),
              found      = pos < input.length ? input.charAt(pos) : null;
    
          if (expected !== null) {
            cleanupExpected(expected);
          }
    
          return new SyntaxError(
            message !== null ? message : buildMessage(expected, found),
            expected,
            found,
            pos,
            posDetails.line,
            posDetails.column
          );
        }
    
        function peg$parsesvg_path() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = [];
          s2 = peg$parsewsp();
          while (s2 !== peg$FAILED) {
            s1.push(s2);
            s2 = peg$parsewsp();
          }
          if (s1 !== peg$FAILED) {
            s2 = peg$parsemoveTo_drawTo_commandGroups();
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s3 = [];
              s4 = peg$parsewsp();
              while (s4 !== peg$FAILED) {
                s3.push(s4);
                s4 = peg$parsewsp();
              }
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c3(s2);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsemoveTo_drawTo_commandGroups() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = peg$parsemoveTo_drawTo_commandGroup();
          if (s1 !== peg$FAILED) {
            s2 = peg$currPos;
            s3 = [];
            s4 = peg$parsewsp();
            while (s4 !== peg$FAILED) {
              s3.push(s4);
              s4 = peg$parsewsp();
            }
            if (s3 !== peg$FAILED) {
              s4 = peg$parsemoveTo_drawTo_commandGroups();
              if (s4 !== peg$FAILED) {
                s3 = [s3, s4];
                s2 = s3;
              } else {
                peg$currPos = s2;
                s2 = peg$c0;
              }
            } else {
              peg$currPos = s2;
              s2 = peg$c0;
            }
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s1 = [s1, s2];
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsemoveTo_drawTo_commandGroup() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = peg$parsemoveto();
          if (s1 !== peg$FAILED) {
            s2 = peg$currPos;
            s3 = [];
            s4 = peg$parsewsp();
            while (s4 !== peg$FAILED) {
              s3.push(s4);
              s4 = peg$parsewsp();
            }
            if (s3 !== peg$FAILED) {
              s4 = peg$parsedrawto_commands();
              if (s4 !== peg$FAILED) {
                s3 = [s3, s4];
                s2 = s3;
              } else {
                peg$currPos = s2;
                s2 = peg$c0;
              }
            } else {
              peg$currPos = s2;
              s2 = peg$c0;
            }
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s1 = [s1, s2];
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsedrawto_commands() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = peg$parsedrawto_command();
          if (s1 !== peg$FAILED) {
            s2 = peg$currPos;
            s3 = [];
            s4 = peg$parsewsp();
            while (s4 !== peg$FAILED) {
              s3.push(s4);
              s4 = peg$parsewsp();
            }
            if (s3 !== peg$FAILED) {
              s4 = peg$parsedrawto_commands();
              if (s4 !== peg$FAILED) {
                s3 = [s3, s4];
                s2 = s3;
              } else {
                peg$currPos = s2;
                s2 = peg$c0;
              }
            } else {
              peg$currPos = s2;
              s2 = peg$c0;
            }
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s1 = [s1, s2];
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsedrawto_command() {
          var s0;
    
          s0 = peg$parseclosepath();
          if (s0 === peg$FAILED) {
            s0 = peg$parselineto();
            if (s0 === peg$FAILED) {
              s0 = peg$parsehorizontal_lineto();
              if (s0 === peg$FAILED) {
                s0 = peg$parsevertical_lineto();
                if (s0 === peg$FAILED) {
                  s0 = peg$parsecurveto();
                  if (s0 === peg$FAILED) {
                    s0 = peg$parsesmooth_curveto();
                    if (s0 === peg$FAILED) {
                      s0 = peg$parsequadratic_bezier_curveto();
                      if (s0 === peg$FAILED) {
                        s0 = peg$parsesmooth_quadratic_bezier_curveto();
                        if (s0 === peg$FAILED) {
                          s0 = peg$parseelliptical_arc();
                        }
                      }
                    }
                  }
                }
              }
            }
          }
    
          return s0;
        }
    
        function peg$parsemoveto() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          if (peg$c4.test(input.charAt(peg$currPos))) {
            s1 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s1 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c5); }
          }
          if (s1 !== peg$FAILED) {
            s2 = [];
            s3 = peg$parsewsp();
            while (s3 !== peg$FAILED) {
              s2.push(s3);
              s3 = peg$parsewsp();
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsemoveto_argument_sequence();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c6(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsemoveto_argument_sequence() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = peg$parsecoordinate_pair();
          if (s1 !== peg$FAILED) {
            s2 = peg$currPos;
            s3 = peg$parsecomma_wsp();
            if (s3 === peg$FAILED) {
              s3 = peg$c2;
            }
            if (s3 !== peg$FAILED) {
              s4 = peg$parselineto_argument_sequence();
              if (s4 !== peg$FAILED) {
                s3 = [s3, s4];
                s2 = s3;
              } else {
                peg$currPos = s2;
                s2 = peg$c0;
              }
            } else {
              peg$currPos = s2;
              s2 = peg$c0;
            }
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              peg$reportedPos = s0;
              s1 = peg$c7(s1, s2);
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parseclosepath() {
          var s0, s1;
    
          s0 = peg$currPos;
          if (peg$c8.test(input.charAt(peg$currPos))) {
            s1 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s1 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c9); }
          }
          if (s1 !== peg$FAILED) {
            peg$reportedPos = s0;
            s1 = peg$c10();
          }
          s0 = s1;
    
          return s0;
        }
    
        function peg$parselineto() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          if (peg$c11.test(input.charAt(peg$currPos))) {
            s1 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s1 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c12); }
          }
          if (s1 !== peg$FAILED) {
            s2 = [];
            s3 = peg$parsewsp();
            while (s3 !== peg$FAILED) {
              s2.push(s3);
              s3 = peg$parsewsp();
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parselineto_argument_sequence();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c13(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parselineto_argument_sequence() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = peg$parsecoordinate_pair();
          if (s1 !== peg$FAILED) {
            s2 = peg$currPos;
            s3 = peg$parsecomma_wsp();
            if (s3 === peg$FAILED) {
              s3 = peg$c2;
            }
            if (s3 !== peg$FAILED) {
              s4 = peg$parselineto_argument_sequence();
              if (s4 !== peg$FAILED) {
                s3 = [s3, s4];
                s2 = s3;
              } else {
                peg$currPos = s2;
                s2 = peg$c0;
              }
            } else {
              peg$currPos = s2;
              s2 = peg$c0;
            }
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              peg$reportedPos = s0;
              s1 = peg$c7(s1, s2);
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsehorizontal_lineto() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          if (peg$c14.test(input.charAt(peg$currPos))) {
            s1 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s1 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c15); }
          }
          if (s1 !== peg$FAILED) {
            s2 = [];
            s3 = peg$parsewsp();
            while (s3 !== peg$FAILED) {
              s2.push(s3);
              s3 = peg$parsewsp();
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsecoordinate_sequence();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c16(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsecoordinate_sequence() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = peg$parsecoordinate();
          if (s1 !== peg$FAILED) {
            s2 = peg$currPos;
            s3 = peg$parsecomma_wsp();
            if (s3 === peg$FAILED) {
              s3 = peg$c2;
            }
            if (s3 !== peg$FAILED) {
              s4 = peg$parsecoordinate_sequence();
              if (s4 !== peg$FAILED) {
                s3 = [s3, s4];
                s2 = s3;
              } else {
                peg$currPos = s2;
                s2 = peg$c0;
              }
            } else {
              peg$currPos = s2;
              s2 = peg$c0;
            }
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              peg$reportedPos = s0;
              s1 = peg$c7(s1, s2);
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsevertical_lineto() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          if (peg$c17.test(input.charAt(peg$currPos))) {
            s1 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s1 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c18); }
          }
          if (s1 !== peg$FAILED) {
            s2 = [];
            s3 = peg$parsewsp();
            while (s3 !== peg$FAILED) {
              s2.push(s3);
              s3 = peg$parsewsp();
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsecoordinate_sequence();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c19(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsecurveto() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          if (peg$c20.test(input.charAt(peg$currPos))) {
            s1 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s1 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c21); }
          }
          if (s1 !== peg$FAILED) {
            s2 = [];
            s3 = peg$parsewsp();
            while (s3 !== peg$FAILED) {
              s2.push(s3);
              s3 = peg$parsewsp();
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsecurveto_argument_sequence();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c22(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsecurveto_argument_sequence() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = peg$parsecurveto_argument();
          if (s1 !== peg$FAILED) {
            s2 = peg$currPos;
            s3 = peg$parsecomma_wsp();
            if (s3 === peg$FAILED) {
              s3 = peg$c2;
            }
            if (s3 !== peg$FAILED) {
              s4 = peg$parsecurveto_argument_sequence();
              if (s4 !== peg$FAILED) {
                s3 = [s3, s4];
                s2 = s3;
              } else {
                peg$currPos = s2;
                s2 = peg$c0;
              }
            } else {
              peg$currPos = s2;
              s2 = peg$c0;
            }
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              peg$reportedPos = s0;
              s1 = peg$c7(s1, s2);
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsecurveto_argument() {
          var s0, s1, s2, s3, s4, s5;
    
          s0 = peg$currPos;
          s1 = peg$parsecoordinate_pair();
          if (s1 !== peg$FAILED) {
            s2 = peg$parsecomma_wsp();
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsecoordinate_pair();
              if (s3 !== peg$FAILED) {
                s4 = peg$parsecomma_wsp();
                if (s4 === peg$FAILED) {
                  s4 = peg$c2;
                }
                if (s4 !== peg$FAILED) {
                  s5 = peg$parsecoordinate_pair();
                  if (s5 !== peg$FAILED) {
                    peg$reportedPos = s0;
                    s1 = peg$c23(s1, s3, s5);
                    s0 = s1;
                  } else {
                    peg$currPos = s0;
                    s0 = peg$c0;
                  }
                } else {
                  peg$currPos = s0;
                  s0 = peg$c0;
                }
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsesmooth_curveto() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          if (peg$c24.test(input.charAt(peg$currPos))) {
            s1 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s1 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c25); }
          }
          if (s1 !== peg$FAILED) {
            s2 = [];
            s3 = peg$parsewsp();
            while (s3 !== peg$FAILED) {
              s2.push(s3);
              s3 = peg$parsewsp();
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsesmooth_curveto_argument_sequence();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c26(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsesmooth_curveto_argument_sequence() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = peg$parsesmooth_curveto_argument();
          if (s1 !== peg$FAILED) {
            s2 = peg$currPos;
            s3 = peg$parsecomma_wsp();
            if (s3 === peg$FAILED) {
              s3 = peg$c2;
            }
            if (s3 !== peg$FAILED) {
              s4 = peg$parsesmooth_curveto_argument_sequence();
              if (s4 !== peg$FAILED) {
                s3 = [s3, s4];
                s2 = s3;
              } else {
                peg$currPos = s2;
                s2 = peg$c0;
              }
            } else {
              peg$currPos = s2;
              s2 = peg$c0;
            }
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              peg$reportedPos = s0;
              s1 = peg$c7(s1, s2);
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsesmooth_curveto_argument() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          s1 = peg$parsecoordinate_pair();
          if (s1 !== peg$FAILED) {
            s2 = peg$parsecomma_wsp();
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsecoordinate_pair();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c27(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsequadratic_bezier_curveto() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          if (peg$c28.test(input.charAt(peg$currPos))) {
            s1 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s1 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c29); }
          }
          if (s1 !== peg$FAILED) {
            s2 = [];
            s3 = peg$parsewsp();
            while (s3 !== peg$FAILED) {
              s2.push(s3);
              s3 = peg$parsewsp();
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsequadratic_bezier_curveto_argument_sequence();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c30(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsequadratic_bezier_curveto_argument_sequence() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = peg$parsequadratic_bezier_curveto_argument();
          if (s1 !== peg$FAILED) {
            s2 = peg$currPos;
            s3 = peg$parsecomma_wsp();
            if (s3 === peg$FAILED) {
              s3 = peg$c2;
            }
            if (s3 !== peg$FAILED) {
              s4 = peg$parsequadratic_bezier_curveto_argument_sequence();
              if (s4 !== peg$FAILED) {
                s3 = [s3, s4];
                s2 = s3;
              } else {
                peg$currPos = s2;
                s2 = peg$c0;
              }
            } else {
              peg$currPos = s2;
              s2 = peg$c0;
            }
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              peg$reportedPos = s0;
              s1 = peg$c7(s1, s2);
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsequadratic_bezier_curveto_argument() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          s1 = peg$parsecoordinate_pair();
          if (s1 !== peg$FAILED) {
            s2 = peg$parsecomma_wsp();
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsecoordinate_pair();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c27(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsesmooth_quadratic_bezier_curveto() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          if (peg$c31.test(input.charAt(peg$currPos))) {
            s1 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s1 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c32); }
          }
          if (s1 !== peg$FAILED) {
            s2 = [];
            s3 = peg$parsewsp();
            while (s3 !== peg$FAILED) {
              s2.push(s3);
              s3 = peg$parsewsp();
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsesmooth_quadratic_bezier_curveto_argument_sequence();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c33(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsesmooth_quadratic_bezier_curveto_argument_sequence() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = peg$parsecoordinate_pair();
          if (s1 !== peg$FAILED) {
            s2 = peg$currPos;
            s3 = peg$parsecomma_wsp();
            if (s3 === peg$FAILED) {
              s3 = peg$c2;
            }
            if (s3 !== peg$FAILED) {
              s4 = peg$parsesmooth_quadratic_bezier_curveto_argument_sequence();
              if (s4 !== peg$FAILED) {
                s3 = [s3, s4];
                s2 = s3;
              } else {
                peg$currPos = s2;
                s2 = peg$c0;
              }
            } else {
              peg$currPos = s2;
              s2 = peg$c0;
            }
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              peg$reportedPos = s0;
              s1 = peg$c7(s1, s2);
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parseelliptical_arc() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          if (peg$c34.test(input.charAt(peg$currPos))) {
            s1 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s1 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c35); }
          }
          if (s1 !== peg$FAILED) {
            s2 = [];
            s3 = peg$parsewsp();
            while (s3 !== peg$FAILED) {
              s2.push(s3);
              s3 = peg$parsewsp();
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parseelliptical_arc_argument_sequence();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c36(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parseelliptical_arc_argument_sequence() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = peg$parseelliptical_arc_argument();
          if (s1 !== peg$FAILED) {
            s2 = peg$currPos;
            s3 = peg$parsecomma_wsp();
            if (s3 === peg$FAILED) {
              s3 = peg$c2;
            }
            if (s3 !== peg$FAILED) {
              s4 = peg$parseelliptical_arc_argument_sequence();
              if (s4 !== peg$FAILED) {
                s3 = [s3, s4];
                s2 = s3;
              } else {
                peg$currPos = s2;
                s2 = peg$c0;
              }
            } else {
              peg$currPos = s2;
              s2 = peg$c0;
            }
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              peg$reportedPos = s0;
              s1 = peg$c7(s1, s2);
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parseelliptical_arc_argument() {
          var s0, s1, s2, s3, s4, s5, s6, s7, s8, s9, s10, s11;
    
          s0 = peg$currPos;
          s1 = peg$parsenonnegative_number();
          if (s1 !== peg$FAILED) {
            s2 = peg$parsecomma_wsp();
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsenonnegative_number();
              if (s3 !== peg$FAILED) {
                s4 = peg$parsecomma_wsp();
                if (s4 === peg$FAILED) {
                  s4 = peg$c2;
                }
                if (s4 !== peg$FAILED) {
                  s5 = peg$parsenumber();
                  if (s5 !== peg$FAILED) {
                    s6 = peg$parsecomma_wsp();
                    if (s6 !== peg$FAILED) {
                      s7 = peg$parseflag();
                      if (s7 !== peg$FAILED) {
                        s8 = peg$parsecomma_wsp();
                        if (s8 === peg$FAILED) {
                          s8 = peg$c2;
                        }
                        if (s8 !== peg$FAILED) {
                          s9 = peg$parseflag();
                          if (s9 !== peg$FAILED) {
                            s10 = peg$parsecomma_wsp();
                            if (s10 === peg$FAILED) {
                              s10 = peg$c2;
                            }
                            if (s10 !== peg$FAILED) {
                              s11 = peg$parsecoordinate_pair();
                              if (s11 !== peg$FAILED) {
                                peg$reportedPos = s0;
                                s1 = peg$c37(s1, s3, s5, s7, s9, s11);
                                s0 = s1;
                              } else {
                                peg$currPos = s0;
                                s0 = peg$c0;
                              }
                            } else {
                              peg$currPos = s0;
                              s0 = peg$c0;
                            }
                          } else {
                            peg$currPos = s0;
                            s0 = peg$c0;
                          }
                        } else {
                          peg$currPos = s0;
                          s0 = peg$c0;
                        }
                      } else {
                        peg$currPos = s0;
                        s0 = peg$c0;
                      }
                    } else {
                      peg$currPos = s0;
                      s0 = peg$c0;
                    }
                  } else {
                    peg$currPos = s0;
                    s0 = peg$c0;
                  }
                } else {
                  peg$currPos = s0;
                  s0 = peg$c0;
                }
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsecoordinate_pair() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          s1 = peg$parsecoordinate();
          if (s1 !== peg$FAILED) {
            s2 = peg$parsecomma_wsp();
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsecoordinate();
              if (s3 !== peg$FAILED) {
                peg$reportedPos = s0;
                s1 = peg$c38(s1, s3);
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsecoordinate() {
          var s0, s1;
    
          s0 = peg$currPos;
          s1 = peg$parsenumber();
          if (s1 !== peg$FAILED) {
            peg$reportedPos = s0;
            s1 = peg$c39(s1);
          }
          s0 = s1;
    
          return s0;
        }
    
        function peg$parsenonnegative_number() {
          var s0;
    
          s0 = peg$parsefloating_point_constant();
          if (s0 === peg$FAILED) {
            s0 = peg$parsedigit_sequence();
          }
    
          return s0;
        }
    
        function peg$parsenumber() {
          var s0, s1, s2;
    
          s0 = peg$currPos;
          s1 = peg$parsesign();
          if (s1 === peg$FAILED) {
            s1 = peg$c2;
          }
          if (s1 !== peg$FAILED) {
            s2 = peg$parsefloating_point_constant();
            if (s2 !== peg$FAILED) {
              s1 = [s1, s2];
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
          if (s0 === peg$FAILED) {
            s0 = peg$currPos;
            s1 = peg$parsesign();
            if (s1 === peg$FAILED) {
              s1 = peg$c2;
            }
            if (s1 !== peg$FAILED) {
              s2 = peg$parsedigit_sequence();
              if (s2 !== peg$FAILED) {
                s1 = [s1, s2];
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          }
    
          return s0;
        }
    
        function peg$parseflag() {
          var s0;
    
          if (input.charCodeAt(peg$currPos) === 48) {
            s0 = peg$c40;
            peg$currPos++;
          } else {
            s0 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c41); }
          }
          if (s0 === peg$FAILED) {
            if (input.charCodeAt(peg$currPos) === 49) {
              s0 = peg$c42;
              peg$currPos++;
            } else {
              s0 = peg$FAILED;
              if (peg$silentFails === 0) { peg$fail(peg$c43); }
            }
          }
    
          return s0;
        }
    
        function peg$parsecomma_wsp() {
          var s0, s1, s2, s3, s4;
    
          s0 = peg$currPos;
          s1 = [];
          s2 = peg$parsewsp();
          if (s2 !== peg$FAILED) {
            while (s2 !== peg$FAILED) {
              s1.push(s2);
              s2 = peg$parsewsp();
            }
          } else {
            s1 = peg$c0;
          }
          if (s1 !== peg$FAILED) {
            s2 = peg$parsecomma();
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s3 = [];
              s4 = peg$parsewsp();
              while (s4 !== peg$FAILED) {
                s3.push(s4);
                s4 = peg$parsewsp();
              }
              if (s3 !== peg$FAILED) {
                s1 = [s1, s2, s3];
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
          if (s0 === peg$FAILED) {
            s0 = peg$currPos;
            s1 = peg$parsecomma();
            if (s1 !== peg$FAILED) {
              s2 = [];
              s3 = peg$parsewsp();
              while (s3 !== peg$FAILED) {
                s2.push(s3);
                s3 = peg$parsewsp();
              }
              if (s2 !== peg$FAILED) {
                s1 = [s1, s2];
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          }
    
          return s0;
        }
    
        function peg$parsecomma() {
          var s0;
    
          if (input.charCodeAt(peg$currPos) === 44) {
            s0 = peg$c44;
            peg$currPos++;
          } else {
            s0 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c45); }
          }
    
          return s0;
        }
    
        function peg$parsefloating_point_constant() {
          var s0, s1, s2;
    
          s0 = peg$currPos;
          s1 = peg$parsefractional_constant();
          if (s1 !== peg$FAILED) {
            s2 = peg$parseexponent();
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s1 = [s1, s2];
              s0 = s1;
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
          if (s0 === peg$FAILED) {
            s0 = peg$currPos;
            s1 = peg$parsedigit_sequence();
            if (s1 !== peg$FAILED) {
              s2 = peg$parseexponent();
              if (s2 !== peg$FAILED) {
                s1 = [s1, s2];
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          }
    
          return s0;
        }
    
        function peg$parsefractional_constant() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          s1 = peg$parsedigit_sequence();
          if (s1 === peg$FAILED) {
            s1 = peg$c2;
          }
          if (s1 !== peg$FAILED) {
            if (input.charCodeAt(peg$currPos) === 46) {
              s2 = peg$c46;
              peg$currPos++;
            } else {
              s2 = peg$FAILED;
              if (peg$silentFails === 0) { peg$fail(peg$c47); }
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsedigit_sequence();
              if (s3 !== peg$FAILED) {
                s1 = [s1, s2, s3];
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
          if (s0 === peg$FAILED) {
            s0 = peg$currPos;
            s1 = peg$parsedigit_sequence();
            if (s1 !== peg$FAILED) {
              if (input.charCodeAt(peg$currPos) === 46) {
                s2 = peg$c46;
                peg$currPos++;
              } else {
                s2 = peg$FAILED;
                if (peg$silentFails === 0) { peg$fail(peg$c47); }
              }
              if (s2 !== peg$FAILED) {
                s1 = [s1, s2];
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          }
    
          return s0;
        }
    
        function peg$parseexponent() {
          var s0, s1, s2, s3;
    
          s0 = peg$currPos;
          if (peg$c48.test(input.charAt(peg$currPos))) {
            s1 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s1 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c49); }
          }
          if (s1 !== peg$FAILED) {
            s2 = peg$parsesign();
            if (s2 === peg$FAILED) {
              s2 = peg$c2;
            }
            if (s2 !== peg$FAILED) {
              s3 = peg$parsedigit_sequence();
              if (s3 !== peg$FAILED) {
                s1 = [s1, s2, s3];
                s0 = s1;
              } else {
                peg$currPos = s0;
                s0 = peg$c0;
              }
            } else {
              peg$currPos = s0;
              s0 = peg$c0;
            }
          } else {
            peg$currPos = s0;
            s0 = peg$c0;
          }
    
          return s0;
        }
    
        function peg$parsesign() {
          var s0;
    
          if (input.charCodeAt(peg$currPos) === 43) {
            s0 = peg$c50;
            peg$currPos++;
          } else {
            s0 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c51); }
          }
          if (s0 === peg$FAILED) {
            if (input.charCodeAt(peg$currPos) === 45) {
              s0 = peg$c52;
              peg$currPos++;
            } else {
              s0 = peg$FAILED;
              if (peg$silentFails === 0) { peg$fail(peg$c53); }
            }
          }
    
          return s0;
        }
    
        function peg$parsedigit_sequence() {
          var s0, s1, s2;
    
          s0 = peg$currPos;
          s1 = [];
          if (peg$c54.test(input.charAt(peg$currPos))) {
            s2 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s2 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c55); }
          }
          if (s2 !== peg$FAILED) {
            while (s2 !== peg$FAILED) {
              s1.push(s2);
              if (peg$c54.test(input.charAt(peg$currPos))) {
                s2 = input.charAt(peg$currPos);
                peg$currPos++;
              } else {
                s2 = peg$FAILED;
                if (peg$silentFails === 0) { peg$fail(peg$c55); }
              }
            }
          } else {
            s1 = peg$c0;
          }
          if (s1 !== peg$FAILED) {
            peg$reportedPos = s0;
            s1 = peg$c56(s1);
          }
          s0 = s1;
    
          return s0;
        }
    
        function peg$parsewsp() {
          var s0;
    
          if (peg$c57.test(input.charAt(peg$currPos))) {
            s0 = input.charAt(peg$currPos);
            peg$currPos++;
          } else {
            s0 = peg$FAILED;
            if (peg$silentFails === 0) { peg$fail(peg$c58); }
          }
    
          return s0;
        }
    
    
          // The last coordinate we are at in the path. In absolute coords.
          var lastCoord = [0, 0];
          // The last control point we encountered in the path. In absolute coords.
          var lastControl = [0, 0];
          // The list of operations we've parsed so far.
          var ops = [];
          // Have we parsed the first sub-path yet?
          var firstSubPath = true;
          // The letter of the last parsed command.
          var lastCh = '';
    
          // Flatten an array.
          function flatten(a) {
            var flat = [];
            for (var i = 0; i < a.length; i++) {
              if (a[i] instanceof Array) {
                flat.push.apply(flat, flatten(a[i]));
              } else {
                flat.push(a[i]);
              }
            }
            return flat;
          }
    
          // Convert a position into an absolute position.
          function makeAbsolute(c, coord) {
            if ('mlazhvcsqt'.indexOf(c) === -1) {
              lastCoord = coord;
            } else {
              lastCoord[0] += coord[0];
              lastCoord[1] += coord[1];
            }
            lastCh = c;
            return lastCoord.slice(0);
          }
    
          // Convert a sequence of coordinates into absolute coordinates.
          //
          // For arguments that take multiple coord pairs, such as bezier.
          function makeAbsoluteMultiple(c, seq) {
            var r = [];
            var lastPosCopy = lastCoord.slice(0);
            for (var i=0; i < seq.length; i+=2) {
              // Only the last point should update lastCoord.
              lastCoord = lastPosCopy.slice(0);
              var coord = makeAbsolute(c, seq.slice(i, i+2));
              r = r.concat(coord);
              // Record the last control point, it might be needed for
              // shorthand operations.
              if (i == seq.length-4) {
                lastControl = coord.slice(0);
              }
            }
            return r;
          }
    
          // Find the reflection of the last control point over
          // the last postion in the path.
          function makeReflected() {
            if ('CcSsQqTt'.indexOf(lastCh) == -1) {
              lastControl = lastCoord.slice(0);
            }
            // reflected = 2*lastCoord - lastControl
            // Note the result is absolute, not relative.
            var r = [0, 0];
            r[0] = 2*lastCoord[0] - lastControl[0];
            r[1] = 2*lastCoord[1] - lastControl[1];
            return r;
          }
    
          function makeAbsoluteFromX(c, x) {
            var coord = [x, 0];
            if (c == 'H') {
              coord[1] = lastCoord[1];
            }
            return makeAbsolute(c, coord);
          }
    
          function makeAbsoluteFromY(c, y) {
            var coord = [0, y];
            if (c == 'V') {
              coord[0] = lastCoord[0];
            }
            return makeAbsolute(c, coord);
          }
    
          function concatSequence(one, rest) {
            var r = [one];
            if (rest && rest.length > 1) {
              var rem = rest[1];
              for (var i = 0; i < rem.length; i++) {
                r.push(rem[i]);
              }
            }
            return r;
          }
    
          function mag(v) {
            return Math.sqrt(Math.pow(v[0], 2) + Math.pow(v[1], 2));
          }
    
          function dot(u, v) {
            return (u[0]*v[0] + u[1]*v[1]);
          }
    
          function ratio(u, v) {
            return dot(u,v) / (mag(u)*mag(v))
          }

          function clamp(value, min, max) {
            return Math.min(Math.max(val, min),max);
          }
    
          function angle(u, v) {
            var sign = 1.0;
            if ((u[0]*v[1] - u[1]*v[0]) < 0) {
              sign = -1.0;
            }
            return sign * Math.acos(clamp(ratio(u,v)), -1, 1);
          }
    
          function rotClockwise(v, angle) {
            var cost = Math.cos(angle);
            var sint = Math.sin(angle);
            return [cost*v[0] + sint*v[1], -1 * sint*v[0] + cost*v[1]];
          }
    
          function rotCounterClockwise(v, angle) {
            var cost = Math.cos(angle);
            var sint = Math.sin(angle);
            return [cost*v[0] - sint*v[1], sint*v[0] + cost*v[1]];
          }
    
          function midPoint(u, v) {
            return [(u[0] - v[0])/2.0, (u[1] - v[1])/2.0];
          }
    
          function meanVec(u, v) {
            return [(u[0] + v[0])/2.0, (u[1] + v[1])/2.0];
          }
    
          function pointMul(u, v) {
            return [u[0]*v[0], u[1]*v[1]];
          }
    
          function scale(c, v) {
            return [c*v[0], c*v[1]];
          }
    
          function sum(u, v) {
            return [u[0] + v[0], u[1] + v[1]];
          }
    
          // Convert an SVG elliptical arc to a series of canvas commands.
          //
          // x1, x2: start and stop coordinates of the ellipse.
          // rx, ry: radii of the ellipse.
          // phi: rotation of the ellipse.
          // fA: large arc flag.
          // fS: sweep flag.
          function ellipseFromEllipticalArc(x1, rx, ry, phi, fA, fS, x2) {
            // Convert from endpoint to center parametrization, as detailed in:
            //   http://www.w3.org/TR/SVG/implnote.html#ArcImplementationNotes
            if (rx == 0 || ry == 0) {
              ops.push({type: 'lineTo', args: x2});
              return;
            }
            var phi = phi * (Math.PI / 180.0);
            rx = Math.abs(rx);
            ry = Math.abs(ry);
            var xPrime = rotClockwise(midPoint(x1, x2), phi);                // F.6.5.1
            var xPrime2 = pointMul(xPrime, xPrime);
            var rx2 = Math.pow(rx, 2);
            var ry2 = Math.pow(ry, 2);
    
            var lambda = Math.sqrt(xPrime2[0]/rx2 + xPrime2[1]/ry2);
            if (lambda > 1) {
              rx *= lambda;
              ry *= lambda;
              rx2 = Math.pow(rx, 2);
              ry2 = Math.pow(ry, 2);
            }
            var factor = Math.sqrt(Math.abs(rx2*ry2 - rx2*xPrime2[1] - ry2*xPrime2[0]) /
              (rx2*xPrime2[1] + ry2*xPrime2[0]));
            if (fA == fS) {
              factor *= -1.0;
            }
            var cPrime = scale(factor, [rx*xPrime[1]/ry, -ry*xPrime[0]/rx]); // F.6.5.2
            var c = sum(rotCounterClockwise(cPrime, phi), meanVec(x1, x2));  // F.6.5.3
            var x1UnitVector = [(xPrime[0] - cPrime[0])/rx, (xPrime[1] - cPrime[1])/ry];
            var x2UnitVector = [(-1.0*xPrime[0] - cPrime[0])/rx, (-1.0*xPrime[1] - cPrime[1])/ry];
            var theta = angle([1, 0], x1UnitVector);                         // F.6.5.5
            var deltaTheta = angle(x1UnitVector, x2UnitVector);              // F.6.5.6
            var start = theta;
            var end = theta+deltaTheta;
            ops.push(
              {type: 'save', args: []},
              {type: 'translate', args: [c[0], c[1]]},
              {type: 'rotate', args: [phi]},
              {type: 'scale', args: [rx, ry]},
              {type: 'arc', args: [0, 0, 1, start, end, 1-fS]},
              {type: 'restore', args: []}
              );
          }
    
    
        peg$result = peg$startRuleFunction();
    
        if (peg$result !== peg$FAILED && peg$currPos === input.length) {
          return peg$result;
        } else {
          if (peg$result !== peg$FAILED && peg$currPos < input.length) {
            peg$fail({ type: "end", description: "end of input" });
          }
    
          throw peg$buildException(null, peg$maxFailExpected, peg$maxFailPos);
        }
      }
    
      return {
        SyntaxError: SyntaxError,
        parse:       parse
      };
    })();

    function Path_(arg) {
      this.ops_ = [];
      if (arg == undefined) {
        return;
      }
      if (typeof arg == 'string') {
        try {
          this.ops_ = parser.parse(arg);
        } catch(e) {
          // Treat an invalid SVG path as an empty path.
        }
      } else if (arg.hasOwnProperty('ops_')) {
        this.ops_ = arg.ops_.slice(0);
      } else {
        throw 'Error: ' + typeof arg + 'is not a valid argument to Path';
      }
    };

    // TODO(jcgregorio) test for arcTo and implement via something.


    // Path methods that map simply to the CanvasRenderingContext2D.
    var simple_mapping = [
      'closePath',
      'moveTo',
      'lineTo',
      'quadraticCurveTo',
      'bezierCurveTo',
      'rect',
      'arc',
      'arcTo',
      'ellipse',
      'isPointInPath',
      'isPointInStroke',
      ];

    function createFunction(name) {
      return function() {
        this.ops_.push({type: name, args: Array.prototype.slice.call(arguments, 0)});
      };
    }

    // Add simple_mapping methods to Path2D.
    for (var i=0; i<simple_mapping.length; i++) {
      var name = simple_mapping[i];
      Path_.prototype[name] = createFunction(name);
    }

    Path_.prototype['addPath'] = function(path, tr) {
      var hasTx = false;
      if (tr
          && tr.a != undefined
          && tr.b != undefined
          && tr.c != undefined
          && tr.d != undefined
          && tr.e != undefined
          && tr.f != undefined) {
        hasTx = true;
        this.ops_.push({type: 'save', args: []});
        this.ops_.push({type: 'transform', args: [tr.a, tr.b, tr.c, tr.d, tr.e, tr.f]});
      }
      this.ops_ = this.ops_.concat(path.ops_);
      if (hasTx) {
        this.ops_.push({type: 'restore', args: []});
      }
    }

    original_fill = CanvasRenderingContext2D.prototype.fill;
    original_stroke = CanvasRenderingContext2D.prototype.stroke;
    original_clip = CanvasRenderingContext2D.prototype.clip;
    original_is_point_in_path = CanvasRenderingContext2D.prototype.isPointInPath;
    original_is_point_in_stroke = CanvasRenderingContext2D.prototype.isPointInStroke;

    // Replace methods on CanvasRenderingContext2D with ones that understand Path2D.
    CanvasRenderingContext2D.prototype.fill = function(arg) {
      if (arg instanceof Path_) {
        this.beginPath();
        for (var i = 0, len = arg.ops_.length; i < len; i++) {
          var op = arg.ops_[i];
          CanvasRenderingContext2D.prototype[op.type].apply(this, op.args);
        }
        original_fill.apply(this, Array.prototype.slice.call(arguments, 1));
      } else {
        original_fill.apply(this, arguments);
      }
    }

    CanvasRenderingContext2D.prototype.stroke = function(arg) {
      if (arg instanceof Path_) {
        this.beginPath();
        for (var i = 0, len = arg.ops_.length; i < len; i++) {
          var op = arg.ops_[i];
          CanvasRenderingContext2D.prototype[op.type].apply(this, op.args);
        }
        original_stroke.call(this);
      } else {
        original_stroke.call(this);
      }
    }

    CanvasRenderingContext2D.prototype.clip = function(arg) {
      if (arg instanceof Path_) {
        // Note that we don't save and restore the context state, since the
        // clip region is part of the state. Not really a problem since the
        // HTML 5 spec doesn't say that clip(path) doesn't affect the current
        // path.
        this.beginPath();
        for (var i = 0, len = arg.ops_.length; i < len; i++) {
          var op = arg.ops_[i];
          CanvasRenderingContext2D.prototype[op.type].apply(this, op.args);
        }
        original_clip.apply(this, Array.prototype.slice.call(arguments, 1));
      } else {
        original_clip.apply(this, arguments);
      }
    }

    CanvasRenderingContext2D.prototype.isPointInPath = function(arg) {
      if (arg instanceof Path_) {
        this.beginPath();
        for (var i = 0, len = arg.ops_.length; i < len; i++) {
          var op = arg.ops_[i];
          CanvasRenderingContext2D.prototype[op.type].apply(this, op.args);
        }
        return original_is_point_in_path.apply(this, Array.prototype.slice.call(arguments, 1));
      } else {
        return original_is_point_in_path.apply(this, arguments);
      }
    }
    CanvasRenderingContext2D.prototype.isPointInStroke = function(arg) {
      if (arg instanceof Path_) {
        this.beginPath();
        for (var i = 0, len = arg.ops_.length; i < len; i++) {
          var op = arg.ops_[i];
          CanvasRenderingContext2D.prototype[op.type].apply(this, op.args);
        }
        return original_is_point_in_stroke.apply(this, Array.prototype.slice.call(arguments, 1));
      } else {
        return original_is_point_in_stroke.apply(this, arguments);
      }
    }

    // Set up externs.
    Path2D = Path_;
  })();
}

})(
  typeof CanvasRenderingContext2D === "undefined" ? undefined : CanvasRenderingContext2D,
  typeof require === "undefined" ? undefined : require
);
;
/* 
    Except for the DrawingObjectType enum constants, this file is identical in the admin and front end 
    so it is safe to copy & paste the whole file when applying changes back and forth.
*/
var DRAWINGOBJECTTYPE_RECTANGLE = 'Rectangle';
var DRAWINGOBJECTTYPE_ROUNDEDRECTANGLE = 'RoundedRectangle';
var DRAWINGOBJECTTYPE_CIRCLE = 'Circle';
var DRAWINGOBJECTTYPE_LINE = 'Line';
var DRAWINGOBJECTTYPE_POLYGON = 'Polygon';
var DRAWINGOBJECTTYPE_TEXT = 'Text';
var DRAWINGOBJECTTYPE_CONTAINER = 'Container';
var DRAWINGOBJECTTYPE_IMAGE = 'Image';
var DRAWINGOBJECTTYPE_PATH = 'Path';
var DRAWINGOBJECTTYPE_POLYLINE = 'Polyline';

/*
    DrawingObject()
*/
function drawingObject(options) {
    this.init(options);
}

drawingObject.prototype = {
    init: function (options) {
        this.context = options.context;
        this.model = options.model;
        this.getColor = options.getColor ? options.getColor : function () { return "white" };
        this.getLineWidth = options.getLineWidth ? options.getLineWidth : null;
    },

    draw: function (drawingObject, isInDetailMode) {
        if ((isInDetailMode && !drawingObject.showInDetailView) ||
            (!isInDetailMode && !drawingObject.showInOverview))
            return;
        var self = this;
        var image = null;

        if (drawingObject.type == DRAWINGOBJECTTYPE_IMAGE) {
            image = this.model.getImage(drawingObject.imageID);
            // ensures that the drawingObject object has the width and height of the image
            if (drawingObject.width <= 0 || drawingObject.height <= 0) {
                if (image && image.htmlImage && image.htmlImage.complete) {
                    drawingObject.width = image.htmlImage.width;
                    drawingObject.height = image.htmlImage.height;
                }
            }
        }

        this.context.save();
        if (drawingObject._matrix === undefined) {
            drawingObject._matrix = [];
            if (drawingObject.matrix != null && drawingObject.matrix.length > 0) {
                var matrix = JSON.parse(drawingObject.matrix);
                if (matrix instanceof Array) {
                    matrix.forEach(function (m) { drawingObject._matrix.push(m) });
                    while (drawingObject._matrix.length < 6)
                        drawingObject._matrix.push(0);
                }
            }
        }
        if (drawingObject.shadow != null) {
            this.context.shadowColor = drawingObject.shadow.color;
            this.context.shadowBlur = drawingObject.shadow.blur;
            this.context.shadowOffsetX = drawingObject.shadow.offsetX;
            this.context.shadowOffsetY = drawingObject.shadow.offsetY;
        }
        
        this.context.fillStyle = drawingObject.fillStyle;
        this.context.strokeStyle = drawingObject.strokeStyle;
        if (drawingObject.type == DRAWINGOBJECTTYPE_TEXT) {
            this.applyTransforms(drawingObject);
            this.context.font = drawingObject.font;
            this.context.fillText(drawingObject.text, 0, 0);
        }
        else if (drawingObject.type == DRAWINGOBJECTTYPE_PATH) {
            this.applyTransforms(drawingObject);
            if (!drawingObject._path2D && (typeof Path2D === 'function'))
                drawingObject._path2D = new Path2D(drawingObject.path);
            if (drawingObject._path2D !== undefined) {
                if (drawingObject.fillStyle && drawingObject.fillStyle != '' && drawingObject.fillStyle != 'none') {
                    var fillStyle = drawingObject.fillStyle.charAt(0) == '{' ? this.getColor(drawingObject.fillStyle) : drawingObject.fillStyle;
                    this.context.fillStyle = fillStyle;
                    this.context.fill(drawingObject._path2D);
                }
                if (drawingObject.strokeStyle && drawingObject.strokeStyle != '') {
                    if (drawingObject.lineWidth > 0)
                        this.context.lineWidth = this.getLineWidth ? this.getLineWidth(drawingObject.lineWidth) : drawingObject.lineWidth;
                    this.context.strokeStyle = drawingObject.strokeStyle;
                    this.context.stroke(drawingObject._path2D);
                }
            }
        }
        else {
            this.createPath(drawingObject);

            if (drawingObject.fillStyle && drawingObject.fillStyle != '') {
                var fillStyle = drawingObject.fillStyle.charAt(0) == '{' ? this.getColor(drawingObject.fillStyle) : drawingObject.fillStyle;
                this.context.fillStyle = fillStyle;
                this.context.fill();
            }

            if (drawingObject.strokeStyle && drawingObject.strokeStyle != '') {
                if (drawingObject.lineWidth > 0)
                    this.context.lineWidth = this.getLineWidth ? this.getLineWidth(drawingObject.lineWidth) : drawingObject.lineWidth;
                this.context.strokeStyle = drawingObject.strokeStyle;
                this.context.stroke();
            }
        }

        if (drawingObject.type == DRAWINGOBJECTTYPE_IMAGE && image && image.htmlImage && image.htmlImage.naturalHeight > 0) {
            this.applyTransforms(drawingObject);
            if (drawingObject.width > 0 && drawingObject.height > 0)
                this.context.drawImage(image.htmlImage, 0, 0, drawingObject.width, drawingObject.height);
            else {
                this.context.drawImage(image.htmlImage, drawingObject.x, drawingObject.y);
            }
        }

        this.context.restore();

        if (drawingObject.children.length > 0) {
            this.context.save();

            this.applyTransforms(drawingObject);
            for (var i = 0; i < drawingObject.children.length; i++)
                this.draw(drawingObject.children[i], isInDetailMode);

            this.context.restore();
        }
    },

    applyTransforms: function (drawingObject) {
        if (drawingObject._matrix !== undefined && drawingObject._matrix.length == 6)
            this.context.transform(drawingObject._matrix[0], drawingObject._matrix[1], drawingObject._matrix[2], drawingObject._matrix[3], drawingObject._matrix[4], drawingObject._matrix[5]);
        this.context.rotate(drawingObject.angle * Math.PI / 180);
        this.context.translate(drawingObject.x, drawingObject.y)
    },

    createPath: function (drawingObject) {
        this.context.save();
        this.applyTransforms(drawingObject);

        this.context.beginPath();
        switch (drawingObject.type) {
            case DRAWINGOBJECTTYPE_CIRCLE:
                this.context.arc(0, 0, drawingObject.radius, 0, 2 * Math.PI);
                break;
            case DRAWINGOBJECTTYPE_RECTANGLE:
            case DRAWINGOBJECTTYPE_IMAGE:
                this.context.rect(0, 0, drawingObject.width, drawingObject.height);
                break;
            case DRAWINGOBJECTTYPE_ROUNDEDRECTANGLE:
                var x = 0;
                var y = 0;

                this.context.moveTo(x + drawingObject.radius, y);
                this.context.lineTo(x + drawingObject.width - drawingObject.radius, y);
                this.context.quadraticCurveTo(x + drawingObject.width, y, x + drawingObject.width, y + drawingObject.radius);
                this.context.lineTo(x + drawingObject.width, y + drawingObject.height - drawingObject.radius);
                this.context.quadraticCurveTo(x + drawingObject.width, y + drawingObject.height, x + drawingObject.width - drawingObject.radius, y + drawingObject.height);
                this.context.lineTo(x + drawingObject.radius, y + drawingObject.height);
                this.context.quadraticCurveTo(x, y + drawingObject.height, x, y + drawingObject.height - drawingObject.radius);
                this.context.lineTo(x, y + drawingObject.radius);
                this.context.quadraticCurveTo(x, y, x + drawingObject.radius, y);
                break;
            case DRAWINGOBJECTTYPE_LINE:
                this.context.moveTo(0, 0);
                this.context.lineTo(drawingObject.coords[0].x - drawingObject.x, drawingObject.coords[0].y - drawingObject.y);
                break;
            case DRAWINGOBJECTTYPE_POLYGON:
            case DRAWINGOBJECTTYPE_POLYLINE:
                this.context.moveTo(drawingObject.coords[0].x - drawingObject.x, drawingObject.coords[0].y - drawingObject.y);
                for (var i = 1; i < drawingObject.coords.length; i++)
                    this.context.lineTo(drawingObject.coords[i].x - drawingObject.x, drawingObject.coords[i].y - drawingObject.y);
                break;
        }
        if (drawingObject.type != DRAWINGOBJECTTYPE_POLYLINE)
            this.context.closePath();

        this.context.restore();
    },
    
    isPointInSection: function (drawingObject, x, y, isInDetailMode, currentSectionID) {
        if (drawingObject.sectionID == null && currentSectionID == null)
            return null;

        if ((isInDetailMode && !drawingObject.showInDetailView) ||
            (!isInDetailMode && !drawingObject.showInOverview))
            return null;

        if (drawingObject.sectionID != null)
            currentSectionID = drawingObject.sectionID;

        var result = null;
        var isPointInPath = false;
        if (drawingObject._path2D) {
            this.context.save();
            this.applyTransforms(drawingObject);
            isPointInPath = this.context.isPointInPath(drawingObject._path2D, x, y);
            this.context.restore();
        }
        else {
            this.createPath(drawingObject);
            isPointInPath = this.context.isPointInPath(x, y);
        }
        if (isPointInPath)
            result = currentSectionID;
        else if (drawingObject.children.length > 0) {
            this.context.save();

            this.applyTransforms(drawingObject);
            for (var i = 0; i < drawingObject.children.length; i++) {
                result = this.isPointInSection(drawingObject.children[i], x, y, isInDetailMode, currentSectionID)
                if (result != null) {
                    result = currentSectionID;
                    break;
                }
            }
            this.context.restore();
        }

        return result;
    },

    /*
        getPositionInfo: returns null if nothing is found. if there is an object at the position, the following properties may be returned:
            - sectionID         - the section ID that the point belongs to.
            - tooltip           - the tooltip text content.
            - imageIDTooltip    - the id of the image that should be shown in the tooltip.
    */
    getPositionInfo: function (drawingObject, x, y, isInDetailMode, info) {
        if (!info)
            info = { hasInfo: false };

        if ((isInDetailMode && !drawingObject.showInDetailView) ||
            (!isInDetailMode && !drawingObject.showInOverview))
            return null;

        var isPointInPath = false;
        if (drawingObject._path2D) {
            this.context.save();
            this.applyTransforms(drawingObject);
            isPointInPath = this.context.isPointInPath(drawingObject._path2D, x, y);
            this.context.restore();
        }
        else {
            this.createPath(drawingObject);
            isPointInPath = this.context.isPointInPath(x, y);
        }
        if (isPointInPath)
        {
            if (info == null)
                info = { hasInfo: false };
            if (drawingObject.sectionID && drawingObject.sectionID > 0) {
                info.hasInfo = true;
                info.sectionID = drawingObject.sectionID;
            }
            if (drawingObject.tooltip && drawingObject.tooltip.length > 0) {
                info.hasInfo = true;
                info.tooltip = drawingObject.tooltip;
            }
            if (drawingObject.imageIDTooltip && drawingObject.imageIDTooltip > 0) {
                info.hasInfo = true;
                info.imageIDTooltip = drawingObject.imageIDTooltip;
            }
        }

        if (drawingObject.children.length > 0) {
            this.context.save();

            this.applyTransforms(drawingObject);
            for (var i = 0; i < drawingObject.children.length; i++)
                this.getPositionInfo(drawingObject.children[i], x, y, isInDetailMode, info)

            this.context.restore();
        }

        return info;
    }

};
;
// Modified:		04.10.2016 ticketporal AG, St Gallen fbe	: (SCR 9459) Bonuscard webshop: error when user has 3rd party cookies disabled
// miniMapController object
function miniMapController(canvas, controller, model, view) {
    this.canvas = canvas;
    this.controller = controller;
    this.model = model;
    this.view = view;

    this.context = this.canvas.getContext('2d');
    this.drawer = new drawingObject({ context: this.context, model: this.model });
    this.cachedImage = null;

    this.logLevel = 0;
    this.scale = this.canvas.width / this.model.seatMap.width;
    this.isDragging = false;
    this.lastMousePosition = null;
    this.singleTouchStartPoint = null;
    this.singleTouchHadMovements = false;
    this.isGesture = false;

    var $canvas = $(this.canvas);
    // events binding
    var self = this;
    $canvas.off('selectstart').on('selectstart', function () { return false; });
    $canvas.off('mouseup').on('mouseup', function (e) { self.onMouseUp(e.originalEvent) });
    $canvas.off('mousedown').on('mousedown', function (e) { self.onMouseDown(e.originalEvent) });
    $canvas.off('mousemouve').on('mousemove', function (e) { self.onMouseMove(e.originalEvent) });
    $canvas.off('click').on('click', function (e) { self.onCanvasClick(e.originalEvent) })
    $canvas.off('dblclick').on('dblclick', function (e) { self.onCanvasDoubleClick(e.originalEvent) });
    $canvas.off('touchstart').on('touchstart', function (e) { self.handleTouchStart(e.originalEvent) });
    $canvas.off('touchend').on('touchend', function (e) { self.handleTouchEnd(e.originalEvent) });
    $canvas.off('touchleave').on('touchleave', function (e) { self.handleTouchLeave(e.originalEvent) });
    $canvas.off('touchmove').on('touchmove', function (e) { self.handleTouchMove(e.originalEvent) });
    $canvas.off('touchcancel').on('touchcancel', function (e) { self.handleTouchCancel(e.originalEvent) });
    $canvas.off('mousewheel').on('mousewheel', function (e) { self.handleMouseWheel(e.originalEvent) });
    $canvas.off('wheel').on('wheel', function (e) { self.handleWheel(e.originalEvent) });
}

// getHighlightArea()
miniMapController.prototype.getHighlightArea = function() {
    var seatMapCanvasSize = this.view.getSize();
    var area = {
        x: this.model.translate.x,
        y: this.model.translate.y,
        width: seatMapCanvasSize.width / this.model.scale,
        height: seatMapCanvasSize.height / this.model.scale
    };

    return area;
}

// draw()
miniMapController.prototype.draw = function () {
    var seatMap = this.model.seatMap;
    this.context.clearRect(0, 0, this.canvas.width, this.canvas.height);
    this.context.save();

    this.context.scale(this.scale, this.scale);

    for (var i = 0; i < seatMap.drawingObjects.length; i++) {
        var drawingObject = seatMap.drawingObjects[i];
        if (drawingObject.showInOverview) {
            this.drawer.draw(drawingObject, false);
        }
    }

    if (this.model.scale != this.model.overviewScale) {
        this.context.globalAlpha = 0.8;

        this.context.strokeStyle = '#565051';
        this.context.fillStyle = '#d1d0ce';
        this.context.lineWidth = 5;

        var highlight = this.getHighlightArea();
        this.context.rect(highlight.x, highlight.y, highlight.width, highlight.height);

        this.context.fill();
        this.context.stroke();
    }
    this.context.restore();
}

// goTo()
miniMapController.prototype.goTo = function (point) {
    var size = this.view.getSize();
    var deltaX = size.width / (this.model.scale * 2);
    var deltaY = size.height / (this.model.scale * 2);
    this.controller.updateTranslate(point.x - deltaX, point.y - deltaY);
    this.view.draw();
}

//---------------------------------------
// Event handlers
//-------------------------------------
miniMapController.prototype.handleMouseWheel = function (event) {
    this.view.handleMouseWheel(event);

    return false;
}
miniMapController.prototype.handleWheel = function (event) {
    this.view.handleWheel(event);

    return false;
}

miniMapController.prototype.onCanvasClick = function (event) {
    if (this.model.scale != this.model.overviewScale) {
        var mouse = this.getMousePosition(event);

        this.goTo(mouse);
    }
}

miniMapController.prototype.onCanvasDoubleClick = function (event) {
    if (this.model.scale != this.model.overviewScale) {
        var mouse = this.getMousePosition(event);

        var section = this.view.getSectionFromPoint(mouse.x, mouse.y);
        if (section != null)
            this.controller.switchToZoomView(section);
    }
}

miniMapController.prototype.onMouseDown = function (event) {
    if (this.model.scale != this.model.overviewScale) {
        var mouse = this.getMousePosition(event);
        this.goTo(mouse);
        this.lastMousePosition = mouse;
        this.isDragging = false;
    }
}

miniMapController.prototype.onMouseUp = function (event) {
    this.lastMousePosition = null;
}

miniMapController.prototype.onMouseMove = function (event) {
    if (this.model.scale != this.model.overviewScale) {
        var mouse = this.getMousePosition(event);
        if (event.which == 1 && this.lastMousePosition != null) {
            this.goTo(mouse);
            return;
        }
    }
}

miniMapController.prototype.handleTouchStart = function (event) {
    this.singleTouchStartPoint = this.getTouchPosition(event);
    this.singleTouchHadMovements = false;

    event.preventDefault();
}

miniMapController.prototype.handleTouchEnd = function (event) {

    if (!this.singleTouchHadMovements && (this.singleTouchStartPoint != null)) {
        this.goTo(this.singleTouchStartPoint);
    }

    this.singleTouchStartPoint = null;
    this.singleTouchHadMovements = false;

    event.preventDefault();
}

miniMapController.prototype.handleTouchLeave = function (event) {

    this.singleTouchStartPoint = null;
    this.singleTouchHadMovements = false;

    event.preventDefault();
}

miniMapController.prototype.handleTouchMove = function (event) {

    this.touchMoveCounter++;
    event.preventDefault();

    var position = this.getTouchPosition(event);
    if ((this.singleTouchStartPoint != null) && (position != null) && !this.isGesture) {
        var deltaX = position.x - this.singleTouchStartPoint.x;
        var deltaY = position.y - this.singleTouchStartPoint.y;

        if (!this.singleTouchHadMovements && (Math.abs(deltaX) > 5 || Math.abs(deltaY) > 5))
            this.singleTouchHadMovements = true;
        if (this.singleTouchHadMovements) {
            this.controller.updateTranslate(this.model.translate.x + deltaX, this.model.translate.y + deltaY);
            this.view.draw();
            this.singleTouchStartPoint = this.getTouchPosition(event);
        }
    }

    if (position != null && this.onMouseMoveEventHandler) {
        this.onMouseMoveEventHandler(position);
    }

    if (this.logLevel >= 2) {
        if (event.targetTouches.length == 1) {
            this.logAction('1 touch move', 'x:' + event.targetTouches[0].pageX + ', y: ' + event.targetTouches[0].pageY);
        } else {
            this.logAction('touch move', event.targetTouches.length + ' touches found');
        }

    }

}

miniMapController.prototype.handleTouchCancel = function (event) {

    event.preventDefault();

    this.singleTouchStartPoint = null;
    this.singleTouchHadMovements = false;

    if (this.logLevel >= 2) {
        logAction('touch cancel', event.targetTouches.length + ' touches found');
    }
}

// ----------------------------------------------------
//      HELPER FUNCTIONS 
// ----------------------------------------------------

miniMapController.prototype.getTouchPosition = function (event) {
    var mouseX = 0;
    var mouseY = 0;

    if (event.targetTouches.length == 1) {
        mouseX = event.targetTouches[0].clientX;
        mouseY = event.targetTouches[0].clientY;
    }
    else if (event.changedTouches.length == 1) {
        mouseX = event.changedTouches[0].clientX;
        mouseY = event.changedTouches[0].clientY;
    }
    else
        return null;

    var canvasPosition = this.canvas.getBoundingClientRect();
    mouseX = (mouseX - canvasPosition.left) / this.scale;
    mouseY = (mouseY - canvasPosition.top) / this.scale;

    return { x: mouseX, y: mouseY };
}

miniMapController.prototype.getMousePosition = function (event) {
    var mouseX = new Number();
    var mouseY = new Number();

    if (event.offsetX != undefined && event.offsetY != undefined) {
        mouseX = event.offsetX;
        mouseY = event.offsetY;
    }

    var seatMapX = parseInt(mouseX / this.scale);
    var seatMapY = parseInt(mouseY / this.scale);

    return { x: seatMapX, y: seatMapY, canvasX: parseInt(mouseX), canvasY: parseInt(mouseY) };
};
var VIEWMODE_OVERVIEW = 1;
var VIEWMODE_DETAIL = 2;

/*
    Seat map model
*/
function seatMapModel(options) {
    this.init(options);
}

seatMapModel.prototype = {
    init: function (options) {
        this.seatMap = options && options.seatMapData ? options.seatMapData : null;
        this.isInFullScreen = false;
        this.viewMode = options && options.viewMode ? options.viewMode : VIEWMODE_OVERVIEW;
        this.translate = { x: 0, y: 0 };
        this.scale = 1;
        this.overviewScale = 1;
        this.detailModeScale = 0.5;
        this.maximumScale = 3;
        this.zoomStep = options && options.zoomStep ? options.zoomStep : 0.01;
        this.imageLocator = options.imageLocator;
        this.selectedPriceCategory = null;
    },

    get aspectRatio() { return this.seatMap.width / this.seatMap.height; },

    get isInDetailMode() { return this.viewMode == VIEWMODE_DETAIL },

    get isInOverviewMode() { return this.viewMode == VIEWMODE_OVERVIEW },

    getPriceCategory: function (priceCategoryID) {
        for (var i = 0; i < this.seatMap.priceCategories.length; i++) {
            var priceCategory = this.seatMap.priceCategories[i];
            if (priceCategory.id == priceCategoryID)
                return priceCategory;
        }
        return null;
    },

    getSection: function (sectionID) {
        for (var i = 0; i < this.seatMap.sections.length; i++) {
            var section = this.seatMap.sections[i];
            if (section.id == sectionID)
                return section;
        }
        return null;
    },

    getSectionByCode: function (sectionCode) {
        for (var i = 0; i < this.seatMap.sections.length; i++) {
            var section = this.seatMap.sections[i];
            if (section.code == sectionCode)
                return section;
        }
        return null;
    },

    getContingent: function (contingentID) {
        for (var i = 0; i < this.seatMap.contingents.length; i++) {
            var contingent = this.seatMap.contingents[i];
            if (contingent.id == contingentID)
                return contingent;
        }
        return null;
    },

    getImage: function(imageID) {
        for (var i = 0; i < this.seatMap.images.length; i++) {
            var image = this.seatMap.images[i];
            if (image.id == imageID) {
                if (this.imageLocator != undefined) {
                    return this.imageLocator(i, imageID);
                } else {
                    return image;
                }
            }
                
        }
        return null;
    }
}

// ticket information model
function ticketInformation(id, rowIdentifier, seatIdentifier, sectionId, sectionCode, sectionName, priceCategoryName, price) {
    this.id = id;
    this.rowIdentifier = rowIdentifier;
    this.seatIdentifier = seatIdentifier;
    this.sectionId = sectionId;
    this.sectionCode = sectionCode;
    this.sectionName = sectionName;
    this.priceCategoryName = priceCategoryName;
    this.price = price;
};
var SECTIONTYPE_GENERALADMISSION = 1;
var SECTIONTYPE_SEAT = 3;
var SECTIONTYPE_GROUP = 4;

function seatingMapController(model, ticketService, options, onreadyCallback) {
    this.init(model, ticketService, options, onreadyCallback);
}

seatingMapController.prototype = {
    init: function (seatMapData, ticketService, options, onreadyCallback) {
        var self = this;

        var finalizeBootstrap = function (seatMapData, ticketService, options, loadedImages) {

            var imageLocator = function (index, imageID) {
                return { htmlImage: loadedImages[index] };
            }

            self.model = new seatMapModel({ seatMapData: seatMapData, imageLocator: imageLocator });
            self.view = new seatingMapView(self, self.model, options);
            self.ticketService = ticketService;
            self.performanceModel = options && options.performanceModel ? options.performanceModel : null;
            self.goToOverviewMode();            
            self.view.draw();

            if (onreadyCallback !== undefined)
                onreadyCallback();
        }

        var imageCounter = 0;
        var imagesToLoad = new Array();
        if (seatMapData.images.length == 0)
            finalizeBootstrap(seatMapData, ticketService, options, imagesToLoad);
        else {
            for (var i = 0; i < seatMapData.images.length; i++) {            
                var image = new Image();
                imagesToLoad.push(image);
                image.onload = function (e) {
                    imageCounter++;
                    if (imageCounter >= seatMapData.images.length)
                        finalizeBootstrap(seatMapData, ticketService, options, imagesToLoad);
                };
                image.onerror = function (e) {
                    imageCounter++;
                    if (imageCounter >= seatMapData.images.length)
                        finalizeBootstrap(seatMapData, ticketService, options, imagesToLoad);
                }

                image.src = seatMapData.images[i].url;
            }
        }
    },

    ensureNotInFullScreenMode: function() {
        if (this.model.isInFullScreen) {
            this.switchToFullScreen();
        }
    },

    switchToFullScreen: function () {
        var container = this.view.getContainer();
        if (container) {
            if (this.model.isInFullScreen) {
                var $placeHolder = $("#seating-map-container-place-holder");
                $(container).insertAfter($placeHolder).removeClass("fullscreen");
                $placeHolder.remove();
                this.model.isInFullScreen = false;
            } else {
                $("<div id='seating-map-container-place-holder'></div>").insertAfter($(container));                
                $(container).appendTo(document.body).addClass("fullscreen");
                this.model.isInFullScreen = true;
            }
            this.view.draw();
        }
    },

    switchToZoomView: function (section) {
        if (this.view.onSectionSelected)
            this.view.onSectionSelected(section);
        if (section.type != SECTIONTYPE_GENERALADMISSION) {
            this.goToDetailMode(section);
            this.view.draw();
        }
    },

    goToSection: function (sectionCode) {
        var seatMap = this.model.seatMap;
        for (var i = 0; i < seatMap.sections.length; i++)
            if (sectionCode == seatMap.sections[i].code) {
                this.switchToZoomView(seatMap.sections[i]);
                break;
            }
    },

    zoomIn: function (position) {
        this.setZoom(this.model.scale + 0.1, position);
    },

    zoomOut: function (position) {
        this.setZoom(this.model.scale - 0.1, position);
    },

    setZoom: function (value, position) {
        var previousScale = this.model.scale;

        this.model.scale = value;
        if (this.model.scale > this.model.maximumScale)
            this.model.scale = this.model.maximumScale;
        else if (this.model.scale < this.model.overviewScale)
            this.model.scale = this.model.overviewScale;

        if (this.model.scale != previousScale) {
            if (position) {
                var newTanslate = { 
                    x: position.x - (position.canvasX / this.model.scale),
                    y: position.y - (position.canvasY / this.model.scale)
                }
                this.updateTranslate(newTanslate.x, newTanslate.y);
            }
            else
                this.updateTranslate(this.model.translate.x, this.model.translate.y);   // ensure that the translation coordinates are still valid

            if (this.model.scale <= this.model.overviewScale && this.model.isInDetailMode)
                this.goToOverviewMode();
            else if (this.model.scale >= this.model.detailModeScale && this.model.isInOverviewMode)
                this.goToDetailMode();

            if (this.view.scaleSlider && this.view.scaleSlider.slider)
                this.view.scaleSlider.slider("option", "value", this.model.scale);

            this.view.draw();
        }
    },

    updateTranslate: function (x, y, canvasSize) {
        this.model.translate.x = x;
        this.model.translate.y = y;
    },

    goToOverviewMode: function (canvasSize) {
        this.model.viewMode = this.model.seatMap.detailModeOnly ? VIEWMODE_DETAIL : VIEWMODE_OVERVIEW;
        this.model.scale = this.model.overviewScale;
        if (!canvasSize)
            canvasSize = this.view.getSize();
        this.updateTranslate(this.model.seatMap.initialCoord.x + (this.model.seatMap.width - canvasSize.width / this.model.scale) / 2, this.model.seatMap.initialCoord.y + (this.model.seatMap.height - canvasSize.height / this.model.scale) / 2, canvasSize);
    },

    goToDetailMode: function (section) {
        this.model.viewMode = VIEWMODE_DETAIL;
        if (section) {
            this.model.scale = section.initialScale > 0 ? section.initialScale : 1;
            this.updateTranslate(section.initialCoord.x, section.initialCoord.y);
        }
    },

    setOverviewScale: function (canvasSize) {
        var scaleX = canvasSize.width / this.model.seatMap.width;
        var scaleY = canvasSize.height / this.model.seatMap.height;

        this.model.overviewScale = Math.min(scaleX, scaleY);

        if (this.model.scale < this.model.overviewScale || this.model.isInOverviewMode)
            this.goToOverviewMode(canvasSize);
    },

    moveLeft: function () {
        var canvasSize = this.view.getSize();

        this.updateTranslate(this.model.translate.x - (canvasSize.width / 5) / this.model.scale, this.model.translate.y, canvasSize);
        this.view.draw();
    },
    moveRight: function () {
        var canvasSize = this.view.getSize();

        this.updateTranslate(this.model.translate.x + (canvasSize.width / 5) / this.model.scale, this.model.translate.y, canvasSize);
        this.view.draw();
    },
    moveUp: function () {
        var canvasSize = this.view.getSize();

        this.updateTranslate(this.model.translate.x, this.model.translate.y - (canvasSize.height / 5) / this.model.scale, canvasSize);
        this.view.draw();
    },
    moveDown: function () {
        var canvasSize = this.view.getSize();

        this.updateTranslate(this.model.translate.x, this.model.translate.y + (canvasSize.height / 5) / this.model.scale, canvasSize);
        this.view.draw();
    },

    getSelectedSeatsId: function () {
        var seats = [];

        for (var i = 0; i < this.view.seatStorage.storedAreas.length; i++) {
            var area = this.view.seatStorage.storedAreas[i];
            for (var j = 0; j < area.seats.length; j++) {
                var seat = area.seats[j];
                if (seat.selected)
                    seats.push(seat.id);
            }
        }
        for (var i = 0; i < this.model.seatMap.selectedSeats.length; i++)
            if (this.model.seatMap.selectedSeats[i].selected)
                seats.push(this.model.seatMap.selectedSeats[i].id);

        return seats;
    },

    getSelectedSeats: function () {
        var seats = [];

        for (var i = 0; i < this.view.seatStorage.storedAreas.length; i++) {
            var area = this.view.seatStorage.storedAreas[i];
            for (var j = 0; j < area.seats.length; j++) {
                var seat = area.seats[j];
                if (seat.selected) {
                    var ticketInfo = this.getTicketInfoFromSeat(seat);
                    if (ticketInfo)
                        seats.push(ticketInfo);
                }
            }
        }
        for (var i = 0; i < this.model.seatMap.selectedSeats.length; i++)
            if (this.model.seatMap.selectedSeats[i].selected) {
                var ticketInfo = this.getTicketInfoFromSeat(this.model.seatMap.selectedSeats[i]);
                if (ticketInfo)
                    seats.push(ticketInfo);
            }

        return seats;
    },

    getSelectedSeatsCounter: function () {
        var seats = 0;

        for (var i = 0; i < this.view.seatStorage.storedAreas.length; i++) {
            var area = this.view.seatStorage.storedAreas[i];
            for (var j = 0; j < area.seats.length; j++) {
                var seat = area.seats[j];
                if (seat.selected)
                    seats++;
            }
        }
        for (var i = 0; i < this.model.seatMap.selectedSeats.length; i++)
            if (this.model.seatMap.selectedSeats[i].selected)
                seats++;

        return seats;
    },

    removeSeat: function (seatID) {
        // first check if the seat is in the selectedSeats collection. If it is there, we just need to set it as not selected, the canvas is not affected.
        for (var i = 0; i < this.model.seatMap.selectedSeats.length; i++) {
            var seat = this.model.seatMap.selectedSeats[i];
            if (seat.id == seatID) {
                seat.selected = false;
                if (this.view.onSeatSelected)
                    this.view.onSeatSelected(this.getTicketInfoFromSeat(seat), seat.selected);
                return;
            }
        }
        // check for canvas seats
        for (var i = 0; i < this.view.seatStorage.storedAreas.length; i++) {
            var area = this.view.seatStorage.storedAreas[i];
            for (var j = 0; j < area.seats.length; j++) {
                var seat = area.seats[j];
                if (seat.id == seatID) {
                    seat.selected = false;
                    if (this.view.onSeatSelected)
                        this.view.onSeatSelected(this.getTicketInfoFromSeat(seat), seat.selected);
                    this.view.draw();
                    return;
                }
            }
        }

    },

    getTicketInfoFromSeat: function (seat) {
        var priceCategory = this.model.getPriceCategory(seat.priceCategoryID);
        var section;
        if (seat.sectionID > 0)
            section = this.model.getSection(seat.sectionID);
        else
            section = this.model.getSectionByCode(seat.sectionCode);
        if (priceCategory && section) {
            var price = this.performanceModel.getPrice(priceCategory.id, this.performanceModel.currencySymbol);
            var ticketInfo = new ticketInformation(seat.id, seat.rowIdentifier, seat.seatIdentifier, section.id, section.code, section.name, priceCategory.name, price);
            return ticketInfo;
        }
        return null;
    },

    refresh: function () {
        this.view.clearSeats();
        this.view.draw();
    },

    selectPriceCategory: function (priceCategory) {
        this.view.selectPriceCategory(priceCategory);
        this.view.draw();
    },

    getPriceCategoryRowStyle: function (priceCategory) {
        return { 'opacity': (this.model == null || this.model.selectedPriceCategory == null || this.model.selectedPriceCategory === priceCategory) ? '1' : '0.4' };
    }
};
var PI_TIMES_2 = 2 * Math.PI;
var pinchDistance = 0;

function seatingMapView(controller, model, options) {
    this.init(controller, model, options);
}

seatingMapView.prototype.init = function (controller, model, options) {
    var self = this;
    this.controller = controller;
    this.canvas = document.getElementById(options.host ? options.host : "canvas");
    this.context = this.canvas.getContext('2d');
    this.minimap = null;
    var miniMapCanvas = document.getElementById(options.minimap ? options.minimap : "minimap");
    if (miniMapCanvas != null) {
        this.miniMap = new miniMapController(miniMapCanvas, controller, model, this);
    }
    this.notAvailableSeatColor = options.notAvailableSeatColor ? options.notAvailableSeatColor : '#c0c0c0';
    this.availableText = options.availableText ? options.availableText : 'Available';
    this.soldText = options.soldText ? options.soldText : 'Sold';
    this.reservedText = options.reservedText ? options.reservedText : 'Reserved';
    this.logLevel = options.logLevel ? options.logLevel : 0;
    this.useSeatSprite = options.useSeatSprite ? options.useSeatSprite : true;
    this.performanceModel = options.performanceModel ? options.performanceModel : null;
    this.model = model;
    this.seatStorage = new seatStorage(controller, this, model);
    this.drawer = new drawingObject({ context: this.context, model: this.model });
    this.controller.setOverviewScale(this.getSize());

    if (options.onDraw) {
        this.onDraw = options.onDraw;
    }
    if (options.onMouseMove) {
        this.onMouseMoveEventHandler = options.onMouseMove;
    }
    if (options.onSectionSelected)
        this.onSectionSelected = options.onSectionSelected;
    if (options.onSeatSelected)
        this.onSeatSelected = options.onSeatSelected;

    this.scaleSlider = $('#' + (options.scale ? options.scale : 'scale'));
    if (this.scaleSlider != null && this.scaleSlider.slider) {
        this.scaleSlider.slider({
            orientation: 'vertical',
            range: 'min',
            min: model.overviewScale,
            max: model.maximumScale,
            step: 0.1,
            value: model.overviewScale,
            slide: function (event, ui) { self.onScaleSlide(event, ui); }
        });
    }
    this.tooltip = $(document.getElementById(options.tooltip ? options.tooltip : 'tooltip'));
    this.tooltip.hide();

    $('.seatmap2-reset').off('click').on('click', function () {
        self.controller.goToOverviewMode();
        self.draw();
        return false;
    });
    $('.seatmap2-moveLeft').off('click').on('click', function () {
        self.controller.moveLeft();
        self.draw();
        return false;
    });
    $('.seatmap2-moveRight').off('click').on('click', function () {
        self.controller.moveRight();
        self.draw();
        return false;
    });
    $('.seatmap2-moveUp').off('click').on('click', function () {
        self.controller.moveUp();
        self.draw();
        return false;
    });
    $('.seatmap2-moveDown').off('click').on('click', function () {
        self.controller.moveDown();
        self.draw();
        return false;
    });
    $('.seatmap2-refresh').off('click').on('click', function () {
        self.controller.refresh();
        return false;
    });
    $('.seatmap2-full-screen-switch').off('click').on('click', function () {
        self.controller.switchToFullScreen();
        self.draw();
    });
    $('.seatmap2-zoom-in').off('click').on('click', function () {
        self.controller.zoomIn(self.centerViewPositionObj());
        self.draw();
    });
    $('.seatmap2-zoom-out').off('click').on('click', function () {
        self.controller.zoomOut(self.centerViewPositionObj());
        self.draw();
    });

    this.seatSprites = null;
    this.lastMousePosition = null;
    this.isDragging = false;
    this.singleTouchStartPoint = null;
    this.singleTouchHadMovements = false;
    this.singleTouchTranslationOffset = null;
    this.currentCirclePreset = null;

    var $canvas = $(this.canvas);

    // bind canvas html events
    $canvas.off('selectstart').on('selectstart', function () { return false; });
    $canvas.off('mouseup').on('mouseup', function (e) { self.onMouseUp(e.originalEvent); });
    $canvas.off('mousedown').on('mousedown', function (e) { self.onMouseDown(e.originalEvent); });
    $canvas.off('mousemouve').on('mousemove', function (e) { self.onMouseMove(e.originalEvent); });
    $canvas.off('click').on('click', function (e) { self.onCanvasClick(e.originalEvent); })
    $canvas.off('dblclick').on('dblclick', function (e) { self.onCanvasDoubleClick(e.originalEvent); });
    $canvas.off('touchstart').on('touchstart', function (e) { self.handleTouchStart(e.originalEvent); });
    $canvas.off('touchend').on('touchend', function (e) { self.handleTouchEnd(e.originalEvent); });
    $canvas.off('touchleave').on('touchleave', function (e) { self.handleTouchLeave(e.originalEvent); });
    $canvas.off('touchmove').on('touchmove', function (e) { self.handleTouchMove(e.originalEvent); });
    $canvas.off('touchcancel').on('touchcancel', function (e) { self.handleTouchCancel(e.originalEvent); });
    $canvas.off('gesturestart').on('gesturestart', function (e) { self.handleGestureStart(e.originalEvent); });
    $canvas.off('gesturechange').on('gesturechange', function (e) { self.handleGestureChange(e.originalEvent); });
    $canvas.off('gestureend').on('gestureend', function (e) { self.handleGestureEnd(e.originalEvent); }); 
    $canvas.bind('mousewheel DOMMouseScroll', function (e) { self.handleMouseWheel(e.originalEvent); });
    window.addEventListener('resize', function (e) { self.canvasResized(e); });
};

seatingMapView.prototype.centerViewPositionObj = function () {
    var seatMapX = parseInt((this.canvas.width  / 2) / this.model.scale + this.model.translate.x);
    var seatMapY = parseInt(this.canvas.height  / 2 / this.model.scale + this.model.translate.y);
    return { x:  seatMapX, y:  seatMapY, canvasX: parseInt(this.canvas.width / 2), canvasY: parseInt(this.canvas.height / 2)};
};

// Gets the container element. Required to switch to fullscreen view
seatingMapView.prototype.getContainer = function () {
    var parent = this.canvas.parentElement;
    while (parent !== undefined && parent != null) {
        if ($(parent).hasClass('canvas-blueprint-container'))
            return parent;

        parent = parent.parentElement;
    }

    return null;
};

seatingMapView.prototype.getSize = function () {
    this.canvas.width = this.canvas.clientWidth;
    this.canvas.height = this.canvas.clientHeight;
    return { width: this.canvas.width, height: this.canvas.height };
};

// clears the cached seats
seatingMapView.prototype.clearSeats = function () {
    this.seatStorage.clear();
};

seatingMapView.prototype.getSeatFromPoint = function (mouse) {
    for (var i = 0; i < this.seatStorage.storedAreas.length; i++) {
        var area = this.seatStorage.storedAreas[i];
        if (area.isPointInArea(mouse)) {
            for (var j = 0; j < area.seats.length; j++) {
                var seat = this.seatStorage.storedAreas[i].seats[j];
                var seatWidth = seat.seatType.width;
                var seatHeight = seat.seatType.height;
                var shapeLeft = seat.x - (seatWidth / 2);
                var shapeTop = seat.y - (seatHeight / 2);

                // check for hover by comparing the mouseX and mouseY to the current shape's x, y, width, and height properties
                if (mouse.x >= shapeLeft && mouse.x <= shapeLeft + seatWidth && mouse.y >= shapeTop && mouse.y <= shapeTop + seatHeight) {
                    if (this.logLevel >= 1) {
                        console.log('Hit test: ' + seat.name);
                    }
                    return seat;
                }
            }
            break;
        }
    }

    if (this.logLevel >= 1) {
        console.log('Hit test: no matches');
    }

    return null;
};

seatingMapView.prototype.logAction = function (category, log) {
    console.log(category + ': ' + log);
};

seatingMapView.prototype.handleTouchStart = function (event) {
    if (event.touches.length > 1) {
        this.pinchDistance = Math.sqrt(Math.pow((event.touches[1].pageX - event.touches[0].pageX), 2) + Math.pow((event.touches[1].pageY - event.touches[0].pageY), 2));
        this.gestureStartScale = this.model.scale;
    } else {
        this.pinchDistance = 0;
        this.singleTouchStartPoint = this.getTouchPosition(event);
        this.singleTouchHadMovements = false;

        if (this.singleTouchStartPoint != null) {
            var seat = null;

            if (this.model.isInDetailMode) {
                seat = this.getSeatFromPoint(this.singleTouchStartPoint);
                this.showSeatTooltip(this.singleTouchStartPoint, seat);
            }

            if (seat == null) {
                var info = this.getInfoFromPoint(this.singleTouchStartPoint);
                this.showInfoTooltip(this.singleTouchStartPoint, info);
            }
        }
    }

    event.preventDefault();
    if (this.logLevel >= 2) {
        this.logAction('touch start', event.targetTouches.length + ' touches found');
    }
};

seatingMapView.prototype.handleTouchEnd = function (event) {
    if (this.pinchDistance <= 0) {
        if (!this.singleTouchHadMovements && (this.singleTouchStartPoint != null)) {
            if (this.model.isInDetailMode) {
                // select seat
                var seat = this.getSeatFromPoint(this.singleTouchStartPoint);
                if (seat != null && seat.status == 1 && seat.hasContingentPermission) {
                    seat.selected = !seat.selected;
                    if (this.onSeatSelected)
                        this.onSeatSelected(this.controller.getTicketInfoFromSeat(seat), seat.selected);
                    this.draw();
                }
            }
            else {
                var section = this.getSectionFromPoint(this.singleTouchStartPoint.x, this.singleTouchStartPoint.y);
                if (section != null)
                    this.controller.switchToZoomView(section);
            }
        }

        this.singleTouchStartPoint = null;
        this.singleTouchHadMovements = false;
    }
    else {
        this.pinchDistance = 0;
    }

    event.preventDefault();
    if (this.logLevel >= 2) {
        this.logAction('touch end', event.targetTouches.length + ' touches found');
    }
};

seatingMapView.prototype.handleTouchLeave = function (event) {
    this.singleTouchStartPoint = null;
    this.singleTouchHadMovements = false;
    this.pinchDistance = 0;

    event.preventDefault();
    if (this.logLevel >= 2) {
        this.logAction('touch leave', event.targetTouches.length + ' touches found');
    }
};

seatingMapView.prototype.handleTouchMove = function (event) {
    if (this.logLevel >= 1) {
        this.logAction('handleTouchMove', 'touchMoveCounter = ' + this.touchMoveCounter);
    }

    if (this.pinchDistance <= 0) {
        event.preventDefault();

        var position = this.getTouchPosition(event);
        if ((this.singleTouchStartPoint != null) && (position != null) && !this.isGesture) {
            var deltaX = position.x - this.singleTouchStartPoint.x;
            var deltaY = position.y - this.singleTouchStartPoint.y;

            if (!this.singleTouchHadMovements && (Math.abs(deltaX) > 5 || Math.abs(deltaY) > 5))
                this.singleTouchHadMovements = true;
            if (this.singleTouchHadMovements) {
                if (this.logLevel >= 1) {
                    this.logAction('handleTouchMove', 'translations before: x:' + this.model.translate.x + ', y: ' + this.model.translate.y);
                }
                this.controller.updateTranslate(this.model.translate.x - deltaX, this.model.translate.y - deltaY);
                this.draw();
                this.singleTouchStartPoint = this.getTouchPosition(event);
                if (this.logLevel >= 1) {
                    this.logAction('handleTouchMove', 'translations after: x:' + this.model.translate.x + ', y: ' + this.model.translate.y);
                }
            }
        }

        if (position != null && this.onMouseMoveEventHandler) {
            this.onMouseMoveEventHandler(position);
        }
    }
    else {
        var newDistance = Math.sqrt(Math.pow((event.touches[1].pageX - event.touches[0].pageX), 2) + Math.pow((event.touches[1].pageY - event.touches[0].pageY), 2));
        this.controller.setZoom(this.gestureStartScale * (newDistance / this.pinchDistance), this.singleTouchStartPoint);
        event.preventDefault();
    }

    if (this.logLevel >= 2) {
        if (event.targetTouches.length == 1) {
            this.logAction('1 touch move', 'x:' + event.targetTouches[0].pageX + ', y: ' + event.targetTouches[0].pageY);
        } else {
            this.logAction('touch move', event.targetTouches.length + ' touches found');
        }
    }
};

seatingMapView.prototype.handleTouchCancel = function (event) {
    event.preventDefault();

    this.singleTouchStartPoint = null;
    this.singleTouchHadMovements = false;

    if (this.logLevel >= 2) {
        logAction('touch cancel', event.targetTouches.length + ' touches found');
    }
};

seatingMapView.prototype.handleMouseWheel = function (event) {
    var wheelDistance = function (evt) {
        if (!evt) evt = event;
        var w = evt.wheelDelta, d = evt.detail;
        if (d) {
            if (w) return w / d / 40 * d > 0 ? 1 : -1; // Opera
            else return -d / 3;              // Firefox;         TODO: do not /3 for OS X
        } else return w / 120;             // IE/Safari/Chrome TODO: /3 for Chrome OS X
    };

    var wheelDirection = function (evt) {
        if (!evt) evt = event;
        return (evt.detail < 0) ? 1 : (evt.wheelDelta > 0) ? 1 : -1;
    };

    var mouse = this.getMousePosition(event);

    if (wheelDistance(event) > 0)
        this.controller.zoomIn(mouse);
    else
        this.controller.zoomOut(mouse);

    event.preventDefault();

    return false;
};

seatingMapView.prototype.handleGestureStart = function (event) {
    this.gestureStartScale = this.model.scale;
    this.isGesture = true;
    event.preventDefault();
};

seatingMapView.prototype.handleGestureEnd = function (event) {
    this.isGesture = false;
    event.preventDefault();
};

seatingMapView.prototype.handleGestureChange = function (event) {
    this.controller.setZoom(this.gestureStartScale * event.scale);
    event.preventDefault();
};

seatingMapView.prototype.getTouchPosition = function (event) {   
    var mouseX = 0;
    var mouseY = 0;

    if (event.targetTouches.length == 1) {
        mouseX = event.targetTouches[0].clientX;
        mouseY = event.targetTouches[0].clientY;
    }
    else if (event.changedTouches.length == 1) {
        mouseX = event.changedTouches[0].clientX;
        mouseY = event.changedTouches[0].clientX;
    }
    else
        return null;

    var canvasPosition = this.canvas.getBoundingClientRect();

    mouseX = mouseX - canvasPosition.left - this.canvas.clientLeft;
    mouseY = mouseY - canvasPosition.top - this.canvas.clientTop;

    var seatMapX = parseInt(mouseX / this.model.scale + this.model.translate.x);
    var seatMapY = parseInt(mouseY / this.model.scale + this.model.translate.y);

    return { x: seatMapX, y: seatMapY, canvasX: parseInt(mouseX), canvasY: parseInt(mouseY) };
};

seatingMapView.prototype.getMousePosition = function (event, useCenterPosition) {
    var mouseX = new Number();
    var mouseY = new Number();

    if (useCenterPosition) {
        mouseX = event.target.clientWidth / 2;
        mouseY = event.target.clientHeight / 2;
    } else if (event.offsetX != undefined && event.offsetY != undefined) {
        mouseX = event.offsetX;
        mouseY = event.offsetY;
    }

    var seatMapX = parseInt(mouseX / this.model.scale + this.model.translate.x);
    var seatMapY = parseInt(mouseY / this.model.scale + this.model.translate.y);

    return { x: seatMapX, y: seatMapY, canvasX: parseInt(mouseX), canvasY: parseInt(mouseY) };
};

seatingMapView.prototype.getSectionFromPoint = function (x, y) {
    var seatMap = this.model.seatMap;

    for (var i = 0; i < seatMap.drawingObjects.length; i++) {
        var sectionID = this.drawer.isPointInSection(seatMap.drawingObjects[i], x, y, this.model.isInDetailMode);
        if (sectionID != null) {
            var section = null;
            for (var j = 0; j < seatMap.sections.length; j++) {
                section = seatMap.sections[j];
                if (section.id == sectionID) {
                    return section;
                }
            }
        }
    }

    return null;
};

seatingMapView.prototype.getInfoFromPoint = function (point) {
    var seatMap = this.model.seatMap;

    for (var i = 0; i < seatMap.drawingObjects.length; i++) {
        var info = this.drawer.getPositionInfo(seatMap.drawingObjects[i], point.x, point.y, this.model.isInDetailMode);
        if (info && info.hasInfo)
            return info;
    }

    return null;
};

seatingMapView.prototype.onCanvasClick = function (event) {
    var mouse = this.getMousePosition(event);

    if (this.model.isInDetailMode) {
        if (!this.isDragging) {
            var seat = this.getSeatFromPoint(mouse);
            if (seat != null) {
                if (seat.status == 1 && seat.hasContingentPermission) {
                    seat.selected = !seat.selected;
                    if (this.onSeatSelected)
                        this.onSeatSelected(this.controller.getTicketInfoFromSeat(seat), seat.selected);
                    this.draw();
                }
            } else {
                if (!this.isDragging) {
                    var section = this.getSectionFromPoint(mouse.x, mouse.y);
                    if (section != null && section.type === 'GeneralAdmission')
                        this.controller.switchToZoomView(section);
                }
            }
        }
    }
    else {
        if (!this.isDragging) {
            var section = this.getSectionFromPoint(mouse.x, mouse.y);
            if (section != null)
                this.controller.switchToZoomView(section);
        }
    }
};

seatingMapView.prototype.onCanvasDoubleClick = function (event) {
    var mouse = this.getMousePosition(event);
    var section = this.getSectionFromPoint(mouse.x, mouse.y);

    var mouseCoordsZoom = () => 
        this.controller.setZoom(this.model.scale + 0.3, mouse);

    /* When the section coords are not imported the initialCoord value will be y === x === 0
    coords are not valide to zoom on the selected section, so we will zoom normally
    */
    if (section !== null)
    {
        var hasValidCoord = 
        section.initialCoord.x === 0
            ? section.initialCoord.y === 0
                ? false
                : true
            : true;

        if (hasValidCoord)
            this.controller.switchToZoomView(section);
        else
            mouseCoordsZoom();
    } else
        mouseCoordsZoom()
};

seatingMapView.prototype.onMouseDown = function (event) {
    var mouse = this.getMousePosition(event);
    this.lastMousePosition = mouse;
    this.isDragging = false;
};

seatingMapView.prototype.onMouseUp = function (event) {
    this.lastMousePosition = null;
};

seatingMapView.prototype.onMouseMove = function (event) {
    var mouse = this.getMousePosition(event);
    var seat = null;

    if (this.model.isInDetailMode) {
        seat = this.getSeatFromPoint(mouse);
        this.showSeatTooltip(mouse, seat);
    }

    if (seat == null) {
        var info = this.getInfoFromPoint(mouse);
        this.showInfoTooltip(mouse, info);
    }

    if (event.which == 1) {
        if (this.lastMousePosition != null) {
            var deltaX = mouse.x - this.lastMousePosition.x;
            var deltaY = mouse.y - this.lastMousePosition.y;
            if (!this.isDragging && (Math.abs(deltaX) > 5 || Math.abs(deltaY) > 5))
                this.isDragging = true;
            if (this.isDragging) {
                this.controller.updateTranslate(this.model.translate.x - deltaX, this.model.translate.y - deltaY);
                this.draw();
                this.lastMousePosition = this.getMousePosition(event);
                if (this.logLevel >= 1) {
                    this.logAction('onMouseMove', 'translations after: x:' + this.model.translate.x + ', y: ' + this.model.translate.y);
                }
            }
        }
    }
    if (this.onMouseMoveEventHandler) {
        this.onMouseMoveEventHandler(mouse);
    }
};

seatingMapView.prototype.canvasResized = function (event) {
    this.controller.setOverviewScale(this.getSize());
    this.draw();
};

seatingMapView.prototype.onScaleSlide = function (event, ui) {
    this.controller.setZoom(ui.value);
};

seatingMapView.prototype.onSeatsLoaded = function (area) {
    if (!this.model.isInDetailMode)
        return;

    if (area != null) {
        for (var i = 0; i < area.seats.length; i++) {
            var seat = area.seats[i];
            this.drawSeatSprite(seat);
        }
    }
    else {
        for (var i = 0; i < this.seatStorage.seats.length; i++) {
            var seat = this.seatStorage.seats[i];
            this.drawSeatSprite(seat);
        }
    }
};

seatingMapView.prototype.draw = function () {
    if (this.miniMap)
        this.miniMap.draw();

    this.context.clearRect(0, 0, this.canvas.width, this.canvas.height);
    this.context.save();

    this.context.scale(this.model.scale, this.model.scale);

    if (this.model.translate)
        this.context.translate(this.model.translate.x ? -this.model.translate.x : 0, this.model.translate.y ? -this.model.translate.y : 0);

    var seatMap = this.model.seatMap;

    for (var i = 0; i < seatMap.drawingObjects.length; i++) {
        var drawingObject = seatMap.drawingObjects[i];
        if ((this.model.isInDetailMode && drawingObject.showInDetailView) ||
            (this.model.isInOverviewMode && drawingObject.showInOverview)) {
            this.drawer.draw(drawingObject, this.model.isInDetailMode);
        }
    }

    this.context.restore();

    // draw seats
    if (this.model.isInDetailMode) {
        var x1 = this.model.translate.x;
        var y1 = this.model.translate.y;
        var x2 = x1 + Math.round(this.canvas.width / this.model.scale);
        var y2 = y1 + Math.round(this.canvas.height / this.model.scale);

        var areas = this.seatStorage.getSeats(x1, y1, x2, y2);
        for (var i = 0; i < areas.length; i++)
            this.onSeatsLoaded(areas[i]);
    }

    if (this.onDraw)
        this.onDraw(this.model);
};

seatingMapView.prototype.drawSeatSprite = function (seat) {
    var spriteMargin = 1;
    var seatSpriteKey = this.model.scale + ';' + seat.seatTypeID + ';';
    if (seat.contingentID) {
        seatSpriteKey += ';' + seat.contingentID;
    }
    if (seat.status == 2 || seat.status == 3) {
        seatSpriteKey += seat.status; // not available seat
    } else {
        seatSpriteKey += seat.colorCode + ';';
        if (seat.selected) {
            seatSpriteKey += '1';
        }
        else {
            seatSpriteKey += '0';
        }
    }

    seatSpriteKey += ';' + (seat.isSelectedPriceCategory === false ? '0' : '1');

    var seatMap = this.model.seatMap;
    if (!seat.seatType) {
        for (var i = 0; i < seatMap.seatTypes.length; i++)
            if (seatMap.seatTypes[i].id == seat.seatTypeID) {
                var seatType = seatMap.seatTypes[i];
                // if the seat type size is not defined in the db, assume the default 15x15 size.
                if (seatType.width == 0)
                    seatType.width = 15;
                if (seatType.height == 0)
                    seatType.height = 15;
                seat.seatType = seatType;
                break;
            }
    }

    if (!this.seatSprites)
        this.seatSprites = [];

    var seatSpriteDefinition = this.seatSprites[seatSpriteKey];
    if (seatSpriteDefinition == null) {
        seatSpriteDefinition = { sprite: null };
        var sprite = document.createElement('canvas');
        // rounding up the canvas height and width (integers) for small resolutions
        // width and height below 1 were rounded down to 0 causing the error "Failed to execute 'drawImage' on 'CanvasRenderingContext2D': The image argument is a canvas element with a width or height of 0."
        sprite.height = Math.ceil((seat.seatType.height + spriteMargin) * this.model.scale);
        sprite.width = Math.ceil((seat.seatType.width + spriteMargin) * this.model.scale);
        var seatSpriteContext = sprite.getContext('2d');
        seatSpriteContext.scale(this.model.scale, this.model.scale);
        seatSpriteDefinition.sprite = sprite;

        if (seat.seatType.drawingObjects.length > 0) {
            seatSpriteContext.translate(spriteMargin / 2, spriteMargin / 2);
            var self = this;
            var spriteDrawer = new drawingObject({
                context: seatSpriteContext,
                model: this.model,
                getColor: function () { return self.getSeatColor(seat) },
                getLineWidth: function (lineWidth) { return seat.selected ? lineWidth + 0 : lineWidth; }
            });
            for (var i = 0; i < seat.seatType.drawingObjects.length; i++) {
                spriteDrawer.draw(seat.seatType.drawingObjects[i], true);
            }
        }
        else {
            // draw seat        
            var drawStroke = false;
            if (seat.isSelectedPriceCategory === undefined || seat.isSelectedPriceCategory) {
                seatSpriteContext.fillStyle = this.getSeatColor(seat);

                // draw hardcoded black circle around selected seat
                if (seat.selected) {
                    seatSpriteContext.strokeStyle = '#000000';
                    seatSpriteContext.lineWidth = 2;
                    drawStroke = true;
                }
                else if (seat.contingentID && seat.contingentID > 0) {
                    var contingent = this.model.getContingent(seat.contingentID);
                    if (contingent && contingent.colorCode != null && contingent.informationPublic) {
                        seatSpriteContext.strokeStyle = contingent.colorCode;
                        seatSpriteContext.lineWidth = 2;
                        drawStroke = true;
                    }
                }
            } else {
                seatSpriteContext.fillStyle = '#ddd';
            }

            seatSpriteContext.beginPath();
            var centerCoordinate = (seat.seatType.width + spriteMargin) / 2;
            seatSpriteContext.arc(centerCoordinate, centerCoordinate, (seat.seatType.width - seatSpriteContext.lineWidth) / 2, 0, PI_TIMES_2);
            seatSpriteContext.closePath();
            seatSpriteContext.fill();
            if (drawStroke) {
                seatSpriteContext.stroke();
            }
        }

        this.seatSprites[seatSpriteKey] = seatSpriteDefinition;
    }

    // finally draw the image
    var drawX = ((seat.x - this.model.translate.x) * this.model.scale);
    var drawY = ((seat.y - this.model.translate.y) * this.model.scale);
    var rotation = seat.rotation * Math.PI / 180;
    this.context.translate(drawX, drawY);
    this.context.rotate(rotation);
    this.context.drawImage(seatSpriteDefinition.sprite, -(seatSpriteDefinition.sprite.width / 2), -(seatSpriteDefinition.sprite.height / 2));
    this.context.rotate(-rotation);
    this.context.translate(-drawX, -drawY);
};

seatingMapView.prototype.getSeatColor = function (seat) {
    var seatMap = this.model.seatMap;
    var seatColor = '';
    if (seat.status == 1 && seat.hasContingentPermission)
        seatColor = seat.selected ? seatMap.selectedSeatColor : seat.colorCode
    else if (!seat.hasContingentPermission) {
        if (seat.contingentID) {
            var contingent = this.model.getContingent(seat.contingentID);
            if (contingent)
                seatColor = contingent.colorCode;
        }
        if (seatColor == null || seatColor == '')
            seatColor = seatMap.seatStatusColor[3];
    }
    else
        seatColor = seatMap.seatStatusColor[seat.status - 1];

    return seatColor;
};

seatingMapView.prototype.drawSeat = function (seat) {
    if (!currentCirclePreset)
        currentCirclePreset = { strokeStyle: null, lineWidth: null, fillStyle: null };

    var originalLineWidth = this.context.lineWidth;

    var newStrokeStyle = 'cornflowerblue';
    var newLineWidth = null;
    var newFillStyle = null;

    if (seat.selected) {
        newStrokeStyle = 'black';
        newLineWidth = 4;
    } else {
        this.context.strokeStyle = 'cornflowerblue';
        newLineWidth = 1;
    }

    var priceCategory = this.model.getPriceCategory(seat.cat);

    if (seat.status == 1) {
        newFillStyle = priceCategory.colorCode;
    }
    else {
        newFillStyle = configuration.notAvailableSeatColor;
    }

    if (!currentCirclePreset.fillStyle || currentCirclePreset.fillStyle != newFillStyle) {
        currentCirclePreset.fillStyle = this.context.fillStyle = newFillStyle;
    }

    if (!currentCirclePreset.strokeStyle || currentCirclePreset.strokeStyle != newStrokeStyle) {
        currentCirclePreset.strokeStyle = this.context.strokeStyle = newStrokeStyle;
    }

    if (!currentCirclePreset.lineWidth || currentCirclePreset.lineWidth != newLineWidth) {
        currentCirclePreset.lineWidth = this.context.lineWidth = newLineWidth;
    }

    this.context.beginPath();
    this.context.arc(seat.x, seat.y, configuration.seatRadius, 0, PI_TIMES_2);
    this.context.stroke();
    this.context.fill();
};

seatingMapView.prototype.showSeatTooltip = function (position, seat) {
    if (seat) {
        var priceCategory = this.model.getPriceCategory(seat.priceCategoryID);
        var separator = '';
        var text = '<div class="tooltip-seat-row">' + ticketportal.webshop.localization.seatRow + ' ' + seat.rowIdentifier + '</div><div class="tooltip-seat-identifier">' + ticketportal.webshop.localization.seatIdentifier + ' ' + seat.seatIdentifier + '</div>';
        if (priceCategory) {
            text += separator + priceCategory.name;
            separator = '<br />';
        }
        if (this.performanceModel) {
            var price = this.performanceModel.getPrice(seat.priceCategoryID, this.performanceModel.currencySymbol);
            if (price) {
                text += separator + price.formattedPrice;
                separator = '<br />';
            }
        }
        if (seat.contingentID) {
            var contingent = this.model.getContingent(seat.contingentID);
            if (contingent && contingent.informationPublic) {
                text += separator + "<span class='tooltip-contingent-'" + contingent.id + "'>" + contingent.name + "</span>";
                separator = '<br />';
            }
        }
        var status = seat.status;
        if (!seat.hasContingentPermission)
            status = 4;
        var statusName = this.model.seatMap.seatStatusName[status - 1];
        text += separator + "<span class='tooltip-seat-status tooltip-seat-status-" + seat.status + "'>" + statusName + "</span>";
        separator = '<br />';
        this.showTooltip(position, text);
    }
    else {
        this.clearTooltip();
    }
};

seatingMapView.prototype.showInfoTooltip = function (position, info) {
    if (info && info.hasInfo) {
        var text = '';
        if (info.sectionID) {
            for (var i = 0; i < this.model.seatMap.sections.length; i++) {
                section = this.model.seatMap.sections[i];
                if (section.id == info.sectionID) {
                    text += '<div class="seatmap-tooltip-section">' + section.name + '</div>';
                    break;
                }
            }
        }
        if (info.tooltip)
            text += '<div class="seatmap-tooltip-text">' + info.tooltip + '</div>';
        if (info.imageIDTooltip) {
            var image = this.model.getImage(info.imageIDTooltip);
            text += '<img class="seatmap-tooltip-image" src="' + image.url + '" />';
        }

        this.showTooltip(position, text);
    }
    else {
        this.clearTooltip();
    }
};

seatingMapView.prototype.showTooltip = function (position, text, title) {
    if (text.length <= 0) {
        this.clearTooltip();
        return;
    }
    var html;
    if (title && title.length > 0)
        html = "<div>" + title + "</div>" + text;
    else
        html = text;
    this.tooltip.show();
    this.tooltip.html(html);
    this.tooltip.css({ left: position.canvasX - (this.tooltip.outerWidth(true) / 2) + 15, top: position.canvasY - this.tooltip.outerHeight(true) - 15 });
};

seatingMapView.prototype.clearTooltip = function () {
    this.tooltip.hide();
};

seatingMapView.prototype.selectPriceCategory = function (priceCategory) {
    if (this.model.selectedPriceCategory === priceCategory) {
        priceCategory = null;
    }

    this.model.selectedPriceCategory = priceCategory;
    this.updatePriceCategoryOnSeats();
};

seatingMapView.prototype.updatePriceCategoryOnSeats = function () {
    for (var i = 0; i < this.seatStorage.storedAreas.length; i++) {
        var area = this.seatStorage.storedAreas[i];
        for (var j = 0; j < area.seats.length; j++) {
            var seat = area.seats[j];
            seat.isSelectedPriceCategory = this.model.selectedPriceCategory == null || seat.priceCategoryID.toString() === this.model.selectedPriceCategory.priceCategoryUniqueId;
            this.drawSeatSprite(seat);
        }
    }
};;
"use strict";

var SeatPicker = function (options) {
    const enums = {
        // Axis will give us strict values for X and Y.
        "Axis": new Enum(["X", "Y"]),
        // Different styles of seat that can be drawn when rendering.
        "SeatIcon": new Enum(["FillCircle", "StrokeCircle", "StrokeCircleWithCross", "StrokeCircleWithSlash", "StrokeCircleWithSpot", "StrokeCircleWithTick"]),
        // Named shapes that are used to describe some plan items.
        "Shape": new Enum(["Basic", "Line", "Oval"])
    };
    const events = {
        "click": "seatPickerClick",
        "renderedFirstFrame": "renderedFirstFrame",
        "mousemove": "seatPickerMouseMove",
        "selectionChanged": "seatPickerSelectionChanged",
        "tap": "seatPickerTap"
    };
    const unassignedCategoryLabel = "default_unassigned";
    var canvas;
    var canvasControls;
    var canvasProperties;
    var categories;
    var colourPalette;
    var hasRenderedFirstFrame;
    var hoveredBlock;
    var hoveredSeat;
    var layout;
    var labelProperties;
    var marqueeSeats;
    var mouse;
    var renderInterval;
    var seatStyles;
    var selectedSeats;
    var tooltip;
    var touch;
    var helpers = {
        // Linear interpolation function for following the line between two points a and b, in step 
        // sizes t, where t ranges from 0 to 1.
        lerp: (a, b, t) => {
            return a + t * (b - a);
        }
    }; // Initialise the requestAnimationFrame method for the current browser.

    requestAnimationFrame = window.requestAnimationFrame || window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || window.msRequestAnimationFrame || setTimeout;

    var initialise = () => {
        initialiseProperties();
        initialiseEvents();
        initialiseLayout();
        initialiseCategories();
        initialiseCanvas();
        startRenderLoop();
    };

    var addUpdateCategory = (changedCategory, isSelectable) => {
        // A category needs an array of layout seats to work properly, so create one if necessary.
        if (changedCategory.seats == null) {
            changedCategory.seats = []; // Find matching seats in the layout json, and add the layout seat to our array for this category.

            changedCategory.seatIdentifiers.forEach(seatIdentifier => {
                let seat = getSeat(seatIdentifier);

                if (seat) {
                    changedCategory.seats.push(seat);
                }
            });
        } // Set the selectability of the seats in this category.


        if (changedCategory.seats != null) {
            changedCategory.seats.forEach(seat => {
                seat.isSelectable = isSelectable != null ? isSelectable : true;
            });
        } // Go through any existing categories and remove any seats that have been set in the changed categories.


        categories.forEach(category => {
            // We don't need to check for the category we're changing, or the unsassigned category.
            if (category.label === changedCategory.label || category.label === unassignedCategoryLabel) {
                return;
            }

            category.seats = category.seats.filter(seat => {
                return !changedCategory.seats.includes(seat);
            });
        }); // Add or replace the changed categories.

        var existingCategoryIndex = categories.findIndex(category => category.label === changedCategory.label);

        if (existingCategoryIndex !== -1) {
            categories[existingCategoryIndex] = changedCategory;
        } else {
            categories.push(changedCategory);
        } // Update the seat identifiers array to match the current array of seats.


        changedCategory.seatIdentifiers = changedCategory.seats.flatMap(seat => seat.identifiers);
        updateUnassignedSeats();
    };

    var animateZoom = () => {
        let hasAnimated = false;
        let offset = canvasProperties.offset;
        let targetOffset = canvasProperties.targetOffset || {
            x: offset.x,
            y: offset.y
        };
        let targetZoom = canvasProperties.targetZoom;
        let zoom = canvasProperties.zoom;

        if (!!targetZoom && zoom !== targetZoom) {
            let zoomDiff = targetZoom - canvasProperties.zoom;

            if (Math.abs(zoomDiff) < 0.001) {
                canvasProperties.zoom = canvasProperties.targetZoom;
                canvasProperties.zoomOffset = getZoomOffset(canvasProperties.zoom);
                canvasProperties.targetZoom = null;
            } else {
                let zoomDelta = zoomDiff / 2;
                doZoom(zoomDelta);
                canvasProperties.zoomOffset = getZoomOffset(canvasProperties.zoom);
            }

            hasAnimated = true;
        } // Prevent the seating plan pulling away from the edges.


        if (hasAnimated && canvasProperties.zoom != canvasProperties.minimumZoom) {
            let defaultOffset = canvasProperties.defaultOffset;
            let planBounds = translateToCanvasBounds(getPlanBounds(true));

            if (planBounds.right - planBounds.left < canvas.width) {
                targetOffset.x = defaultOffset.x;
            } else if (planBounds.left > 0) {
                targetOffset.x = offset.x - planBounds.left;
            } else if (planBounds.right < canvas.width) {
                targetOffset.x = offset.x - (planBounds.right - canvas.width);
            }

            if (planBounds.bottom - planBounds.top < canvas.height) {
                targetOffset.y = defaultOffset.y;
            } else if (planBounds.top > 0) {
                targetOffset.y = targetOffset.y - planBounds.top;
            } else if (planBounds.bottom < canvas.height) {
                targetOffset.y = targetOffset.y - (planBounds.bottom - canvas.height);
            }
        }

        if (!!targetOffset && offset !== targetOffset) {
            let offset = canvasProperties.offset;
            let offsetDiff = {
                "x": targetOffset.x - offset.x,
                "y": targetOffset.y - offset.y
            };

            if (Math.abs(offsetDiff.x) < 0.001 && Math.abs(offsetDiff.y) < 0.001) {
                canvasProperties.offset = targetOffset;
                canvasProperties.targetOffset = null;
            } else {
                let offsetDelta = {
                    "x": offsetDiff.x / 2,
                    "y": offsetDiff.y / 2
                };
                canvasProperties.offset.x += offsetDelta.x;
                canvasProperties.offset.y += offsetDelta.y;
            }

            hasAnimated = true;
        }

        if (!hasAnimated && zoom == canvasProperties.defaultZoom) {
            canvasProperties.offset.x = canvasProperties.defaultOffset.x;
            canvasProperties.offset.y = canvasProperties.defaultOffset.y;
        }
    }; // Calculates the bounds of the given vertices.


    var calculateBlockBounds = vertices => {
        let bounds = {
            "bottom": null,
            "left": null,
            "right": null,
            "top": null
        }; // Calculate the bounds of this block.

        for (let i = 0; i < vertices.length; i++) {
            let vertex = vertices[i];
            bounds.left = bounds.left == null ? vertex.x : Math.min(bounds.left, vertex.x);
            bounds.right = bounds.right == null ? vertex.x : Math.max(bounds.right, vertex.x);
            bounds.top = bounds.top == null ? vertex.y : Math.min(bounds.top, vertex.y);
            bounds.bottom = bounds.bottom == null ? vertex.y : Math.max(bounds.bottom, vertex.y);
        }

        return bounds;
    };

    var checkSeatHover = event => {
        if (hoveredSeat) {
            if (!!tooltip && (!tooltip.overrides || !tooltip.overrides.content)) {
                setDefaultTooltipContent(hoveredSeat);
            }

            if (tooltip != null && (!tooltip.overrides || !tooltip.overrides.position)) {
                setDefaultTooltipPosition(hoveredSeat);
            }

            if (!!tooltip && (!tooltip.overrides || !tooltip.overrides.display)) {
                displayTooltip();
            }
        } // Despatch custom event for external libraries to detect.


        let mouseMoveEvent = new CustomEvent(events.mousemove);
        mouseMoveEvent.originalEvent = event;
        mouseMoveEvent.block = hoveredBlock;
        mouseMoveEvent.seat = hoveredSeat;
        document.dispatchEvent(mouseMoveEvent);
    };

    var clearSelectedSeats = () => {
        selectedSeats = [];
    }; // Sets alpha value on css syntax colour. Will convert a hex colour value to rgba.


    var colourToRgba = (colour, alpha) => {
        let r, g, b;

        if (colour.startsWith("#")) {
            colour = colour.replace(/#/, "");

            if (colour.length === 3) {
                r = parseInt(colour.charAt(0), 16) * 0x11;
                g = parseInt(colour.charAt(1), 16) * 0x11;
                b = parseInt(colour.charAt(2), 16) * 0x11;
            } else {
                r = parseInt(colour.substring(0, 2), 16);
                g = parseInt(colour.substring(2, 4), 16);
                b = parseInt(colour.substring(4, 6), 16);
            }
        } else if (colour.startsWith("rgb")) {
            colour = colour.replace(/rgba|rgb/i, "");
            colour = colour.replace(/\(|\)/g, "");
            colour = colour.replace(/\s/g, "");
            let colourParts = colour.split(",");
            r = colourParts[0];
            g = colourParts[1];
            b = colourParts[2];
        } else {
            // If the colour was in a named format such as "white", then we can't change 
            // the brightness so just return the unaltered value.
            return colour;
        }

        return "rgba(" + r + ", " + g + ", " + b + ", " + alpha + ")";
    }; // Display the tooltip over the currently hovered seat.


    var displayTooltip = () => {
        if (!tooltip || !tooltip.container) {
            return;
        }

        let container = tooltip.container; // Now we can position the tooltip in the correct place.

        container.style.left = `${tooltip.x}px`;
        container.style.top = `${tooltip.y}px`;
        container.classList.remove("hidden");
    };

    var doZoom = delta => {
        let zoom = canvasProperties.zoom + delta;

        if (zoom < canvasProperties.minimumZoom) {
            zoom = canvasProperties.minimumZoom;
            endAnimateZoom();
        } else if (zoom > canvasProperties.maximumZoom) {
            zoom = canvasProperties.maximumZoom;
            endAnimateZoom();
        } else if (canvasProperties.constrainZoomThreshold != null && zoom > canvasProperties.constrainZoomThreshold) {
            let planBounds = getPlanBounds(layout);
            let planWidth = planBounds.right - planBounds.left;
            let maxZoom = canvas.width / planWidth / canvasProperties.devicePixelRatio;

            if (maxZoom < canvasProperties.constrainZoomThreshold) {
                zoom = canvasProperties.constrainZoomThreshold;
                endAnimateZoom();
            } else if (zoom > maxZoom) {
                zoom = maxZoom;
                endAnimateZoom();
            }
        }

        canvasProperties.zoom = zoom;
    };

    var endAnimateZoom = () => {
        if (canvasProperties.targetZoom) {
            canvasProperties.zoom = canvasProperties.targetZoom;
            canvasProperties.zoomOffset = getZoomOffset(canvasProperties.zoom);
            canvasProperties.targetZoom = null;
        }

        if (canvasProperties.targetOffset) {
            canvasProperties.offset.x = canvasProperties.targetOffset.x;
            canvasProperties.offset.y = canvasProperties.targetOffset.y;
            canvasProperties.targetOffset = null;
        }
    };

    var getAllBlocks = () => {
        return layout.blocks;
    };

    var getAllTexts = () => {
        return layout.texts;
    };

    var getBlocks = blockLabel => {
        let blocks = layout.blocks.filter(block => {
            if (block.label.trim().toLowerCase() === blockLabel.trim().toLowerCase()) {
                return true;
            } // It's possible this block has a range of pipe separated values.


            let labels = block.label.split("|");
            return labels.some(label => label.trim().toLowerCase() === blockLabel.trim().toLowerCase());
        });
        return blocks;
    };

    var getBlockOfSeat = seat => {
        return layout.blocks.filter(block => block.label === seat.blockLabel).find(block => {
            return block.rows.filter(row => row.label === seat.rowLabel).some(row => {
                return row.seats.some(checkSeat => checkSeat.label === seat.label);
            });
        });
    };

    var getMarqueeSeats = () => {
        return marqueeSeats;
    };

    var getPlanBounds = includePadding => {
        let planBounds = {
            "bottom": null,
            "left": null,
            "right": null,
            "top": null
        }; // Get the extremities of the plan design vertices.

        for (let i = 0; i < layout.blocks.length; i++) {
            let block = layout.blocks[i]; // Check the vertices of the block outlines.

            for (let j = 0; j < block.layoutProperties.vertices.length; j++) {
                let vertex = block.layoutProperties.vertices[j];

                if (planBounds.left == null || vertex.x < planBounds.left) {
                    planBounds.left = vertex.x;
                }

                if (planBounds.right == null || vertex.x > planBounds.right) {
                    planBounds.right = vertex.x;
                }

                if (planBounds.top == null || vertex.y < planBounds.top) {
                    planBounds.top = vertex.y;
                }

                if (planBounds.bottom == null || vertex.y > planBounds.bottom) {
                    planBounds.bottom = vertex.y;
                }
            } // There may be features such as seats on the outside of the block outlines.


            for (let j = 0; j < block.rows.length; j++) {
                let row = block.rows[j];

                for (let k = 0; k < row.seats.length; k++) {
                    let centre = row.seats[k].layoutProperties.centre;
                    let seatSize = row.seats[0].layoutProperties.size;

                    if (planBounds.left == null || centre.x < planBounds.left) {
                        planBounds.left = centre.x - seatSize;
                    }

                    if (planBounds.right == null || centre.x > planBounds.right) {
                        planBounds.right = centre.x + seatSize;
                    }

                    if (planBounds.top == null || centre.y < planBounds.top) {
                        planBounds.top = centre.y - seatSize;
                    }

                    if (planBounds.bottom == null || centre.y > planBounds.bottom) {
                        planBounds.bottom = centre.y + seatSize;
                    }
                }
            }
        } // Check the shapes array too.


        if (layout.shapes != null) {
            layout.shapes.forEach(shape => {
                shape.layoutProperties.vertices.forEach(vertex => {
                    if (planBounds.left == null || vertex.x < planBounds.left) {
                        planBounds.left = vertex.x;
                    }

                    if (planBounds.right == null || vertex.x > planBounds.right) {
                        planBounds.right = vertex.x;
                    }

                    if (planBounds.top == null || vertex.y < planBounds.top) {
                        planBounds.top = vertex.y;
                    }

                    if (planBounds.bottom == null || vertex.y > planBounds.bottom) {
                        planBounds.bottom = vertex.y;
                    }
                });
            });
        }

        if (layout.texts != null) {
            // Get the context as we'll need it for measure the text size of text elements.
            let context = canvas.getContext("2d");

            for (let i = 0; i < layout.texts.length; i++) {
                let text = layout.texts[i];
                let props = text.layoutProperties;
                let dpr = canvasProperties.devicePixelRatio;
                let zoom = canvasProperties.zoom;
                context.font = `${props.fontSize * zoom * dpr}px ${props.fontFamily}`;
                context.textAlign = "center";
                context.textBaseline = "middle";
                let metrics = context.measureText(text.label);

                if (!planBounds.left || props.centre.x - metrics.width / 2 < planBounds.left) {
                    planBounds.left = props.centre.x - metrics.width / 2;
                }

                if (!planBounds.right || props.centre.x + metrics.width / 2 > planBounds.right) {
                    planBounds.right = props.centre.x + metrics.width / 2;
                }

                if (!planBounds.top || props.centre.y - props.fontSize / 2 < planBounds.top) {
                    planBounds.top = props.centre.y - props.fontSize / 2;
                }

                if (!planBounds.bottom || props.centre.y + props.fontSize / 2 > planBounds.bottom) {
                    planBounds.bottom = props.centre.y + props.fontSize / 2;
                }
            }
        } // If the includePadding flag is set then we want to move the padding values to allow for the padding.


        if (includePadding) {
            let padding = canvasProperties.planPadding;
            planBounds.left -= padding;
            planBounds.right += padding;
            planBounds.top -= padding;
            planBounds.bottom += padding;
        }

        return planBounds;
    };

    var getHasRenderedFirstFrame = () => {
        return hasRenderedFirstFrame;
    };

    var getRenderInterval = () => {
        return renderInterval;
    };

    var getRowOfSeat = seat => {
        return layout.blocks.filter(block => block.label === seat.blockLabel).flatMap(block => block.rows).filter(row => row.label === seat.rowLabel).find(row => {
            return row.seats.some(checkSeat => checkSeat.label === seat.label);
        });
    };

    var getSeat = seatIdentifier => {
        if (!seatIdentifier) {
            return null;
        }

        let blockLabel, rowLabel, label;
        let seatParts = seatIdentifier.split("-");

        if (seatParts.length !== 3) {
            return null;
        }

        blockLabel = seatParts[0];
        rowLabel = seatParts[1];
        label = seatParts[2];

        for (let i = 0; i < layout.blocks.length; i++) {
            let block = layout.blocks[i]; // Blocks can have multiple labels separated by a pipe.

            let blockLabels = block.label.split("|").map(item => item.trim().toLowerCase());

            if (blockLabels.find(item => item === blockLabel.trim().toLowerCase()) == null) {
                continue;
            }

            for (let j = 0; j < block.rows.length; j++) {
                let row = block.rows[j];

                if (!new RegExp("^" + row.label + "$", "i").test(rowLabel)) {
                    continue;
                }

                for (let k = 0; k < row.seats.length; k++) {
                    let seat = row.seats[k];

                    if (new RegExp("^" + seat.label + "$", "i").test(label)) {
                        return seat;
                    }
                }
            }
        }

        return null;
    };

    var getSeatIndexInRow = seat => {
        let row = getRowOfSeat(seat);

        for (let i = 0; i < row.seats.length; i++) {
            if (row.seats[i].label === seat.label) {
                return i;
            }
        }

        return null;
    };

    var getSeats = seatIdentifiers => {
        if (!seatIdentifiers || seatIdentifiers.length === 0) {
            return null;
        }

        let seats = [];
        seatIdentifiers.forEach(seatIdentifier => {
            seats.push(getSeat(seatIdentifier));
        });
        return seats;
    };

    var getSelectedSeats = () => {
        return selectedSeats;
    };

    var getTexts = textIdentifier => {
        if (!layout || layout.texts.length === 0) {
            return null;
        }

        let texts = [];

        for (let i = 0; i < layout.texts.length; i++) {
            let text = layout.texts[i];

            if (!textIdentifier || new RegExp("^" + text.label + "$", "i").test(textIdentifier)) {
                texts.push(text);
            }
        }

        return texts;
    };

    var getZoom = () => {
        return canvasProperties.zoom;
    }; // Calculate the zoom offset so that we can give the effect of the zoom focussing on the centre of the canvas.


    var getZoomOffset = zoom => {
        let dpr = canvasProperties.devicePixelRatio;
        let centre = {
            "x": canvas.width / 2,
            "y": canvas.height / 2
        };
        let centrePostZoom = {
            "x": centre.x * zoom,
            "y": centre.y * zoom
        };
        let zoomDiff = {
            "x": centrePostZoom.x - centre.x,
            "y": centrePostZoom.y - centre.y
        };
        return {
            "x": -zoomDiff.x / dpr * 1 / zoom,
            "y": -zoomDiff.y / dpr * 1 / zoom
        };
    };

    var hideTooltip = () => {
        if (!tooltip || !tooltip.container || !tooltip.container.offsetParent) {
            return;
        }

        let container = tooltip.container;
        container.classList.add("hidden");
        container.style.left = "-99rem";
        container.style.top = "-99rem";
    };

    var isSeatSelected = seat => {
        if (selectedSeats.find(selectedSeat => selectedSeat === seat) != null) {
            return true;
        }

        return false;
    }; // Prepare the html5 canvas for the seat picker.


    var initialiseCanvas = () => {
        let dpr = canvasProperties.devicePixelRatio; // High density displays require scaling of the canvas.

        if (dpr !== 1) {
            let context = canvas.getContext("2d");
            context.scale(dpr, dpr);
            canvas.style.height = `${canvas.height}px`;
            canvas.style.width = `${canvas.width}px`;
            canvas.setAttribute("height", canvas.height * dpr);
            canvas.setAttribute("width", canvas.width * dpr);
        }

        let planPadding = canvasProperties.planPadding;
        let planBounds = getPlanBounds(layout);
        let planWidth = planBounds.right - planBounds.left;
        let planHeight = planBounds.bottom - planBounds.top;
        planWidth += planPadding;
        planHeight += planPadding; // Initialise the zoom values.

        let planRatio = planWidth / planHeight;
        let canvasRatio = canvas.width / canvas.height;

        if (planRatio > canvasRatio) {
            canvasProperties.defaultZoom = canvas.width / planWidth / dpr;
        } else {
            canvasProperties.defaultZoom = canvas.height / planHeight / dpr;
        }

        canvasProperties.minimumZoom = canvasProperties.defaultZoom;
        doZoom(-(1 - canvasProperties.defaultZoom));
        canvasProperties.zoomOffset = getZoomOffset(canvasProperties.zoom);
        canvasProperties.defaultZoomOffset = {
            "x": canvasProperties.zoomOffset.x,
            "y": canvasProperties.zoomOffset.y
        }; // Initialise the canvas offsets.

        let canvasHalfHeight = canvas.height / 2 / canvasProperties.defaultZoom / dpr;
        let canvasHalfWidth = canvas.width / 2 / canvasProperties.defaultZoom / dpr;
        let planHalfHeight = planHeight / 2;
        let planHalfWidth = planWidth / 2;
        canvasProperties.defaultOffset = {
            "x": canvasHalfWidth - planHalfWidth - planBounds.left - canvasProperties.zoomOffset.x + planPadding,
            "y": canvasHalfHeight - planHalfHeight - planBounds.top - canvasProperties.zoomOffset.y + planPadding
        };
        canvasProperties.offset = {
            "x": canvasProperties.defaultOffset.x,
            "y": canvasProperties.defaultOffset.y
        };
    };

    var initialiseCategories = () => {
        // Create a category to handle layout seats that aren't mapped to anything in the categories array. This will allow us to 
        // render seats even if they're not in use so that the seat picker doesn't have empty spaces where seats should be.
        let unassignedCategory = {
            "colour": colourPalette.default,
            "label": unassignedCategoryLabel,
            "seats": [],
            "seatStyle": seatStyles.default
        };
        categories.push(unassignedCategory);
        updateUnassignedSeats(); // If any categories were passed in the options object then we can get those added to the categories array.

        if (options.categories != null) {
            options.categories.forEach(category => {
                addUpdateCategory(category, category.isSelectable);
            });
        }
    };

    var initialiseEvents = () => {
        canvas.addEventListener("contextmenu", event => {
            if (event.preventDefault && event.cancelable) {
                event.preventDefault();
            }
        });
        canvas.addEventListener("mousemove", event => {
            if (event.preventDefault && event.cancelable) {
                event.preventDefault();
            }

            updateMouse(event.clientX, event.clientY, event);
        }); // If the user manages to hover over a visible tooltip within 1 render frame then the logic to hide the tooltip 
        // won't run and it will continue to display. To avoid this we can monitor the mouse move at document level.

        document.addEventListener("mousemove", event => {
            if (event.target !== canvas && !!tooltip && (!tooltip.overrides || !tooltip.overrides.hide)) {
                // Hide the tooltip because we'll re-check whether it needs to be displayed.
                hideTooltip();
            }
        });
        canvas.addEventListener("mouseout", event => {
            mouse.x = 0;
            mouse.y = 0;
            mouse.isDragged = false;
        });
        canvas.addEventListener("mousedown", event => {
            if (event.preventDefault && event.cancelable) {
                event.preventDefault();
            } // End any zoom animation so that the renderer doesn't keep trying to reach targets after letting go of the mouse.


            endAnimateZoom();

            if (event.button === mouse.buttons.drag) {
                mouse.isDragged = true;
            }

            if (event.button === mouse.buttons.marquee) {
                updateMarqueePoint(event, event.clientX, event.clientY);
            }
        });
        canvas.addEventListener("mouseup", event => {
            if (event.preventDefault && event.cancelable) {
                event.preventDefault();
            }

            let changedSeats = [];

            if (!mouse.moved) {
                if (hoveredSeat != null) {
                    if (event.button === mouse.buttons.marquee) {
                        if (!event.shiftKey && !event.altKey) {
                            changedSeats = selectedSeats;
                            selectedSeats = [];
                        }
                    }

                    if (event.button === mouse.buttons.select || event.button === mouse.buttons.marquee) {
                        if (selectedSeats.indexOf(hoveredSeat) === -1) {
                            selectSeat(hoveredSeat);
                        } else {
                            unselectSeat(hoveredSeat);
                        }

                        changedSeats = [hoveredSeat];
                    }
                } // Despatch custom event for external libraries to detect.


                let clickEvent = new CustomEvent(events.click);
                clickEvent.originalEvent = event;
                clickEvent.block = hoveredBlock;
                clickEvent.seat = hoveredSeat;
                document.dispatchEvent(clickEvent);
            }

            if (event.button === mouse.buttons.drag) {
                mouse.isDragged = false;
            }

            if (event.button === mouse.buttons.marquee) {
                mouse.marqueePoint = null;

                if (marqueeSeats.length > 0) {
                    if (event.shiftKey) {
                        let seatsToAdd = marqueeSeats.filter(marqueeSeat => {
                            return selectedSeats.filter(selectionSeat => selectionSeat === marqueeSeat).length === 0;
                        });
                        selectedSeats = selectedSeats.concat(seatsToAdd);
                        changedSeats = marqueeSeats;
                    } else if (event.altKey) {
                        let length = selectedSeats.length;
                        let matchedSeats = marqueeSeats.filter(marqueeSeat => {
                            return selectedSeats.filter(selectionSeat => selectionSeat === marqueeSeat).length > 0;
                        });
                        selectedSeats = selectedSeats.filter(selectionSeat => {
                            return marqueeSeats.filter(marqueeSeat => marqueeSeat === selectionSeat).length === 0;
                        });

                        if (selectedSeats.length !== length) {
                            changedSeats = matchedSeats;
                        }
                    } else {
                        changedSeats = marqueeSeats;
                        selectedSeats = marqueeSeats;
                    }
                }
            }

            let selectionChangedEvent = new CustomEvent(events.selectionChanged);
            selectionChangedEvent.changedSeats = changedSeats;
            selectionChangedEvent.marqueeSeats = marqueeSeats;
            selectionChangedEvent.originalEvent = event;
            selectionChangedEvent.selectedSeats = selectedSeats;
            document.dispatchEvent(selectionChangedEvent);
            marqueeSeats = [];
            mouse.moved = false;
        });
        canvas.addEventListener("mousewheel", event => {
            processMouseWheel(event);
        });
        canvas.addEventListener("wheel", event => {
            processMouseWheel(event);
        });
        canvas.addEventListener("touchstart", event => {
            // End any lerping so that the renderer doesn't keep tryign to reach targets after letting go of the mouse.
            endAnimateZoom();
            touch.isTouched = true; // Update the touch panning start point.

            if (mouse.buttons.marquee == null || event.targetTouches.length === 2) {
                if (event.preventDefault && event.cancelable) {
                    event.preventDefault();
                }

                let targetTouch = event.targetTouches[0];
                let targetTouch2 = event.targetTouches.length === 2 ? event.targetTouches[1] : null;
                touch.origin = {
                    "x": targetTouch.clientX,
                    "y": targetTouch.clientY
                };

                if (targetTouch2 != null) {
                    touch.origin.x = (touch.origin.x + targetTouch2.clientX) / 2;
                    touch.origin.y = (touch.origin.y + targetTouch2.clientY) / 2;
                }

                touch.initialCanvasOffset.x = canvasProperties.offset.x;
                touch.initialCanvasOffset.y = canvasProperties.offset.y;
            } // Updat the "mouse" position, as used by the rest of the seat picker logic.


            if (event.targetTouches.length === 1) {
                let targetTouch = event.targetTouches[0];
                updateMouse(targetTouch.clientX, targetTouch.clientY, event);

                if (mouse.buttons.marquee != null) {
                    updateMarqueePoint(event, targetTouch.clientX, targetTouch.clientY);
                }
            } // Update the touch settings for zooming.


            if (event.targetTouches.length === 2) {
                // If we have a currently active marquee and a second touch is involved, cancel the marquee.
                if (mouse.marqueePoint) {
                    mouse.marqueePoint = null;
                    marqueeSeats = [];
                }

                let firstTouch = event.targetTouches[0];
                let secondTouch = event.targetTouches[1]; // Work out the relative distance between this second finger touchstart and the current  
                // single finger position.

                touch.originDiffX = secondTouch.clientX - firstTouch.clientX;
                touch.originDiffY = secondTouch.clientY - firstTouch.clientY; // Set the current zoom level of the canvas.

                touch.initialCanvasZoom = canvasProperties.zoom;
            }
        });
        canvas.addEventListener("touchmove", event => {
            let allowPageScroll = true;
            let targetTouch1 = event.targetTouches[0];
            let targetTouch2 = event.targetTouches.length > 1 ? event.targetTouches[1] : null;
            let targetTouchPosition = {
                x: targetTouch1.clientX,
                y: targetTouch1.clientY
            };

            if (targetTouch2 != null) {
                targetTouchPosition.x = (targetTouchPosition.x + targetTouch2.clientX) / 2;
                targetTouchPosition.y = (targetTouchPosition.y + targetTouch2.clientY) / 2;
            }

            let deltaX = touch.origin.x - targetTouchPosition.x;
            let deltaY = touch.origin.y - targetTouchPosition.y;

            if (mouse.moved || Math.abs(deltaX) > touch.deadZone || Math.abs(deltaY) > touch.deadZone) {
                mouse.moved = true;

                if (event.targetTouches.length === 2 && !mouse.marqueePoint) {
                    // Do a zoom if two finger touch.
                    if (document.elementFromPoint(targetTouch2.clientX, targetTouch2.clientY) === canvas) {
                        let diffX = targetTouch2.clientX - targetTouch1.clientX;
                        let diffY = targetTouch2.clientY - targetTouch1.clientY;
                        let originD = Math.sqrt(Math.pow(touch.originDiffX, 2) + Math.pow(touch.originDiffY, 2));
                        let currentD = Math.sqrt(Math.pow(diffX, 2) + Math.pow(diffY, 2));
                        let deltaD = (currentD - originD) / canvasProperties.touchZoomRatio * canvasProperties.zoom;
                        canvasProperties.targetZoom = canvasProperties.zoom + deltaD; // Update the origin values so our reference point on the next pass is the current position.

                        touch.originDiffX = diffX;
                        touch.originDiffY = diffY;

                        if (canvasProperties.targetZoom <= canvasProperties.minimumZoom) {
                            canvasProperties.targetZoom = canvasProperties.minimumZoom;
                        }
                    }
                } else if (mouse.marqueePoint && event.targetTouches.length === 1) {
                    let targetTouch = event.targetTouches[0];
                    updateMouse(targetTouch.clientX, targetTouch.clientY, event);
                }

                if (mouse.buttons.marquee == null || event.targetTouches.length === 2 && !mouse.marqueePoint) {
                    // Do canvas panning.
                    let planBounds = translateToCanvasBounds(getPlanBounds(layout));

                    if (deltaX < 0 && planBounds.left < 0 || deltaX > 0 && planBounds.right > canvas.width) {
                        canvasProperties.offset.x = touch.initialCanvasOffset.x - deltaX / canvasProperties.zoom;
                    }

                    if (deltaY < 0 && planBounds.top < 0 || deltaY > 0 && planBounds.bottom > canvas.height) {
                        canvasProperties.offset.y = touch.initialCanvasOffset.y - deltaY / canvasProperties.zoom;
                        allowPageScroll = false;
                    }
                }
            }

            if (event.preventDefault && event.cancelable && (canvasControls.touchPanLock || !allowPageScroll)) {
                event.preventDefault();
            }
        });
        canvas.addEventListener("touchend", event => {
            if (event.preventDefault && event.cancelable) {
                event.preventDefault();
            }

            let changedSeats = []; // Detect taps.

            if (event.targetTouches.length === 0 && !mouse.moved) {
                // Perform seat actions if the default select mode isn't disabled.
                if (mouse.buttons.select != null && hoveredSeat != null) {
                    if (selectedSeats.indexOf(hoveredSeat) === -1) {
                        selectSeat(hoveredSeat);
                    } else {
                        unselectSeat(hoveredSeat);
                    }

                    changedSeats = [hoveredSeat];
                } // Despatch custom event for external libraries to detect.


                let tap = new CustomEvent(events.tap);
                tap.originalEvent = event;
                tap.block = hoveredBlock;
                tap.seat = hoveredSeat;
                document.dispatchEvent(tap);
            } // Else if we're on a touch device and a finger is removed but leaves one touching, then we need to 
            // update the origin of the movements to prevent the rendered seats from jumping around.
            else if (event.targetTouches.length === 1 && event.changedTouches.length === 1) {
                let targetTouch = event.targetTouches[0];
                touch.origin = {
                    "x": targetTouch.clientX,
                    "y": targetTouch.clientY
                };
                touch.initialCanvasOffset.x = canvasProperties.offset.x;
                touch.initialCanvasOffset.y = canvasProperties.offset.y;
                touch.initialCanvasZoom = canvasProperties.zoom;
            } else {
                mouse.isDragged = false;
                mouse.moved = false;
            }

            if (mouse.buttons.marquee != null) {
                mouse.marqueePoint = null;

                if (marqueeSeats.length > 0) {
                    let changedSeats = marqueeSeats;
                    selectedSeats = changedSeats;
                }
            } // If there are no touches left, then update the status.


            if (event.targetTouches.length === 0) {
                touch.isTouched = false;
            }

            let selectionChangedEvent = new CustomEvent(events.selectionChanged);
            selectionChangedEvent.changedSeats = changedSeats;
            selectionChangedEvent.marqueeSeats = marqueeSeats;
            selectionChangedEvent.originalEvent = event;
            selectionChangedEvent.selectedSeats = selectedSeats;
            document.dispatchEvent(selectionChangedEvent);
            marqueeSeats = [];
        });

        if (canvasControls.zoomInButton) {
            canvasControls.zoomInButton.addEventListener("click", event => {
                if (event.preventDefault && event.cancelable) {
                    event.preventDefault();
                }

                let delta = canvasControls.zoomAmount * canvasProperties.zoom;
                canvasProperties.targetZoom = canvasProperties.zoom + delta;
            });
        }

        if (canvasControls.zoomOutButton) {
            canvasControls.zoomOutButton.addEventListener("click", event => {
                if (event.preventDefault && event.cancelable) {
                    event.preventDefault();
                }

                let delta = canvasControls.zoomAmount * canvasProperties.zoom;
                canvasProperties.targetZoom = canvasProperties.zoom - delta;
            });
        }

        if (canvasControls.zoomResetButton) {
            canvasControls.zoomResetButton.addEventListener("click", event => {
                if (event.preventDefault && event.cancelable) {
                    event.preventDefault();
                }

                canvasProperties.targetZoom = canvasProperties.defaultZoom;
                canvasProperties.targetOffset = {
                    "x": canvasProperties.defaultOffset.x,
                    "y": canvasProperties.defaultOffset.y
                };
            });
        }
    }; // Step through the layout json objects and initialise some values.


    var initialiseLayout = () => {
        for (let i = 0; i < layout.blocks.length; i++) {
            let block = layout.blocks[i];

            for (let j = 0; j < block.rows.length; j++) {
                let row = block.rows[j];
                row.blockLabel = block.label;

                for (let k = 0; k < row.seats.length; k++) {
                    let seat = row.seats[k];

                    if (block.label == null || row.label == null || seat.label == null) {
                        continue;
                    }

                    seat.blockLabel = block.label;
                    seat.identifiers = block.label.split("|").map(blockLabel => `${blockLabel.trim()}-${row.label}-${seat.label}`);
                    seat.isSelectable = false;
                    seat.rowLabel = row.label;
                }
            } // The serialization rules of the seating plan layout manager omit properties with null or default values, meaning
            // things like integers with 0 value won't exist. We need to make sure certain ones do.


            if (!block.layoutProperties.order) {
                block.layoutProperties.order = 0;
            }

            for (let j = 0; j < block.layoutProperties.vertices.length; j++) {
                let v = block.layoutProperties.vertices[j];

                if (!v.x) {
                    v.x = 0;
                }

                if (!v.y) {
                    v.y = 0;
                }
            }
        }

        if (layout.images != null) {
            for (let i = 0; i < layout.images.length; i++) {
                let image = layout.images[i];
                image.loaded = false;
                image.obj = new Image();
                image.obj.src = image.imageLocation;

                image.obj.onload = function () {
                    image.loaded = true;
                };
            }
        } // Sort the layout blocks by their order to make sure we render in the correct order.


        layout.blocks.sort((a, b) => {
            return a.layoutProperties.order - b.layoutProperties.order;
        });
    };

    var initialiseProperties = () => {
        canvas = options.canvas || document.querySelector("canvas");
        categories = [];
        hasRenderedFirstFrame = false;
        hoveredBlock = null;
        hoveredSeat = null;
        layout = options.layout;
        renderInterval = Math.floor(1000 / 30);
        marqueeSeats = [];
        selectedSeats = []; // Initialise configurable options.

        canvasControls = {
            "zoomAmount": 0.20,
            "zoomAmountWheel": 0.20,
            "zoomInButton": document.querySelector(".zoom-in"),
            "zoomOutButton": document.querySelector(".zoom-out"),
            "zoomResetButton": document.querySelector(".zoom-reset"),
            ...options.canvasControls
        };
        canvasProperties = {
            "constrainZoomThreshold": null,
            "defaultZoom": 1,
            "devicePixelRatio": window.devicePixelRatio || 1,
            "maximumZoom": 5,
            "minimumZoom": 0.05,
            "offsetX": 0,
            "offsetY": 0,
            "planPadding": 20,
            "touchZoomRatio": 100,
            "zoom": 1,
            ...options.canvasProperties
        };
        labelProperties = {
            "colour": "#000",
            "fontFamily": "Arial",
            "fontSizeRatio": 0.8,
            "seatLabelsVisible": false,
            ...options.labelProperties
        };
        colourPalette = {
            "default": "#ccc",
            "marquee": "#ef5b5b",
            "marqueeSeat": "#ef5b5b",
            "selectedSeat": "#ffba49",
            ...options.colourPalette
        };
        seatStyles = {};
        seatStyles.default = {
            "colour": colourPalette.default,
            "icon": enums.SeatIcon.FillCircle,
            ...(options.seatStyles ? options.seatStyles.default : {})
        };
        seatStyles.marquee = {
            "colour": colourPalette.marqueeSeat,
            ...(options.seatStyles ? options.seatStyles.marquee : {})
        };
        seatStyles.selected = {
            "colour": colourPalette.selectedSeat,
            ...(options.seatStyles ? options.seatStyles.selected : {})
        };
        mouse = {
            "isDragged": false,
            "lastX": 0,
            "lastY": 0,
            "marqueePoint": null,
            "moved": false,
            "x": 0,
            "y": 0,
            ...options.mouse
        };
        mouse.buttons = {
            "drag": 0,
            "marquee": null,
            "select": 0,
            ...(options.mouse ? options.mouse.buttons : {})
        };
        tooltip = {
            "container": document.querySelector(".tooltip"),
            "x": 0,
            "y": 0,
            ...options.tooltip
        };
        touch = {
            "deadZone": 2,
            "initialCanvasOffset": {
                "x": 0,
                "y": 0
            },
            "isTouched": false,
            "origin": {
                "x": 0,
                "y": 0
            },
            ...options.touch
        }; // Scale the minimum zoom to the device pixel ratio.

        canvasProperties.minimumZoom *= canvasProperties.devicePixelRatio;
        canvasProperties.defaultOffset = {
            "x": 0,
            "y": 0
        };
        canvasProperties.defaultZoom = 1;
        canvasProperties.offset = {
            "x": canvasProperties.defaultOffset.x,
            "y": canvasProperties.defaultOffset.y
        };
        canvasProperties.targetOffset = null;
        canvasProperties.zoom = canvasProperties.defaultZoom;
        canvasProperties.targetZoom = null; // Initialise a separate offset for managing the focus point of a zoom action.

        canvasProperties.zoomOffset = {
            "x": 0,
            "y": 0
        };
        canvasProperties.defaultZoomOffset = {
            "x": 0,
            "y": 0
        };
    };

    var planSeatIsVisible = layoutSeat => {
        let coords = translateToCanvasCoordinates(layoutSeat.layoutProperties.centre);
        let dpr = canvasProperties.devicePixelRatio;
        let radius = layoutSeat.layoutProperties.size / 2 * canvasProperties.zoom;
        return coords.x + radius * dpr > 0 && coords.x - radius * dpr < canvas.width * dpr && coords.y + radius * dpr > 0 && coords.y - radius * dpr < canvas.height * dpr;
    };

    var processMouseWheel = event => {
        // We want to zoom so prevent the page from scrolling.
        if (event.preventDefault && event.cancelable) {
            event.preventDefault();
        }

        if (event.deltaY != null) {
            let delta = canvasControls.zoomAmountWheel * canvasProperties.zoom;

            if (event.deltaY > 0) {
                delta = -delta;
            }

            let targetZoom = canvasProperties.zoom + delta;

            if (targetZoom < canvasProperties.minimumZoom) {
                targetZoom = canvasProperties.minimumZoom;
            }

            canvasProperties.targetZoom = targetZoom;
        }
    };

    var removeCategory = categoryLabel => {
        let categoryIndex = categories.findIndex(category => category.label === categoryLabel);
        categories.splice(categoryIndex, 1);
        updateUnassignedSeats();
    };

    var unselectSeat = seat => {
        if (!seat || !seat.isSelectable) {
            return;
        }

        let index = selectedSeats.indexOf(seat);

        if (index !== -1) {
            selectedSeats.splice(index, 1);
        }
    }; // The main render loop redraws the canvas n times a second.


    var render = () => {
        requestAnimationFrame(render);
        let context = canvas.getContext("2d"); // Clear the current canvas ready to redraw the frame.

        context.clearRect(0, 0, canvas.width, canvas.height); // Initialise the canvas font rendering text baseline to top to match up with other object coordinates. 

        context.textBaseline = "top"; // Render the seating plan layout.

        renderPlanLayoutImages(context);
        renderPlanLayoutBlocks(context);
        renderPlanLayoutSeats(context);
        renderPlanLayoutShapes(context);
        renderPlanLayoutTexts(context);
        renderSelectionMarquee(context); // Check the label properties to see if we need to show any.

        if (labelProperties.seatLabelsVisible) {
            renderSeatLabels(context);
        } // Set the cursor mode based on the hover states.


        if (!hoveredSeat && !hoveredBlock) {
            canvas.style.cursor = "default";
        } else {
            canvas.style.cursor = "pointer";
        } // Despatch an event to state that the canvas now has an image.


        if (!hasRenderedFirstFrame) {
            hasRenderedFirstFrame = true;
            let renderedFirstFrameEvent = new CustomEvent(events.renderedFirstFrame);
            document.dispatchEvent(renderedFirstFrameEvent);
        }
    }; // Render the block outlines of the plan layout.


    var renderPlanLayoutBlocks = context => {
        if (layout == null || layout.blocks == null || layout.blocks.length === 0) {
            return;
        }

        let hoveredBlockThisFrame; // Plot the plan items out.

        context.lineWidth = 1;
        context.fillStyle = "#ffffff";
        let defaultStrokeType = "#aaaaaa";

        for (let i = 0; i < layout.blocks.length; i++) {
            let layoutBlock = layout.blocks[i];
            let shape = layoutBlock.layoutProperties.shape;
            context.beginPath();

            if (shape === enums.Shape.Oval) {
                let vertices = layoutBlock.layoutProperties.vertices;
                let vertex1 = translateToCanvasCoordinates(vertices[0]);
                let vertex2 = translateToCanvasCoordinates(vertices[1]);
                let vertex3 = translateToCanvasCoordinates(vertices[2]);
                let vertex4 = translateToCanvasCoordinates(vertices[3]);
                let startPoint = {
                    "x": helpers.lerp(vertex4.x, vertex1.x, 0.5),
                    "y": helpers.lerp(vertex4.y, vertex1.y, 0.5)
                };
                let endPoint = {
                    "x": helpers.lerp(vertex2.x, vertex3.x, 0.5),
                    "y": helpers.lerp(vertex2.y, vertex3.y, 0.5)
                };
                context.moveTo(startPoint.x, startPoint.y); // Draw a curve to the other side of the oval...

                context.bezierCurveTo(vertex1.x, vertex1.y, vertex2.x, vertex2.y, endPoint.x, endPoint.y); // ...and back to the start again.

                context.bezierCurveTo(vertex3.x, vertex3.y, vertex4.x, vertex4.y, startPoint.x, startPoint.y);
            } else if (shape === enums.Shape.Basic) {
                let firstVertex = translateToCanvasCoordinates(layoutBlock.layoutProperties.vertices[0]);
                context.moveTo(firstVertex.x, firstVertex.y); // Plot the lines between the vertices.

                for (let j = 0; j < layoutBlock.layoutProperties.vertices.length; j++) {
                    let vertex = translateToCanvasCoordinates(layoutBlock.layoutProperties.vertices[j]);
                    context.lineTo(vertex.x, vertex.y);
                } // Draw a line from the last vertext to the first.


                context.lineTo(firstVertex.x, firstVertex.y);
            } // Set the hovered block to this one if the mouse is over it.


            if (context.isPointInPath(mouse.x, mouse.y)) {
                hoveredBlockThisFrame = layoutBlock;
            }

            if (layoutBlock.layoutProperties.hideBlockOutline) {
                context.closePath();
            } else {
                context.fill();
                context.setLineDash([]);
                context.strokeStyle = layoutBlock.strokeStyle || defaultStrokeType;
                context.stroke();
            }
        }

        if (hoveredBlockThisFrame != null) {
            hoveredBlock = hoveredBlockThisFrame;
        } else {
            hoveredBlock = null;
        }
    }; // Render the seats of the plan layout.


    var renderPlanLayoutSeats = context => {
        if (layout == null) {
            return;
        } // Get some dimensions from the canvas properties.   


        let props = canvasProperties;
        let dpr = props.devicePixelRatio;
        let hoveredSeatThisFrame; // Organise all of our renderable seats into groups of mutual properties, so that we have to do less render passes.

        let seatGroups = {};

        for (let i = 0; i < categories.length; i++) {
            let category = categories[i];
            let seats = category.seats;

            for (let j = 0; j < seats.length; j++) {
                let seat = seats[j];

                if (seat.layoutProperties.isDisabled) {
                    continue;
                }

                let seatInMarquee = marqueeSeats.filter(marqueeSeat => marqueeSeat === seat).length > 0;
                let seatInSelection = selectedSeats.filter(selectedSeat => selectedSeat === seat).length > 0; // Prepare the renderstyle for this seat.

                let renderStyle = {
                    "colour": seatStyles.default.colour,
                    "icon": seatStyles.default.icon
                }; // Set renderstyle based on category style.

                if (category.seatStyle) {
                    renderStyle.colour = category.seatStyle.colour || renderStyle.colour;
                    renderStyle.icon = category.seatStyle.icon || renderStyle.icon;
                } // Override style based on selection status.

                if (seat.style) {
                    renderStyle.colour = seat.style.colour || renderStyle.colour;
                    renderStyle.icon = seat.style.icon || renderStyle.icon;
                }

                if (seatInMarquee) {
                    renderStyle.colour = seatStyles.marquee.colour || renderStyle.colour;
                    renderStyle.icon = seatStyles.marquee.icon || renderStyle.icon;
                } else if (seatInSelection) {
                    renderStyle.colour = seatStyles.selected.colour || renderStyle.colour;
                    renderStyle.icon = seatStyles.selected.icon || renderStyle.icon;
                } // Override style if specific values are set on the individual seat.

                let groupKey = `${renderStyle.icon}-${renderStyle.colour}`;
                let group = seatGroups[groupKey] || {
                    style: renderStyle,
                    seats: []
                };
                group.seats.push(seat);

                if (!seatGroups[groupKey]) {
                    seatGroups[groupKey] = group;
                }
            }
        }

        let getScaledRadius = diameter => {
            return diameter / 2 * canvasProperties.zoom * dpr;
        };

        let getLineWidth = radius => {
            return radius / 4;
        }; // Render the items from each group.


        for (let groupKey in seatGroups) {
            let renderGroup = seatGroups[groupKey];
            let seatIcon = renderGroup.style.icon;

            if (seatIcon === enums.SeatIcon.FillCircle) {
                context.beginPath();

                for (let i = 0; i < renderGroup.seats.length; i++) {
                    let seat = renderGroup.seats[i];
                    let seatCentre = translateToCanvasCoordinates(seat.layoutProperties.centre); // Before we do anything else, lets make sure the seat is actually visible before we try to render it.

                    if (!planSeatIsVisible(seat)) {
                        continue;
                    }

                    let scaledSeatRadius = getScaledRadius(seat.layoutProperties.size); // Render the seat to the canvas.

                    context.moveTo(seatCentre.x + scaledSeatRadius, seatCentre.y);
                    context.arc(seatCentre.x, seatCentre.y, scaledSeatRadius, 0, Math.PI * 2, false); // Check to see if we're hovering over this seat.

                    if (mouse.x > seatCentre.x - scaledSeatRadius && mouse.x < seatCentre.x + scaledSeatRadius && mouse.y > seatCentre.y - scaledSeatRadius && mouse.y < seatCentre.y + scaledSeatRadius) {
                        hoveredSeatThisFrame = seat;
                    }
                }

                context.fillStyle = renderGroup.style.colour;
                context.fill();
            } else if (seatIcon === enums.SeatIcon.StrokeCircle) {
                context.beginPath();

                if (!renderGroup.seats || renderGroup.seats.length === 0) {
                    break;
                }

                let scaledSeatRadius = getScaledRadius(renderGroup.seats[0].layoutProperties.size);
                let seatLineWidth = getLineWidth(scaledSeatRadius);
                scaledSeatRadius -= seatLineWidth / 2;

                for (let i = 0; i < renderGroup.seats.length; i++) {
                    let seat = renderGroup.seats[i];
                    let seatCentre = translateToCanvasCoordinates(seat.layoutProperties.centre); // Before we do anything else, lets make sure the seat is actually visible before we try to render it.

                    if (!planSeatIsVisible(seat)) {
                        continue;
                    } // Render the seat to the canvas.


                    context.moveTo(seatCentre.x + scaledSeatRadius, seatCentre.y);
                    context.arc(seatCentre.x, seatCentre.y, scaledSeatRadius, 0, Math.PI * 2, false); // Check to see if we're hovering over this seat.

                    if (mouse.x > seatCentre.x - scaledSeatRadius && mouse.x < seatCentre.x + scaledSeatRadius && mouse.y > seatCentre.y - scaledSeatRadius && mouse.y < seatCentre.y + scaledSeatRadius) {
                        hoveredSeatThisFrame = seat;
                    }
                }

                context.lineWidth = seatLineWidth;
                context.setLineDash([]);
                context.strokeStyle = renderGroup.style.colour;
                context.stroke();
            } else if (seatIcon === enums.SeatIcon.StrokeCircleWithCross) {
                // Draw the cross.
                context.beginPath();

                if (!renderGroup.seats || renderGroup.seats.length === 0) {
                    break;
                }

                let scaledSeatRadius = getScaledRadius(renderGroup.seats[0].layoutProperties.size);
                let seatLineWidth = getLineWidth(scaledSeatRadius);
                scaledSeatRadius -= seatLineWidth / 2;
                let crossRadius = scaledSeatRadius * 0.45;

                for (let i = 0; i < renderGroup.seats.length; i++) {
                    let seat = renderGroup.seats[i];
                    let seatCentre = translateToCanvasCoordinates(seat.layoutProperties.centre);

                    if (!planSeatIsVisible(seat)) {
                        continue;
                    }

                    context.moveTo(seatCentre.x - crossRadius, seatCentre.y + crossRadius);
                    context.lineTo(seatCentre.x + crossRadius, seatCentre.y - crossRadius);
                    context.moveTo(seatCentre.x - crossRadius, seatCentre.y - crossRadius);
                    context.lineTo(seatCentre.x + crossRadius, seatCentre.y + crossRadius);
                }

                context.lineWidth = seatLineWidth;
                context.setLineDash([]);
                context.strokeStyle = renderGroup.style.colour;
                context.stroke(); // Draw the seat stroke.

                context.beginPath();

                for (let i = 0; i < renderGroup.seats.length; i++) {
                    let seat = renderGroup.seats[i];
                    let seatCentre = translateToCanvasCoordinates(seat.layoutProperties.centre); // Before we do anything else, lets make sure the seat is actually visible before we try to render it.

                    if (!planSeatIsVisible(seat)) {
                        continue;
                    } // Render the seat to the canvas.


                    context.moveTo(seatCentre.x + scaledSeatRadius, seatCentre.y);
                    context.arc(seatCentre.x, seatCentre.y, scaledSeatRadius, 0, Math.PI * 2, false); // Check to see if we're hovering over this seat.

                    if (mouse.x > seatCentre.x - scaledSeatRadius && mouse.x < seatCentre.x + scaledSeatRadius && mouse.y > seatCentre.y - scaledSeatRadius && mouse.y < seatCentre.y + scaledSeatRadius) {
                        hoveredSeatThisFrame = seat;
                    }
                }

                context.lineWidth = scaledSeatRadius / 4;
                context.setLineDash([]);
                context.strokeStyle = renderGroup.style.colour;
                context.stroke();
            } else if (seatIcon === enums.SeatIcon.StrokeCircleWithSlash) {
                // Draw the slash.
                context.beginPath();
                let scaledSeatRadius = getScaledRadius(renderGroup.seats[0].layoutProperties.size);
                let seatLineWidth = getLineWidth(scaledSeatRadius);
                scaledSeatRadius -= seatLineWidth / 2;
                let slashRadius = scaledSeatRadius * 0.7;
                let slashLineWidth = seatLineWidth * 2;

                for (let i = 0; i < renderGroup.seats.length; i++) {
                    let seat = renderGroup.seats[i];
                    let seatCentre = translateToCanvasCoordinates(seat.layoutProperties.centre); // Before we do anything else, lets make sure the seat is actually visible before we try to render it.

                    if (!planSeatIsVisible(seat)) {
                        continue;
                    }

                    context.moveTo(seatCentre.x - slashRadius, seatCentre.y + slashRadius);
                    context.lineTo(seatCentre.x + slashRadius, seatCentre.y - slashRadius);
                }

                context.lineWidth = slashLineWidth;
                context.setLineDash([]);
                context.strokeStyle = renderGroup.style.colour;
                context.stroke(); // Draw the seat stroke.

                context.beginPath();

                for (let i = 0; i < renderGroup.seats.length; i++) {
                    let seat = renderGroup.seats[i];

                    if (seat.layoutProperties.isDisabled) {
                        continue;
                    }

                    let seatCentre = translateToCanvasCoordinates(seat.layoutProperties.centre); // Render the seat to the canvas.

                    context.moveTo(seatCentre.x + scaledSeatRadius, seatCentre.y);
                    context.arc(seatCentre.x, seatCentre.y, scaledSeatRadius, 0, Math.PI * 2, false); // Check to see if we're hovering over this seat.

                    if (mouse.x > seatCentre.x - scaledSeatRadius && mouse.x < seatCentre.x + scaledSeatRadius && mouse.y > seatCentre.y - scaledSeatRadius && mouse.y < seatCentre.y + scaledSeatRadius) {
                        hoveredSeatThisFrame = seat;
                    }
                }

                context.lineWidth = scaledSeatRadius / 4;
                context.setLineDash([]);
                context.strokeStyle = renderGroup.style.colour;
                context.stroke();
            } else if (seatIcon === enums.SeatIcon.StrokeCircleWithSpot) {
                // Draw the spot.
                context.beginPath();

                for (let i = 0; i < renderGroup.seats.length; i++) {
                    let seat = renderGroup.seats[i];
                    let seatCentre = translateToCanvasCoordinates(seat.layoutProperties.centre);

                    if (!planSeatIsVisible(seat)) {
                        continue;
                    }

                    let scaledSeatRadius = getScaledRadius(seat.layoutProperties.size) / 3; // Render the seat to the canvas.

                    context.moveTo(seatCentre.x + scaledSeatRadius, seatCentre.y);
                    context.arc(seatCentre.x, seatCentre.y, scaledSeatRadius, 0, Math.PI * 2, false);
                }

                context.fillStyle = renderGroup.style.colour;
                context.fill(); // Draw the seat stroke.

                context.beginPath();
                let scaledSeatRadius = getScaledRadius(renderGroup.seats[0].layoutProperties.size);
                let seatLineWidth = getLineWidth(scaledSeatRadius);
                scaledSeatRadius -= seatLineWidth / 2;

                for (let i = 0; i < renderGroup.seats.length; i++) {
                    let seat = renderGroup.seats[i];
                    let seatCentre = translateToCanvasCoordinates(seat.layoutProperties.centre); // Before we do anything else, lets make sure the seat is actually visible before we try to render it.

                    if (!planSeatIsVisible(seat)) {
                        continue;
                    } // Render the seat to the canvas.


                    context.moveTo(seatCentre.x + scaledSeatRadius, seatCentre.y);
                    context.arc(seatCentre.x, seatCentre.y, scaledSeatRadius, 0, Math.PI * 2, false); // Check to see if we're hovering over this seat.

                    if (mouse.x > seatCentre.x - scaledSeatRadius && mouse.x < seatCentre.x + scaledSeatRadius && mouse.y > seatCentre.y - scaledSeatRadius && mouse.y < seatCentre.y + scaledSeatRadius) {
                        hoveredSeatThisFrame = seat;
                    }
                }

                context.lineWidth = seatLineWidth;
                context.setLineDash([]);
                context.strokeStyle = renderGroup.style.colour;
                context.stroke();
            } else if (seatIcon === enums.SeatIcon.StrokeCircleWithTick) {
                // Draw the tick.
                context.beginPath();
                let scaledSeatRadius = getScaledRadius(renderGroup.seats[0].layoutProperties.size);
                let seatLineWidth = getLineWidth(scaledSeatRadius);
                scaledSeatRadius -= seatLineWidth / 2;
                let tickRadius = scaledSeatRadius * 0.4;
                let tickYOffset = tickRadius * 0.33;

                for (let i = 0; i < renderGroup.seats.length; i++) {
                    let seat = renderGroup.seats[i];
                    let seatCentre = translateToCanvasCoordinates(seat.layoutProperties.centre); // Before we do anything else, lets make sure the seat is actually visible before we try to render it.

                    if (!planSeatIsVisible(seat)) {
                        continue;
                    } // Render the seat to the canvas.


                    context.moveTo(seatCentre.x - tickRadius * 1.4, tickYOffset + (seatCentre.y - tickRadius * 0.4));
                    context.lineTo(seatCentre.x - tickRadius / 2, tickYOffset + (seatCentre.y + tickRadius / 2));
                    context.lineTo(seatCentre.x + tickRadius * 1.2, tickYOffset + (seatCentre.y - tickRadius * 1.2));
                }

                context.lineWidth = seatLineWidth;
                context.setLineDash([]);
                context.strokeStyle = renderGroup.style.colour;
                context.stroke(); // Draw the seat stroke.

                context.beginPath();

                for (let i = 0; i < renderGroup.seats.length; i++) {
                    let seat = renderGroup.seats[i];

                    if (seat.layoutProperties.isDisabled) {
                        continue;
                    }

                    let seatCentre = translateToCanvasCoordinates(seat.layoutProperties.centre); // Render the seat to the canvas.

                    context.moveTo(seatCentre.x + scaledSeatRadius, seatCentre.y);
                    context.arc(seatCentre.x, seatCentre.y, scaledSeatRadius, 0, Math.PI * 2, false); // Check to see if we're hovering over this seat.

                    if (mouse.x > seatCentre.x - scaledSeatRadius && mouse.x < seatCentre.x + scaledSeatRadius && mouse.y > seatCentre.y - scaledSeatRadius && mouse.y < seatCentre.y + scaledSeatRadius) {
                        hoveredSeatThisFrame = seat;
                    }
                }

                context.lineWidth = scaledSeatRadius / 4;
                context.setLineDash([]);
                context.strokeStyle = renderGroup.style.colour;
                context.stroke();
            }
        } // If we have a current hovered seat in this render frame, set it.
        // Note: the check for mouse x:y is to prevent a hover state on seats if they're positioned in the very
        // top left of the canvas, as the mouse resets to 0:0 when the canvas isn't hovered.


        if (!!hoveredSeatThisFrame && mouse.x !== 0 && mouse.y !== 0) {
            hoveredSeat = hoveredSeatThisFrame;
        } else {
            hoveredSeat = null;
        }
    }; // Render the shapes of the plan layout.


    var renderPlanLayoutShapes = context => {
        if (layout == null || layout.shapes == null || layout.shapes.length === 0) {
            return;
        }

        context.lineWidth = 1;
        context.fillStyle = "#ffffff";
        let defaultStrokeType = "#aaaaaa";
        layout.shapes.forEach(layoutShape => {
            let shape = layoutShape.layoutProperties.shape;
            context.beginPath();

            if (shape === enums.Shape.Line && layoutShape.layoutProperties.vertices.length == 2) {
                let firstVertex = translateToCanvasCoordinates(layoutShape.layoutProperties.vertices[0]);
                let secondVertex = translateToCanvasCoordinates(layoutShape.layoutProperties.vertices[1]);
                context.moveTo(firstVertex.x, firstVertex.y);
                context.lineTo(secondVertex.x, secondVertex.y);
                context.strokeStyle = layoutShape.strokeStyle || defaultStrokeType;
                context.stroke();
            }
        });
    }; // Render the text items from the plan layout.


    var renderPlanLayoutTexts = context => {
        if (layout == null || layout.texts == null || layout.texts.length === 0) {
            return;
        }

        let selectedLayoutBlock = null;

        if (hoveredBlock != null) {
            let layoutBlocks = getBlocks(hoveredBlock.label);

            for (let i = 0; i < layoutBlocks.length; i++) {
                if (layoutBlocks[i] === hoveredBlock) {
                    selectedLayoutBlock = layoutBlocks[i].layoutProperties;
                }
            }
        }

        for (let i = 0; i < layout.texts.length; i++) {
            let dpr = canvasProperties.devicePixelRatio;
            let zoom = canvasProperties.zoom;
            let text = layout.texts[i];
            let textProps = text.layoutProperties;
            let centre = translateToCanvasCoordinates(textProps.centre);
            context.font = `${textProps.fontSize * zoom * dpr}px ${textProps.fontFamily}`;
            context.textAlign = "center";
            context.textBaseline = "middle"; // Rotate the canvas if the text has any rotation.

            if (!!textProps.rotation && textProps.rotation !== 0) {
                context.translate(centre.x, centre.y);
                context.rotate(textProps.rotation * Math.PI / 180);
                context.translate(-centre.x, -centre.y);
            }

            let hideText = false; // Do a check to see if this text block is on top of any selected block. Ideally we'd check for 
            // all intersecting axes here, but mostly we could assume that the text centre is within the 
            // block bounds, e.g. for people labelling the blocks with text, so we'll just check that for simplicity.

            if (!!selectedLayoutBlock && !!text.layoutProperties.isHideable) {
                // Plot the block so we can use the canvas to check for the text centre in the path.
                context.beginPath();
                let firstVertex = translateToCanvasCoordinates(selectedLayoutBlock.vertices[0]);
                context.moveTo(firstVertex.x, firstVertex.y);

                for (let j = 1; j < selectedLayoutBlock.vertices.length; j++) {
                    let vertex = translateToCanvasCoordinates(selectedLayoutBlock.vertices[j]);
                    context.lineTo(vertex.x, vertex.y);
                }

                if (context.isPointInPath(centre.x, centre.y)) {
                    // Don't render this text item.
                    hideText = true;
                }

                context.closePath();
            }

            if (hideText) {
                context.fillStyle = colourToRgba(textProps.fontColour, 0.25);
            } else {
                context.fillStyle = textProps.fontColour;
            }

            context.fillText(text.label, centre.x, centre.y); // Rotate the canvas back if we have it currently rotated.

            if (!!textProps.rotation && textProps.rotation !== 0) {
                context.translate(centre.x, centre.y);
                context.rotate(-textProps.rotation * Math.PI / 180);
                context.translate(-centre.x, -centre.y);
            }
        }
    }; // Render the images items from the plan layout.


    var renderPlanLayoutImages = context => {
        if (layout == null || layout.images == null || layout.images.length === 0) {
            return;
        }

        for (let i = 0; i < layout.images.length; i++) {
            let image = layout.images[i];

            if (!image.loaded) {
                continue;
            }

            let dpr = canvasProperties.devicePixelRatio;
            let zoom = canvasProperties.zoom;
            let imageprops = image.layoutProperties;
            let centre = translateToCanvasCoordinates(imageprops.centre);
            let width = image.width * zoom * dpr;
            let height = image.height * zoom * dpr;
            let x = centre.x - width / 2;
            let y = centre.y - height / 2; // Rotate the canvas if the image has any rotation.

            if (!!imageprops.rotation && imageprops.rotation !== 0) {
                context.translate(centre.x, centre.y);
                context.rotate(imageprops.rotation * Math.PI / 180);
                context.translate(-centre.x, -centre.y);
            }

            context.drawImage(image.obj, x, y, width, height); // Rotate the canvas back if we have it currently rotated.

            if (!!imageprops.rotation && imageprops.rotation !== 0) {
                context.translate(centre.x, centre.y);
                context.rotate(-imageprops.rotation * Math.PI / 180);
                context.translate(-centre.x, -centre.y);
            }
        }
    };

    var renderSeatLabels = context => {
        let seats = layout.blocks.flatMap(b => b.rows).flatMap(r => r.seats).filter(s => planSeatIsVisible(s) && !s.layoutProperties.isDisabled);
        seats.forEach(seat => {
            let fontSize = seat.layoutProperties.size * labelProperties.fontSizeRatio * canvasProperties.zoom * canvasProperties.devicePixelRatio;
            let scaledCentre = translateToCanvasCoordinates(seat.layoutProperties.centre);
            scaledCentre.y += fontSize / 20;
            context.font = `${fontSize}px ${labelProperties.fontFamily}`;
            context.fillStyle = labelProperties.colour;
            context.textAlign = "center";
            context.textBaseline = "middle";
            context.fillText(seat.label, scaledCentre.x, scaledCentre.y);
        });
    };

    var renderSelectionMarquee = context => {
        if (!mouse.marqueePoint) {
            return;
        }

        context.beginPath(); // Note: using mouse.lastX/Y here because mouse.x/y will get nullified on mouseout, which will happen if the pointer hovers over the 
        // tooltip during dragging, which in turn will cause the selection marquee to jump about in an ugly way.

        context.rect(mouse.marqueePoint.x, mouse.marqueePoint.y, mouse.lastX - mouse.marqueePoint.x, mouse.lastY - mouse.marqueePoint.y);
        let dpr = canvasProperties.devicePixelRatio;
        context.lineWidth = 2 * dpr;
        context.setLineDash([7 * dpr, 5 * dpr]);
        context.strokeStyle = colourPalette.marquee;
        context.stroke();
    };

    var selectSeat = seat => {
        if (!seat || !seat.isSelectable) {
            return;
        }

        var index = selectedSeats.indexOf(seat);

        if (index === -1) {
            selectedSeats.push(seat);
        }
    };

    var setDefaultTooltipContent = seat => {
        tooltip.container.innerHTML = `
            <div class="tooltip__block">
                ${seat.blockLabel}
            </div>
            <div class="tooltip__row-seat">
                ${seat.rowLabel}${seat.label}
            </div>`;
    };

    var setDefaultTooltipPosition = seat => {
        if (touch.isTouched) {
            // Display tooltip in a centre of the canvas. Set position for the top left vertex of the tooltip.
            let container = tooltip.container;
            let width = container.offsetWidth;
            let height = container.offsetHeight;
            var canvasCentre = {
                x: canvas.offsetWidth / 2,
                y: canvas.offsetHeight / 2
            };
            var tooltipHalfWidth = width / 2;
            var tooltipHalfHeight = height / 2;
            var positionLeftTopTooltipPoint = {
                x: canvasCentre.x - tooltipHalfWidth,
                y: canvasCentre.y - tooltipHalfHeight
            };
            tooltip.x = Math.max(0, positionLeftTopTooltipPoint.x);
            tooltip.y = Math.max(0, positionLeftTopTooltipPoint.y);
        } else {
            let scaledCoords = {
                "x": 0,
                "y": 0
            };
            let scaledRadius = 0;
            let zoom = canvasProperties.zoom;
            scaledCoords = translateToCanvasCoordinates(seat.layoutProperties.centre, false);
            scaledRadius = seat.layoutProperties.size / 2 * zoom;
            let position = {
                "x": scaledCoords.x + scaledRadius,
                "y": scaledCoords.y + scaledRadius
            }; // Set up the positioning of the tooltip.

            let container = tooltip.container;
            let width = container.offsetWidth;
            let height = container.offsetHeight; // Make sure the tooltip doesn't go off the canvas in the x dimension...

            if (position.x + width > canvas.offsetWidth) {
                position.x = Math.max(0, scaledCoords.x - scaledRadius - width);
            } // ... and in the y dimension.


            if (position.y + height > canvas.offsetHeight) {
                position.y = Math.max(0, scaledCoords.y - scaledRadius - height);
            }

            tooltip.x = position.x;
            tooltip.y = position.y;
        }
    };

    var setZoom = zoom => {
        canvasProperties.zoom = zoom;
        canvasProperties.zoomOffset = getZoomOffset(canvasProperties.zoom);
    };

    var setZoomToTargets = (offset, zoom, zoomOffset) => {
        // Store current offsets and zoom levels as we need to use the working canvas to do some calculations.
        let currentOffsetX = canvasProperties.offset.x;
        let currentOffsetY = canvasProperties.offset.y;
        let currentZoom = canvasProperties.zoom;
        let currentZoomOffset = canvasProperties.zoomOffset; // Set the new values on the working canvas to effectively find the end result.

        canvasProperties.offset.x = offset.x;
        canvasProperties.offset.y = offset.y;
        canvasProperties.zoom = zoom;
        canvasProperties.zoomOffset = zoomOffset; // Now get the plan bounds and work out if we've pulled the plan away from the edges, so that we can then calculate
        // how much we need to pull it back.

        let planBounds = translateToCanvasBounds(getPlanBounds(true));

        if (planBounds.left > 0 && planBounds.right < canvas.width) {
            canvasProperties.offset.x = canvasProperties.defaultOffset.x;
        } else if (planBounds.left > 0) {
            canvasProperties.offset.x = canvasProperties.offset.x - planBounds.left / canvasProperties.zoom;
        } else if (planBounds.right < canvas.width) {
            canvasProperties.offset.x = canvasProperties.offset.x - (planBounds.right - canvas.width) / canvasProperties.zoom;
        }

        if (planBounds.top > 0 && planBounds.bottom < canvas.height) {
            canvasProperties.offset.y = canvasProperties.defaultOffset.y;
        } else if (planBounds.top > 0) {
            canvasProperties.offset.y = canvasProperties.offset.y - planBounds.top / canvasProperties.zoom;
        } else if (planBounds.bottom < canvas.height) {
            canvasProperties.offset.y = canvasProperties.offset.y - (planBounds.bottom - canvas.height) / canvasProperties.zoom;
        } // Set the target offset and zoom to the final values calculated above, then reset the working canvas.


        canvasProperties.targetOffset = {
            "x": canvasProperties.offset.x,
            "y": canvasProperties.offset.y
        };
        canvasProperties.targetZoom = zoom;
        canvasProperties.offset.x = currentOffsetX;
        canvasProperties.offset.y = currentOffsetY;
        canvasProperties.zoom = currentZoom;
        canvasProperties.zoomOffset = currentZoomOffset;
    };

    var startRenderLoop = () => {
        // Set off an interval to call the zoom animation functions.
        setInterval(() => {
            animateZoom();
        }, renderInterval); // Kick off the main render loop.

        requestAnimationFrame(render);
    };

    var translateToCanvasBounds = bounds => {
        return {
            left: translateToCanvasPoint(bounds.left, enums.Axis.X),
            right: translateToCanvasPoint(bounds.right, enums.Axis.X),
            top: translateToCanvasPoint(bounds.top, enums.Axis.Y),
            bottom: translateToCanvasPoint(bounds.bottom, enums.Axis.Y)
        };
    };

    var translateToCanvasCoordinates = (coords, applyDpr = true) => {
        if (isNaN(coords.x)) {
            coords.x = 0;
        }

        if (isNaN(coords.y)) {
            coords.y = 0;
        }

        return {
            x: translateToCanvasPoint(coords.x, enums.Axis.X, applyDpr),
            y: translateToCanvasPoint(coords.y, enums.Axis.Y, applyDpr)
        };
    }; // Methods to translate given values or coordinates from plan level to determine the 
    // relative coordinates on the canvas. This is for rendering plan items to the html 5 canvas.


    var translateToCanvasPoint = (value, axis, applyDpr = true) => {
        let dpr = canvasProperties.devicePixelRatio;
        let offset = 0;
        let zoom = canvasProperties.zoom;
        let zoomOffset = 0;

        switch (axis) {
            case enums.Axis.X:
                offset = canvasProperties.offset.x;
                zoomOffset = canvasProperties.zoomOffset.x;
                break;

            case enums.Axis.Y:
                offset = canvasProperties.offset.y;
                zoomOffset = canvasProperties.zoomOffset.y;
                break;

            default:
                break;
        }

        var resultWithZoom = (value + offset + zoomOffset) * zoom;
        return applyDpr ? resultWithZoom * dpr : resultWithZoom;
    }; // Translate given coordinates from canvas level to determine the relative coordinates on the plan grid.
    // This is for calculating what plan grid point the mouse is currently over.


    var translateToPlanCoordinates = coords => {
        let dpr = canvasProperties.devicePixelRatio;
        let offset = canvasProperties.offset;
        let zoom = canvasProperties.zoom;
        let zoomOffset = canvasProperties.zoomOffset;
        return {
            x: coords.x / zoom / dpr - offset.x - zoomOffset.x,
            y: coords.y / zoom / dpr - offset.y - zoomOffset.y
        };
    };

    var updateMouse = (x, y, event) => {
        if (!!tooltip && (!tooltip.overrides || !tooltip.overrides.hide)) {
            // Hide the tooltip because we'll re-check whether it needs to be displayed.
            hideTooltip();
        }

        let canvasPosition = canvas.getBoundingClientRect();
        let dpr = canvasProperties.devicePixelRatio;
        mouse.x = parseInt(x - canvasPosition.left) * dpr;
        mouse.y = parseInt(y - canvasPosition.top) * dpr;

        if (mouse.isDragged) {
            let planBounds = translateToCanvasBounds(getPlanBounds(true));
            let zoom = canvasProperties.zoom;
            let deltaX = (mouse.lastX - mouse.x) / zoom;
            let deltaY = (mouse.lastY - mouse.y) / zoom; // Round the values we use for comparison to stop the seat picker dancing when
            // dragged near an edge.

            deltaX = Math.round(deltaX);
            deltaY = Math.round(deltaY);
            planBounds.top = Math.round(planBounds.top);
            planBounds.right = Math.round(planBounds.right);
            planBounds.bottom = Math.round(planBounds.bottom);
            planBounds.left = Math.round(planBounds.left); // Prevent the plan from scrolling in the x axis if the bounds have come away from the edges.

            if (deltaX < 0 && planBounds.left < 0 || deltaX > 0 && planBounds.right > canvas.width) {
                canvasProperties.offset.x -= deltaX;
            } // Correct for plan bounds leaving the edges in the x axis.


            if (planBounds.left > 0 && planBounds.right > canvas.width) {
                canvasProperties.offset.x -= planBounds.left / zoom;
            } else if (planBounds.right < canvas.width && planBounds.left < 0) {
                canvasProperties.offset.x += (canvas.width - planBounds.right) / zoom;
            } // Prevent the plan from scrolling in the y axis if the bounds have come away from the edges.


            if (deltaY < 0 && planBounds.top < 0 || deltaY > 0 && planBounds.bottom > canvas.height) {
                canvasProperties.offset.y -= deltaY;
            } // Correct for plan bounds leaving the edges in the y axis.


            if (planBounds.top > 0 && planBounds.bottom > canvas.height) {
                canvasProperties.offset.y -= planBounds.top / zoom;
            } else if (planBounds.bottom < canvas.height && planBounds.top < 0) {
                canvasProperties.offset.y += (canvas.height - planBounds.bottom) / zoom;
            }
        }

        if ((mouse.isDragged || !!mouse.marqueePoint) && (mouse.lastX !== mouse.x || mouse.lastY !== mouse.y)) {
            mouse.moved = true;
        }

        mouse.lastX = mouse.x;
        mouse.lastY = mouse.y;

        if (mouse.marqueePoint) {
            updateSeatsInMarquee();
        } // Check seat hover state to display default tooltip if necessary.


        checkSeatHover(event);
    };

    var updateSeatsInMarquee = () => {
        if (!mouse.marqueePoint) {
            return;
        }

        marqueeSeats = [];
        let marqueePointPlanCoords = translateToPlanCoordinates(mouse.marqueePoint);
        let mousePlanCoords = translateToPlanCoordinates(mouse);
        let marqueeBounds = {
            minX: Math.min(marqueePointPlanCoords.x, mousePlanCoords.x),
            minY: Math.min(marqueePointPlanCoords.y, mousePlanCoords.y),
            maxX: Math.max(marqueePointPlanCoords.x, mousePlanCoords.x),
            maxY: Math.max(marqueePointPlanCoords.y, mousePlanCoords.y)
        };

        for (let i = 0; i < layout.blocks.length; i++) {
            let block = layout.blocks[i];

            for (let j = 0; j < block.rows.length; j++) {
                let row = block.rows[j];

                for (let k = 0; k < row.seats.length; k++) {
                    let seat = row.seats[k];

                    if (seat.layoutProperties.isDisabled || !seat.isSelectable) {
                        continue;
                    }

                    let seatCentre = seat.layoutProperties.centre;

                    if (seatCentre.x > marqueeBounds.minX && seatCentre.x < marqueeBounds.maxX && seatCentre.y > marqueeBounds.minY && seatCentre.y < marqueeBounds.maxY) {
                        marqueeSeats.push(seat);
                    }
                }
            }
        }
    };

    var updateMarqueePoint = (event, x, y) => {
        let canvasPosition = canvas.getBoundingClientRect();
        let dpr = canvasProperties.devicePixelRatio;
        mouse.marqueePoint = {
            "x": parseInt(x - canvasPosition.left) * dpr,
            "y": parseInt(y - canvasPosition.top) * dpr
        };

        if (!event.shiftKey && !event.altKey) {
            let changedSeats = selectedSeats;
            selectedSeats = [];
            let selectionChangedEvent = new CustomEvent(events.selectionChanged);
            selectionChangedEvent.changedSeats = changedSeats;
            selectionChangedEvent.marqueeSeats = marqueeSeats;
            selectionChangedEvent.originalEvent = event;
            selectionChangedEvent.selectedSeats = selectedSeats;
            document.dispatchEvent(selectionChangedEvent);
        }
    };

    var updateUnassignedSeats = () => {
        // Get the unassigned category and reset its seats.
        let unassignedCategory = categories.find(category => category.label === unassignedCategoryLabel);
        unassignedCategory.seats = []; // Get an array of all seats in the layout.

        let layoutSeats = [];

        for (let i = 0; i < layout.blocks.length; i++) {
            let block = layout.blocks[i];

            for (let j = 0; j < block.rows.length; j++) {
                let row = block.rows[j];

                for (let k = 0; k < row.seats.length; k++) {
                    layoutSeats.push(row.seats[k]);
                }
            }
        } // Get an array of all seats that are in a category.


        let categorySeats = [];
        categories.forEach(category => {
            categorySeats = categorySeats.concat(category.seats);
        }); // Filter out a list of layout seats that aren't in a category and put them back in the unassigned category.

        unassignedCategory.seats = layoutSeats.filter(layoutSeat => {
            return !categorySeats.find(categorySeat => categorySeat === layoutSeat);
        }); // Make sure the unassigned category seats aren't selectable.

        unassignedCategory.seats.forEach(seat => {
            seat.isSelectable = false;
        });
    }; // Zoom the viewport to make the selected block fill the window. Note: there will be no zoom if the 
    // block is already at least filling the window.


    var zoomToBlock = layoutBlock => {
        if (!layoutBlock) {
            return;
        }

        let blockBounds = calculateBlockBounds(layoutBlock.layoutProperties.vertices);
        let dpr = canvasProperties.devicePixelRatio;
        let planPadding = canvasProperties.planPadding;
        let height = blockBounds.bottom - blockBounds.top;
        let width = blockBounds.right - blockBounds.left;
        height += planPadding * 2;
        width += planPadding * 2;
        let blockRatio = width / height;
        let canvasRatio = canvas.width / canvas.height;
        let zoom = 1;

        if (blockRatio > canvasRatio) {
            zoom = canvas.width / width;
        } else {
            zoom = canvas.height / height;
        } // Scale the zoom to the screen's pixel density.


        zoom /= dpr; // Perform the zoom if we're zooming in, and not out.

        if (canvasProperties.zoom < zoom) {
            // Calculate the offsets to centre the target block.
            let canvasHalfWidth = canvas.width / 2 / zoom / dpr;
            let blockHalfWidth = width / 2;
            let zoomOffset = getZoomOffset(zoom);
            let zoomToOffset = {
                "x": canvasHalfWidth - blockHalfWidth - blockBounds.left - zoomOffset.x + planPadding,
                "y": -blockBounds.top - zoomOffset.y + planPadding
            };
            setZoomToTargets(zoomToOffset, zoom, zoomOffset);
        }
    };

    var zoomToSeat = (layoutSeat, zoomLevel) => {
        if (!layoutSeat) {
            return;
        }

        let dpr = canvasProperties.devicePixelRatio;
        let zoom = zoomLevel || 3;
        let zoomOffset = getZoomOffset(zoom);
        let canvasHalfWidth = canvas.width / 2 / zoom / dpr;
        let canvasHalfHeight = canvas.height / 2 / zoom / dpr;
        let seatRadius = layoutSeat.layoutProperties.size / 2 / zoom / dpr;
        let zoomToOffset = {
            "x": canvasHalfWidth - layoutSeat.layoutProperties.centre.x - seatRadius - zoomOffset.x,
            "y": canvasHalfHeight - layoutSeat.layoutProperties.centre.y - seatRadius - zoomOffset.y
        };
        setZoomToTargets(zoomToOffset, zoom, zoomOffset);
    };

    initialise();
    return {
        "addUpdateCategory": addUpdateCategory,
        "canvas": canvas,
        "canvasControls": canvasControls,
        "canvasProperties": canvasProperties,
        "categories": categories,
        "clearSelectedSeats": clearSelectedSeats,
        "colourPalette": colourPalette,
        "enums": enums,
        "displayTooltip": displayTooltip,
        "events": events,
        "getAllBlocks": getAllBlocks,
        "getAllTexts": getAllTexts,
        "getBlockOfSeat": getBlockOfSeat,
        "getBlocks": getBlocks,
        "getHasRenderedFirstFrame": getHasRenderedFirstFrame,
        "getRenderInterval": getRenderInterval,
        "getRowOfSeat": getRowOfSeat,
        "getSeat": getSeat,
        "getSeatIndexInRow": getSeatIndexInRow,
        "getSeats": getSeats,
        "getMarqueeSeats": getMarqueeSeats,
        "getSelectedSeats": getSelectedSeats,
        "getTexts": getTexts,
        "getZoom": getZoom,
        "helpers": helpers,
        "hideTooltip": hideTooltip,
        "isSeatSelected": isSeatSelected,
        "labelProperties": labelProperties,
        "layout": layout,
        "mouse": mouse,
        "removeCategory": removeCategory,
        "seatStyles": seatStyles,
        "selectSeat": selectSeat,
        "setDefaultTooltipPosition": setDefaultTooltipPosition,
        "setZoom": setZoom,
        "tooltip": tooltip,
        "touch": touch,
        "translateToCanvasCoordinates": translateToCanvasCoordinates,
        "translateToPlanCoordinates": translateToPlanCoordinates,
        "unselectSeat": unselectSeat,
        "zoomToBlock": zoomToBlock,
        "zoomToSeat": zoomToSeat
    };
}; // ----- Helpers ----- //


SeatPicker.generateBase64SeatPickerIcon = async function (size, icon, colour) {
    return new Promise(resolve => {
        let iconCanvas = document.createElement("canvas");
        iconCanvas.id = "icon-canvas";
        iconCanvas.width = size;
        iconCanvas.height = size;
        document.querySelector("body").appendChild(iconCanvas);
        let iconLayout = {
            "blocks": [{
                "layoutProperties": {
                    "vertices": [{
                        "x": 0,
                        "y": 0
                    }, {
                        "x": size,
                        "y": 0
                    }, {
                        "x": size,
                        "y": size
                    }, {
                        "x": 0,
                        "y": size
                    }]
                },
                "rows": [{
                    "seats": [{
                        "layoutProperties": {
                            "centre": {
                                "x": size / 2,
                                "y": size / 2
                            },
                            "size": size
                        },
                        "style": {
                            "colour": colour,
                            "icon": icon
                        }
                    }]
                }]
            }]
        };
        let iconSeatPicker = new SeatPicker({
            "canvas": iconCanvas,
            "canvasControls": {
                "zoomAmountWheel": 0
            },
            "canvasProperties": {
                "planPadding": 0
            },
            "layout": iconLayout,
            "mouse": {
                "buttons": {
                    "drag": null,
                    "marquee": null,
                    "select": null
                }
            },
            "tooltip": {
                "overrides": {
                    "content": true,
                    "display": true,
                    "position": true
                }
            }
        });

        if (iconSeatPicker.getHasRenderedFirstFrame()) {
            resolve(iconCanvas.toDataURL("image/png"));
            iconCanvas.remove();
        } else {
            // Listen for the first render frame event
            document.addEventListener(iconSeatPicker.events.renderedFirstFrame, () => {
                resolve(iconCanvas.toDataURL("image/png"));
                iconCanvas.remove();
            });
        }
    });
}; // Emulate enums in javascript.


function Enum(args) {
    for (var i = 0; i < args.length; i++) {
        this[args[i]] = args[i];
    }

    Object.freeze(this);
} // ----- Polyfills/Prototypes ----- //
// CustomEvent


(() => {
    if (typeof window.CustomEvent === "function") return false;

    function CustomEvent(event, params) {
        params = params || {
            bubbles: false,
            cancelable: false,
            detail: null
        };
        var evt = document.createEvent('CustomEvent');
        evt.initCustomEvent(event, params.bubbles, params.cancelable, params.detail);
        return evt;
    }

    window.CustomEvent = CustomEvent;
})(); // ----- Module Exports ----- //;
const DEFAULT_AREA_SIZE = 50000;
function seatStorage(controller, view, model) {
    this.controller = controller;
    this.view = view;
    this.model = model;
    this.storedAreas = [];
    this.loadedAreasCounter = 0;
};

seatStorage.prototype.clear = function () {
    this.storedAreas.length = 0;
}

seatStorage.prototype.getSeats = function (x1, y1, x2, y2) {
    var areas = this.getNormalizedAreas(x1, y1, x2, y2);
    var existentAreas = [];

    for (var i = 0; i < areas.length;) {
        var area = areas[i];
        var areaExists = false;
        for (var j = 0; j < this.storedAreas.length; j++)
            if (area.x1 == this.storedAreas[j].x1 && area.y1 == this.storedAreas[j].y1) {
                areaExists = true;
                area = this.storedAreas[j];
                break;
            }

        if (areaExists) {
            existentAreas.push(area);
            areas.splice(i, 1);
        }
        else
            i++;
    }

//    while (this.getNewAreas(areas));

    if (areas.length > 0) {
        for (var i = 0; i < areas.length; i++)
            this.storedAreas.push(areas[i]);
        var seatMap = this.model.seatMap;
        var self = this;
        this.controller.ticketService.getSeats(seatMap.performanceID, areas).then(function (responseData) {
            var data = responseData.data;
            var responseAreas = responseData.areas;

            if (data.succeeded) {
                for (var i = 0; i < responseAreas.length; i++) {
                    for (var j = 0; j < seatMap.selectedSeats.length; j++) {
                        var seat = seatMap.selectedSeats[j];
                        if (responseAreas[i].isPointInArea({ x: seat.x, y: seat.y }))
                            for (var seatIndex = 0; seatIndex < data.areas[i].seats.length; seatIndex++) {
                                if (data.areas[i].seats[seatIndex].id == seat.id) {
                                    data.areas[i].seats[seatIndex].selected = seat.selected;
                                    seatMap.selectedSeats.splice(j, 1);
                                    j--;
                                    break;
                                }
                            }
                    }
                    responseAreas[i].loaded = true;
                    responseAreas[i].seats = data.areas[i].seats;
                    self.loadedAreasCounter++;
                    self.view.onSeatsLoaded(responseAreas[i]);
                }
            }
        });
    }

    return existentAreas;
}

seatStorage.prototype.getNewAreas = function (areas) {
    var newAreasCreated = false;

    for (var i = 0; i < this.storedAreas.length; i++)
    {
        for (var j = 0; j < areas.length;) {
            var newAreas = this.storedAreas[i].getNewAreas(areas[j]);
            if (newAreas != null) {
                areas.splice(j, 1);
                if (newAreas.length > 0) {
                    newAreasCreated = true;
                    for (var areaIndex = 0; areaIndex < newAreas.length; areaIndex++)
                        areas.push(newAreas[areaIndex]);
                }
            }
            else
                j++;
        }
    }
    return newAreasCreated;
}

seatStorage.prototype.getNormalizedAreas = function (x1, y1, x2, y2) {
    var fullArea = {
        x1: Math.floor(x1 / DEFAULT_AREA_SIZE) * DEFAULT_AREA_SIZE,
        y1: Math.floor(y1 / DEFAULT_AREA_SIZE) * DEFAULT_AREA_SIZE,
        x2: Math.ceil(x2 / DEFAULT_AREA_SIZE) * DEFAULT_AREA_SIZE,
        y2: Math.ceil(y2 / DEFAULT_AREA_SIZE) * DEFAULT_AREA_SIZE
    };
    var xCounter = (fullArea.x2 - fullArea.x1) / DEFAULT_AREA_SIZE;
    var yCounter = (fullArea.y2 - fullArea.y1) / DEFAULT_AREA_SIZE;
    var areas = [];
    var x = fullArea.x1;
    for (var xIndex = 0; xIndex < xCounter; xIndex++) {
        var y = fullArea.y1;
        for (var yIndex = 0; yIndex < yCounter; yIndex++) {
            areas.push(new seatStorageArea(x, y, x + DEFAULT_AREA_SIZE, y + DEFAULT_AREA_SIZE));
            y += DEFAULT_AREA_SIZE;
        }
        x += DEFAULT_AREA_SIZE;
    }
    return areas;
}

function seatStorageArea(x1, y1, x2, y2) {
    this.x1 = Math.floor(x1 / DEFAULT_AREA_SIZE) * DEFAULT_AREA_SIZE;
    this.y1 = Math.floor(y1 / DEFAULT_AREA_SIZE) * DEFAULT_AREA_SIZE;
    this.x2 = Math.ceil(x2 / DEFAULT_AREA_SIZE) * DEFAULT_AREA_SIZE;
    this.y2 = Math.ceil(y2 / DEFAULT_AREA_SIZE) * DEFAULT_AREA_SIZE;
    this.loaded = false;
    this.seats = [];
}

seatStorageArea.prototype.isPointInArea = function (point) {
    if (point.x >= this.x1 && point.x < this.x2 &&
        point.y >= this.y1 && point.y < this.y2)
        return true;

    return false;
}

seatStorageArea.prototype.getNewAreas = function (area) {
    // handle when areas do not intersect
    if ((this.x2 < area.x1) || (this.x1 >= area.x2) ||
        (this.y2 < area.y1) || (this.y1 >= area.y2))
        return null;

    var newAreas = [];

    if (this.x1 > area.x1)
        newAreas.push(new seatStorageArea(area.x1, area.y1, this.x1, Math.max(area.y2, this.y2)));
    if (this.y1 > area.y1)
        newAreas.push(new seatStorageArea(Math.max(this.x1, area.x1), area.y1, area.x2, this.y1));
    if (this.x2 < area.x2)
        newAreas.push(new seatStorageArea(this.x2, Math.max(this.y1, area.y1), area.x2, area.y2));
    if (this.y2 < area.y2)
        newAreas.push(new seatStorageArea(Math.max(this.x1, area.x1), this.y2, Math.min(this.x2, area.x2), area.y2));

    return newAreas;
}


;
(function() {
    'use strict';

    angular
        .module('webShop2App')
        .directive('priceCategoryBooking', priceCategoryBookingDirective);

    priceCategoryBookingDirective.$inject = ['$window'];
    
    function priceCategoryBookingDirective($window) {
        var directive = {
            link: link,
            restrict: 'EA',
            scope: {
                priceCategory: '=',
                currencyCode: '&'
            },
            templateUrl: '/Templates/PriceCategoryBooking.aspx'
        };
        return directive;

        function link(scope, element, attrs) {
            if (scope.fullPriceText === undefined)
                scope.fullPriceText = ticketportal.webshop.localization.fullPriceText;
            if (scope.priceCategory.selectedAmount === undefined)
                scope.priceCategory.selectedAmount = 0;
            if (scope.price === undefined) {
                scope.priceCategory.prices.some(function (price, index, array) {
                    if (price.currencySymbol == scope.currencyCode()) {
                        scope.price = price;
                        return true;
                    }
                    return false;
                });
                if (scope.price === undefined)
                    scope.price = {};
            }
        }
    }
})();;
(function () {
    'use strict';

    angular
        .module('webShop2App')
        .controller('sectionBookingController', controller);

    controller.$inject = ['$scope', '$uibModalInstance', 'section', 'currencyCode'];

    function controller($scope, $uibModalInstance, section, currencyCode) {
        $scope.section = section;
        $scope.currencyCode = currencyCode;
        $scope.cancel = function () {
            $uibModalInstance.dismiss('cancel');
        }

        $scope.addToBasket = function () {
            $uibModalInstance.close($scope.section);
        }

        activate();

        function activate() { }
    }
})();
;
// Modified:		31.08.2018 Starticket AG, St. Gallen rha    : #11324 Fix webshop checkout
(function () {
    'use strict';

    var controllerId = 'checkoutController';
    angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', 'ecommerceService', 'checkoutService', 'checkoutModel', 'pageService', '$modal', 'customizationService', controller]);

    function controller($rootScope, $scope, ecommerceService, checkoutService, checkoutModel, pageService, $modal, customizationService) {
        $scope.checkoutService = checkoutService;
        $scope.model = checkoutModel;
        $scope.modal = $modal,
        $scope.pageService = pageService;
        $scope.customizationService = customizationService;
        $scope.ecommerceService = ecommerceService;

        $scope.$watch('pageService.pageModel.currentStep', function () {
            if ($scope.pageService.getStep() === CURRENT_STEP_CHECKOUT) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.checkoutService.get().then(function (data) {
                    if (data.succeeded) {
                        $rootScope.$broadcast('event:BasketChanged', data);
                        $scope.customizationService.updateCheckoutModel($scope, data, $scope.model);
                        $scope.pageService.pageModel.isLoading = false;

                        // in iFrame post a message saying a delivery method was selected
                        $scope.pageService.postMessageToIFrameParent("checkoutIsReady");
                    }
                    else
                        $scope.handleCheckoutUpdateError(data);
                });
            }
        });

        $scope.$watch('model.successMessage', function () {
            if ($scope.model.isRedeemGiftCertificateAction)
                window.setTimeout(function () { $("#alternativePaymentRedeemedAlert").alert('close'); }, 2000);
        });

        // called whenever the delivery option selection changes
        $scope.deliveryOptionSelected = function (deliveryOption) {
            $scope.pageService.pageModel.isLoading = true;
            $scope.checkoutService.updateDelivery(deliveryOption.id).then(function (data) {
                if (data.succeeded) {
                    $rootScope.$broadcast('event:BasketChanged', data);
                    $scope.customizationService.updateCheckoutModel($scope, data, $scope.model);

                    for (var i = 0; i < $scope.model.deliveryOptions.length; i++) {
                        var currentDeliveryOption = $scope.model.deliveryOptions[i];
                        currentDeliveryOption.isSelected = (currentDeliveryOption.id === deliveryOption.id);
                    }

                    $scope.pageService.pageModel.isLoading = false;

                    // in iFrame post a message saying a delivery method was selected
                    $scope.pageService.postMessageToIFrameParent("deliveryMethodSelected");


                } else {
                    $scope.pageService.pageModel.isLoading = false;

                    $scope.handleCheckoutUpdateError(data);
                }
            });
        };

        $scope.handleCheckoutUpdateError = function (data) {
            if (data.error.code === ERROR_BASKETEXPIRED) {

                // reset the amount of items known to be in the basket
                $scope.pageService.setBasketItemCount(0);
                $scope.pageService.showModal({
                    message: data.error.message,
                    onClose: function () { window.location.reload(true); }
                });

            } else if (data.error.code === ERROR_GENERALBASKETERROR) {
                $scope.pageService.setStep(CURRENT_STEP_BASKET);
                $scope.pageService.pageModel.isLoading = false;
                $rootScope.$broadcast('event:ShowAlert', { title: '', message: data.error.message });
            } else if (data.error.code === ERROR_GENERALPATRONERROR) {
                $scope.pageService.setStep(CURRENT_STEP_PATRON);
                $scope.pageService.pageModel.isLoading = false;
                $rootScope.$broadcast('event:ShowAlert', { title: '', message: data.error.message });
            }
            else
                $rootScope.$broadcast('event:ShowAlert', { title: '', message: data.error.message });
        };

        $scope.displayDeliveryDetail = function (deliveryOption) {
            $rootScope.$broadcast('event:ShowAlert', { title: deliveryOption.name, message: deliveryOption.description });

        };

        // called whenever an alternative payment method is selected
        $scope.alternativePaymentMethodSelected = function (alternativePayment) {
            // Clear payment method code and error message
            if ($scope.selectedAlternativePaymentMethod) {
                $scope.selectedAlternativePaymentMethod.code = null;
            }
            alternativePayment.errorMessage = null;

            $scope.model.selectedAlternativePaymentMethod = alternativePayment;

            // in iFrame post a message saying an alternative payment method was selected
            $scope.pageService.postMessageToIFrameParent("alternativePaymentMethodSelected");
        };

        // called whenever the alternative payment is applied
        $scope.applyAlternativePayment = function () {
            $scope.pageService.pageModel.isLoading = true;
            $scope.checkoutService.applyAlternativePayment($scope.model.selectedAlternativePaymentMethod.id, $("#alternativePaymentMethodUserCode").val()).then(function (data) {
                $rootScope.$broadcast('event:BasketChanged', data);
                $scope.customizationService.updateCheckoutModel($scope, data, $scope.model);
                if ($scope.model.succeeded) {
                    $scope.model.selectedAlternativePaymentMethod = null;

                    // in iFrame post a message saying an alternative payment method was selected
                    $scope.pageService.postMessageToIFrameParent("alternativePaymentMethodApplied");
                }
                $scope.pageService.pageModel.isLoading = false;
            });
        };

        // called whenever the alternative payment is removed
        $scope.removeAlternativePayment = function (appliedAlternativePayment) {
            $scope.pageService.pageModel.isLoading = true;
            $scope.checkoutService.removeAlternativePayment(appliedAlternativePayment).then(function (data) {
                $rootScope.$broadcast('event:BasketChanged', data);
                $scope.customizationService.updateCheckoutModel($scope, data, $scope.model);
                $scope.pageService.pageModel.isLoading = false;

                // in iFrame post a message saying an alternative payment method was selected
                $scope.pageService.postMessageToIFrameParent("alternativePaymentMethodRemoved");
            });
        };

        // called whenever the alternative payment is cancelled
        $scope.cancelAlternativePayment = function () {
            // update the reference
            $scope.model.selectedAlternativePaymentMethod.errorMessage = '';
            $scope.model.selectedAlternativePaymentMethod.code = '';

            // cancel the reference to hide the html element
            $scope.model.selectedAlternativePaymentMethod = null;

            // in iFrame post a message saying an alternative payment method selection was canceled
            $scope.pageService.postMessageToIFrameParent("alternativePaymentMethodSelectionCanceled");
        };

        // payment option filter
        $scope.paymentOptionFilter = function (paymentOption) {
            if ($scope.model.displaySavedCreditCardPayments)
                return paymentOption.savedCreditCardId > 0;
            else
                return paymentOption.savedCreditCardId === 0;
        };

        // payment option filter change
        $scope.paymentOptionFilterChange = function (displaySavedCreditCards) {
            $scope.model.displaySavedCreditCardPayments = displaySavedCreditCards;
            $scope.model.selectedPaymentOption = null;
            $scope.model.selectedPaymentOptionId = null;
        };

        // called whenever a payment option is selected
        $scope.paymentOptionSelected = function (paymentOption) {
            $scope.pageService.pageModel.isLoading = true;
            $scope.checkoutService.updateDeliveryAndPayment($scope.model.selectedDeliveryOptionId, paymentOption).then(function (data) {
                if (data.succeeded) {
                    $rootScope.$broadcast('event:BasketChanged', data);
                    $scope.customizationService.updateCheckoutModel($scope, data, $scope.model);
                    $scope.pageService.pageModel.isLoading = false;

                    // in iFrame post a message saying a payment method was selected
                    $scope.pageService.postMessageToIFrameParent("paymentMethodSelected");

                } else {
                    $scope.handleCheckoutUpdateError(data);
                }
            });
        };

        // called whenever the terms and conditions has been clicked
        $scope.onAcceptedGeneralTermsAndConditionsChange = function () {
            $scope.model.validateOptions();
        };

        $scope.onAcceptedInsuranceTermsAndConditionsChange = function () {
            $scope.model.validateOptions();
        };

        $scope.onAcceptedExtraConfirmationChange = function () {
            $scope.model.validateOptions();
        };

        $scope.back = function () {
            if ($scope.model.bookingSelected)
                $scope.model.bookingSelected = false;
            else if ($scope.model.reservationSelected)
                $scope.model.reservationSelected = false;
            else
                $scope.pageService.previousStep();
        };

        $scope.placeOrder = function (isValid) {
            $scope.model.checkoutFormSubmitted = true;
            if (isValid && $scope.model.canPlaceBasket) {
                $scope.model.errorMessage = '';
                if (!$scope.model.hasPersonalization || $scope.model.validatePersonalization($scope.model.basket.itemGroups)) {
                    $scope.pageService.pageModel.isLoading = true;
                    var placeOrderOptions = { delivery: $scope.model.selectedDeliveryOptionId, payment: $scope.model.selectedPaymentOption, additionalTicketEmail: $scope.model.additionalTicketEmail };
                    var beforePlaceOrderResult = $scope.customizationService.onBeforePlaceOrder(placeOrderOptions);

                    if (!beforePlaceOrderResult || beforePlaceOrderResult.succeeded) {
                        $scope.checkoutService.placeOrder(placeOrderOptions).then(function (data) {
                            if (data.succeeded) {
                                $scope.ecommerceService.pushCheckout($scope.model);
                                window.location = data.nextPageUrl;
                            } else {
                                if (data.error.code === ERROR_REQUIRESLOGIN) {
                                    window.location = data.loginUrl;
                                } else if (data.error.code === ERROR_BASKETEXPIRED) {
                                    $scope.pageService.pageModel.isLoading = false;
                                    $scope.handleCheckoutUpdateError(data);
                                } else if (data.error.code === ERROR_REQUIRESPATRONLOGIN) {
                                    $scope.pageService.pageModel.isLoading = false;
                                    $scope.pageService.pageModel.requiresPatronLogin = true;
                                } else if (data.error.code === ERROR_BASKETRULEERROR) {
                                    $scope.pageService.pageModel.isLoading = false;
                                    $scope.pageService.updateHasPendingBasketValidation(true);
                                    $scope.pageService.showModal({
                                        message: data.error.message,
                                        onClose: function () { $scope.pageService.setStep(CURRENT_STEP_BASKET); }
                                    });
                                } else {
                                    $scope.model.errorMessage = data.error.message;
                                    $scope.pageService.pageModel.isLoading = false;
                                }
                            }
                        });
                    } else {
                        $scope.pageService.pageModel.isLoading = false;
                        $rootScope.$broadcast('event:ShowAlert', { title: ticketportal.webshop.localization.errorTitle, message: beforePlaceOrderResult.errorMessage });
                    }
                } else {
                    $rootScope.$broadcast('event:ShowAlert', { title: ticketportal.webshop.localization.errorTitle, message: ticketportal.webshop.localization.personalizationMissing });
                }
            }
        };

        $scope.reserveOrder = function () {
            $scope.model.errorMessage = '';
            $scope.pageService.pageModel.isLoading = true;
            $scope.checkoutService.reserveOrder().then(function (data) {
                if (data.succeeded) {
                    window.location = data.nextPageUrl;
                } else {
                    if (data.error.code === ERROR_REQUIRESLOGIN) {
                        window.location = data.loginUrl;
                    } else if (data.error.code === ERROR_BASKETEXPIRED) {
                        $scope.pageService.pageModel.isLoading = false;
                        $scope.handleCheckoutUpdateError(data);
                    } else if (data.error.code === ERROR_REQUIRESPATRONLOGIN) {
                        $scope.pageService.pageModel.isLoading = false;
                        $scope.pageService.pageModel.requiresPatronLogin = true;
                    } else if (data.error.code === ERROR_BASKETRULEERROR) {
                        $scope.pageService.pageModel.isLoading = false;
                        $scope.pageService.updateHasPendingBasketValidation(true);
                        $scope.pageService.showModal({
                            message: data.error.message,
                            onClose: function () { $scope.pageService.setStep(CURRENT_STEP_BASKET); }
                        });
                    } else {
                        $scope.model.errorMessage = data.error.message;
                        $scope.pageService.pageModel.isLoading = false;
                    }
                }
            });
        };

        $scope.backToRegisterStepBilling = function () {
            $scope.pageService.pageModel.editAddress = true;
            $scope.pageService.setStep(CURRENT_STEP_PATRON);
        };

        $scope.backToBasket = function () {
            $scope.pageService.setStep(CURRENT_STEP_BASKET);
        }
    }
})();;
// Modified:		31.08.2018 Starticket AG, St. Gallen rha    : #11324 Fix webshop checkout
// Modified:		24.05.2017 Starticket AG, St. Gallen fbe    : #3591 Optimize ticket insurance workflow in web shop 2
(function () {
    'use strict';

    var modelId = 'checkoutModel';
    angular.module('webShop2App').factory(modelId, ['commonService', model]);

    function model(commonService) {
        var model = {
            deliveryOptions: [],
            paymentOptions: [],
            alternativePayments: [],
            appliedAlternativePayments: [],
            currencySymbol: '',
            displaySavedCreditCardPayments: false,
            hasSavedCreditCardPayments: false,
            formattedTotal: '',
            formattedTotalToPay: '',
            formattedSubTotal: '',
            formattedDeliveryAndPaymentFeeTotal: '',
            formattedTicketInsuranceTotal: '',
            hasAlternativePaymentMethods: false,
            isTicketInsuranceSelected: false,
            needsPaymentSelection: true,
            selectedDeliveryOption: null,
            selectedDeliveryOptionId: null,
            additionalTicketEmail: '', // additional email for print@home / mobile-ticket
            selectedPaymentOption: null,
            selectedPaymentOptionId: null,
            termsAndConditions: [],
            ticketInsuranceTermsAndConditionsURL: '',
            canDisplayTicketInsuranceTermsAndConditions: false,
            acceptedInsuranceTermsAndConditions: false,
            ticketInsuranceRequiresCheckoutConfirmation: true,
            canDisplayPlaceBasketSteps: false,
            acceptedGeneralTermsAndConditions: false,
            requiredExtraConfirmation: false,
            acceptedExtraConfirmation: false,
            deliveryAndPaymentFeeTotal: 0,
            subTotal: 0,
            total: 0,
            totalToPay: 0,
            ticketInsuranceTotal: 0,
            canPlaceBasket: false,
            errorMessage: '',
            isAddressIncomplete: false,
            selectedAlternativePaymentMethod: null,
            isRedeemGiftCertificateAction: false,
            successMessage: '',
            succeeded: true,
            orderFees: [],
            checkoutFormSubmitted: false,
            bookingAllowed: true,
            reservationAllowed: false,
            bookingSelected: false,
            reservationSelected: false,
            bookingOnly: false,
            reservationOnly: false,
            deliveryAddress: null,
            billingAddress: null,
            isDeliveryAddressSameAsBillingAddress: false,
            basket: null,
            hasPersonalization: false,

            // child models
            paymentOptionModel: paymentOptionModel,
            termsAndConditionsModel: termsAndConditionsModel,
            deliveryOptionModel: deliveryOptionModel,
            alternativePaymentMethodModel: alternativePaymentMethodModel,
            appliedAlternativePaymentModel: appliedAlternativePaymentModel,
            orderFeeModel: orderFeeModel,
            deliveryAddressModel: deliveryAddressModel,
            billingAddressModel: billingAddressModel,

            // functions
            update: update,
            validateOptions: validateOptions,
            validatePersonalization: validatePersonalization
        };

        return model;

        function update(source) {
            if (source.successMessage !== undefined) this.successMessage = source.successMessage;
            if (source.succeeded !== undefined) this.succeeded = source.succeeded;
            if (source.checkoutOptions) {
                var checkoutOptions = source.checkoutOptions;
                if (checkoutOptions.total !== undefined) this.total = checkoutOptions.total;
                if (checkoutOptions.totalToPay !== undefined) this.totalToPay = checkoutOptions.totalToPay;
                if (checkoutOptions.formattedTotal !== undefined) this.formattedTotal = checkoutOptions.formattedTotal;
                if (checkoutOptions.formattedTotalToPay !== undefined) this.formattedTotalToPay = checkoutOptions.formattedTotalToPay;
                if (checkoutOptions.formattedSubTotal !== undefined) this.formattedSubTotal = checkoutOptions.formattedSubTotal;
                if (checkoutOptions.formattedDeliveryAndPaymentFeeTotal !== undefined) this.formattedDeliveryAndPaymentFeeTotal = checkoutOptions.formattedDeliveryAndPaymentFeeTotal;
                if (checkoutOptions.formattedTicketInsuranceTotal !== undefined) this.formattedTicketInsuranceTotal = checkoutOptions.formattedTicketInsuranceTotal;
                if (checkoutOptions.isTicketInsuranceSelected !== undefined) this.isTicketInsuranceSelected = checkoutOptions.isTicketInsuranceSelected;
                if (checkoutOptions.needsPaymentSelection !== undefined) this.needsPaymentSelection = checkoutOptions.needsPaymentSelection;
                if (checkoutOptions.deliveryAndPaymentFeeTotal !== undefined) this.deliveryAndPaymentFeeTotal = checkoutOptions.deliveryAndPaymentFeeTotal;
                if (checkoutOptions.subTotal !== undefined) this.subTotal = checkoutOptions.subTotal;
                if (checkoutOptions.ticketInsuranceTotal !== undefined) this.ticketInsuranceTotal = checkoutOptions.ticketInsuranceTotal;
                if (checkoutOptions.isRedeemGiftCertificateAction !== undefined) this.isRedeemGiftCertificateAction = checkoutOptions.isRedeemGiftCertificateAction;
                if (checkoutOptions.currencySymbol !== undefined) this.currencySymbol = checkoutOptions.currencySymbol;
                if (checkoutOptions.bookingAllowed !== undefined) this.bookingAllowed = checkoutOptions.bookingAllowed;
                if (checkoutOptions.reservationAllowed !== undefined) this.reservationAllowed = checkoutOptions.reservationAllowed;
                if (checkoutOptions.ticketInsuranceRequiresCheckoutConfirmation !== undefined) this.ticketInsuranceRequiresCheckoutConfirmation = checkoutOptions.ticketInsuranceRequiresCheckoutConfirmation;
                if (checkoutOptions.deliveryAddress) this.deliveryAddress = new deliveryAddressModel(checkoutOptions.deliveryAddress);
                if (checkoutOptions.billingAddress) this.billingAddress = new billingAddressModel(checkoutOptions.billingAddress);
                if (checkoutOptions.basket !== undefined) {
                    this.basket = checkoutOptions.basket;
                    for (var groupIndex = 0; groupIndex < this.basket.itemGroups.length; groupIndex++) {
                        if (this.basket.itemGroups[groupIndex].hasPersonalization) {
                            this.hasPersonalization = true;
                            break;
                        }
                    }
                }
                this.bookingOnly = this.bookingAllowed && !this.reservationAllowed;
                this.reservationOnly = this.reservationAllowed && !this.bookingAllowed;
                if (checkoutOptions.deliveryOptions) {
                    this.deliveryOptions.length = 0;
                    for (var i = 0; i < checkoutOptions.deliveryOptions.length; i++)
                        this.deliveryOptions.push(new deliveryOptionModel(checkoutOptions.deliveryOptions[i]));
                }

                this.paymentOptions.length = 0;
                this.alternativePayments.length = 0;
                this.appliedAlternativePayments.length = 0;
                
                this.isDeliveryAddressSameAsBillingAddress = this.deliveryAddress.id === this.billingAddress.id;

                // has a selected delivery option
                var deliveryOption = checkoutOptions.delivery ? checkoutOptions.delivery : checkoutOptions.selectedDeliveryOption;
                if (deliveryOption) {
                    this.selectedDeliveryOption = new deliveryOptionModel(deliveryOption);
                    this.selectedDeliveryOptionId = deliveryOption.id;

                    // set the alternative payments
                    this.alternativePayments.length = 0;
                    for (var alternativePaymentIndex = 0; alternativePaymentIndex < checkoutOptions.alternativePaymentMethods.length; ++alternativePaymentIndex) {
                        var alternativePaymentModel = new alternativePaymentMethodModel(checkoutOptions.alternativePaymentMethods[alternativePaymentIndex]);
                        if (this.selectedAlternativePaymentMethod && this.selectedAlternativePaymentMethod.id === alternativePaymentModel.id)
                            this.selectedAlternativePaymentMethod = alternativePaymentModel;
                        this.alternativePayments.push(alternativePaymentModel);

                        for (var appliedAlternativePaymentIndex = 0; appliedAlternativePaymentIndex < alternativePaymentModel.appliedPayments.length; ++appliedAlternativePaymentIndex) {
                            this.appliedAlternativePayments.push(alternativePaymentModel.appliedPayments[appliedAlternativePaymentIndex]);
                        }
                    }
                    this.hasAlternativePaymentMethods = this.alternativePayments.length > 0;

                    // set the payment options
                    this.paymentOptions.length = 0;
                    this.selectedPaymentOption = null;
                    this.selectedPaymentOptionId = null;
                    for (var paymentIndex = 0; paymentIndex < deliveryOption.payments.length; ++paymentIndex) {
                        var paymentModel = new paymentOptionModel(deliveryOption.payments[paymentIndex]);
                        if (paymentModel.savedCreditCardId > 0)
                            this.hasSavedCreditCardPayments = true;
                        this.paymentOptions.push(paymentModel);
                        if (paymentModel.isSelected) {
                            this.selectedPaymentOption = paymentModel;
                            this.selectedPaymentOptionId = paymentModel.id;
                        }
                    }
                    if (this.selectedPaymentOption) {
                        this.displaySavedCreditCardPayments = this.selectedPaymentOption.savedCreditCardId > 0 ? true : false;

                        // set saved credit card session storage item
                        commonService.setSessionStorageItem("savedCreditCardId", this.selectedPaymentOption.savedCreditCardId);
                    }
                    else {
                        this.displaySavedCreditCardPayments = this.hasSavedCreditCardPayments;

                        // remove saved credit card session storage item
                        commonService.removeSessionStorageItem("savedCreditCardId");
                    }
                }
                else {
                    // reset delivery option and payment option
                    this.selectedDeliveryOption = null;
                    this.selectedDeliveryOptionId = null;
                    this.selectedPaymentOption = null;
                    this.selectedPaymentOptionId = null;
                    this.selectedAlternativePaymentMethod = null;
                }

                this.orderFees.length = 0;
                if (checkoutOptions.orderFees) {
                    for (var orderFeeIndex = 0; orderFeeIndex < checkoutOptions.orderFees.length; ++orderFeeIndex) {
                        this.orderFees.push(new orderFeeModel(checkoutOptions.orderFees[orderFeeIndex]));
                    }
                }

                if (checkoutOptions.termsAndConditions) {
                    this.termsAndConditions.length = 0;
                    for (var termsIndex = 0; termsIndex < checkoutOptions.termsAndConditions.length; ++termsIndex) {
                        this.termsAndConditions.push(new termsAndConditionsModel(checkoutOptions.termsAndConditions[termsIndex]));
                    }
                }
            } else {
                if (source.total !== undefined) this.total = source.total;
                if (source.totalToPay !== undefined) this.totalToPay = source.totalToPay;
                if (source.formattedTotal !== undefined) this.formattedTotal = source.formattedTotal;
                if (source.formattedTotalToPay !== undefined) this.formattedTotalToPay = source.formattedTotalToPay;
            }

            if (source.isCheckoutResponse) {
                this.errorMessage = source.error && source.error.message ? source.error.message : '';
                this.isAddressIncomplete = source.error && source.error.code ? source.error.code === ERROR_ADDRESSMISSING : false;
            }

            this.validateOptions();
        }

        function validateOptions() {
            // evaluate possibilities
            this.canDisplayPlaceBasketSteps = (this.selectedDeliveryOptionId !== null) && (!this.needsPaymentSelection || (this.selectedPaymentOptionId !== null && this.selectedPaymentOptionId.length > 0));
            this.canPlaceBasket =
                this.canDisplayPlaceBasketSteps &&
                this.acceptedGeneralTermsAndConditions &&
                (!this.isTicketInsuranceSelected || !this.ticketInsuranceRequiresCheckoutConfirmation || this.acceptedInsuranceTermsAndConditions) &&
                (!this.requiredExtraConfirmation || this.acceptedExtraConfirmation) &&
                (!this.errorMessage.length > 0);
        }

        function paymentOptionModel(source) {
            var model = {
                id: source ? source.id : 0,
                name: source ? source.name : '',
                description: source ? source.description : '',
                feeValue: source ? source.feeValue : '',
                currencyCode: source ? source.currencyCode : '',
                isSelected: source ? source.isSelected : false,
                savedCreditCardId: source ? source.savedCreditCardId : 0,
                savedCreditCardExpirationDate: source ? source.savedCreditCardExpirationDate : '',
                savedCreditCardCCNumber: source ? source.savedCreditCardCCNumber : '',
                savedCreditCardTypeName: source ? source.savedCreditCardTypeName : '',
                formattedFeeValue: source ? source.formattedFeeValue : '',
                nameAndFee: '',
                imageURL: ''
            };

            if (source) {
                if (source.name) {
                    var nameAndFee = source.name;
                    if (source.formattedFeeValue.length > 0)
                        nameAndFee += ' (' + source.formattedFeeValue + ')';
                    model.nameAndFee = nameAndFee;
                }
                if (source.imageURL && source.imageURL.length > 0)
                    model.imageURL = source.imageURL;
                else if (this.savedCreditCardId > 0)
                    model.imageURL = "/image/cc/" + model.savedCreditCardTypeName + ".png";
            }
            return model;
        }

        function termsAndConditionsModel(source) {
            var model = {
                companyName: source ? source.companyName : '',
                url: source ? source.url : ''
            };
            return model;
        }

        function deliveryOptionModel(source) {
            var model = {
                id: source ? source.id : '',
                ticketMediaId: source ? source.ticketMediaId : 0,
                name: source ? source.name : '',
                description: source ? source.description : '',
                feeValue: source ? source.feeValue : '',
                itemFeeValue: source ? source.itemFeeValue : '',
                maxItemFeeValue: source ? source.maxItemFeeValue : '',
                currencyCode: source ? source.currencyCode : '',
                isSelected: source ? source.isSelected : false,
                formattedFeeValue: source ? source.formattedFeeValue : '',
                nameAndFee: ''
            };

            if (source && source.name) {
                var nameAndFee = source.name;
                if (source.formattedFeeValue.length > 0)
                    nameAndFee += ' (' + source.formattedFeeValue + ')';
                model.nameAndFee = nameAndFee;
            }
            return model;
        }

        function alternativePaymentMethodModel(source) {
            var model = {
                id: source ? source.id : 0,
                name: source ? source.name : '',
                description: source ? source.description : '',
                imageURL: source ? source.imageURL : '',
                inputHint: source ? source.inputHint : '',
                type: source ? source.type : '',
                errorMessage: source ? source.errorMessage : '',
                appliedPayments: []
            };
            
            if (source && source.appliedPayments && source.appliedPayments.length > 0) {
                for (var i = 0; i < source.appliedPayments.length ; i++) {
                    model.appliedPayments.push(new appliedAlternativePaymentModel(source.appliedPayments[i], source.name, source.id));
                }
            }
            return model;
        }

        function appliedAlternativePaymentModel(source, name, paymentMethodId) {
            var model = {
                id: source.id,
                code: source.code,
                usedValue: source.usedValue,
                formattedUsedValue: source.formattedUsedValue,
                name: name,
                paymentMethodId: paymentMethodId
            };
            return model;
        }

        function orderFeeModel(source) {
            var model = {
                name: source.name,
                formattedValue: source.formattedValue
            };
            return model;
        }

        function deliveryAddressModel(source) {
            var model = {
                id: source.id,
                salutationName: source.salutationName,
                companyName: source.companyName,
                firstName: source.firstName,
                lastName: source.lastName,
                address1: source.addressLine1,
                address2: source.addressLine2,
                city: source.city,
                zipCode: source.zipCode,
                stateName: source.stateName,
                countryName: source.countryName,
                email: source.email,
            };

            return model;
        }

        function billingAddressModel(source) {
            var model = {
                id: source.id,
                salutationName: source.salutationName,
                companyName: source.companyName,
                firstName: source.firstName,
                lastName: source.lastName,
                address1: source.addressLine1,
                address2: source.addressLine2,
                city: source.city,
                zipCode: source.zipCode,
                stateName: source.stateName,
                countryName: source.countryName,
                email: source.email,
            };

            return model;
        }

        function validatePersonalization(itemGroups) {
            let valid = true;
            validationLoop:
            for (let i = 0; i < itemGroups.length; i++) {
                let itemGroup = itemGroups[i];
                if (itemGroup.hasPersonalization) {
                    for (let j = 0; j < itemGroup.items.length; j++) {
                        let item = itemGroup.items[j];
                        if (item.personalization !== null) {
                            for (let x = 0; x < item.personalization.fields.length; x++) {
                                if (!item.personalization.fields[x].value) {
                                    valid = false;
                                    break validationLoop;
                                }
                            }
                        }
                    }
                }
            }

            return valid;
        }
    }
})();;
(function () {
    'use strict';

    var serviceId = 'checkoutService';
    angular.module('webShop2App').factory(serviceId, ['$http', '$sce', 'commonService', 'webShop2AppSettings', service]);

    function service($http, $sce, commonService, webShop2AppSettings) {
        return {
            get: function () {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/checkout');
                var savedCreditCardId = commonService.getSessionStorageItem("savedCreditCardId");
                var formData = { ajaxRequest: 1, savedCreditCardId: savedCreditCardId != null ? savedCreditCardId : 0 };

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // updated the delivery in the basket
            updateDelivery: function (deliveryId) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/checkout');
                var formData = { ajaxRequest: 1, isUpdateDelivery: true, delivery: deliveryId };

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // apply the alternative payment in the basket
            applyAlternativePayment: function (alternativePaymentMethodId, userCode) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/checkout');
                var formData = { ajaxRequest: 1, alternativePaymentMethodId: alternativePaymentMethodId, alternativePaymentMethodUserCode: userCode };

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // remove the alternative payment in the basket
            removeAlternativePayment: function (alternativePayment) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/checkout');
                var formData = { ajaxRequest: 1, alternativePaymentMethodIdToRemove: alternativePayment.paymentMethodId, alternativePaymentIdToRemove: alternativePayment.id };

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // update the delivery and payment in the basket
            updateDeliveryAndPayment: function (deliveryId, paymentOption) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/checkout');
                var formData = { ajaxRequest: 1, isUpdatePayment: true, delivery: deliveryId, payment: paymentOption.id, savedCreditCardId: paymentOption.savedCreditCardId };

                // set saved credit card session storage item
                commonService.setSessionStorageItem("savedCreditCardId", paymentOption.savedCreditCardId);

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            placeOrder: function (options) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/checkout');
                var formData = { action: 'place', insuranceTerms: 'on', agb: 'on', delivery: options.delivery, additionalTicketEmail: options.additionalTicketEmail };
                if (options.payment) {
                    if (options.payment.savedCreditCardId) formData.savedCreditCardId = options.payment.savedCreditCardId;
                    if (options.payment.id) formData.payment = options.payment.id;
                    if (options.payment.coopSupercardPoints) formData.coopSupercardPoints = options.payment.coopSupercardPoints;
                }

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            reserveOrder: function () {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/checkout');
                var formData = { action: 'reserve' };
     
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(formData),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            }

        }
    }

})();;
//! moment.js
//! version : 2.10.2
//! authors : Tim Wood, Iskren Chernev, Moment.js contributors
//! license : MIT
//! momentjs.com
!function (a, b) { "object" == typeof exports && "undefined" != typeof module ? module.exports = b() : "function" == typeof define && define.amd ? define(b) : a.moment = b() }(this, function () {
    "use strict"; function a() { return Ac.apply(null, arguments) } function b(a) { Ac = a } function c() { return { empty: !1, unusedTokens: [], unusedInput: [], overflow: -2, charsLeftOver: 0, nullInput: !1, invalidMonth: null, invalidFormat: !1, userInvalidated: !1, iso: !1 } } function d(a) { return "[object Array]" === Object.prototype.toString.call(a) } function e(a) { return "[object Date]" === Object.prototype.toString.call(a) || a instanceof Date } function f(a, b) { var c, d = []; for (c = 0; c < a.length; ++c) d.push(b(a[c], c)); return d } function g(a, b) { return Object.prototype.hasOwnProperty.call(a, b) } function h(a, b) { for (var c in b) g(b, c) && (a[c] = b[c]); return g(b, "toString") && (a.toString = b.toString), g(b, "valueOf") && (a.valueOf = b.valueOf), a } function i(a, b, c, d) { return ya(a, b, c, d, !0).utc() } function j(a) { return null == a._isValid && (a._isValid = !isNaN(a._d.getTime()) && a._pf.overflow < 0 && !a._pf.empty && !a._pf.invalidMonth && !a._pf.nullInput && !a._pf.invalidFormat && !a._pf.userInvalidated, a._strict && (a._isValid = a._isValid && 0 === a._pf.charsLeftOver && 0 === a._pf.unusedTokens.length && void 0 === a._pf.bigHour)), a._isValid } function k(a) { var b = i(0 / 0); return null != a ? h(b._pf, a) : b._pf.userInvalidated = !0, b } function l(a, b) { var c, d, e; if ("undefined" != typeof b._isAMomentObject && (a._isAMomentObject = b._isAMomentObject), "undefined" != typeof b._i && (a._i = b._i), "undefined" != typeof b._f && (a._f = b._f), "undefined" != typeof b._l && (a._l = b._l), "undefined" != typeof b._strict && (a._strict = b._strict), "undefined" != typeof b._tzm && (a._tzm = b._tzm), "undefined" != typeof b._isUTC && (a._isUTC = b._isUTC), "undefined" != typeof b._offset && (a._offset = b._offset), "undefined" != typeof b._pf && (a._pf = b._pf), "undefined" != typeof b._locale && (a._locale = b._locale), Cc.length > 0) for (c in Cc) d = Cc[c], e = b[d], "undefined" != typeof e && (a[d] = e); return a } function m(b) { l(this, b), this._d = new Date(+b._d), Dc === !1 && (Dc = !0, a.updateOffset(this), Dc = !1) } function n(a) { return a instanceof m || null != a && g(a, "_isAMomentObject") } function o(a) { var b = +a, c = 0; return 0 !== b && isFinite(b) && (c = b >= 0 ? Math.floor(b) : Math.ceil(b)), c } function p(a, b, c) { var d, e = Math.min(a.length, b.length), f = Math.abs(a.length - b.length), g = 0; for (d = 0; e > d; d++) (c && a[d] !== b[d] || !c && o(a[d]) !== o(b[d])) && g++; return g + f } function q() { } function r(a) { return a ? a.toLowerCase().replace("_", "-") : a } function s(a) { for (var b, c, d, e, f = 0; f < a.length;) { for (e = r(a[f]).split("-"), b = e.length, c = r(a[f + 1]), c = c ? c.split("-") : null; b > 0;) { if (d = t(e.slice(0, b).join("-"))) return d; if (c && c.length >= b && p(e, c, !0) >= b - 1) break; b-- } f++ } return null } function t(a) { var b = null; if (!Ec[a] && "undefined" != typeof module && module && module.exports) try { b = Bc._abbr, require("./locale/" + a), u(b) } catch (c) { } return Ec[a] } function u(a, b) { var c; return a && (c = "undefined" == typeof b ? w(a) : v(a, b), c && (Bc = c)), Bc._abbr } function v(a, b) { return null !== b ? (b.abbr = a, Ec[a] || (Ec[a] = new q), Ec[a].set(b), u(a), Ec[a]) : (delete Ec[a], null) } function w(a) { var b; if (a && a._locale && a._locale._abbr && (a = a._locale._abbr), !a) return Bc; if (!d(a)) { if (b = t(a)) return b; a = [a] } return s(a) } function x(a, b) { var c = a.toLowerCase(); Fc[c] = Fc[c + "s"] = Fc[b] = a } function y(a) { return "string" == typeof a ? Fc[a] || Fc[a.toLowerCase()] : void 0 } function z(a) { var b, c, d = {}; for (c in a) g(a, c) && (b = y(c), b && (d[b] = a[c])); return d } function A(b, c) { return function (d) { return null != d ? (C(this, b, d), a.updateOffset(this, c), this) : B(this, b) } } function B(a, b) { return a._d["get" + (a._isUTC ? "UTC" : "") + b]() } function C(a, b, c) { return a._d["set" + (a._isUTC ? "UTC" : "") + b](c) } function D(a, b) { var c; if ("object" == typeof a) for (c in a) this.set(c, a[c]); else if (a = y(a), "function" == typeof this[a]) return this[a](b); return this } function E(a, b, c) { for (var d = "" + Math.abs(a), e = a >= 0; d.length < b;) d = "0" + d; return (e ? c ? "+" : "" : "-") + d } function F(a, b, c, d) { var e = d; "string" == typeof d && (e = function () { return this[d]() }), a && (Jc[a] = e), b && (Jc[b[0]] = function () { return E(e.apply(this, arguments), b[1], b[2]) }), c && (Jc[c] = function () { return this.localeData().ordinal(e.apply(this, arguments), a) }) } function G(a) { return a.match(/\[[\s\S]/) ? a.replace(/^\[|\]$/g, "") : a.replace(/\\/g, "") } function H(a) { var b, c, d = a.match(Gc); for (b = 0, c = d.length; c > b; b++) d[b] = Jc[d[b]] ? Jc[d[b]] : G(d[b]); return function (e) { var f = ""; for (b = 0; c > b; b++) f += d[b] instanceof Function ? d[b].call(e, a) : d[b]; return f } } function I(a, b) { return a.isValid() ? (b = J(b, a.localeData()), Ic[b] || (Ic[b] = H(b)), Ic[b](a)) : a.localeData().invalidDate() } function J(a, b) { function c(a) { return b.longDateFormat(a) || a } var d = 5; for (Hc.lastIndex = 0; d >= 0 && Hc.test(a) ;) a = a.replace(Hc, c), Hc.lastIndex = 0, d -= 1; return a } function K(a, b, c) { Yc[a] = "function" == typeof b ? b : function (a) { return a && c ? c : b } } function L(a, b) { return g(Yc, a) ? Yc[a](b._strict, b._locale) : new RegExp(M(a)) } function M(a) { return a.replace("\\", "").replace(/\\(\[)|\\(\])|\[([^\]\[]*)\]|\\(.)/g, function (a, b, c, d, e) { return b || c || d || e }).replace(/[-\/\\^$*+?.()|[\]{}]/g, "\\$&") } function N(a, b) { var c, d = b; for ("string" == typeof a && (a = [a]), "number" == typeof b && (d = function (a, c) { c[b] = o(a) }), c = 0; c < a.length; c++) Zc[a[c]] = d } function O(a, b) { N(a, function (a, c, d, e) { d._w = d._w || {}, b(a, d._w, d, e) }) } function P(a, b, c) { null != b && g(Zc, a) && Zc[a](b, c._a, c, a) } function Q(a, b) { return new Date(Date.UTC(a, b + 1, 0)).getUTCDate() } function R(a) { return this._months[a.month()] } function S(a) { return this._monthsShort[a.month()] } function T(a, b, c) { var d, e, f; for (this._monthsParse || (this._monthsParse = [], this._longMonthsParse = [], this._shortMonthsParse = []), d = 0; 12 > d; d++) { if (e = i([2e3, d]), c && !this._longMonthsParse[d] && (this._longMonthsParse[d] = new RegExp("^" + this.months(e, "").replace(".", "") + "$", "i"), this._shortMonthsParse[d] = new RegExp("^" + this.monthsShort(e, "").replace(".", "") + "$", "i")), c || this._monthsParse[d] || (f = "^" + this.months(e, "") + "|^" + this.monthsShort(e, ""), this._monthsParse[d] = new RegExp(f.replace(".", ""), "i")), c && "MMMM" === b && this._longMonthsParse[d].test(a)) return d; if (c && "MMM" === b && this._shortMonthsParse[d].test(a)) return d; if (!c && this._monthsParse[d].test(a)) return d } } function U(a, b) { var c; return "string" == typeof b && (b = a.localeData().monthsParse(b), "number" != typeof b) ? a : (c = Math.min(a.date(), Q(a.year(), b)), a._d["set" + (a._isUTC ? "UTC" : "") + "Month"](b, c), a) } function V(b) { return null != b ? (U(this, b), a.updateOffset(this, !0), this) : B(this, "Month") } function W() { return Q(this.year(), this.month()) } function X(a) { var b, c = a._a; return c && -2 === a._pf.overflow && (b = c[_c] < 0 || c[_c] > 11 ? _c : c[ad] < 1 || c[ad] > Q(c[$c], c[_c]) ? ad : c[bd] < 0 || c[bd] > 24 || 24 === c[bd] && (0 !== c[cd] || 0 !== c[dd] || 0 !== c[ed]) ? bd : c[cd] < 0 || c[cd] > 59 ? cd : c[dd] < 0 || c[dd] > 59 ? dd : c[ed] < 0 || c[ed] > 999 ? ed : -1, a._pf._overflowDayOfYear && ($c > b || b > ad) && (b = ad), a._pf.overflow = b), a } function Y(b) { a.suppressDeprecationWarnings === !1 && "undefined" != typeof console && console.warn && console.warn("Deprecation warning: " + b) } function Z(a, b) { var c = !0; return h(function () { return c && (Y(a), c = !1), b.apply(this, arguments) }, b) } function $(a, b) { hd[a] || (Y(b), hd[a] = !0) } function _(a) { var b, c, d = a._i, e = id.exec(d); if (e) { for (a._pf.iso = !0, b = 0, c = jd.length; c > b; b++) if (jd[b][1].exec(d)) { a._f = jd[b][0] + (e[6] || " "); break } for (b = 0, c = kd.length; c > b; b++) if (kd[b][1].exec(d)) { a._f += kd[b][0]; break } d.match(Vc) && (a._f += "Z"), sa(a) } else a._isValid = !1 } function aa(b) { var c = ld.exec(b._i); return null !== c ? void (b._d = new Date(+c[1])) : (_(b), void (b._isValid === !1 && (delete b._isValid, a.createFromInputFallback(b)))) } function ba(a, b, c, d, e, f, g) { var h = new Date(a, b, c, d, e, f, g); return 1970 > a && h.setFullYear(a), h } function ca(a) { var b = new Date(Date.UTC.apply(null, arguments)); return 1970 > a && b.setUTCFullYear(a), b } function da(a) { return ea(a) ? 366 : 365 } function ea(a) { return a % 4 === 0 && a % 100 !== 0 || a % 400 === 0 } function fa() { return ea(this.year()) } function ga(a, b, c) { var d, e = c - b, f = c - a.day(); return f > e && (f -= 7), e - 7 > f && (f += 7), d = za(a).add(f, "d"), { week: Math.ceil(d.dayOfYear() / 7), year: d.year() } } function ha(a) { return ga(a, this._week.dow, this._week.doy).week } function ia() { return this._week.dow } function ja() { return this._week.doy } function ka(a) { var b = this.localeData().week(this); return null == a ? b : this.add(7 * (a - b), "d") } function la(a) { var b = ga(this, 1, 4).week; return null == a ? b : this.add(7 * (a - b), "d") } function ma(a, b, c, d, e) { var f, g, h = ca(a, 0, 1).getUTCDay(); return h = 0 === h ? 7 : h, c = null != c ? c : e, f = e - h + (h > d ? 7 : 0) - (e > h ? 7 : 0), g = 7 * (b - 1) + (c - e) + f + 1, { year: g > 0 ? a : a - 1, dayOfYear: g > 0 ? g : da(a - 1) + g } } function na(a) { var b = Math.round((this.clone().startOf("day") - this.clone().startOf("year")) / 864e5) + 1; return null == a ? b : this.add(a - b, "d") } function oa(a, b, c) { return null != a ? a : null != b ? b : c } function pa(a) { var b = new Date; return a._useUTC ? [b.getUTCFullYear(), b.getUTCMonth(), b.getUTCDate()] : [b.getFullYear(), b.getMonth(), b.getDate()] } function qa(a) { var b, c, d, e, f = []; if (!a._d) { for (d = pa(a), a._w && null == a._a[ad] && null == a._a[_c] && ra(a), a._dayOfYear && (e = oa(a._a[$c], d[$c]), a._dayOfYear > da(e) && (a._pf._overflowDayOfYear = !0), c = ca(e, 0, a._dayOfYear), a._a[_c] = c.getUTCMonth(), a._a[ad] = c.getUTCDate()), b = 0; 3 > b && null == a._a[b]; ++b) a._a[b] = f[b] = d[b]; for (; 7 > b; b++) a._a[b] = f[b] = null == a._a[b] ? 2 === b ? 1 : 0 : a._a[b]; 24 === a._a[bd] && 0 === a._a[cd] && 0 === a._a[dd] && 0 === a._a[ed] && (a._nextDay = !0, a._a[bd] = 0), a._d = (a._useUTC ? ca : ba).apply(null, f), null != a._tzm && a._d.setUTCMinutes(a._d.getUTCMinutes() - a._tzm), a._nextDay && (a._a[bd] = 24) } } function ra(a) { var b, c, d, e, f, g, h; b = a._w, null != b.GG || null != b.W || null != b.E ? (f = 1, g = 4, c = oa(b.GG, a._a[$c], ga(za(), 1, 4).year), d = oa(b.W, 1), e = oa(b.E, 1)) : (f = a._locale._week.dow, g = a._locale._week.doy, c = oa(b.gg, a._a[$c], ga(za(), f, g).year), d = oa(b.w, 1), null != b.d ? (e = b.d, f > e && ++d) : e = null != b.e ? b.e + f : f), h = ma(c, d, e, g, f), a._a[$c] = h.year, a._dayOfYear = h.dayOfYear } function sa(b) { if (b._f === a.ISO_8601) return void _(b); b._a = [], b._pf.empty = !0; var c, d, e, f, g, h = "" + b._i, i = h.length, j = 0; for (e = J(b._f, b._locale).match(Gc) || [], c = 0; c < e.length; c++) f = e[c], d = (h.match(L(f, b)) || [])[0], d && (g = h.substr(0, h.indexOf(d)), g.length > 0 && b._pf.unusedInput.push(g), h = h.slice(h.indexOf(d) + d.length), j += d.length), Jc[f] ? (d ? b._pf.empty = !1 : b._pf.unusedTokens.push(f), P(f, d, b)) : b._strict && !d && b._pf.unusedTokens.push(f); b._pf.charsLeftOver = i - j, h.length > 0 && b._pf.unusedInput.push(h), b._pf.bigHour === !0 && b._a[bd] <= 12 && (b._pf.bigHour = void 0), b._a[bd] = ta(b._locale, b._a[bd], b._meridiem), qa(b), X(b) } function ta(a, b, c) { var d; return null == c ? b : null != a.meridiemHour ? a.meridiemHour(b, c) : null != a.isPM ? (d = a.isPM(c), d && 12 > b && (b += 12), d || 12 !== b || (b = 0), b) : b } function ua(a) { var b, d, e, f, g; if (0 === a._f.length) return a._pf.invalidFormat = !0, void (a._d = new Date(0 / 0)); for (f = 0; f < a._f.length; f++) g = 0, b = l({}, a), null != a._useUTC && (b._useUTC = a._useUTC), b._pf = c(), b._f = a._f[f], sa(b), j(b) && (g += b._pf.charsLeftOver, g += 10 * b._pf.unusedTokens.length, b._pf.score = g, (null == e || e > g) && (e = g, d = b)); h(a, d || b) } function va(a) { if (!a._d) { var b = z(a._i); a._a = [b.year, b.month, b.day || b.date, b.hour, b.minute, b.second, b.millisecond], qa(a) } } function wa(a) { var b, c = a._i, e = a._f; return a._locale = a._locale || w(a._l), null === c || void 0 === e && "" === c ? k({ nullInput: !0 }) : ("string" == typeof c && (a._i = c = a._locale.preparse(c)), n(c) ? new m(X(c)) : (d(e) ? ua(a) : e ? sa(a) : xa(a), b = new m(X(a)), b._nextDay && (b.add(1, "d"), b._nextDay = void 0), b)) } function xa(b) { var c = b._i; void 0 === c ? b._d = new Date : e(c) ? b._d = new Date(+c) : "string" == typeof c ? aa(b) : d(c) ? (b._a = f(c.slice(0), function (a) { return parseInt(a, 10) }), qa(b)) : "object" == typeof c ? va(b) : "number" == typeof c ? b._d = new Date(c) : a.createFromInputFallback(b) } function ya(a, b, d, e, f) { var g = {}; return "boolean" == typeof d && (e = d, d = void 0), g._isAMomentObject = !0, g._useUTC = g._isUTC = f, g._l = d, g._i = a, g._f = b, g._strict = e, g._pf = c(), wa(g) } function za(a, b, c, d) { return ya(a, b, c, d, !1) } function Aa(a, b) { var c, e; if (1 === b.length && d(b[0]) && (b = b[0]), !b.length) return za(); for (c = b[0], e = 1; e < b.length; ++e) b[e][a](c) && (c = b[e]); return c } function Ba() { var a = [].slice.call(arguments, 0); return Aa("isBefore", a) } function Ca() { var a = [].slice.call(arguments, 0); return Aa("isAfter", a) } function Da(a) { var b = z(a), c = b.year || 0, d = b.quarter || 0, e = b.month || 0, f = b.week || 0, g = b.day || 0, h = b.hour || 0, i = b.minute || 0, j = b.second || 0, k = b.millisecond || 0; this._milliseconds = +k + 1e3 * j + 6e4 * i + 36e5 * h, this._days = +g + 7 * f, this._months = +e + 3 * d + 12 * c, this._data = {}, this._locale = w(), this._bubble() } function Ea(a) { return a instanceof Da } function Fa(a, b) { F(a, 0, 0, function () { var a = this.utcOffset(), c = "+"; return 0 > a && (a = -a, c = "-"), c + E(~~(a / 60), 2) + b + E(~~a % 60, 2) }) } function Ga(a) { var b = (a || "").match(Vc) || [], c = b[b.length - 1] || [], d = (c + "").match(qd) || ["-", 0, 0], e = +(60 * d[1]) + o(d[2]); return "+" === d[0] ? e : -e } function Ha(b, c) { var d, f; return c._isUTC ? (d = c.clone(), f = (n(b) || e(b) ? +b : +za(b)) - +d, d._d.setTime(+d._d + f), a.updateOffset(d, !1), d) : za(b).local(); return c._isUTC ? za(b).zone(c._offset || 0) : za(b).local() } function Ia(a) { return 15 * -Math.round(a._d.getTimezoneOffset() / 15) } function Ja(b, c) { var d, e = this._offset || 0; return null != b ? ("string" == typeof b && (b = Ga(b)), Math.abs(b) < 16 && (b = 60 * b), !this._isUTC && c && (d = Ia(this)), this._offset = b, this._isUTC = !0, null != d && this.add(d, "m"), e !== b && (!c || this._changeInProgress ? Za(this, Ua(b - e, "m"), 1, !1) : this._changeInProgress || (this._changeInProgress = !0, a.updateOffset(this, !0), this._changeInProgress = null)), this) : this._isUTC ? e : Ia(this) } function Ka(a, b) { return null != a ? ("string" != typeof a && (a = -a), this.utcOffset(a, b), this) : -this.utcOffset() } function La(a) { return this.utcOffset(0, a) } function Ma(a) { return this._isUTC && (this.utcOffset(0, a), this._isUTC = !1, a && this.subtract(Ia(this), "m")), this } function Na() { return this._tzm ? this.utcOffset(this._tzm) : "string" == typeof this._i && this.utcOffset(Ga(this._i)), this } function Oa(a) { return a = a ? za(a).utcOffset() : 0, (this.utcOffset() - a) % 60 === 0 } function Pa() { return this.utcOffset() > this.clone().month(0).utcOffset() || this.utcOffset() > this.clone().month(5).utcOffset() } function Qa() { if (this._a) { var a = this._isUTC ? i(this._a) : za(this._a); return this.isValid() && p(this._a, a.toArray()) > 0 } return !1 } function Ra() { return !this._isUTC } function Sa() { return this._isUTC } function Ta() { return this._isUTC && 0 === this._offset } function Ua(a, b) { var c, d, e, f = a, h = null; return Ea(a) ? f = { ms: a._milliseconds, d: a._days, M: a._months } : "number" == typeof a ? (f = {}, b ? f[b] = a : f.milliseconds = a) : (h = rd.exec(a)) ? (c = "-" === h[1] ? -1 : 1, f = { y: 0, d: o(h[ad]) * c, h: o(h[bd]) * c, m: o(h[cd]) * c, s: o(h[dd]) * c, ms: o(h[ed]) * c }) : (h = sd.exec(a)) ? (c = "-" === h[1] ? -1 : 1, f = { y: Va(h[2], c), M: Va(h[3], c), d: Va(h[4], c), h: Va(h[5], c), m: Va(h[6], c), s: Va(h[7], c), w: Va(h[8], c) }) : null == f ? f = {} : "object" == typeof f && ("from" in f || "to" in f) && (e = Xa(za(f.from), za(f.to)), f = {}, f.ms = e.milliseconds, f.M = e.months), d = new Da(f), Ea(a) && g(a, "_locale") && (d._locale = a._locale), d } function Va(a, b) { var c = a && parseFloat(a.replace(",", ".")); return (isNaN(c) ? 0 : c) * b } function Wa(a, b) { var c = { milliseconds: 0, months: 0 }; return c.months = b.month() - a.month() + 12 * (b.year() - a.year()), a.clone().add(c.months, "M").isAfter(b) && --c.months, c.milliseconds = +b - +a.clone().add(c.months, "M"), c } function Xa(a, b) { var c; return b = Ha(b, a), a.isBefore(b) ? c = Wa(a, b) : (c = Wa(b, a), c.milliseconds = -c.milliseconds, c.months = -c.months), c } function Ya(a, b) { return function (c, d) { var e, f; return null === d || isNaN(+d) || ($(b, "moment()." + b + "(period, number) is deprecated. Please use moment()." + b + "(number, period)."), f = c, c = d, d = f), c = "string" == typeof c ? +c : c, e = Ua(c, d), Za(this, e, a), this } } function Za(b, c, d, e) { var f = c._milliseconds, g = c._days, h = c._months; e = null == e ? !0 : e, f && b._d.setTime(+b._d + f * d), g && C(b, "Date", B(b, "Date") + g * d), h && U(b, B(b, "Month") + h * d), e && a.updateOffset(b, g || h) } function $a(a) { var b = a || za(), c = Ha(b, this).startOf("day"), d = this.diff(c, "days", !0), e = -6 > d ? "sameElse" : -1 > d ? "lastWeek" : 0 > d ? "lastDay" : 1 > d ? "sameDay" : 2 > d ? "nextDay" : 7 > d ? "nextWeek" : "sameElse"; return this.format(this.localeData().calendar(e, this, za(b))) } function _a() { return new m(this) } function ab(a, b) { var c; return b = y("undefined" != typeof b ? b : "millisecond"), "millisecond" === b ? (a = n(a) ? a : za(a), +this > +a) : (c = n(a) ? +a : +za(a), c < +this.clone().startOf(b)) } function bb(a, b) { var c; return b = y("undefined" != typeof b ? b : "millisecond"), "millisecond" === b ? (a = n(a) ? a : za(a), +a > +this) : (c = n(a) ? +a : +za(a), +this.clone().endOf(b) < c) } function cb(a, b, c) { return this.isAfter(a, c) && this.isBefore(b, c) } function db(a, b) { var c; return b = y(b || "millisecond"), "millisecond" === b ? (a = n(a) ? a : za(a), +this === +a) : (c = +za(a), +this.clone().startOf(b) <= c && c <= +this.clone().endOf(b)) } function eb(a) { return 0 > a ? Math.ceil(a) : Math.floor(a) } function fb(a, b, c) { var d, e, f = Ha(a, this), g = 6e4 * (f.utcOffset() - this.utcOffset()); return b = y(b), "year" === b || "month" === b || "quarter" === b ? (e = gb(this, f), "quarter" === b ? e /= 3 : "year" === b && (e /= 12)) : (d = this - f, e = "second" === b ? d / 1e3 : "minute" === b ? d / 6e4 : "hour" === b ? d / 36e5 : "day" === b ? (d - g) / 864e5 : "week" === b ? (d - g) / 6048e5 : d), c ? e : eb(e) } function gb(a, b) { var c, d, e = 12 * (b.year() - a.year()) + (b.month() - a.month()), f = a.clone().add(e, "months"); return 0 > b - f ? (c = a.clone().add(e - 1, "months"), d = (b - f) / (f - c)) : (c = a.clone().add(e + 1, "months"), d = (b - f) / (c - f)), -(e + d) } function hb() { return this.clone().locale("en").format("ddd MMM DD YYYY HH:mm:ss [GMT]ZZ") } function ib() { var a = this.clone().utc(); return 0 < a.year() && a.year() <= 9999 ? "function" == typeof Date.prototype.toISOString ? this.toDate().toISOString() : I(a, "YYYY-MM-DD[T]HH:mm:ss.SSS[Z]") : I(a, "YYYYYY-MM-DD[T]HH:mm:ss.SSS[Z]") } function jb(b) { var c = I(this, b || a.defaultFormat); return this.localeData().postformat(c) } function kb(a, b) { return Ua({ to: this, from: a }).locale(this.locale()).humanize(!b) } function lb(a) { return this.from(za(), a) } function mb(a) { var b; return void 0 === a ? this._locale._abbr : (b = w(a), null != b && (this._locale = b), this) } function nb() { return this._locale } function ob(a) { switch (a = y(a)) { case "year": this.month(0); case "quarter": case "month": this.date(1); case "week": case "isoWeek": case "day": this.hours(0); case "hour": this.minutes(0); case "minute": this.seconds(0); case "second": this.milliseconds(0) } return "week" === a && this.weekday(0), "isoWeek" === a && this.isoWeekday(1), "quarter" === a && this.month(3 * Math.floor(this.month() / 3)), this } function pb(a) { return a = y(a), void 0 === a || "millisecond" === a ? this : this.startOf(a).add(1, "isoWeek" === a ? "week" : a).subtract(1, "ms") } function qb() { return +this._d - 6e4 * (this._offset || 0) } function rb() { return Math.floor(+this / 1e3) } function sb() { return this._offset ? new Date(+this) : this._d } function tb() { var a = this; return [a.year(), a.month(), a.date(), a.hour(), a.minute(), a.second(), a.millisecond()] } function ub() { return j(this) } function vb() { return h({}, this._pf) } function wb() { return this._pf.overflow } function xb(a, b) { F(0, [a, a.length], 0, b) } function yb(a, b, c) { return ga(za([a, 11, 31 + b - c]), b, c).week } function zb(a) { var b = ga(this, this.localeData()._week.dow, this.localeData()._week.doy).year; return null == a ? b : this.add(a - b, "y") } function Ab(a) { var b = ga(this, 1, 4).year; return null == a ? b : this.add(a - b, "y") } function Bb() { return yb(this.year(), 1, 4) } function Cb() { var a = this.localeData()._week; return yb(this.year(), a.dow, a.doy) } function Db(a) { return null == a ? Math.ceil((this.month() + 1) / 3) : this.month(3 * (a - 1) + this.month() % 3) } function Eb(a, b) { if ("string" == typeof a) if (isNaN(a)) { if (a = b.weekdaysParse(a), "number" != typeof a) return null } else a = parseInt(a, 10); return a } function Fb(a) { return this._weekdays[a.day()] } function Gb(a) { return this._weekdaysShort[a.day()] } function Hb(a) { return this._weekdaysMin[a.day()] } function Ib(a) { var b, c, d; for (this._weekdaysParse || (this._weekdaysParse = []), b = 0; 7 > b; b++) if (this._weekdaysParse[b] || (c = za([2e3, 1]).day(b), d = "^" + this.weekdays(c, "") + "|^" + this.weekdaysShort(c, "") + "|^" + this.weekdaysMin(c, ""), this._weekdaysParse[b] = new RegExp(d.replace(".", ""), "i")), this._weekdaysParse[b].test(a)) return b } function Jb(a) { var b = this._isUTC ? this._d.getUTCDay() : this._d.getDay(); return null != a ? (a = Eb(a, this.localeData()), this.add(a - b, "d")) : b } function Kb(a) { var b = (this.day() + 7 - this.localeData()._week.dow) % 7; return null == a ? b : this.add(a - b, "d") } function Lb(a) { return null == a ? this.day() || 7 : this.day(this.day() % 7 ? a : a - 7) } function Mb(a, b) { F(a, 0, 0, function () { return this.localeData().meridiem(this.hours(), this.minutes(), b) }) } function Nb(a, b) { return b._meridiemParse } function Ob(a) { return "p" === (a + "").toLowerCase().charAt(0) } function Pb(a, b, c) { return a > 11 ? c ? "pm" : "PM" : c ? "am" : "AM" } function Qb(a) { F(0, [a, 3], 0, "millisecond") } function Rb() { return this._isUTC ? "UTC" : "" } function Sb() { return this._isUTC ? "Coordinated Universal Time" : "" } function Tb(a) { return za(1e3 * a) } function Ub() { return za.apply(null, arguments).parseZone() } function Vb(a, b, c) { var d = this._calendar[a]; return "function" == typeof d ? d.call(b, c) : d } function Wb(a) { var b = this._longDateFormat[a]; return !b && this._longDateFormat[a.toUpperCase()] && (b = this._longDateFormat[a.toUpperCase()].replace(/MMMM|MM|DD|dddd/g, function (a) { return a.slice(1) }), this._longDateFormat[a] = b), b } function Xb() { return this._invalidDate } function Yb(a) { return this._ordinal.replace("%d", a) } function Zb(a) { return a } function $b(a, b, c, d) { var e = this._relativeTime[c]; return "function" == typeof e ? e(a, b, c, d) : e.replace(/%d/i, a) } function _b(a, b) { var c = this._relativeTime[a > 0 ? "future" : "past"]; return "function" == typeof c ? c(b) : c.replace(/%s/i, b) } function ac(a) { var b, c; for (c in a) b = a[c], "function" == typeof b ? this[c] = b : this["_" + c] = b; this._ordinalParseLenient = new RegExp(this._ordinalParse.source + "|" + /\d{1,2}/.source) } function bc(a, b, c, d) { var e = w(), f = i().set(d, b); return e[c](f, a) } function cc(a, b, c, d, e) { if ("number" == typeof a && (b = a, a = void 0), a = a || "", null != b) return bc(a, b, c, e); var f, g = []; for (f = 0; d > f; f++) g[f] = bc(a, f, c, e); return g } function dc(a, b) { return cc(a, b, "months", 12, "month") } function ec(a, b) { return cc(a, b, "monthsShort", 12, "month") } function fc(a, b) { return cc(a, b, "weekdays", 7, "day") } function gc(a, b) { return cc(a, b, "weekdaysShort", 7, "day") } function hc(a, b) { return cc(a, b, "weekdaysMin", 7, "day") } function ic() { var a = this._data; return this._milliseconds = Od(this._milliseconds), this._days = Od(this._days), this._months = Od(this._months), a.milliseconds = Od(a.milliseconds), a.seconds = Od(a.seconds), a.minutes = Od(a.minutes), a.hours = Od(a.hours), a.months = Od(a.months), a.years = Od(a.years), this } function jc(a, b, c, d) { var e = Ua(b, c); return a._milliseconds += d * e._milliseconds, a._days += d * e._days, a._months += d * e._months, a._bubble() } function kc(a, b) { return jc(this, a, b, 1) } function lc(a, b) { return jc(this, a, b, -1) } function mc() { var a, b, c, d = this._milliseconds, e = this._days, f = this._months, g = this._data, h = 0; return g.milliseconds = d % 1e3, a = eb(d / 1e3), g.seconds = a % 60, b = eb(a / 60), g.minutes = b % 60, c = eb(b / 60), g.hours = c % 24, e += eb(c / 24), h = eb(nc(e)), e -= eb(oc(h)), f += eb(e / 30), e %= 30, h += eb(f / 12), f %= 12, g.days = e, g.months = f, g.years = h, this } function nc(a) { return 400 * a / 146097 } function oc(a) { return 146097 * a / 400 } function pc(a) { var b, c, d = this._milliseconds; if (a = y(a), "month" === a || "year" === a) return b = this._days + d / 864e5, c = this._months + 12 * nc(b), "month" === a ? c : c / 12; switch (b = this._days + Math.round(oc(this._months / 12)), a) { case "week": return b / 7 + d / 6048e5; case "day": return b + d / 864e5; case "hour": return 24 * b + d / 36e5; case "minute": return 24 * b * 60 + d / 6e4; case "second": return 24 * b * 60 * 60 + d / 1e3; case "millisecond": return Math.floor(24 * b * 60 * 60 * 1e3) + d; default: throw new Error("Unknown unit " + a) } } function qc() { return this._milliseconds + 864e5 * this._days + this._months % 12 * 2592e6 + 31536e6 * o(this._months / 12) } function rc(a) { return function () { return this.as(a) } } function sc(a) { return a = y(a), this[a + "s"]() } function tc(a) { return function () { return this._data[a] } } function uc() { return eb(this.days() / 7) } function vc(a, b, c, d, e) { return e.relativeTime(b || 1, !!c, a, d) } function wc(a, b, c) { var d = Ua(a).abs(), e = ce(d.as("s")), f = ce(d.as("m")), g = ce(d.as("h")), h = ce(d.as("d")), i = ce(d.as("M")), j = ce(d.as("y")), k = e < de.s && ["s", e] || 1 === f && ["m"] || f < de.m && ["mm", f] || 1 === g && ["h"] || g < de.h && ["hh", g] || 1 === h && ["d"] || h < de.d && ["dd", h] || 1 === i && ["M"] || i < de.M && ["MM", i] || 1 === j && ["y"] || ["yy", j]; return k[2] = b, k[3] = +a > 0, k[4] = c, vc.apply(null, k) } function xc(a, b) { return void 0 === de[a] ? !1 : void 0 === b ? de[a] : (de[a] = b, !0) } function yc(a) { var b = this.localeData(), c = wc(this, !a, b); return a && (c = b.pastFuture(+this, c)), b.postformat(c) } function zc() { var a = ee(this.years()), b = ee(this.months()), c = ee(this.days()), d = ee(this.hours()), e = ee(this.minutes()), f = ee(this.seconds() + this.milliseconds() / 1e3), g = this.asSeconds(); return g ? (0 > g ? "-" : "") + "P" + (a ? a + "Y" : "") + (b ? b + "M" : "") + (c ? c + "D" : "") + (d || e || f ? "T" : "") + (d ? d + "H" : "") + (e ? e + "M" : "") + (f ? f + "S" : "") : "P0D" } var Ac, Bc, Cc = a.momentProperties = [], Dc = !1, Ec = {}, Fc = {}, Gc = /(\[[^\[]*\])|(\\)?(Mo|MM?M?M?|Do|DDDo|DD?D?D?|ddd?d?|do?|w[o|w]?|W[o|W]?|Q|YYYYYY|YYYYY|YYYY|YY|gg(ggg?)?|GG(GGG?)?|e|E|a|A|hh?|HH?|mm?|ss?|S{1,4}|x|X|zz?|ZZ?|.)/g, Hc = /(\[[^\[]*\])|(\\)?(LTS|LT|LL?L?L?|l{1,4})/g, Ic = {}, Jc = {}, Kc = /\d/, Lc = /\d\d/, Mc = /\d{3}/, Nc = /\d{4}/, Oc = /[+-]?\d{6}/, Pc = /\d\d?/, Qc = /\d{1,3}/, Rc = /\d{1,4}/, Sc = /[+-]?\d{1,6}/, Tc = /\d+/, Uc = /[+-]?\d+/, Vc = /Z|[+-]\d\d:?\d\d/gi, Wc = /[+-]?\d+(\.\d{1,3})?/, Xc = /[0-9]*['a-z\u00A0-\u05FF\u0700-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]+|[\u0600-\u06FF\/]+(\s*?[\u0600-\u06FF]+){1,2}/i, Yc = {}, Zc = {}, $c = 0, _c = 1, ad = 2, bd = 3, cd = 4, dd = 5, ed = 6; F("M", ["MM", 2], "Mo", function () { return this.month() + 1 }), F("MMM", 0, 0, function (a) { return this.localeData().monthsShort(this, a) }), F("MMMM", 0, 0, function (a) { return this.localeData().months(this, a) }), x("month", "M"), K("M", Pc), K("MM", Pc, Lc), K("MMM", Xc), K("MMMM", Xc), N(["M", "MM"], function (a, b) { b[_c] = o(a) - 1 }), N(["MMM", "MMMM"], function (a, b, c, d) { var e = c._locale.monthsParse(a, d, c._strict); null != e ? b[_c] = e : c._pf.invalidMonth = a }); var fd = "January_February_March_April_May_June_July_August_September_October_November_December".split("_"), gd = "Jan_Feb_Mar_Apr_May_Jun_Jul_Aug_Sep_Oct_Nov_Dec".split("_"), hd = {}; a.suppressDeprecationWarnings = !1; var id = /^\s*(?:[+-]\d{6}|\d{4})-(?:(\d\d-\d\d)|(W\d\d$)|(W\d\d-\d)|(\d\d\d))((T| )(\d\d(:\d\d(:\d\d(\.\d+)?)?)?)?([\+\-]\d\d(?::?\d\d)?|\s*Z)?)?$/, jd = [["YYYYYY-MM-DD", /[+-]\d{6}-\d{2}-\d{2}/], ["YYYY-MM-DD", /\d{4}-\d{2}-\d{2}/], ["GGGG-[W]WW-E", /\d{4}-W\d{2}-\d/], ["GGGG-[W]WW", /\d{4}-W\d{2}/], ["YYYY-DDD", /\d{4}-\d{3}/]], kd = [["HH:mm:ss.SSSS", /(T| )\d\d:\d\d:\d\d\.\d+/], ["HH:mm:ss", /(T| )\d\d:\d\d:\d\d/], ["HH:mm", /(T| )\d\d:\d\d/], ["HH", /(T| )\d\d/]], ld = /^\/?Date\((\-?\d+)/i; a.createFromInputFallback = Z("moment construction falls back to js Date. This is discouraged and will be removed in upcoming major release. Please refer to https://github.com/moment/moment/issues/1407 for more info.", function (a) { a._d = new Date(a._i + (a._useUTC ? " UTC" : "")) }), F(0, ["YY", 2], 0, function () { return this.year() % 100 }), F(0, ["YYYY", 4], 0, "year"), F(0, ["YYYYY", 5], 0, "year"), F(0, ["YYYYYY", 6, !0], 0, "year"), x("year", "y"), K("Y", Uc), K("YY", Pc, Lc), K("YYYY", Rc, Nc), K("YYYYY", Sc, Oc), K("YYYYYY", Sc, Oc), N(["YYYY", "YYYYY", "YYYYYY"], $c), N("YY", function (b, c) { c[$c] = a.parseTwoDigitYear(b) }), a.parseTwoDigitYear = function (a) { return o(a) + (o(a) > 68 ? 1900 : 2e3) }; var md = A("FullYear", !1); F("w", ["ww", 2], "wo", "week"), F("W", ["WW", 2], "Wo", "isoWeek"), x("week", "w"), x("isoWeek", "W"), K("w", Pc), K("ww", Pc, Lc), K("W", Pc), K("WW", Pc, Lc), O(["w", "ww", "W", "WW"], function (a, b, c, d) { b[d.substr(0, 1)] = o(a) }); var nd = { dow: 0, doy: 6 }; F("DDD", ["DDDD", 3], "DDDo", "dayOfYear"), x("dayOfYear", "DDD"), K("DDD", Qc), K("DDDD", Mc), N(["DDD", "DDDD"], function (a, b, c) { c._dayOfYear = o(a) }), a.ISO_8601 = function () { }; var od = Z("moment().min is deprecated, use moment.min instead. https://github.com/moment/moment/issues/1548", function () { var a = za.apply(null, arguments); return this > a ? this : a }), pd = Z("moment().max is deprecated, use moment.max instead. https://github.com/moment/moment/issues/1548", function () { var a = za.apply(null, arguments); return a > this ? this : a }); Fa("Z", ":"), Fa("ZZ", ""), K("Z", Vc), K("ZZ", Vc), N(["Z", "ZZ"], function (a, b, c) { c._useUTC = !0, c._tzm = Ga(a) }); var qd = /([\+\-]|\d\d)/gi; a.updateOffset = function () { }; var rd = /(\-)?(?:(\d*)\.)?(\d+)\:(\d+)(?:\:(\d+)\.?(\d{3})?)?/, sd = /^(-)?P(?:(?:([0-9,.]*)Y)?(?:([0-9,.]*)M)?(?:([0-9,.]*)D)?(?:T(?:([0-9,.]*)H)?(?:([0-9,.]*)M)?(?:([0-9,.]*)S)?)?|([0-9,.]*)W)$/; Ua.fn = Da.prototype; var td = Ya(1, "add"), ud = Ya(-1, "subtract"); a.defaultFormat = "YYYY-MM-DDTHH:mm:ssZ"; var vd = Z("moment().lang() is deprecated. Instead, use moment().localeData() to get the language configuration. Use moment().locale() to change languages.", function (a) { return void 0 === a ? this.localeData() : this.locale(a) }); F(0, ["gg", 2], 0, function () { return this.weekYear() % 100 }), F(0, ["GG", 2], 0, function () { return this.isoWeekYear() % 100 }), xb("gggg", "weekYear"), xb("ggggg", "weekYear"), xb("GGGG", "isoWeekYear"), xb("GGGGG", "isoWeekYear"), x("weekYear", "gg"), x("isoWeekYear", "GG"), K("G", Uc), K("g", Uc), K("GG", Pc, Lc), K("gg", Pc, Lc), K("GGGG", Rc, Nc), K("gggg", Rc, Nc), K("GGGGG", Sc, Oc), K("ggggg", Sc, Oc), O(["gggg", "ggggg", "GGGG", "GGGGG"], function (a, b, c, d) { b[d.substr(0, 2)] = o(a) }), O(["gg", "GG"], function (b, c, d, e) { c[e] = a.parseTwoDigitYear(b) }), F("Q", 0, 0, "quarter"), x("quarter", "Q"), K("Q", Kc), N("Q", function (a, b) { b[_c] = 3 * (o(a) - 1) }), F("D", ["DD", 2], "Do", "date"), x("date", "D"), K("D", Pc), K("DD", Pc, Lc), K("Do", function (a, b) { return a ? b._ordinalParse : b._ordinalParseLenient }), N(["D", "DD"], ad), N("Do", function (a, b) { b[ad] = o(a.match(Pc)[0], 10) }); var wd = A("Date", !0); F("d", 0, "do", "day"), F("dd", 0, 0, function (a) { return this.localeData().weekdaysMin(this, a) }), F("ddd", 0, 0, function (a) { return this.localeData().weekdaysShort(this, a) }), F("dddd", 0, 0, function (a) { return this.localeData().weekdays(this, a) }), F("e", 0, 0, "weekday"), F("E", 0, 0, "isoWeekday"), x("day", "d"), x("weekday", "e"), x("isoWeekday", "E"), K("d", Pc), K("e", Pc), K("E", Pc), K("dd", Xc), K("ddd", Xc), K("dddd", Xc), O(["dd", "ddd", "dddd"], function (a, b, c) { var d = c._locale.weekdaysParse(a); null != d ? b.d = d : c._pf.invalidWeekday = a }), O(["d", "e", "E"], function (a, b, c, d) { b[d] = o(a) }); var xd = "Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday".split("_"), yd = "Sun_Mon_Tue_Wed_Thu_Fri_Sat".split("_"), zd = "Su_Mo_Tu_We_Th_Fr_Sa".split("_"); F("H", ["HH", 2], 0, "hour"), F("h", ["hh", 2], 0, function () { return this.hours() % 12 || 12 }), Mb("a", !0), Mb("A", !1), x("hour", "h"), K("a", Nb), K("A", Nb), K("H", Pc), K("h", Pc), K("HH", Pc, Lc), K("hh", Pc, Lc), N(["H", "HH"], bd), N(["a", "A"], function (a, b, c) { c._isPm = c._locale.isPM(a), c._meridiem = a }), N(["h", "hh"], function (a, b, c) { b[bd] = o(a), c._pf.bigHour = !0 }); var Ad = /[ap]\.?m?\.?/i, Bd = A("Hours", !0); F("m", ["mm", 2], 0, "minute"), x("minute", "m"), K("m", Pc), K("mm", Pc, Lc), N(["m", "mm"], cd); var Cd = A("Minutes", !1); F("s", ["ss", 2], 0, "second"), x("second", "s"), K("s", Pc), K("ss", Pc, Lc), N(["s", "ss"], dd); var Dd = A("Seconds", !1); F("S", 0, 0, function () { return ~~(this.millisecond() / 100) }), F(0, ["SS", 2], 0, function () { return ~~(this.millisecond() / 10) }), Qb("SSS"), Qb("SSSS"), x("millisecond", "ms"), K("S", Qc, Kc), K("SS", Qc, Lc), K("SSS", Qc, Mc), K("SSSS", Tc), N(["S", "SS", "SSS", "SSSS"], function (a, b) { b[ed] = o(1e3 * ("0." + a)) }); var Ed = A("Milliseconds", !1); F("z", 0, 0, "zoneAbbr"), F("zz", 0, 0, "zoneName"); var Fd = m.prototype; Fd.add = td, Fd.calendar = $a, Fd.clone = _a, Fd.diff = fb, Fd.endOf = pb, Fd.format = jb, Fd.from = kb, Fd.fromNow = lb, Fd.get = D, Fd.invalidAt = wb, Fd.isAfter = ab, Fd.isBefore = bb, Fd.isBetween = cb, Fd.isSame = db, Fd.isValid = ub, Fd.lang = vd, Fd.locale = mb, Fd.localeData = nb, Fd.max = pd, Fd.min = od, Fd.parsingFlags = vb, Fd.set = D, Fd.startOf = ob, Fd.subtract = ud, Fd.toArray = tb, Fd.toDate = sb, Fd.toISOString = ib, Fd.toJSON = ib, Fd.toString = hb, Fd.unix = rb, Fd.valueOf = qb, Fd.year = md, Fd.isLeapYear = fa, Fd.weekYear = zb, Fd.isoWeekYear = Ab, Fd.quarter = Fd.quarters = Db, Fd.month = V, Fd.daysInMonth = W, Fd.week = Fd.weeks = ka, Fd.isoWeek = Fd.isoWeeks = la, Fd.weeksInYear = Cb, Fd.isoWeeksInYear = Bb, Fd.date = wd, Fd.day = Fd.days = Jb, Fd.weekday = Kb, Fd.isoWeekday = Lb, Fd.dayOfYear = na, Fd.hour = Fd.hours = Bd, Fd.minute = Fd.minutes = Cd, Fd.second = Fd.seconds = Dd, Fd.millisecond = Fd.milliseconds = Ed, Fd.utcOffset = Ja, Fd.utc = La, Fd.local = Ma, Fd.parseZone = Na, Fd.hasAlignedHourOffset = Oa, Fd.isDST = Pa, Fd.isDSTShifted = Qa, Fd.isLocal = Ra, Fd.isUtcOffset = Sa, Fd.isUtc = Ta, Fd.isUTC = Ta, Fd.zoneAbbr = Rb, Fd.zoneName = Sb, Fd.dates = Z("dates accessor is deprecated. Use date instead.", wd), Fd.months = Z("months accessor is deprecated. Use month instead", V), Fd.years = Z("years accessor is deprecated. Use year instead", md), Fd.zone = Z("moment().zone is deprecated, use moment().utcOffset instead. https://github.com/moment/moment/issues/1779", Ka); var Gd = Fd, Hd = { sameDay: "[Today at] LT", nextDay: "[Tomorrow at] LT", nextWeek: "dddd [at] LT", lastDay: "[Yesterday at] LT", lastWeek: "[Last] dddd [at] LT", sameElse: "L" }, Id = { LTS: "h:mm:ss A", LT: "h:mm A", L: "MM/DD/YYYY", LL: "MMMM D, YYYY", LLL: "MMMM D, YYYY LT", LLLL: "dddd, MMMM D, YYYY LT" }, Jd = "Invalid date", Kd = "%d", Ld = /\d{1,2}/, Md = { future: "in %s", past: "%s ago", s: "a few seconds", m: "a minute", mm: "%d minutes", h: "an hour", hh: "%d hours", d: "a day", dd: "%d days", M: "a month", MM: "%d months", y: "a year", yy: "%d years" }, Nd = q.prototype; Nd._calendar = Hd, Nd.calendar = Vb, Nd._longDateFormat = Id, Nd.longDateFormat = Wb, Nd._invalidDate = Jd, Nd.invalidDate = Xb, Nd._ordinal = Kd, Nd.ordinal = Yb, Nd._ordinalParse = Ld,
    Nd.preparse = Zb, Nd.postformat = Zb, Nd._relativeTime = Md, Nd.relativeTime = $b, Nd.pastFuture = _b, Nd.set = ac, Nd.months = R, Nd._months = fd, Nd.monthsShort = S, Nd._monthsShort = gd, Nd.monthsParse = T, Nd.week = ha, Nd._week = nd, Nd.firstDayOfYear = ja, Nd.firstDayOfWeek = ia, Nd.weekdays = Fb, Nd._weekdays = xd, Nd.weekdaysMin = Hb, Nd._weekdaysMin = zd, Nd.weekdaysShort = Gb, Nd._weekdaysShort = yd, Nd.weekdaysParse = Ib, Nd.isPM = Ob, Nd._meridiemParse = Ad, Nd.meridiem = Pb, u("en", { ordinalParse: /\d{1,2}(th|st|nd|rd)/, ordinal: function (a) { var b = a % 10, c = 1 === o(a % 100 / 10) ? "th" : 1 === b ? "st" : 2 === b ? "nd" : 3 === b ? "rd" : "th"; return a + c } }), a.lang = Z("moment.lang is deprecated. Use moment.locale instead.", u), a.langData = Z("moment.langData is deprecated. Use moment.localeData instead.", w); var Od = Math.abs, Pd = rc("ms"), Qd = rc("s"), Rd = rc("m"), Sd = rc("h"), Td = rc("d"), Ud = rc("w"), Vd = rc("M"), Wd = rc("y"), Xd = tc("milliseconds"), Yd = tc("seconds"), Zd = tc("minutes"), $d = tc("hours"), _d = tc("days"), ae = tc("months"), be = tc("years"), ce = Math.round, de = { s: 45, m: 45, h: 22, d: 26, M: 11 }, ee = Math.abs, fe = Da.prototype; fe.abs = ic, fe.add = kc, fe.subtract = lc, fe.as = pc, fe.asMilliseconds = Pd, fe.asSeconds = Qd, fe.asMinutes = Rd, fe.asHours = Sd, fe.asDays = Td, fe.asWeeks = Ud, fe.asMonths = Vd, fe.asYears = Wd, fe.valueOf = qc, fe._bubble = mc, fe.get = sc, fe.milliseconds = Xd, fe.seconds = Yd, fe.minutes = Zd, fe.hours = $d, fe.days = _d, fe.weeks = uc, fe.months = ae, fe.years = be, fe.humanize = yc, fe.toISOString = zc, fe.toString = zc, fe.toJSON = zc, fe.locale = mb, fe.localeData = nb, fe.toIsoString = Z("toIsoString() is deprecated. Please use toISOString() instead (notice the capitals)", zc), fe.lang = vd, F("X", 0, 0, "unix"), F("x", 0, 0, "valueOf"), K("x", Uc), K("X", Wc), N("X", function (a, b, c) { c._d = new Date(1e3 * parseFloat(a, 10)) }), N("x", function (a, b, c) { c._d = new Date(o(a)) }), a.version = "2.10.2", b(za), a.fn = Gd, a.min = Ba, a.max = Ca, a.utc = i, a.unix = Tb, a.months = dc, a.isDate = e, a.locale = u, a.invalid = k, a.duration = Ua, a.isMoment = n, a.weekdays = fc, a.parseZone = Ub, a.localeData = w, a.isDuration = Ea, a.monthsShort = ec, a.weekdaysMin = hc, a.defineLocale = v, a.weekdaysShort = gc, a.normalizeUnits = y, a.relativeTimeThreshold = xc; var ge = a; return ge
});;

/**
 * basketDetails directive use case :
 * 
 *  <basket-details
 *      model="model"
 *      id="basket-details"
 *      ng-show='model.basketItemCount > 0'
 *      imgurl="'basket32x32.png'" >
 *  </basket-details>
 *
 *  id {string} it has to be always the same
 *  imgurl {string} full basket image uploaded path included file name
 *  model {object}
*/
angular.module('webShop2App').directive('basketDetails', ['pageService', function(pageService) {
    return {
        restrict: 'EA',
        scope: {
            model: '=',
            imgurl: '='
        },
        templateUrl: '/Templates/basketDetailsModal.html',
        controller : function($scope) {
            $scope.labelTotal = ticketportal.webshop.localization.labelTotal;
            $scope.goToBasketText = ticketportal.webshop.localization.goToBasket;
            $scope.labelTicketInsurance = ticketportal.webshop.localization.labelTicketInsurance;
        },
        link: function(scope, elements) {

            /*
            * directive scope functions
            */
            scope.toBasketPage = function () {
                if (pageService.pageModel.currentStep !== 'Basket')
                    scope.$parent.goToBasket();
            };
            scope.cartToggle = function ($event) {
                /*
                * When basket icon clicked, add positioning styling
                * to the cart (basket) container & top arrow, for the small devices width
                */
                if ($event.view.innerWidth < 768) {
                    elements.find('#toparrow').css('left', `${$event.pageX-25}px`);
                    elements.find('#cart-summary-modal').css('top', `${$event.pageY+25}px`);
                };

                scope.$parent.getCart();
            };
        }
    }
}]);

/* 
* Hidding the elemenet with off-click attribut when clicking outside 
*/
angular.module('webShop2App').directive('offClick', function($document, $parse, $timeout) {
    return {
        restrict: 'A',
        compile: function(tElement, tAttrs) {
        var fn = $parse(tAttrs.offClick);
        
        return function(scope, iElement, iAttrs) {
            function eventHandler(ev) {
                if (iElement[0].contains(ev.target)) {
                    $document.one('click touchend', eventHandler);
                } else {
                    scope.$apply(function() {
                        fn(scope);
                    });
                }
            }
            scope.$watch(iAttrs.offClickActivator, function(activate) {
                if (activate) {
                    $timeout(function() {
                        $document.one('click touchend', eventHandler);
                    });
                } else {
                    $document.off('click touchend', eventHandler);
                }
            });
        };
    }};
});;
(function () {
    'use strict';

    let controllerId = 'packageController';
    angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', 'packageService', 'packageModel', 'pageService', 'ecommerceService', 'ticketService', 'commonService', 'selectedPerformancePlacesFinderService', '$timeout', '$modal', '$location', '$anchorScroll', 'webShop2AppSettings', controller]);

    function controller($rootScope, $scope, packageService, packageModel, pageService, ecommerceService, ticketService, commonService, selectedPerformancePlacesFinderService, $timeout, $modal, $location, $anchorScroll, webShop2AppSettings) {
        $scope.packageService = packageService;
        $scope.model = packageModel;
        $scope.pageService = pageService;
        $scope.modal = $modal;
        $scope.webShop2AppSettings = webShop2AppSettings;
        $scope.pageService.setStartStep(CURRENT_STEP_SHOP);
        $scope.ecommerceService = ecommerceService;
        $scope.ticketService = ticketService;
        $scope.model.selectedSeats = [];
        $scope.selectedPlacesFinderService = selectedPerformancePlacesFinderService;

        $scope.createSectionTooltip = function (sectionCode) {
            let sectionData = $.grep($scope.model.performance.sections, function (e) { return e.code === sectionCode; })[0];
            if (!sectionData) return null;
            let content = '<h1>' + sectionData.name + '</h2>';

            for (let i = 0; i < sectionData.priceCategories.length; i++) {
                let categoryAvailability = sectionData.priceCategories[i];
                content += '<p><strong>' + categoryAvailability.name + '</strong>';
                if (categoryAvailability.ticketAvailabilityText.length > 0)
                    content += ': ' + categoryAvailability.ticketAvailabilityText;
                content += '</p>';
            }

            return content;
        };

        // Change the current currency
        $scope.changeCurrency = function (currency) {

            $scope.pageService.pageModel.isLoading = true;
            $scope.packageService.setBasketCurrency(currency.symbol).then(function (data) {
                $rootScope.$broadcast('event:BasketChanged', data);

                // update selected performance model with new currency
                if (data.allowedCurrencies)
                    $scope.model.updateAvailableCurrencies(data.allowedCurrencies);

                $scope.pageService.pageModel.isLoading = false;
            });
        };

        $scope.bindImageMap = function () {
            // TODO: create a directive out of it
            let mapSeatingMapName = $scope.model.isSeatingMapOpen ? '#mapSeatingMapModal' : '#mapSeatingMap';
            let imgSeatingMapName = $scope.model.isSeatingMapOpen ? '#imgSeatingMapModal' : '#imgSeatingMap';

            $(mapSeatingMapName).children().remove();
            for (let i = 0; i < $scope.model.selectedSeatingImage.areas.length; ++i) {
                let area = $scope.model.selectedSeatingImage.areas[i];
                let name = area.isImageUrl ? 'perspective-' + i : area.key;
                let imageUrl = area.isImageUrl ? area.key : '';
                let imageName = area.imageName ? area.imageName : '';
                $(mapSeatingMapName).append('<area shape="' + area.shape + '" coords="' + area.coordinates + '" name="' + name + '" imageUrl="' + imageUrl + '" imageName="' + imageName + '" href="#" />');
            }

            let areas = new Array();
            let htmlAreas = $(mapSeatingMapName + ' > area');
            for (let i = htmlAreas.length - 1; i >= 0; i--) {
                let name = $(htmlAreas[i]).attr('name');
                let toolTipContent = '';
                if (name.indexOf('perspective-') >= 0) {
                    let imageUrl = $(htmlAreas[i]).attr('imageUrl');
                    toolTipContent = '<img class="image" src="' + imageUrl + '"></img>';
                }
                else {
                    toolTipContent = '<div class="mapster-tooltip" style="border: 2px solid #CB076D; background: #FFF; width:160px; padding:4px; margin: 4px; -moz-box-shadow: 3px 3px 5px #535353; ' +
                        '-webkit-box-shadow: 3px 3px 5px #535353; box-shadow: 3px 3px 5px #535353; -moz-border-radius: 6px 6px 6px 6px; -webkit-border-radius: 6px; ' +
                        'border-radius: 6px 6px 6px 6px; color: #555;">' + $scope.createSectionTooltip(name) + '</div>';
                }

                if (toolTipContent && toolTipContent.length > 0)
                    areas.push({ key: name, toolTip: toolTipContent });
            }

            let $img = $(imgSeatingMapName);
            $img.attr('src', $scope.model.selectedSeatingImage.imageUrl);


            // customize the seating map based on the webShop2AppSettings.customization
            let fillOpacity = 0.4;
            let fillColor = 'd42e16';
            let strokeColor = '3320FF';
            let strokeOpacity = 0.8;
            let strokeWidth = 4;
            let displayStroke = true;

            if ($scope.webShop2AppSettings.customization !== undefined && $scope.webShop2AppSettings.customization.seatingMap !== undefined) {
                let seatingMapCustomization = $scope.webShop2AppSettings.customization.seatingMap;

                if (seatingMapCustomization.imageMapFillColor !== undefined)
                    fillColor = seatingMapCustomization.imageMapFillColor;
                if (seatingMapCustomization.imageMapFillOpacity !== undefined)
                    fillOpacity = seatingMapCustomization.imageMapFillOpacity;
                if (seatingMapCustomization.imageMapStrokeColor !== undefined)
                    strokeColor = seatingMapCustomization.imageMapStrokeColor;
                if (seatingMapCustomization.imageMapStrokeOpacity !== undefined)
                    strokeOpacity = seatingMapCustomization.imageMapStrokeOpacity;
                if (seatingMapCustomization.imageMapStrokeWidth !== undefined)
                    strokeWidth = seatingMapCustomization.imageMapStrokeWidth;
                if (seatingMapCustomization.imageMapDisplayStroke !== undefined)
                    displayStroke = seatingMapCustomization.imageMapDisplayStroke;
            }

            if (typeof $img.mapster === 'function') {
                $img.mapster({
                    fillOpacity: fillOpacity,
                    fillColor: fillColor,
                    stroke: displayStroke,
                    strokeColor: strokeColor,
                    strokeOpacity: strokeOpacity,
                    strokeWidth: strokeWidth,
                    singleSelect: true,
                    mapKey: 'name',
                    listKey: 'name',
                    onClick: $scope.selectImageArea,
                    toolTipContainer: '<div class="seating-map-tooltip"></div>',
                    toolTipClose: ["tooltip-click", "area-click", "img-mouseout", "area-mouseout"],
                    areas: areas,
                    showToolTip: true,
                    onShowToolTip: function (e) {
                        let imagePos = $(imgSeatingMapName).offset();
                        let left = imagePos.left - e.toolTip.width() - 40;
                        left = Math.max(0, left) + 'px';
                        let top = imagePos.top + $(imgSeatingMapName).height() / 2 - e.toolTip.height() / 2;
                        top = Math.max(0, top) + 'px';
                        e.toolTip.css({ left: left, top: top });
                    }
                });
            }

            if ($scope.model.isSeatingMapOpen)
                $scope.setSeatingMap(false);
        };
        
        $scope.containerLoaded = function () {
            if ($scope.model.performance && $scope.model.mustWaitForContentToBeLoaded) {
                $scope.updatePerformanceUI();
                $scope.model.mustWaitForContentToBeLoaded = false;
            }
        };

        $scope.updatePerformanceUI = function () {
            if ($scope.model.performance.isReservationAllowed) {

                // Reset seating map
                $scope.setSeatingMap(true);
                if ($scope.model.performance.canvasModel && $scope.model.performance.canvasModel !== undefined && $scope.model.performance.canvasModel.useCanvas) {
                    $scope.safeBindCanvasSeatingMap($scope.model.performance.canvasModel);
                }
                else {
                    // set the current seating map
                    $scope.model.performance.seatingMapImage.isTopLevel = true;
                    $scope.model.selectedSeatingImage = $scope.model.performance.seatingMapImage;
                    if ($scope.model.selectedSeatingImage.imageUrl.length > 0) {
                        $scope.bindImageMap();
                    }

                    // display seating map help
                    if (!$scope.model.seatingMapReservationHelpAlreadyDisplayed) {
                        $scope.showSeatingMapBookingHelp();

                        $scope.model.seatingMapReservationHelpAlreadyDisplayed = true;
                    }

                    // show section directly
                    if ($scope.model.performance.showSectionDirectly) {
                        // 1. select the section                                                
                        if ($scope.model.performance.sections.length > 0) {
                            $scope.selectSection($scope.model.performance.sections[0].code, false);
                        }
                    }
                }
            }

            if ($scope.model.performance.seatingPlanCode) {
                $scope.ticketService.getSeatingPlan($scope.model.performance.tag, $scope.model.performance.seatingPlanCode).then(function (data) {
                    var seatingPlan = JSON.parse(data.seatingPlan.jsonString);
                    var seatPicker = new SeatPicker({
                        "canvas": document.querySelector("canvas.canvas-seatpicker"),
                        "tooltip": {
                            "container": document.querySelector(".tooltip-seatpicker"),
                            "overrides": {
                                "content": true,
                                "display": false,
                                "hide": false,
                                "position": true
                            }
                        },
                        "categories": data.categories,
                        "layout": seatingPlan.SeatingPlan.Layout,
                        "colourPalette": {
                            "selectedSeat": data.selectedSeatColor
                        },
                        "canvasControls": {
                            "zoomInButton": document.querySelector(".seatingPlan-zoomIn"),
                            "zoomOutButton": document.querySelector(".seatingPlan-zoomOut"),
                            "zoomResetButton": document.querySelector(".seatingPlan-reset")
                        },
                        "seatStyles": {
                            "selected": {
                                "icon": "StrokeCircleWithTick"
                            }
                        }
                    });

                    // Set reserved/sold/not available seats
                    seatPicker.categories.forEach(function (category) {
                        if (category.status && category.status !== 1) {
                            // Reserved
                            if (category.status == 2) {
                                category.seatStyle.icon = seatPicker.enums.SeatIcon.StrokeCircleWithSlash;
                            }
                            // Sold or not available
                            else if (category.status == 3 || category.status == 4) {
                                category.seatStyle.icon = seatPicker.enums.SeatIcon.StrokeCircleWithCross;
                            }
                            seatPicker.addUpdateCategory(category, false);
                        }
                    });

                    // Add sectionMappings to seatmap controller
                    seatPicker.sectionMappings = data.sections;

                    // Add seat mappings to seatpicker
                    seatPicker.seatMappings = data.seats;

                    // Remove seat function
                    seatPicker.removeSeat = function (seat) {
                        seatPicker.unselectSeat(seat);
                        var index = $scope.model.selectedSeats.indexOf(seat);
                        if (index !== -1) {
                            $scope.model.selectedSeats.splice(index, 1);
                        }
                    }

                    // Zoom to section function
                    seatPicker.goToSection = function (sectionCode, seat) {
                        var blocks = seatPicker.getBlocks(sectionCode);
                        var block = !!seat ? blocks.find(function (b) {
                            return b.rows.find(function (r) {
                                return r.seats.find(function (s) {
                                    return s.identifiers.includes(seat.identifiers[0])
                                })
                            })
                        }) : blocks[0];
                        seatPicker.zoomToBlock(block)
                    }

                    // Set selected seats
                    seatPicker.setSelectedSeats = function (selectedSeats) {
                        selectedSeats.forEach(function (selectedSeat) {
                            seatPicker.setAdditionalSeatProperties(selectedSeat);
                            $scope.model.selectedSeats.push(selectedSeat);
                        });
                    }

                    // Set additional seat properties
                    seatPicker.setAdditionalSeatProperties = function (seat) {
                        var seatMapping = $scope.seatMapController.seatMappings.find(function (m) { return m.seatMappingId === seat.identifiers[0] });
                        if (!!seatMapping) {
                            var section = $scope.seatMapController.sectionMappings.find(function (s) { return s.code === seatMapping.sectionCode });
                            var priceCategory = $scope.seatMapController.categories.find(function (c) { return c.seatIdentifiers && c.seatIdentifiers.includes(seat.identifiers[0]) });
                            var ticketPrice = $scope.model.performance.priceCategories.find(function (c) { return c.uniqueId === seatMapping.priceCategoryId });
                            seat.id = seatMapping.id;
                            seat.sectionCode = seatMapping.sectionCode;
                            seat.sectionName = !!section ? section.name : '';
                            seat.priceCategoryName = !!priceCategory ? priceCategory.name : '';
                            seat.rowIdentifier = seatMapping.rowIdentifier;
                            seat.seatIdentifier = seatMapping.seatIdentifier;
                            seat.price = !!ticketPrice ? ticketPrice.prices[0].formattedPriceWithFees : '';
                            seat.status = seatMapping.status;
                            seat.statusName = seatMapping.statusName;
                            seat.color = !!priceCategory ? priceCategory.seatStyle.colour : '';;
                        } else {
                            var section = $scope.seatMapController.sectionMappings.find(function (s) { return s.code === seat.blockLabel });
                            var priceCategory = $scope.seatMapController.categories.find(function (c) { return c.seatIdentifiers && c.seatIdentifiers.includes(seat.identifiers[0]) });
                            seat.sectionCode = seat.blockLabel;
                            seat.sectionName = !!section ? section.name : '';
                            seat.priceCategoryName = !!priceCategory ? priceCategory.name : '';
                            seat.rowIdentifier = seat.rowLabel;
                            seat.seatIdentifier = seat.label;
                            seat.price = !!priceCategory ? priceCategory.price : '';
                            seat.color = !!priceCategory ? priceCategory.seatStyle.colour : '';;
                        }
                    }

                    // Set and display tooltip
                    seatPicker.setTooltip = function (seat) {
                        var html = '';
                        if (!!seat) {
                            seatPicker.setAdditionalSeatProperties(seat);
                            if (seat.color) {
                                seatPicker.tooltip.container.style.background = seat.color;
                                seatPicker.tooltip.container.style.color = textColorGenerator(seat.color);
                            } else {
                                seatPicker.tooltip.container.style.background = '';
                                seatPicker.tooltip.container.style.color = '';
                            }

                            html += '<div class="tooltip-seat-location">';
                            if (seat.sectionName) {
                                html += '<span class="tooltip-seat-section">' + seat.sectionName + '</span>';
                            }
                            html += '<span class="tooltip-seat-identifier">' + ticketportal.webshop.localization.seatRow + ": " + seat.rowIdentifier + '</span>';
                            html += '<span class="tooltip-seat-identifier">' + ticketportal.webshop.localization.seatIdentifier + ': ' + seat.seatIdentifier + '</span>';
                            html += '</div>';
                            html += '<div class="tooltip-seat-information">';
                            if (seat.priceCategoryName) {
                                html += '<span class="tooltip-seat-category">' + seat.priceCategoryName + '</span>';
                            }
                            if (seat.price) {
                                html += '<span class="tooltip-seat-price">' + seat.price + '</span>';
                            }
                            if (seat.status) {
                                html += '<span class="tooltip-seat-status">' + seat.statusName + '</span>';
                            }
                            html += '</div>';

                            var seatCoords = seat.layoutProperties.centre;
                            var canvasCoords = seatPicker.translateToCanvasCoordinates(seatCoords);
                            seatPicker.tooltip.container.innerHTML = html;
                            seatPicker.tooltip.x = canvasCoords.x + seatPicker.canvas.offsetLeft + 10;
                            seatPicker.tooltip.y = canvasCoords.y + seatPicker.canvas.offsetTop + 10;
                            seatPicker.displayTooltip();
                        }
                    }

                    // Add events
                    document.addEventListener(seatPicker.events.selectionChanged, function (event) {
                        $scope.model.selectedSeats = [];
                        seatPicker.setSelectedSeats(event.selectedSeats);
                        $scope.$apply();
                    });

                    document.addEventListener(seatPicker.events.click, function (event) {
                        seatPicker.zoomToBlock(event.block);
                    });

                    document.addEventListener(seatPicker.events.mousemove, function (event) {
                        seatPicker.setTooltip(event.seat);
                    });

                    document.addEventListener(seatPicker.events.tap, function (event) {
                        seatPicker.setTooltip(event.seat);
                    });

                    $scope.seatMapController = seatPicker;

                    // Set checked seats
                    $scope.seatMapController.seatMappings.forEach(function (seatMapping) {
                        if (seatMapping.isSeatChecked) {
                            var seat = seatPicker.getSeat(seatMapping.seatMappingId);
                            seatPicker.selectSeat(seat);
                            if (!$scope.model.selectedSeats.some(function (s) { return s.identifiers[0] === seatMapping.seatMappingId })) {
                                seatPicker.setSelectedSeats([seat]);
                            }
                        }
                    });
                });
            }

            // in iFrame post a message saying a performance was selected
            $scope.pageService.postMessageToIFrameParent("performanceSelected");
        };

        // sets the current performance
        $scope.setPerformanceStep = function (performance, restrictions) {
            let selectedPerformanceData = new performanceModel(performance);
            $scope.model.performance = selectedPerformanceData;
            $scope.model.selectedSection = null;
            $scope.model.selectedSeatingImage = null;
            $scope.model.bestPlaceSelection.reset();
            $scope.model.displayBackToMainViewButton = false;
            $scope.model.stepRestrictions = restrictions;

            $scope.model.performance.init();

            if (!$scope.model.mustWaitForContentToBeLoaded)
                $scope.updatePerformanceUI();
        };

        // Called by the image map
        $scope.selectImageArea = function (area) {
            if (area.key) {
                if (area.key.indexOf('perspective-') >= 0) {
                    let imageUrl = area.e.currentTarget.getAttribute('imageUrl');
                    let imageName = area.e.currentTarget.getAttribute('imageName');
                    let img = $('<img />', { src: imageUrl, 'class': 'image' });
                    $('#showImageModal .modal-title').html(imageName);
                    $('#showImageModal .modal-body').html(img);
                    $('#showImageModal').modal('show');
                }
                else
                    $scope.selectSection(area.key, true);
            }
        };

        // Show seating map booking help
        $scope.showSeatingMapBookingHelp = function () {
            if ($scope.pageService.showSeatingMapHelp())
            $('#seating-map-help').modal('show');
        };

        $('#seatingMapModal').on('hidden.bs.modal', function () {
            if ($scope.model.isSeatingMapOpen) {
                $scope.backToMainSeatingMapView(true);
                $scope.model.isSeatingMapOpen = false;
            }
        });

        $('.modal').on('hidden.bs.modal', function () {
            let area = $('area');
            if (area.length > 0)
                $('area').mapster('deselect');
        });

        $scope.buildBestPlaceSelectionModel = function (isAsync) {
            let itemOptions = new Array();
            for (let itemAmount = 0; itemAmount <= $scope.model.amountOfItemsToSelect; ++itemAmount) {
                itemOptions.push(itemAmount);
            }

            if ($scope.model.selectedSection) {
                $scope.model.bestPlaceSelection.sectionId = $scope.model.selectedSection.id;
                $scope.model.bestPlaceSelection.sectionName = $scope.model.selectedSection.name;
                $scope.model.bestPlaceSelection.sectionCode = $scope.model.selectedSection.code;

                let priceCategoryList = new Array();
                for (let priceCategoryIndex = 0; priceCategoryIndex < $scope.model.selectedSection.priceCategories.length; priceCategoryIndex++) {
                    let priceCategory = $scope.model.selectedSection.priceCategories[priceCategoryIndex];
                    itemOptions = itemOptions;
                    priceCategoryList.push(new $scope.model.bestPlaceSelectionCategoryModel(priceCategory.uniqueId, priceCategory.name, priceCategory.fullPriceName, priceCategory.colorCode, priceCategory.priority, priceCategory.isSoldOut, priceCategory.prices, $scope.model.performance.currencySymbol, itemOptions, priceCategory.ticketAvailabilityText));
                }
                $scope.model.bestPlaceSelection.setPriceCategories(priceCategoryList);

            } else {
                $scope.model.bestPlaceSelection.sectionId = '';
                $scope.model.bestPlaceSelection.sectionName = '';
                $scope.model.bestPlaceSelection.sectionCode = '';

                // best place among all sections
                let priceCategoryList = new Array();
                for (let priceCategoryIndex = 0; priceCategoryIndex < $scope.model.performance.priceCategories.length; priceCategoryIndex++) {
                    let priceCategory = $scope.model.performance.priceCategories[priceCategoryIndex];
                    if (!priceCategory.isSoldOut) {
                        priceCategoryList.push(new $scope.model.bestPlaceSelectionCategoryModel(priceCategory.uniqueId, priceCategory.name, priceCategory.fullPriceName, priceCategory.colorCode, priceCategory.priority, priceCategory.isSoldOut, priceCategory.prices, $scope.model.performance.currencySymbol, itemOptions, priceCategory.ticketAvailabilityText));
                    }
                }
                $scope.model.bestPlaceSelection.setPriceCategories(priceCategoryList);
            }

            if (isAsync) {
                $scope.$apply(function () {
                    $scope.model.bestPlaceSelection.isValid = true;
                });
            } else {
                $scope.model.bestPlaceSelection.isValid = true;
            }
        };

        $scope.backToMainSeatingMapView = function (isModalClose) {

            // 1. reset any best place being displayed
            $scope.model.bestPlaceSelection.reset();

            // 2. reset current section
            $scope.model.selectedSection = null;
            $scope.model.displayBackToMainViewButton = false;

            // 3. update the current displayed seating map
            if (!isModalClose) {
                $scope.model.selectedSeatingImage = $scope.model.performance.seatingMapImage;
                $scope.bindImageMap();
            }

            if ($scope.model.isSeatingMapOpen) {
                $scope.setSeatingMap(true);
                $scope.model.sectionNavigationItems.length = 0;
                $('#imgSeatingMapRenderModal').show();
            }

            $scope.$apply();
        };

        // Display the current section
        $scope.selectSection = function (sectionCode, isAsync) {
            let selectedSection = $.grep($scope.model.performance.sections, function (e) { return e.code === sectionCode; })[0];
            if (selectedSection) {
                // check if it has no place
                if (!selectedSection.isGroupingSection && selectedSection.isSoldOut) {
                    $rootScope.$broadcast('event:ShowAlert', { title: selectedSection.name, message: ticketportal.webshop.localization.noTicketsAvailableInThisSector });
                    return;
                }

                if ($scope.pageService.isMobile())
                    selectedSection.isBestPlace = true;

                $scope.model.selectedSection = selectedSection;
                $scope.model.selectedSeatingImage = selectedSection.seatingMapImage;

                if (selectedSection.isGroupingSection) {
                    $scope.model.bestPlaceSelection.reset();
                    if (!$scope.pageService.isMobile()) {
                        $scope.model.isSeatingMapOpen = true;
                        $('#seatingMapModal').modal({ keyboard: true });
                        $('#imgSeatingMapRenderModal').show();
                    }
                    $scope.bindImageMap();
                    $scope.setSeatingMap(true);
                } else if (selectedSection.isBestPlace) {
                    $scope.buildBestPlaceSelectionModel(isAsync);
                    if ($scope.model.isSeatingMapOpen) {
                        $('#seatingMapRenderModal').html("");
                        $('#seatingMapLegendModal').html("");
                        $('#seatingMapBestPlaceRenderModal').html($('#seatingMapBestPlace').html());
                        $('#imgSeatingMapRenderModal').hide();
                    }
                } else {
                    $scope.model.bestPlaceSelection.reset();
                    $scope.pageService.pageModel.isLoading = true;
                    // display seating map booking process
                    $scope.packageService.getPerformanceSeatingMapHtml($scope.model.tag, $scope.model.performance.tag, $scope.model.selectedSection.code).then(function (data) {
                        if ($scope.model.performance.showSectionDirectly) {
                            let elements = $(data);
                            let seatMap = $('#seatingMapRenderModal', elements);
                            let legend = jQuery.grep(elements, function (element) { return element.id === 'seatingMapLegendModal'; });
                            $('#seatingMapRender').html(seatMap);
                            $('#seatingMapLegend').html(legend);
                            $scope.pageService.pageModel.isLoading = false;
                            $scope.bindImageMap();

                            // allow iframe to recalculate it's size
                            $scope.pageService.postMessageToIFrameParent("performance-showing-section-" + sectionCode);
                        }
                        else {
                            $scope.model.isSeatingMapOpen = true;
                            // TODO: use directives to display the seating map modal                    
                            $('#seatingMapModal .modal-body-seatingMapRender').html(data);
                            $scope.pageService.pageModel.isLoading = false;
                            $('#seatingMapModal').modal({ keyboard: true });
                            $('#imgSeatingMapRenderModal').hide();
                            $scope.toggleAddPlaceToBasketButton();
                            $scope.buildSectionNavigation();
                        }
                    });
                }

                if (isAsync) {
                    $scope.$apply(function () {
                        if ($scope.model.isSeatingMapOpen) {
                            $scope.buildSectionNavigation();
                            $scope.toggleAddPlaceToBasketButton();
                        }

                        if ($scope.pageService.isMobile())
                            $scope.model.displayBackToMainViewButton = !$scope.model.performance.showSectionDirectly;
                    });
                }
            }
        };

        $scope.buildSectionNavigation = function () {
            $scope.model.sectionNavigationItems.length = 0;
            if ($scope.model.selectedSection.parentSectionCode !== null && $scope.model.selectedSection.parentSectionCode.length > 0) {
                let sectionCode = $scope.model.selectedSection.parentSectionCode;
                while (sectionCode !== null && sectionCode.length > 0) {
                    let selectedSection = $.grep($scope.model.performance.sections, function (e) { return e.code === sectionCode; })[0];
                    $scope.model.sectionNavigationItems.push({
                        code: selectedSection.code,
                        name: selectedSection.name,
                        imageUrl: selectedSection.isGroupingSection ? selectedSection.seatingMapImage.imageUrl : '',
                        active: false
                    });
                    sectionCode = selectedSection.parentSectionCode;
                }
            }
            $scope.model.sectionNavigationItems.push({
                code: $scope.model.selectedSection.code,
                name: $scope.model.selectedSection.name,
                imageUrl: $scope.model.selectedSection.isGroupingSection ? $scope.model.selectedSeatingImage.imageUrl : '',
                active: true
            });
        };

        $scope.closeSeatingMap = function () {
            $('#seatingMapModal').modal('hide');
            $scope.model.isSeatingMapOpen = false;
        };

        $scope.toggleAddPlaceToBasketButton = function () {
            if ($scope.model.selectedSection.isBestPlace) {
                $('#seatingMapModal .add-places-to-basket').hide();
                $('#seatingMapModal .add-bestplaces-to-basket').show();
            }
            else {
                $('#seatingMapModal .add-places-to-basket').show();
                $('#seatingMapModal .add-bestplaces-to-basket').hide();
            }
        };

        $scope.setSeatingMap = function (reset) {
            if (reset) {
                $('#seatingMapRenderModal').html("");
                $('#seatingMapBestPlaceRenderModal').html("");
                $('#seatingMapLegendModal').html("");
            }
        };

        $scope.addModalBestPlacesToBasket = function () {
            for (let i = 0; i < $scope.model.bestPlaceSelection.priceCategories.length; i++) {
                let priceCategory = $scope.model.bestPlaceSelection.priceCategories[i];
                let priceCategoryElement = $('#pricecategory-' + priceCategory.id);
                priceCategory.ticketAmount = $('#pricecategory-' + priceCategory.id).val();
            }

            $scope.addBestPlacesToBasket();
        };

        $scope.addBestPlacesToBasket = function () {
            let ticketsToReserve = new Array();

            if ($scope.model.bestPlaceSelection.isValid) {
                for (let i = 0; i < $scope.model.bestPlaceSelection.priceCategories.length; i++) {
                    let priceCategory = $scope.model.bestPlaceSelection.priceCategories[i];
                    if (priceCategory.ticketAmount !== '') {
                        ticketsToReserve.push({ ticketId: 'bsp_sec-' + priceCategory.id + '-' + $scope.model.bestPlaceSelection.sectionCode, amount: priceCategory.ticketAmount });
                    }
                }
            }

            if (ticketsToReserve.length > 0) {
                $scope.pageService.pageModel.isLoading = true;

                $scope.packageService.addBestPlacesToBasket($scope.model.tag, $scope.model.performance.tag, ticketsToReserve, $scope.model.bestPlaceSelection.sectionCode).then(function (data) {
                    $scope.handleReservationResult(data);
                });
            }
        };

        $scope.handleReservationResult = function (data) {
            if (data.succeeded) {
                $rootScope.$broadcast('event:BasketChanged', data);
                $scope.pageService.pageModel.isLoading = false;
                if ($scope.model.isSeatingMapOpen) {
                    $scope.closeSeatingMap();
                }

                if (data.isPackageBookingFinished) {
                    // ecommerce service datalayer push
                    $scope.ecommerceService.pushAddPackageTicketToCart($scope.model, data);

                    // go to the basket
                    $scope.pageService.setStep(CURRENT_STEP_BASKET);
                } else if (data.nextStep) {
                    $scope.initNextBookingProcess(data.nextStep);
                }


            } else {
                $scope.pageService.pageModel.isLoading = false;
                $rootScope.$broadcast('event:ShowAlert', { title: $scope.model.selectedSection && $scope.model.selectedSection.name, message: data.errorMessage });
            }
        };

        $scope.addPlacesToBasket = function () {
            var selectedPlaces = $scope.selectedPlacesFinderService.find($scope.seatMapController, $scope.model.performance.canvasModel.useCanvas);

            if (selectedPlaces.length > 0) {
                if ($scope.model.amountOfItemsToSelect) {
                    if (selectedPlaces.length !== $scope.model.amountOfItemsToSelect) {
                        $rootScope.$broadcast('event:ShowAlert', { title: $scope.model.selectedSection && $scope.model.selectedSection.name, message: ticketportal.webshop.localization.seatingMapExactAmountOfPlaceAlertFormat.format($scope.model.amountOfItemsToSelect) });
                        return;
                    }
                }

                $scope.pageService.pageModel.isLoading = true;
                var section = null;
                if ($scope.model.selectedSection && $scope.model.selectedSection.code)
                    section = $scope.model.selectedSection.code;
                $scope.packageService.addPlacesToBasket($scope.model.tag, $scope.model.performance.tag, section, selectedPlaces).then(function (data) {
                    $scope.handleReservationResult(data);
                });
            }
        };

        $scope.updateCalendarMonth = function () {
            $scope.model.calendarUrl = $scope.getCalendarUrl();
        };

        $scope.performanceListPageChange = function (pageIndex) {
            $scope.model.calendarUrl = $scope.getCalendarUrl(pageIndex);
        };

        $scope.getCalendarUrl = function (pageIndex) {
            var newCalendarUrl = $scope.model.baseCalendarUrl + '?date=' + $scope.model.selectedCalendarMonth;

            if (pageIndex)
                newCalendarUrl += '&pageIndex=' + pageIndex;

            return newCalendarUrl;
        }

        // initialize the booking process
        $scope.initNextBookingProcess = function (bookingProcess) {
            $scope.model.processStepStatusText = bookingProcess.processStepStatusText;
            $scope.model.pendingStepCount = bookingProcess.pendingStepCount;
            $scope.model.totalStepCount = bookingProcess.totalStepCount;
            $scope.model.selectItemText = bookingProcess.selectItemText;
            $scope.model.amountOfItemsToSelect = bookingProcess.amountOfItemsToSelect;
            $scope.model.selectedSeats = [];

            if (bookingProcess.isNextContentTicket) {
                $scope.setPerformanceStep(bookingProcess.currentStep, bookingProcess.currentStepRestrictions);
                let nextStepUrl = 'packageTicketStep.html';
                if (nextStepUrl !== $scope.model.packageStepUrl) {
                    $scope.model.mustWaitForContentToBeLoaded = true;
                    $scope.model.packageStepUrl = nextStepUrl;
                }
            }
        };

        $scope.selectPackageContent = function () {
            if ($scope.model.isValid()) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.packageService.selectPackageContent($scope.model).then(function (data) {
                    $rootScope.$broadcast('event:BasketChanged', data);
                    $scope.pageService.pageModel.isLoading = false;
                    if (data.errorMessage) {
                        $rootScope.$broadcast('event:ShowAlert', { title: '', message: data.errorMessage });
                    } else if (data.isPackageBookingFinished) {
                        // go to the basket
                        $scope.pageService.setStep(CURRENT_STEP_BASKET);
                    } else if (data.bookingProcess) {
                        $scope.initNextBookingProcess(data.bookingProcess);
                    }
                });
            }
        };

        // display all options of an item
        $scope.displayAllOptions = function (item) {
            for (let i = 0; i < item.options.length; i++) {
                let option = item.options[i];
                option.hidden = false;
            }

            item.isDisplayingAllOptions = true;
        };

        $scope.onSeatSelected = function (seat, selected) {
            if (selected)
                $scope.model.selectedSeats.push(seat);
            else {
                for (var i = 0; i < $scope.model.selectedSeats.length; i++)
                    if ($scope.model.selectedSeats[i].id == seat.id) {
                        $scope.model.selectedSeats.splice(i, 1);
                        break;
                    }
            }
        }

        $scope.safeBindCanvasSeatingMap = function (canvasModel) {
            if ($("#canvas")[0].clientWidth > 0) {

                var seatMapControllerOptions = {
                    host: "canvas",
                    scale: "scaleSlider",
                    reset: "reset",
                    moveUp: 'moveUp',
                    moveDown: 'moveDown',
                    moveLeft: 'moveLeft',
                    moveRight: 'moveRight',
                    refresh: 'refresh',
                    performanceModel: $scope.model.performance,
                    onSeatSelected: function (seat, selected) {
                        $timeout(function () {
                            $scope.onSeatSelected(seat, selected);
                        }, 0);
                    }
                };

                var seatMapController = new seatingMapController(canvasModel, ticketService, seatMapControllerOptions, function () {
                    $scope.seatMapController = seatMapController;
                    if ($scope.seatMapController != null) {
                        $scope.model.selectedSeats.length = 0;
                        for (var i = 0; i < canvasModel.selectedSeats.length; i++)
                            $scope.model.selectedSeats.push($scope.seatMapController.getTicketInfoFromSeat(canvasModel.selectedSeats[i]));
                    }
                });
                if ($scope.seatMapController == null)
                    $scope.seatMapController = seatMapController;
            } else {
                $timeout(function () { $scope.safeBindCanvasSeatingMap(canvasModel); }, 0);
            }
        }
    }

})();;

(function () {
    'use strict';

    var modelId = 'packageModel';
    angular.module('webShop2App').factory(modelId, [model]);

    function model() {
        var model = {
            amount: 1,
            packageStepUrl: 'packageInitial.html',
            mustWaitForContentToBeLoaded: true,
            processStepStatusText: '',
            pendingStepCount: 0,
            totalStepCount: 0,
            selectItemText: 0,
            amountOfItemsToSelect: 0,
            performance: null,
            bestPlaceSelection: new bestPlaceSelectionModel(),
            stepRestrictions: [],
            optionalItemOptions: [],
            visibleItems: [],
            maximumInitialVisibleOptions: 10,
            sectionNavigationItems: [],

            // methods
            init: init,
            isValid: isValid,
            bestPlaceSelectionCategoryModel: bestPlaceSelectionCategoryModel

        };
        model.init();
        return model;

        function init() {
            if (__startupModel != undefined)
                angular.extend(this, __startupModel);

            for (var itemIndex = 0; itemIndex < this.items.length; itemIndex++) {
                var item = this.items[itemIndex];
                item.isSingleSelection = false;
                item.singleSelectedOptionId = 0;
                if (item.discounts) {
                    for (var discountIndex = item.discounts.length - 1; discountIndex >= 0; --discountIndex) {
                        var discount = item.discounts[discountIndex];
                        if (!discount.selectedAmount)
                            discount.selectedAmount = discount.minimum;
                    }
                }

                // build visible & optional items 
                if (item.needsToDisplay) {
                    // optional?
                    if (item.minimumAmountOfOptionsToSelect == 0) {
                        for (var j = 0; j < item.options.length; j++) {
                            this.optionalItemOptions.push(item.options[j]);
                        }
                    } else {
                        this.visibleItems.push(item);
                        item.isSingleSelection = item.maximumAmountOfOptionsToSelect == 1 && item.minimumAmountOfOptionsToSelect == 1;
                        // set in the singleSelectedItemId the value of the selected item
                        if (item.isSingleSelection) {
                            for (var j = 0; j < item.options.length; j++) {
                                var option = item.options[j];
                                if (option.isSelected) {
                                    item.singleSelectedOptionId = option.id;
                                    break;
                                }
                            }
                        }
                    }
                }

                // fix items with too many options
                item.isDisplayingAllOptions = true;
                if (item.options.length > this.maximumInitialVisibleOptions) {
                    for (var optionIndex = this.maximumInitialVisibleOptions; optionIndex < item.options.length; ++optionIndex) {
                        var option = item.options[optionIndex];
                        if (!option.isSelected) {
                            option.hidden = true;
                            item.isDisplayingAllOptions = false;
                        }
                    }
                }
            }
        }

        function isValid() {
            return true;
        }
    }
})();;
// Modified:		19.02.2016 ticketporal AG, St Gallen fbe	: (SCR 8833) Web shop 2: Cannot display seating map section if performance name has a pipe "|"
(function () {
    'use strict';

    var serviceId = 'packageService';
    angular.module('webShop2App').factory(serviceId, ['$http', '$sce', 'webShop2AppSettings', service]);

    function service($http, $sce, webShop2AppSettings) {
        return {
           
            // gets the html with the seating map
            getPerformanceSeatingMapHtml: function (tag, performanceTag, sectionCode) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/package/' + encodeURIComponent(tag.replace('|', '-')) + '/packageticketseatingmap?performanceTag=' + performanceTag + '&section=' + encodeURIComponent(sectionCode));
                return $http.post(postUrl).then(function (response) {
                    return response.data;
                });
            },

            // adds seating places to the basket
            addPlacesToBasket: function (tag, performanceTag, sectionCode, places) {
                var data = { placeList: places, section: sectionCode, tag: tag, performanceTag: performanceTag, json: 1 };

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/package/' + tag + '/packageticketstep?performanceTag=' + performanceTag + '&section=' + encodeURIComponent(sectionCode));

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: data
                }).then(function (response) {
                    return response.data;
                });                
            },

            // adds best places to the basket
            addBestPlacesToBasket: function (tag, performanceTag, tickets, sectionCode) {
                var data = { tag: tag, performanceTag: performanceTag, json: 1 };
                for (var i = 0; i < tickets.length; i++) {
                    data[tickets[i].ticketId] = tickets[i].amount;
                }

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/package/' + tag + '/packageticketstep?performanceTag=' + performanceTag + '&section=' + encodeURIComponent(sectionCode));

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // Sets the basket currency
            setBasketCurrency: function (currencySymbol) {
                var data = {
                    basketAction: 'changeCurrency',
                    json: 1,
                    basketActionParameter: currencySymbol
                };

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/');

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            selectPackageContent: function (model) {
                var data = {
                    amount: model.amount,
                    json: '1'
                };

                
                for (var itemIndex = model.items.length - 1; itemIndex >= 0; --itemIndex) {
                    var item = model.items[itemIndex];

                    // look for discounts
                    for (var discountIndex = item.discounts.length - 1; discountIndex >= 0; --discountIndex) {
                        var discount = item.discounts[discountIndex];
                        if (discount.selectedAmount > 0)
                            data['item-discount-' + item.id + '-' + discount.discountTypeId] = discount.selectedAmount;
                    }

                    if (item.isSingleSelection) {
                        data['option-' + item.id + '-' + item.singleSelectedOptionId.toString()] = 'on';
                    } else {
                        // look for item options
                        for (var optionIndex = item.options.length - 1; optionIndex >= 0; --optionIndex) {
                            var option = item.options[optionIndex];
                            if (option.isSelected)
                                data['option-' + item.id + '-' + option.id] = 'on';
                        }
                    }
                }                

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/package/' + model.tag);

                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            }
        }
    }

})();;
(function () {
    'use strict';

    var controllerId = 'admissionTicketController';
    angular.module('webShop2App')
        .controller(controllerId, ['$rootScope', '$scope', '$timeout', '$window', 'ecommerceService', 'admissionTicketService', 'admissionTicketModel', 'pageService', 'customizationService', 'homeModel', '$sce', '$uibModal', controller]);

    function controller($rootScope, $scope, $timeout, $window, ecommerceService, admissionTicketService, admissionTicketModel, pageService, customizationService, homeModel, $sce, $uibModal) {
        $scope.admissionTicketService = admissionTicketService;
        $scope.model = admissionTicketModel;
        $scope.pageService = pageService;
        $scope.model.isMobile = pageService.isMobile();
        $scope.customizationService = customizationService;
        $scope.mainModel = homeModel;
        $scope.ecommerceService = ecommerceService;

        // for admission tickets there is no need to start with an event description
        // we display the description and ticket reservation together
        //$scope.pageService.setStartStep(CURRENT_STEP_SHOP);       

        // Used to start the time definition selection again
        $scope.resetTimeSelection = function () {
            if ($scope.model.selectedDayDefinition) {
                $scope.model.displayTimeDefinitionSelection = true;
                $scope.model.displayPriceList = false;
                $scope.model.selectedTimeDefinition = null;
                $scope.model.isBasketLink = false;

                if ($scope.model.availableTimeDefinitions == null) {
                    $scope.checkCapacity();
                }

                // in iFrame post a message notifying that the UI changed
                $scope.pageService.postMessageToIFrameParent("admissionTicket/resetTimeSelection");
            }
        }

        // Used to start the date selection again
        $scope.resetDateSelection = function () {
            $scope.model.selectedDayDefinition = null;
            $scope.model.selectedDate = null;
            $scope.model.availableTimeDefinitions = null;
            $scope.model.displayTimeDefinitionSelection = false;
            $scope.model.displayPriceList = false;
            $scope.model.selectedTimeDefinition = null;
            $scope.model.isBasketLink = false;

            // in iFrame post a message notifying that the UI changed
            $scope.pageService.postMessageToIFrameParent("admissionTicket/resetDateSelection");
        }

        $scope.backFromPriceList = function () {
            if ($scope.model.mustSelectTimeDefinition)
                $scope.resetTimeSelection();
            else
                $scope.resetDateSelection();
        }

        // Builds the available time definitions
        // Optional parameter statusByTimeDefinitionDictionary: contains the availability per time slot. If undefined then there is not capacity check
        $scope.buildAvailableTimeDefinitions = function (statusByTimeDefinitionDictionary) {
            // create a copy of the time definitions with the respective availability
            var newAvailableTimes = new Array();
            var grouppedAvailableTimes = new Array();
            var currentGroupTime = null;
            for (var timeDefinitionIndex = 0; timeDefinitionIndex < $scope.model.selectedDayDefinition.timeDefinitions.length; timeDefinitionIndex++) {
                var timeDefinitionToShow = angular.copy($scope.model.selectedDayDefinition.timeDefinitions[timeDefinitionIndex], {});
                if (statusByTimeDefinitionDictionary !== undefined) {
                    var statusForCurrentTimeDefinition = statusByTimeDefinitionDictionary[timeDefinitionToShow.id];
                    if (statusForCurrentTimeDefinition !== undefined) {
                        if (statusForCurrentTimeDefinition.inPast || !statusForCurrentTimeDefinition.isValid)
                            continue;
                        timeDefinitionToShow.availability = statusForCurrentTimeDefinition;
                        timeDefinitionToShow.availability.status = timeDefinitionToShow.availability.status.toLowerCase();
                        timeDefinitionToShow.availability.isBookable = timeDefinitionToShow.availability.status != 'soldout';
                        timeDefinitionToShow.timeSlotTypeId = statusForCurrentTimeDefinition.timeSlotTypeId;
                    }
                }

                if (timeDefinitionToShow.availability === undefined)
                    timeDefinitionToShow.availability = { status: 'available', isBookable: true, inPast: false, isValid: true };

                timeDefinitionToShow.prices = [];
                var timeSlotType = $scope.model.findTimeSlotType(timeDefinitionToShow.timeSlotTypeId);
                if (timeSlotType != null) {
                    if (timeSlotType.name !== undefined && timeSlotType.name.length > 0)
                        timeDefinitionToShow.name += ' - ' + timeSlotType.name;
                    timeDefinitionToShow.hasPermission = timeDefinitionToShow.hasPermission && timeSlotType.hasPermission;
                    timeDefinitionToShow.displayIfNotAllowed = timeSlotType.displayIfNotAllowed;
                    timeDefinitionToShow.notAllowedMessage = timeSlotType.notAllowedMessage;
                    timeDefinitionToShow.prices = timeSlotType.prices;
                }

                newAvailableTimes.push(timeDefinitionToShow);

                if (timeDefinitionToShow.hasPermission || timeDefinitionToShow.displayIfNotAllowed) {
                    if (currentGroupTime == null || currentGroupTime.startHour !== timeDefinitionToShow.startTime.substring(0, 2)) {

                        currentGroupTime = {
                            startHour: timeDefinitionToShow.startTime.substring(0, 2),
                            name: timeDefinitionToShow.startTime.substring(0, 2) + ':00',
                            items: new Array()
                        };

                        grouppedAvailableTimes.push(currentGroupTime);
                    }

                    currentGroupTime.items.push(timeDefinitionToShow);
                }
            }

            $scope.model.displayTimeDefinitionSelection = true;
            $scope.model.displayPriceList = false;
            $scope.model.availableTimeDefinitions = newAvailableTimes;
            $scope.model.grouppedAvailableTimeDefinitions = grouppedAvailableTimes;
        }

        $scope.selectedDateChanged = function () {

            if (!$scope.model.selectedDate || $scope.model.isBasketLink)
                return;

            // reset any time definition that was previously selected
            $scope.model.selectedTimeDefinition = null;
            
            // find the day definition for the selected date
            $scope.model.selectedDayDefinition = $scope.model.findDayDefinition($scope.model.selectedDate);
            if ($scope.model.selectedDayDefinition != null) {
                if ($scope.model.mustSelectTimeDefinition) {

                    if ($scope.model.selectedDayDefinition.hasPermission) {
                        if ($scope.model.hasCapacityCheck) {
                            // Capacity check is enabled
                            // get for all time definitions the current status for the selected date
                            // possible status : 'Available', 'NotAvailable'
                            $scope.checkCapacity();
                        } else {
                            // No capacity check: display all time definitions as 'available'
                            $scope.buildAvailableTimeDefinitions();
                        }
                    }

                } else {

                    $scope.loadAndDisplayPriceList();
                }
            } else {
                $scope.model.displayTimeDefinitionSelection = false;
                $scope.model.displayPriceList = false;
                $scope.model.availablePrices = null;
                $scope.model.availableTimeDefinitions = null;
            }

            // in iFrame post a message notifying that the UI changed
            // we can only update the IFrame size if the pageModel is properly loaded,
            // therefore, if it is not yet loaded we try again later
            function watitForPageLoadAndEmitEvent() {
                if ($scope.$$phase === null && $scope.pageService.pageModel.isLoading === false) {
                    $scope.pageService.postMessageToIFrameParent("admissionTicket/selectedDateChanged");
                } else {
                    // wait for 10 ms an try again
                    $timeout(watitForPageLoadAndEmitEvent, 10);
                }
            }
            watitForPageLoadAndEmitEvent();

        }

        // Loads the price list based on the selected date & time
        $scope.loadAndDisplayPriceList = function () {
            // user does not have to choose a time definition (not time slot based)
            // the list of possible prices is based on all price categories of all time definitions

            // get for all time definitions the current status for the selected date
            // possible status : 'Available', 'NotAvailable'
            $scope.pageService.loading(true);
            $scope.admissionTicketService.getPrices($scope.model.tag, $scope.model.selectedDate, $scope.model.selectedDayDefinition.id, 0).then(function (result) {

                $scope.model.availableTimeDefinitions = [$scope.model.selectedDayDefinition.timeDefinitions];
                $scope.model.availablePrices = new Array();
                for (var i = 0; i < result.availabilities.length; ++i) {
                    var timeDefinitionPrices = result.availabilities[i];
                    var flattenPriceCategories = $scope.customizationService.buildFlattenPriceCategory(timeDefinitionPrices);
                    for (var flattenPriceCategoryIndex = 0; flattenPriceCategoryIndex < flattenPriceCategories.length; flattenPriceCategoryIndex++) {
                        var flattenPriceCategory = flattenPriceCategories[flattenPriceCategoryIndex];
                        flattenPriceCategory.timeDefinitionId = timeDefinitionPrices.timeDefinitionId;
                        flattenPriceCategory.timeDefinitionName = timeDefinitionPrices.timeDefinitionName;

                        $scope.model.availablePrices.push(flattenPriceCategory)
                    }
                }

                $scope.model.displayTimeDefinitionSelection = false;
                $scope.model.displayPriceList = true;
                $scope.pageService.loading(false);

            });

        }

        // called when the user selects a time definition
        // only happens with time slot based performances
        $scope.selectTime = function (timeDefinition) {
            if ($scope.model.isBasketLink || timeDefinition.availability.availablePlaces > 0) {
                // the list of available prices is collection from the time definition
                $scope.model.selectedTimeDefinition = timeDefinition;
                $scope.model.availablePrices = null;

                if ($scope.model.selectedTimeDefinition.hasPermission) {

                    // Capacity check is enabled
                    // get for all time definitions the current status for the selected date
                    // possible status : 'Available', 'NotAvailable'            
                    $scope.pageService.loading(true);

                    $scope.admissionTicketService.getPrices($scope.model.tag, $scope.model.selectedDate, $scope.model.selectedDayDefinition.id, $scope.model.selectedTimeDefinition.id).then(function (result) {
                        if (!result.isTimeDefinitionReservationAllowed) {
                            $scope.model.selectedTimeDefinition.hasPermission = false;
                            $scope.model.selectedTimeDefinition.notAllowedMessage = $sce.trustAsHtml(result.message);

                        } else {
                            $scope.model.availablePrices = new Array();
                            for (var i = 0; i < result.availabilities.length; ++i) {
                                var timeDefinitionPrices = result.availabilities[i];
                                var flattenPriceCategories = $scope.customizationService.buildFlattenPriceCategory(timeDefinitionPrices);
                                for (var flattenPriceCategoryIndex = 0; flattenPriceCategoryIndex < flattenPriceCategories.length; flattenPriceCategoryIndex++) {
                                    var flattenPriceCategory = flattenPriceCategories[flattenPriceCategoryIndex];
                                    flattenPriceCategory.timeDefinitionId = timeDefinitionPrices.timeDefinitionId;
                                    flattenPriceCategory.timeDefinitionName = timeDefinitionPrices.timeDefinitionName;

                                    $scope.model.availablePrices.push(flattenPriceCategory)
                                }
                            }

                            $scope.model.displayPriceList = true;
                        }

                        $scope.pageService.loading(false);
                        // in iFrame post a message notifying that the UI changed
                        $scope.pageService.postMessageToIFrameParent("admissionTicket/timeSelected");

                    });
                }
            }
            else {
                $scope.pageService.showModal({
                    message: ticketportal.webshop.localization.timeslotSoldOut
                });
            }
        }

        $scope.isDateDisabled = function (date, mode) {
            return !$scope.model.isDateEnabled(date);
        }


        $scope.$watch('model.selectedDate', function () {
            $scope.selectedDateChanged();

        });

        // Starts the booking process
        $scope.startBookingProcess = function () {
            if (!$scope.pageService.getStep())
                $scope.pageService.setStep(CURRENT_STEP_SHOP);
        }


        $scope.addToBasket = function (isValid, fromAttributesSelection) {

            if (fromAttributesSelection == undefined)
                fromAttributesSelection = false;

            if (isValid) {
                var ticketsToReserve = new Array();
                var attributesToSelect = [];
                for (var priceIndex = 0; priceIndex < $scope.model.availablePrices.length; priceIndex++) {
                    var price = $scope.model.availablePrices[priceIndex];
                    var ticketAmount = parseInt(price.ticketAmount);
                    if (ticketAmount > 0) {
                        var attributesSelection = [];
                        for (var i = priceIndex; i >= 0; i--) {
                            var fullPrice = $scope.model.availablePrices[i];
                            if (price.priceCategoryUniqueId != fullPrice.priceCategoryUniqueId)
                                break;
                            if (fullPrice.attributes != null) {
                                for (var j = 0; j < fullPrice.attributes.length; j++) {
                                    var attribute = fullPrice.attributes[j];
                                    if (fromAttributesSelection) {
                                        if (attribute.selectedValue && attribute.selectedValue.length > 0) {
                                            attributesSelection.push({
                                                id: attribute.id,
                                                value: attribute.selectedValue
                                            })
                                        }
                                    }
                                    else
                                        attributesToSelect.push(attribute);
                                }
                            }
                        }
                        ticketsToReserve.push({
                            amount: ticketAmount,
                            timeDefinitionId: price.timeDefinitionId,
                            dayDefinitionId: $scope.model.selectedDayDefinition.id,
                            priceCategoryId: price.priceCategoryUniqueId,
                            discountId: price.discountId,
                            attributes: attributesSelection
                        });
                    }
                }
                if (attributesToSelect.length > 0) {
                    // if there are any attribute configured for selection, show the modal
                    $scope.showAttributeSelection(attributesToSelect);
                }
                else if (ticketsToReserve.length > 0) {
                    $scope.pageService.loading(true);

                    $scope.admissionTicketService.addToBasket($scope.model.tag, ticketsToReserve, $scope.model.selectedDate).then(function (data) {
                        $scope.handleReservationResult(data);
                        if (data.succeeded) {
                            $scope.ecommerceService.pushAdmissionTicketToCart($scope.mainModel, $scope.model, ticketsToReserve);
                        }
                    });

                }
            }
        }

        $scope.showAttributeSelection = function (attributesToSelect) {
            var modalInstance = $uibModal.open({
                templateUrl: "attributesSelectionModalContent.html",
                controller: 'attributesSelectionController',
                resolve: {
                    attributes: function () {
                        return attributesToSelect;
                    },
                    parentScope: function () {
                        return $scope;
                    }
                }
            });
        }

        $scope.handleReservationResult = function (data) {
            if (data.succeeded) {
                $rootScope.$broadcast('event:BasketChanged', data);
                $scope.pageService.loading(false);

                if ($scope.pageService.pageModel.hasCrossSelling)
                    $scope.pageService.setStep(CURRENT_STEP_CROSSSELLING);
                else
                    $scope.pageService.setStep(CURRENT_STEP_BASKET);
            } else {
                $scope.pageService.loading(false);
                $rootScope.$broadcast('event:ShowAlert', { title: ticketportal.webshop.localization.errorTitle, message: data.errorMessage });
            }
        }

        if (window.__admissionTicketModel) {
            $scope.model.init(window.__admissionTicketModel, $sce);
            if (!$scope.model.mustSelectDate) {

                if ($scope.model.mustSelectTimeDefinition) {
                    // do not have to choose a date
                    // display the price list                
                    $scope.model.displayPriceList = false;
                    $scope.model.displayTimeDefinitionSelection = true;
                    $scope.buildAvailableTimeDefinitions();
                } else {
                    // display the price list
                    $scope.model.selectedDayDefinition = $scope.model.dayDefinitions[0];
                    $scope.loadAndDisplayPriceList();
                }
            }
        }

        $scope.checkCapacity = function () {
            $scope.pageService.loading(true);
            $scope.admissionTicketService.checkCapacity($scope.model.tag, $scope.model.selectedDate, $scope.model.selectedDayDefinition.id).then(function (result) {

                // no permission for the selected day definition
                if (!result.isDayDefinitionReservationAllowed) {
                    $scope.model.selectedDayDefinition.hasPermission = false;
                    $scope.model.selectedDayDefinition.notAllowedMessage = $sce.trustAsHtml(result.message);

                } else {
                    $scope.buildAvailableTimeDefinitions(result.statusByTimeDefinition);
                }

                $scope.pageService.loading(false);
            });
        }

        // handle 'display time slot if item was clicked in basket'
        $scope.$watch('pageService.pageModel.currentStep', function () {
            if ($scope.pageService.getStep() == CURRENT_STEP_SHOP) {
                if (getQueryStringValue('step') == 'shop') {
                    if ($scope.mainModel.basketItemCount > 0) {
                        var performance = getQueryStringValue('performance');
                        if (performance && performance.length > 0) {
                            var date = getQueryStringValue('date');
                            if (date && date.length > 0) {
                                $scope.model.isBasketLink = true;
                                var dateToSelect = new Date(date);
                                dateToSelect.setHours(0, 0, 0, 0);
                                $scope.model.selectedDate = dateToSelect;
                                var dayDefinitionId = getQueryStringValue('dayDefinitionId');
                                if (dayDefinitionId && dayDefinitionId.length > 0) {
                                    $scope.model.selectedDayDefinition = $.grep($scope.model.dayDefinitions, function (n, i) { return n.id == parseInt(dayDefinitionId); })[0];
                                    var timeDefinitionId = getQueryStringValue('timeDefinitionId');
                                    if (timeDefinitionId && timeDefinitionId.length > 0) {
                                        var timeDefinition = $.grep($scope.model.selectedDayDefinition.timeDefinitions, function (n, i) { return n.id == parseInt(timeDefinitionId); })[0];
                                        $scope.selectTime(timeDefinition);
                                        return;
                                    }
                                }
                            }
                        }
                    }
                }
            }
        });
    }

    angular.module('webShop2App').controller('attributesSelectionController', ['$scope', '$modalInstance', 'attributes', 'parentScope', attributesSelectionController])
    function attributesSelectionController($scope, $modalInstance, attributes, parentScope) {
        attributes.forEach(function (attribute) { attribute.selectedValue = null });
        $scope.attributes = attributes;
        $scope.cancel = function () {
            $modalInstance.dismiss('cancel');
        }
        $scope.addToBasket = function (valid) {
            parentScope.addToBasket(valid, true);
            $modalInstance.close();
        }
    }

})();;
// Modified:		06.11.2017 Starticket AG, St. Gallen fbe	: #6155 Admission ticket: cannot select day definition using "not valid dates" filter
(function () {
    'use strict';

    var modelId = 'admissionTicketModel';
    angular.module('webShop2App').factory(modelId, [model]);

    function model() {
        var model = {
            tag: null,
            reservationText: null,
            isBasketLink: false,
            selectedDate: null,
            selectedDayDefinition: null,
            selectedTimeDefinition: null,
            startDate: new Date(),
            endDate: new Date(),
            mustSelectDate: false,
            mustSelectTimeDefinition: false,
            availableTimeDefinitions: null,
            grouppedAvailableTimeDefinitions: null,
            availablePrices: null,
            displayTimeDefinitionSelection: false,
            displayPriceList: false,
            hasCapacityCheck: false,
            dayDefinitions: [],
            timeSlotTypes: [],
            closedDays: { specificDays: [], specificNotValidDays: [], daysOfTheWeek: [] },

            init: init,
            isDateEnabled: isDateEnabled,
            findDayDefinition: findDayDefinition,
            isSameDate: isSameDate,
            findTimeSlotType: findTimeSlotType
        };

        function isSameDate(date1, date2) {
            return date1.getDate() === date2.getDate() &&
                date1.getMonth() === date2.getMonth() &&
                date1.getYear() === date2.getYear();
        }

        // Check if the date is valid in the calendar
        function isDateEnabled(date) {
            if (this.closedDays !== null) {
                if (this.closedDays.specificNotValidDays !== null && this.closedDays.specificNotValidDays.length > 0) {
                    for (var i = 0; i < this.closedDays.specificNotValidDays.length; ++i) {
                        if (this.isSameDate(date, this.closedDays.specificNotValidDays[i])) {
                            return true;
                        }
                    }
                }

                if (this.closedDays.daysOfTheWeek !== null && this.closedDays.daysOfTheWeek.length > 0) {
                    if (this.closedDays.daysOfTheWeek.indexOf(date.getDay()) !== -1) {
                        return false;
                    }
                }

                if (this.closedDays.specificDays !== null && this.closedDays.specificDays.length > 0) {
                    for (var i = 0; i < this.closedDays.specificDays.length; ++i) {
                        if (this.isSameDate(date, this.closedDays.specificDays[i])) {
                            return false;
                        }
                    }
                }
            }

            return true;
        }

        function init(source, $sce) {
            this.tag = source.tag;
            this.reservationText = source.reservationText;
            this.mustSelectDate = source.mustSelectDate;
            this.mustSelectTimeDefinition = source.mustSelectTimeDefinition;
            this.hasCapacityCheck = source.hasCapacityCheck;
            this.startDate = new Date(source.startDate);
            var today = new Date();
            if (this.startDate < today) {
                this.startDate = today;
            }
            this.endDate = new Date(source.endDate);
            this.timeSlotSelectionMode = source.timeSlotSelectionMode;

            
            if (source.closedDays !== null) {
                if (source.closedDays.specificDates !== null) {
                    for (var dateIndex = 0; dateIndex < source.closedDays.specificDates.length; ++dateIndex) {
                        this.closedDays.specificDays.push(new Date(source.closedDays.specificDates[dateIndex]));
                    }
                }

                if (source.closedDays.specificNotValidDates !== null) {
                    for (var dateIndex = 0; dateIndex < source.closedDays.specificNotValidDates.length; ++dateIndex) {
                        this.closedDays.specificNotValidDays.push(new Date(source.closedDays.specificNotValidDates[dateIndex]));
                    }
                }

                if (source.closedDays.daysOfTheWeek !== null) {
                    for (var dayOfTheWeekIndex = 0; dayOfTheWeekIndex < source.closedDays.daysOfTheWeek.length; ++dayOfTheWeekIndex) {
                        this.closedDays.daysOfTheWeek.push(source.closedDays.daysOfTheWeek[dayOfTheWeekIndex]);
                    }
                }
            }


            if (source.dayDefinitions !== null) {
                for (var i = 0; i < source.dayDefinitions.length; ++i) {
                    var sourceDayDefinition = source.dayDefinitions[i];                    
                    var dayDefinitionModel = {
                        daysOfTheWeek: sourceDayDefinition.daysOfTheWeek,
                        id: sourceDayDefinition.id,
                        specificDays: null,
                        specificNotValidDays: null,
                        timeDefinitions: null,
                        displayIfNotAllowed: sourceDayDefinition.displayIfNotAllowed,
                        notAllowedMessage: $sce.trustAsHtml(sourceDayDefinition.notAllowedMessage),
                        hasPermission: sourceDayDefinition.hasPermission,
                        timeDefinitionSelectionText: $sce.trustAsHtml(sourceDayDefinition.timeDefinitionSelectionText)
                    };

                    if (sourceDayDefinition.timeDefinitions !== null) {
                        dayDefinitionModel.timeDefinitions = [];

                        for (var timeDefinitionIndex = 0; timeDefinitionIndex < sourceDayDefinition.timeDefinitions.length; ++timeDefinitionIndex) {
                            var sourceTimeDefinition = sourceDayDefinition.timeDefinitions[timeDefinitionIndex];
                            var timeDefinitionModel = {
                                id: sourceTimeDefinition.id,
                                name: sourceTimeDefinition.name,
                                startTime: sourceTimeDefinition.startTime,
                                hasPermission: sourceTimeDefinition.hasPermission,
                                timeSlotTypeId: sourceTimeDefinition.timeSlotTypeId
                            };

                            dayDefinitionModel.timeDefinitions.push(timeDefinitionModel);
                        }
                    }

                    if (sourceDayDefinition.specificDays !== null) {
                        dayDefinitionModel.specificDays = [];
                        for (var dateIndex = 0; dateIndex < sourceDayDefinition.specificDays.length; ++dateIndex) {
                            dayDefinitionModel.specificDays.push(new Date(sourceDayDefinition.specificDays[dateIndex]));
                        }

                    }


                    if (sourceDayDefinition.specificNotValidDays !== null) {
                        dayDefinitionModel.specificNotValidDays = [];
                        for (var dateIndex = 0; dateIndex < sourceDayDefinition.specificNotValidDays.length; ++dateIndex) {
                            dayDefinitionModel.specificNotValidDays.push(new Date(sourceDayDefinition.specificNotValidDays[dateIndex]));
                        }

                    }

                    this.dayDefinitions.push(dayDefinitionModel);
                }
            }

            if (source.timeSlotTypes !== null) {
                for (var i = 0; i < source.timeSlotTypes.length; i++) {
                    var timeSlotType = source.timeSlotTypes[i];

                    var timeSlotTypeModel = angular.copy(timeSlotType);
                    timeSlotTypeModel.notAllowedMessage = $sce.trustAsHtml(timeSlotType.notAllowedMessage);
                    timeSlotTypeModel.prices = [];
                    for (var priceIndex = 0; priceIndex < timeSlotType.prices.length; priceIndex++) {
                        var sourcePrice = timeSlotType.prices[priceIndex];
                        timeSlotTypeModel.prices.push(new timeDefinitionPriceSelectionModel(sourcePrice));

                        for (var discountIndex = 0; discountIndex < sourcePrice.discounts.length; discountIndex++) {
                            var sourceDiscount = sourcePrice.discounts[discountIndex];
                            timeSlotTypeModel.prices.push(new timeDefinitionPriceSelectionModel(sourcePrice, sourceDiscount));
                        }
                    }

                    this.timeSlotTypes.push(timeSlotTypeModel);
                }
            }
        }

        // finds the day definition based on a specific date
        function findDayDefinition(date) {
            for (var i = 0; i < this.dayDefinitions.length; i++) {
                var dayDefinition = this.dayDefinitions[i];

                // check not valid dates
                if (dayDefinition.specificNotValidDays) {
                    for (var dayIndex = 0; dayIndex < dayDefinition.specificNotValidDays.length; dayIndex++) {
                        if (this.isSameDate(dayDefinition.specificNotValidDays[dayIndex], date)) {
                            continue;
                        }
                    }
                }

                // check specific days
                if (dayDefinition.specificDays) {
                
                    for (var dayIndex = 0; dayIndex < dayDefinition.specificDays.length; dayIndex++) {
                        if (this.isSameDate(dayDefinition.specificDays[dayIndex], date)) {
                            return dayDefinition;
                        }
                    }

                    // did not find by specific date == does not match
                    continue;
                }

                // check days of the week
                if (dayDefinition.daysOfTheWeek && dayDefinition.daysOfTheWeek.length > 0) {
                    if (dayDefinition.daysOfTheWeek.indexOf(date.getDay()) != -1)
                        return dayDefinition;
                    else
                        continue;
                }

                return dayDefinition;
            }

            return null;
        }

        return model;
        
    }

    function findTimeSlotType(timeSlotTypeId) {
        for (var i = 0; i < this.timeSlotTypes.length; i++) {
            var timeSlotType = this.timeSlotTypes[i];
            if (timeSlotType.id === timeSlotTypeId)
                return timeSlotType;
        }

        return null;
    }

    function timeDefinitionPriceSelectionModel(category, discount) {
        var model = {
            id: category.priceCategoryUniqueId + (discount !== undefined ? discount.id.toString() : '0'),
            name: discount === undefined ? category.priceCategoryName : discount.name,
            priceCategoryUniqueId: category.priceCategoryUniqueId,
            discountId: discount !== undefined ? discount.id : 0,
            price: category.price,
            priority: category.priceCategoryPriority,
            formattedPrice: category.formattedPrice,
            priceWithFees: category.priceWithFees,
            formattedPriceWithFees: category.formattedPriceWithFees,
            isSoldOut: category.isSoldOut,
            ticketAmount: ''
        };

        

        return model;

    }

})();;
// Modified:		05.04.2017 starticket AG, St. Gallen fbe  : #2918 Admission ticket booking in a different timezone causes the ticket to be in a wrong date
(function () {
    'use strict';

    var serviceId = 'admissionTicketService';
    angular.module('webShop2App').factory(serviceId, ['$http', '$sce', 'webShop2AppSettings', service]);

    function service($http, $sce, webShop2AppSettings) {
        return {
            safeDate: function(date) {
                var month = date.getMonth() + 1;
                var safeMonth = (month < 10) ? '0' + month.toString() : month.toString();

                var day = date.getDate();
                var safeDay = (day < 10) ? '0' + day.toString() : day.toString();
                return date.getFullYear().toString() + "-" + safeMonth + "-" + safeDay;
            },

            checkCapacity: function(performanceTag, date, dayDefinitionId){
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + performanceTag + '/checkadmissionticketcapacity');
                var data = {
                    performanceTag: performanceTag,
                    date: this.safeDate(date),
                    dayDefinitionId: dayDefinitionId
                };


                var postResult = $http.post(postUrl, data);
                return postResult.then(function (result) {
                    return result.data;
                });
            },

            // gets the prices given a performance, date, day & time definition
            getPrices: function (performanceTag, date, dayDefinitionId, timeDefinitionId) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + performanceTag + '/getadmissionticketprices');
                var data = {
                    performanceTag: performanceTag,                    
                    dayDefinitionId: dayDefinitionId,
                    timeDefinitionId: timeDefinitionId
                };

                if (date !== undefined && date)
                    data.date = this.safeDate(date);                


                var postResult = $http.post(postUrl, data);
                return postResult.then(function (result) {
                    return result.data;
                });
            },

            // adds best places to the basket
            addToBasket: function (tag, tickets, selectedDate) {
                var data = {
                    json: 1,
                    tag: tag,
                    admissionTicket: 1,                    
                    admissionTickets: tickets
                };

                if (selectedDate !== undefined && selectedDate)
                    data.selectedDate = this.safeDate(selectedDate);                

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/ticket/' + tag + '/reservation');
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: data,
                }).then(function (response) {
                    return response.data;
                });
            }
        }
    }

})();;
(function () {
    'use strict';

    var controllerId = 'productController';
    angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', 'productService', 'productModel', 'ecommerceService', 'pageService', 'commonService', '$modal', controller]);

    function controller($rootScope, $scope, productService, productModel, ecommerceService, pageService, commonService, $modal) {
        $scope.productService = productService;
        $scope.model = productModel;
        $scope.pageService = pageService;
        $scope.modal = $modal;
        $scope.ecommerce = ecommerceService;


        $scope.pageService.setStartStep(CURRENT_STEP_SHOP);

        $scope.addProductToBasket = function (productItem) {
            $scope.pageService.pageModel.isLoading = true;
            var products = new Array();
            if (productItem.amount > 0) {
                var addProductModel = new $scope.model.addProductModel();
                addProductModel.id = productItem.id;
                addProductModel.amount = productItem.amount;
                products.push(addProductModel);
            }

            ecommerceService.pushAddToCartProduct($scope.model, products);

            $scope.productService.addProductsToBasket($scope.model.productDefinitionModel.tag, products).then(function (data) {
                $scope.handleReservationResult(data);
            });
        }

        $scope.handleReservationResult = function (data) {
            if (data.succeeded) {
                $rootScope.$broadcast('event:BasketChanged', data);
                $scope.pageService.pageModel.isLoading = false;

                // go to the basket
                $scope.pageService.setStep(CURRENT_STEP_BASKET);
            } else {
                $scope.pageService.pageModel.isLoading = false;
            }
        }
    }

})();;
(function () {
    'use strict';

    var modelId = 'productModel';
    angular.module('webShop2App').factory(modelId, ['$sce', model]);

    function model($sce) {
        var model = {
            addProducts: [],

            // child models
            addProductModel: addProductModel,

            // functions
            init: init,
            getProductOrDefinitionImage: getProductOrDefinitionImage
        };
        model.init($sce);
        return model;

        function init($sce) {
            if (__startupModel != undefined) {
                angular.extend(this, __startupModel);
                for (var productIndex = 0; productIndex < this.productDefinitionModel.products.length; productIndex++) {
                    var productItem = this.productDefinitionModel.products[productIndex];
                    productItem.descriptionSafe = $sce.trustAsHtml(productItem.description);
                    productItem.amount = 0;
                    var itemOptions = new Array();
                    for (var itemAmount = 0; itemAmount <= this.productDefinitionModel.maximumToSelect; ++itemAmount) {
                        itemOptions.push(itemAmount);
                    }
                    productItem.itemOptions = itemOptions;
                }
            }
        }

        function getProductOrDefinitionImage(productItem) {
            if (productItem.imageCollection.length > 0)
                return productItem.imageCollection[0];
            else if (this.productDefinitionModel.imageCollection.length > 0)
                return this.productDefinitionModel.imageCollection[0];
        }

        function addProductModel() {
            var model = { id: 0, amount: 0 };
            return model;
        }
    }

})();;
(function () {
    'use strict';

    var serviceId = 'productService';
    angular.module('webShop2App').factory(serviceId, ['$http', 'webShop2AppSettings', service]);

    function service($http, webShop2AppSettings) {
        return {
            // adds products to the basket
            addProductsToBasket: function (tag, products) {
                var data = { tag: tag, products: JSON.stringify(products), json: 1 };

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/product/' + tag);
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            }
        };
    }

})();;
// Modified:		08.12.2015 ticketporal AG, St Gallen fbe	: (SCR 8649) Gift certificate: Error-message when value not between minimum and maximum amount
(function () {
    'use strict';

    var controllerId = 'giftCertificateController';
    angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', 'giftCertificateService', 'giftCertificateModel', 'ecommerceService', 'pageService', 'commonService', '$modal', '$sce', controller]);

    function controller($rootScope, $scope, giftCertificateService, giftCertificateModel, ecommerceService, pageService, commonService, $modal, $sce) {
        $scope.giftCertificateService = giftCertificateService;
        $scope.model = giftCertificateModel;
        $scope.pageService = pageService;
        $scope.modal = $modal;
        $scope.ecommerce = ecommerceService;

        $scope.pageService.setStartStep(CURRENT_STEP_SHOP);

        $scope.addGiftCertificateToBasket = function (giftCertificateItem) {
            $scope.pageService.pageModel.isLoading = true;
            var giftCertificates = new Array();
            var value = giftCertificateItem.value ? (giftCertificateItem.value.value ? giftCertificateItem.value.value : giftCertificateItem.value) : 0;
            if (giftCertificateItem.amount > 0 && value > 0) {
                var addGiftCertificateModel = new $scope.model.addGiftCertificateModel();
                addGiftCertificateModel.id = giftCertificateItem.id;
                addGiftCertificateModel.amount = giftCertificateItem.amount;
                addGiftCertificateModel.value = value;
                addGiftCertificateModel.message = giftCertificateItem.message;
                addGiftCertificateModel.name = giftCertificateItem.name;
                giftCertificates.push(addGiftCertificateModel);
            }

            $scope.ecommerce.pushAddToCartGiftCertificates($scope.model, giftCertificates);

            $scope.giftCertificateService.addGiftCertificatesToBasket(giftCertificates).then(function (data) {
                $scope.handleReservationResult(data);
            });
        }

        $scope.handleReservationResult = function (data) {
            if (data.succeeded) {
                $rootScope.$broadcast('event:BasketChanged', data);
                $scope.pageService.pageModel.isLoading = false;

                // go to the basket
                $scope.pageService.setStep(CURRENT_STEP_BASKET);
            } else {
                $scope.pageService.showModal({ message: $sce.trustAsHtml(data.error.message) });
                $scope.pageService.pageModel.isLoading = false;
            }
        }
    }

})();;
(function () {
    'use strict';

    var modelId = 'giftCertificateModel';
    angular.module('webShop2App').factory(modelId, ['$sce', model]);

    function model($sce) {
        var model = {
            addGiftCertificates: [],

            // child models
            addGiftCertificateModel: addGiftCertificateModel,

            // functions
            init: init
        };
        model.init($sce);
        return model;

        function init($sce) {
            if (__startupModel != undefined) {
                angular.extend(this, __startupModel);
                for (var giftCertificateIndex = 0; giftCertificateIndex < this.giftCertificates.length; giftCertificateIndex++) {
                    var giftCertificateItem = this.giftCertificates[giftCertificateIndex];
                    giftCertificateItem.descriptionSafe = $sce.trustAsHtml(giftCertificateItem.description);
                    giftCertificateItem.amount = 1;
                    if (giftCertificateItem.values && giftCertificateItem.values.length > 0) {
                        giftCertificateItem.value = giftCertificateItem.values.find(v =>  v.value === __startupModel.initialValue);
                        if (giftCertificateItem.value === undefined) {
                            giftCertificateItem.value = giftCertificateItem.values[0];
                        }
                    }
                    else
                        giftCertificateItem.value = '';
                    giftCertificateItem.message = '';
                }
            }
        }

        function addGiftCertificateModel() {
            var model = {
                id: 0,
                amount: 0,
                value: 0,
                message: ''
            };
            return model;
        }
    }

})();;
(function () {
    'use strict';

    var serviceId = 'giftCertificateService';
    angular.module('webShop2App').factory(serviceId, ['$http', 'webShop2AppSettings', service]);

    function service($http, webShop2AppSettings) {
        return {
            // adds products to the basket
            addGiftCertificatesToBasket: function (giftCertificates) {
                var data = { giftCertificates: JSON.stringify(giftCertificates), json: 1 };

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/giftcertificate/');
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            }
        };
    }

})();;
// Modified:		01.02.2017 Starticket AG, St. Gallen fbe  : #1496 Webshop 2: PopUp Buchungsregeln wird nicht dargestellt
(function () {
    'use strict';

    var controllerId = 'crossSellingController';
    angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', '$window', 'ecommerceService', 'ticketService', 'productService', 'giftCertificateService', 'crossSellingModel', 'pageService', controller]);

    function controller($rootScope, $scope, $window, ecommerceService, ticketService, productService, giftCertificateService, crossSellingModel, pageService) {
        $scope.model = crossSellingModel;
        $scope.pageService = pageService;
        $scope.ecommerceService = ecommerceService;
        $scope.model.hasCrossSellingAdded = false;

        $scope.crossSellingHasDiscount = function (priceCategory) {
            return priceCategory.priceCategoryAndDiscountId != priceCategory.priceCategoryUniqueId;
        };

        $scope.continueWithoutCrossSelling = function () {
            return $scope.model.hasCrossSellingAdded
        }

        $scope.addCrossSellingPerformanceToBasket = function (performance) {
            var ticketsToReserve = new Array();
            for (var i = 0; i < performance.bookingCategories.length; i++) {
                var bookingCategory = performance.bookingCategories[i];
                if (bookingCategory.amount > 0) {
                    ticketsToReserve.push({ ticketId: bookingCategory.bookingCategoryId, amount: bookingCategory.amount });
                }
            }

            if (ticketsToReserve.length > 0) {
                $scope.pageService.pageModel.isLoading = true;

                ticketService.addBestPlacesToBasket(performance.tag, ticketsToReserve, '').then(function (data) {
                    $scope.handleReservationResult(data);
                    if (data.succeeded) {
                        $scope.ecommerceService.pushCrossSellingPerformanceToCart($scope.model, performance);
                        $scope.model.hasCrossSellingAdded = true;
                    }
                });
            }
        };

        $scope.addCrossSellingProductToBasket = function (productDefinition) {
            $scope.pageService.pageModel.isLoading = true;
            var products = new Array();
            for (var i = 0; i < productDefinition.products.length; ++i) {
                var productItem = productDefinition.products[i];
                if (productItem.amount > 0) {
                    var addProductModel = { id: 0, amount: 0 };
                    addProductModel.id = productItem.id;
                    addProductModel.amount = productItem.amount;
                    products.push(addProductModel);
                }
            }

            productService.addProductsToBasket(productDefinition.tag, products).then(function (data) {
                $scope.handleReservationResult(data);
                if (data.succeeded) {
                    $scope.ecommerceService.pushCrossSellingProductsToCart($scope.model, productDefinition);
                    $scope.model.hasCrossSellingAdded = true;
                }
            });
        };

        $scope.addCrossSellingGiftCertificateToBasket = function (giftCertificateItem) {
            $scope.pageService.pageModel.isLoading = true;
            var giftCertificates = new Array();
            var value = giftCertificateItem.value ? (giftCertificateItem.value.value ? giftCertificateItem.value.value : giftCertificateItem.value) : 0;
            if (giftCertificateItem.amount > 0 && value > 0) {
                var addGiftCertificateModel = { id: 0, amount: 0, value: 0, message: '' };
                addGiftCertificateModel.id = giftCertificateItem.id;
                addGiftCertificateModel.amount = giftCertificateItem.amount;
                addGiftCertificateModel.value = value;
                addGiftCertificateModel.message = giftCertificateItem.message;
                giftCertificates.push(addGiftCertificateModel);
            }

            giftCertificateService.addGiftCertificatesToBasket(giftCertificates).then(function (data) {
                $scope.handleReservationResult(data);
                if (data.succeeded) {
                    $scope.ecommerceService.pushCrossSellingGiftCertificateToCart(giftCertificateItem);
                    $scope.model.hasCrossSellingAdded = true;
                }
            });
        };

        $scope.handleReservationResult = function (data) {
            if (data.succeeded) {
                data.isCrossSelling = true;
                $rootScope.$broadcast('event:BasketChanged', data);
                $scope.pageService.pageModel.isLoading = false;
            } else {
                $scope.pageService.pageModel.isLoading = false;
            }
            if (data.errorMessage) {
                $scope.pageService.showModal({
                    message: data.errorMessage
                });
            }
            else if (data.error && data.error.message) {
                $scope.pageService.showModal({
                    message: data.error.message
                });
            }
        };

        $scope.moveToNextStep = function () {
            $scope.pageService.nextStep();
        };

        $scope.moveToPreviousStep = function () {
            $scope.pageService.previousStep();
        };
        
    }

})();;
// Modified:		01.01.2016 ticketportal AG, St. Gallen FBE	: (SCR 9045) Web shop 2: If the amount of cross selling in the drop down is too high (1000+) the cross selling booking step is unresponsive on IE11
(function () {
    'use strict';

    var modelId = 'crossSellingModel';
    angular.module('webShop2App').factory(modelId, ['$sce', model]);

    function model($sce) {
        var model = {
            crossSellingPerformanceGroupOptions: null,
            crossSellingProductGroupOptions: null,
            crossSellingGiftCertificateGroupOption: null,
            hasCrossSelling: false,

            // functions
            updateFromService: updateFromService,
        }
        return model;

        // Updates the model based on the contents from the server
        function updateFromService(data) {
            var MAXITEMAMOUNT = 10;
            var MAXITEMAMOUNTTORENDER = 20;

            if (data.crossSelling && (data.crossSelling.performances || data.crossSelling.products || data.crossSelling.giftCertificates)) {
                this.crossSellingPerformanceGroupOptions = null;
                if (data.crossSelling.performances) {
                    this.crossSellingPerformanceGroupOptions = new Array();
                    for (var optionIndex = 0; optionIndex < data.crossSelling.performances.length; ++optionIndex) {
                        var optionGroup = data.crossSelling.performances[optionIndex];
                        for (var itemIndex = 0; itemIndex < optionGroup.performances.length; ++itemIndex) {
                            var optionItem = optionGroup.performances[itemIndex];
                            optionItem.headlineSafe = $sce.trustAsHtml(optionItem.headline);
                            optionItem.descriptionSafe = $sce.trustAsHtml(optionItem.description);
                            if (optionItem.maxItemAmount == 0)
                                optionItem.maxItemAmount = MAXITEMAMOUNT;
                            else if (optionItem.maxItemAmount > MAXITEMAMOUNTTORENDER)
                                optionItem.maxItemAmount = MAXITEMAMOUNTTORENDER;

                            optionItem.itemOptions = new Array();
                            optionItem.itemOptions.push('');
                            for (var amountIndex = 1; amountIndex <= optionItem.maxItemAmount; ++amountIndex) {
                                optionItem.itemOptions.push(amountIndex);
                            }

                            for (var bookingCategoryIndex = 0; bookingCategoryIndex < optionItem.bookingCategories.length; ++bookingCategoryIndex) {
                                var bookingCategory = optionItem.bookingCategories[bookingCategoryIndex];
                                bookingCategory.amount = '';
                            }
                        }
                        this.crossSellingPerformanceGroupOptions.push(optionGroup);
                    }
                }

                this.crossSellingProductGroupOptions = null;
                if (data.crossSelling.products) {
                    this.crossSellingProductGroupOptions = new Array();
                    for (var optionIndex = 0; optionIndex < data.crossSelling.products.length; ++optionIndex) {
                        var optionGroup = data.crossSelling.products[optionIndex];
                        for (var itemIndex = 0; itemIndex < optionGroup.productDefinitions.length; ++itemIndex) {
                            var optionItem = optionGroup.productDefinitions[itemIndex];
                            optionItem.descriptionSafe = $sce.trustAsHtml(optionItem.description);
                            if (optionItem.maxItemAmount == 0)
                                optionItem.maxItemAmount = MAXITEMAMOUNT;
                            else if (optionItem.maxItemAmount > MAXITEMAMOUNTTORENDER)
                                optionItem.maxItemAmount = MAXITEMAMOUNTTORENDER;

                            optionItem.itemOptions = new Array();
                            optionItem.itemOptions.push('');
                            for (var amountIndex = 1; amountIndex <= optionItem.maxItemAmount; ++amountIndex) {
                                optionItem.itemOptions.push(amountIndex);
                            }

                            // set the select amount of each product to ''
                            for (var productIndex = optionItem.products.length - 1; productIndex >= 0; --productIndex)
                                optionItem.products[productIndex].amount = '';
                        }
                        this.crossSellingProductGroupOptions.push(optionGroup);
                    }
                }

                this.crossSellingGiftCertificateGroupOption = null;
                if (data.crossSelling.giftCertificates) {
                    var optionGroup = data.crossSelling.giftCertificates;
                    for (var itemIndex = 0; itemIndex < optionGroup.giftCertificateDefinitions.length; ++itemIndex) {
                        var optionItem = optionGroup.giftCertificateDefinitions[itemIndex];
                        optionItem.descriptionSafe = $sce.trustAsHtml(optionItem.description);
                        optionItem.amount = 1;
                        if (optionItem.values && optionItem.values.length > 0)
                            optionItem.value = optionItem.values[0];
                        else
                            optionItem.value = '';
                        optionItem.message = '';
                    }
                    this.crossSellingGiftCertificateGroupOption = optionGroup;
                }

                if ((this.crossSellingPerformanceGroupOptions && this.crossSellingPerformanceGroupOptions.length > 0) ||
                    (this.crossSellingProductGroupOptions && this.crossSellingProductGroupOptions.length > 0) ||
                    (this.crossSellingGiftCertificateGroupOption && this.crossSellingGiftCertificateGroupOption.giftCertificateDefinitions.length > 0))
                    this.hasCrossSelling = true;
                else
                    this.hasCrossSelling = false;
            }
        }
    }

})();;
(function () {
    'use strict';

    var controllerId = 'subscriptionController';
    angular.module('webShop2App').controller(controllerId, ['$rootScope', '$scope', '$timeout', 'subscriptionService', 'subscriptionModel', 'pageService', 'commonService', 'homeModel', '$modal', 'selectedSubscriptionPlacesFinderService', 'customizationService', 'webShop2AppSettings', 'ecommerceService', controller]);

    function controller($rootScope, $scope, $timeout, subscriptionService, subscriptionModel, pageService, commonService, homeModel, $modal, selectedSubscriptionPlacesFinderService, customizationService, webShop2AppSettings, ecommerceService) {
        $scope.subscriptionService = subscriptionService;
        $scope.model = subscriptionModel;
        $scope.pageService = pageService;
        $scope.modal = $modal;
        $scope.mainModel = homeModel;
        $scope.selectedPlacesFinderService = selectedSubscriptionPlacesFinderService;
        $scope.customizationService = customizationService;
        $scope.model.selectedSeats = [];
        $scope.model.selectedSeats.expanded = false;
        $scope.webShop2AppSettings = webShop2AppSettings;
        $scope.ecommerceService = ecommerceService;

        $scope.pageService.setStartStep(CURRENT_STEP_SHOP);

        $scope.init = function () {

            if ($scope.model.isReservationAllowed) {
                $scope.customizationService.buildFlattenPriceCategory($scope.model);

                if ($scope.model.selectedVenue.canvasModel && $scope.model.selectedVenue.canvasModel !== undefined && $scope.model.selectedVenue.canvasModel.useCanvas) {
                    $scope.safeBindCanvasSeatingMap($scope.model.selectedVenue.canvasModel);
                }
                else {
                    // Set the current seating map
                    if ($scope.model.selectedVenue.seatingMapImage) {
                        $scope.model.selectedVenue.seatingMapImage.isTopLevel = true;
                        $scope.model.selectedSeatingImage = $scope.model.selectedVenue.seatingMapImage;
                        $scope.bindImageMap();
                    }
                }

                if ($scope.model.selectedVenue.seatingPlanCode) {
                    $scope.subscriptionService.getSeatingPlan($scope.model.tag, $scope.model.selectedVenue.seatingPlanCode).then(function (data) {
                        var seatingPlan = JSON.parse(data.seatingPlan.jsonString);
                        var seatPicker = new SeatPicker({
                            "canvas": document.querySelector("canvas.canvas-seatpicker"),
                            "tooltip": {
                                "container": document.querySelector(".tooltip-seatpicker"),
                                "overrides": {
                                    "content": true,
                                    "display": false,
                                    "hide": false,
                                    "position": true
                                }
                            },
                            "categories": data.categories,
                            "layout": seatingPlan.SeatingPlan.Layout,
                            "colourPalette": {
                                "selectedSeat": data.selectedSeatColor
                            },
                            "canvasControls": {
                                "zoomInButton": document.querySelector(".seatingPlan-zoomIn"),
                                "zoomOutButton": document.querySelector(".seatingPlan-zoomOut"),
                                "zoomResetButton": document.querySelector(".seatingPlan-reset")
                            },
                            "seatStyles": {
                                "selected": {
                                    "icon": "StrokeCircleWithTick"
                                }
                            }
                        });

                        // Set reserved/sold/not available seats
                        seatPicker.categories.forEach(function (category) {
                            if (category.status && category.status !== 1) {
                                // Reserved
                                if (category.status == 2) {
                                    category.seatStyle.icon = seatPicker.enums.SeatIcon.StrokeCircleWithSlash;
                                }
                                // Sold or not available
                                else if (category.status == 3 || category.status == 4) {
                                    category.seatStyle.icon = seatPicker.enums.SeatIcon.StrokeCircleWithCross;
                                }
                                seatPicker.addUpdateCategory(category, false);
                            }
                        });

                        // Add sectionMappings to seatmap controller
                        seatPicker.sectionMappings = data.sections;

                        // Add seat mappings to seatpicker
                        seatPicker.seatMappings = data.seats;

                        // Remove seat function
                        seatPicker.removeSeat = function (seat) {
                            seatPicker.unselectSeat(seat);
                            var index = $scope.model.selectedSeats.indexOf(seat);
                            if (index !== -1) {
                                $scope.model.selectedSeats.splice(index, 1);
                            }
                        }

                        // Zoom to section function
                        seatPicker.goToSection = function (sectionCode, seat) {
                            var blocks = seatPicker.getBlocks(sectionCode);
                            var block = !!seat ? blocks.find(function (b) {
                                return b.rows.find(function (r) {
                                    return r.seats.find(function (s) {
                                        return s.identifiers.includes(seat.identifiers[0])
                                    })
                                })
                            }) : blocks[0];
                            seatPicker.zoomToBlock(block)
                        }

                        // Set selected seats
                        seatPicker.setSelectedSeats = function (selectedSeats) {
                            selectedSeats.forEach(function (selectedSeat) {
                                seatPicker.setAdditionalSeatProperties(selectedSeat);
                                $scope.model.selectedSeats.push(selectedSeat);
                            });
                        }

                        // Set additional seat properties
                        seatPicker.setAdditionalSeatProperties = function (seat) {
                            var seatMapping = $scope.seatMapController.seatMappings.find(function (m) {
                                return m.seatMappingId === seat.identifiers[0]
                            });
                            if (!!seatMapping) {
                                var section = $scope.model.selectedVenue.sections.find(function (s) { return s.code === seatMapping.sectionCode });
                                var priceCategory = $scope.seatMapController.categories.find(function (c) { return c.seatIdentifiers && c.seatIdentifiers.includes(seat.identifiers[0]) });
                                var ticketPrice = section.priceCategories.find(function (c) { return c.uniqueId === seatMapping.priceCategoryId });
                                seat.id = seatMapping.id;
                                seat.sectionCode = seatMapping.sectionCode;
                                seat.sectionName = !!section ? section.name : '';
                                seat.priceCategoryName = !!priceCategory ? priceCategory.name : '';
                                seat.rowIdentifier = seatMapping.rowIdentifier;
                                seat.seatIdentifier = seatMapping.seatIdentifier;
                                seat.price = !!ticketPrice ? ticketPrice.prices[0].formattedPrice : '';
                                seat.status = seatMapping.status;
                                seat.statusName = seatMapping.statusName;
                                seat.color = !!priceCategory ? priceCategory.seatStyle.colour : '';;
                            } else {
                                var section = $scope.seatMapController.sectionMappings.find(function (s) { return s.code === seat.blockLabel });
                                var priceCategory = $scope.seatMapController.categories.find(function (c) { return c.seatIdentifiers && c.seatIdentifiers.includes(seat.identifiers[0]) });
                                seat.sectionCode = seat.blockLabel;
                                seat.sectionName = !!section ? section.name : '';
                                seat.priceCategoryName = !!priceCategory ? priceCategory.name : '';
                                seat.rowIdentifier = seat.rowLabel;
                                seat.seatIdentifier = seat.label;
                                seat.color = !!priceCategory ? priceCategory.seatStyle.colour : '';;
                            }
                        }

                        // Set and display tooltip
                        seatPicker.setTooltip = function (seat) {
                            var html = '';
                            if (!!seat) {
                                seatPicker.setAdditionalSeatProperties(seat);
                                if (seat.color) {
                                    seatPicker.tooltip.container.style.background = seat.color;
                                    seatPicker.tooltip.container.style.color = textColorGenerator(seat.color);
                                } else {
                                    seatPicker.tooltip.container.style.background = '';
                                    seatPicker.tooltip.container.style.color = '';
                                }

                                html += '<div class="tooltip-seat-location">';
                                if (seat.sectionName) {
                                    html += '<span class="tooltip-seat-section">' + seat.sectionName + '</span>';
                                }
                                html += '<span class="tooltip-seat-identifier">' + ticketportal.webshop.localization.seatRow + ": " + seat.rowIdentifier + '</span>';
                                html += '<span class="tooltip-seat-identifier">' + ticketportal.webshop.localization.seatIdentifier + ': ' + seat.seatIdentifier + '</span>';
                                html += '</div>';
                                html += '<div class="tooltip-seat-information">';
                                if (seat.priceCategoryName) {
                                    html += '<span class="tooltip-seat-category">' + seat.priceCategoryName + '</span>';
                                }
                                if (seat.price) {
                                    html += '<span class="tooltip-seat-price">' + seat.price + '</span>';
                                }
                                if (seat.status) {
                                    html += '<span class="tooltip-seat-status">' + seat.statusName + '</span>';
                                }
                                html += '</div>';

                                var seatCoords = seat.layoutProperties.centre;
                                var canvasCoords = seatPicker.translateToCanvasCoordinates(seatCoords);
                                seatPicker.tooltip.container.innerHTML = html;
                                seatPicker.tooltip.x = canvasCoords.x + seatPicker.canvas.offsetLeft + 10;
                                seatPicker.tooltip.y = canvasCoords.y + seatPicker.canvas.offsetTop + 10;
                                seatPicker.displayTooltip();
                            }
                        }

                        // Add events
                        document.addEventListener(seatPicker.events.selectionChanged, function (event) {
                            $scope.model.selectedSeats = [];
                            seatPicker.setSelectedSeats(event.selectedSeats);
                            $scope.$apply();
                        });

                        document.addEventListener(seatPicker.events.click, function (event) {
                            seatPicker.zoomToBlock(event.block);
                        });

                        document.addEventListener(seatPicker.events.mousemove, function (event) {
                            seatPicker.setTooltip(event.seat);
                        });

                        document.addEventListener(seatPicker.events.tap, function (event) {
                            seatPicker.setTooltip(event.seat);
                        });

                        $scope.seatMapController = seatPicker;

                        // Set checked seats
                        $scope.seatMapController.seatMappings.forEach(function (seatMapping) {
                            if (seatMapping.isSeatChecked) {
                                var seat = seatPicker.getSeat(seatMapping.seatMappingId);
                                seatPicker.selectSeat(seat);
                                if (!$scope.model.selectedSeats.some(function (s) { return s.identifiers[0] === seatMapping.seatMappingId })) {
                                    seatPicker.setSelectedSeats([seat]);
                                }
                            }
                        });
                    });
                }

                // Display seating map help
                if (!$scope.model.seatingMapReservationHelpAlreadyDisplayed) {
                    if (getQueryStringValue('step') != 'shop')
                        $scope.showSeatingMapBookingHelp();
                    $scope.model.seatingMapReservationHelpAlreadyDisplayed = true;
                }

                // Show section directly
                if ($scope.model.selectedVenue.showSectionDirectly) {
                    if ($scope.model.selectedVenue.sections.length > 0) {
                        $scope.selectSection($scope.model.selectedVenue.sections[0].code, false);
                    }
                }
            }
        }

        $scope.onSeatSelected = function (seat, selected) {
            if (selected)
                $scope.model.selectedSeats.push(seat);
            else {
                for (var i = 0; i < $scope.model.selectedSeats.length; i++)
                    if ($scope.model.selectedSeats[i].id == seat.id) {
                        $scope.model.selectedSeats.splice(i, 1);
                        break;
                    }
            }
        }

        $scope.safeBindCanvasSeatingMap = function (canvasModel, initialSectionCode) {
            if ($("#canvas")[0].clientWidth > 0) {

                var seatMapControllerOptions = {
                    host: "canvas",
                    scale: "scaleSlider",
                    reset: "reset",
                    moveUp: 'moveUp',
                    moveDown: 'moveDown',
                    moveLeft: 'moveLeft',
                    moveRight: 'moveRight',
                    refresh: 'refresh',
                    performanceModel: $scope.model.selectedPerformance,
                    onSeatSelected: function (seat, selected) {
                        $timeout(function () {
                            $scope.onSeatSelected(seat, selected);
                        }, 0);
                    }
                };

                var seatMapController = new seatingMapController(canvasModel, ticketService, seatMapControllerOptions, function () {
                    $scope.seatMapController = seatMapController;
                    if (initialSectionCode)
                        $scope.seatMapController.goToSection(initialSectionCode);

                    $scope.model.selectedSeats.length = 0;
                    for (var i = 0; i < canvasModel.selectedSeats.length; i++)
                        $scope.model.selectedSeats.push($scope.seatMapController.getTicketInfoFromSeat(canvasModel.selectedSeats[i]));
                });
            } else {
                $timeout(function () { $scope.safeBindCanvasSeatingMap(canvasModel, initialSectionCode); }, 0);
            }
        }

        $scope.getSeats = function (performanceID, areas) {
            return subscriptionService.getSeats($scope.model.id, areas);
        }

        $scope.showSelectedSeats = function () {
            if ($scope.model.selectedSeats.length > 0)
                $scope.model.selectedSeats.expanded = true;
        }

        $scope.hideSelectedSeats = function () {
            $scope.model.selectedSeats.expanded = false;
        }

        $scope.createSectionTooltip = function (sectionCode) {
            var sectionData = $.grep($scope.model.selectedVenue.sections, function (e) { return e.code == sectionCode; })[0];
            if (!sectionData) return null;
            var content = '<h1>' + (sectionData.name || sectionData.code) + '</h1>';

            if (sectionData.isSoldOut) {
                content += '<p class="section-tooltip no-tickets-available section-"' + sectionData.code + ' >' + ticketportal.webshop.localization.noTicketsAvailableInThisSector + "</p>";
                return content;
            }

            if (sectionData.priceCategories.length > 0) {
                for (var i = 0; i < sectionData.priceCategories.length; i++) {
                    var categoryAvailability = sectionData.priceCategories[i];
                    content += '<p><strong>' + categoryAvailability.name + '</strong>';
                    if (categoryAvailability.ticketAvailabilityText.length > 0)
                        content += ': ' + categoryAvailability.ticketAvailabilityText;
                    content += '</p>';
                }
            }

            return content;
        }

        // Change the current currency
        $scope.changeCurrency = function (currency) {
            $scope.pageService.pageModel.isLoading = true;
            $scope.subscriptionService.setBasketCurrency(currency.symbol).then(function (data) {
                $rootScope.$broadcast('event:BasketChanged', data);

                // update selected performance model with new currency
                if (data.allowedCurrencies)
                    $scope.model.updateAvailableCurrencies(data.allowedCurrencies);

                $scope.pageService.pageModel.isLoading = false;
            });
        }

        $scope.bindImageMap = function () {
            // TODO: create a directive out of it
            var mapSeatingMapName = $scope.model.isSeatingMapOpen ? '#mapSeatingMapModal' : '#mapSeatingMap';
            var imgSeatingMapName = $scope.model.isSeatingMapOpen ? '#imgSeatingMapModal' : '#imgSeatingMap';

            $(mapSeatingMapName).children().remove();
            for (var i = 0; i < $scope.model.selectedSeatingImage.areas.length; ++i) {
                var area = $scope.model.selectedSeatingImage.areas[i];
                var name = area.isImageUrl ? 'perspective-' + i : area.key;
                var imageUrl = area.isImageUrl ? area.key : '';
                var imageName = area.imageName ? area.imageName : '';
                $(mapSeatingMapName).append('<area shape="' + area.shape + '" coords="' + area.coordinates + '" name="' + name + '" imageUrl="' + imageUrl + '" imageName="' + imageName + '" href="#" />');
            }

            var areas = new Array();
            var htmlAreas = $(mapSeatingMapName + ' > area');
            for (var i = htmlAreas.length - 1; i >= 0; i--) {
                var name = $(htmlAreas[i]).attr('name');
                var toolTipContent = '';
                if (name.indexOf('perspective-') >= 0) {
                    var imageUrl = $(htmlAreas[i]).attr('imageUrl');
                    toolTipContent = '<img class="image" src="' + imageUrl + '"></img>';
                }
                else {
                    toolTipContent = '<div class="mapster-tooltip" style="border: 2px solid #CB076D; background: #FFF; width:160px; padding:4px; margin: 4px; -moz-box-shadow: 3px 3px 5px #535353; ' +
                        '-webkit-box-shadow: 3px 3px 5px #535353; box-shadow: 3px 3px 5px #535353; -moz-border-radius: 6px 6px 6px 6px; -webkit-border-radius: 6px; ' +
                        'border-radius: 6px 6px 6px 6px; color: #555;">' + $scope.createSectionTooltip(name) + '</div>';
                }

                if (toolTipContent && toolTipContent.length > 0)
                    areas.push({ key: name, toolTip: toolTipContent });
            }

            var $img = $(imgSeatingMapName);
            $img.attr('src', $scope.model.selectedSeatingImage.imageUrl);

            // customize the seating map based on the webShop2AppSettings.customization
            var fillOpacity = 0.4;
            var fillColor = 'd42e16';
            var strokeColor = '3320FF';
            var strokeOpacity = 0.8;
            var strokeWidth = 4;
            var displayStroke = true;

            if ($scope.webShop2AppSettings.customization !== undefined && $scope.webShop2AppSettings.customization.seatingMap !== undefined) {
                var seatingMapCustomization = $scope.webShop2AppSettings.customization.seatingMap;

                if (seatingMapCustomization.imageMapFillColor != undefined)
                    fillColor = seatingMapCustomization.imageMapFillColor;
                if (seatingMapCustomization.imageMapFillOpacity != undefined)
                    fillOpacity = seatingMapCustomization.imageMapFillOpacity;
                if (seatingMapCustomization.imageMapStrokeColor != undefined)
                    strokeColor = seatingMapCustomization.imageMapStrokeColor;
                if (seatingMapCustomization.imageMapStrokeOpacity != undefined)
                    strokeOpacity = seatingMapCustomization.imageMapStrokeOpacity;
                if (seatingMapCustomization.imageMapStrokeWidth != undefined)
                    strokeWidth = seatingMapCustomization.imageMapStrokeWidth;
                if (seatingMapCustomization.imageMapDisplayStroke != undefined)
                    displayStroke = seatingMapCustomization.imageMapDisplayStroke;
            }

            if (typeof $img.mapster === 'function') {
                $img.mapster({
                    fillOpacity: fillOpacity,
                    fillColor: fillColor,
                    stroke: displayStroke,
                    strokeColor: strokeColor,
                    strokeOpacity: strokeOpacity,
                    strokeWidth: strokeWidth,
                    singleSelect: true,
                    mapKey: 'name',
                    listKey: 'name',
                    onClick: $scope.selectImageArea,
                    toolTipContainer: '<div class="seating-map-tooltip"></div>',
                    toolTipClose: ["tooltip-click", "area-click", "img-mouseout", "area-mouseout"],
                    areas: areas,
                    showToolTip: true,
                    onShowToolTip: function (e) {
                        var imagePos = $(imgSeatingMapName).offset();
                        var left = ((imagePos.left - e.toolTip.width()) - 40);
                        left = Math.max(0, left) + 'px';
                        var top = (imagePos.top + ($(imgSeatingMapName).height() / 2)) - (e.toolTip.height() / 2);
                        top = Math.max(0, top) + 'px';
                        e.toolTip.css({ left: left, top: top });
                    }
                });
            }

            if ($scope.model.isSeatingMapOpen)
                $scope.setSeatingMap(false);
        }

        $scope.showTab = function (tab) {
            if (tab == 'reservationinfo') {
                $scope.model.isDisplayingReservationInfoTab = true;
                $scope.model.isDisplayingPriceInfoTab = false;
            } else if (tab == 'priceinfo') {
                $scope.model.isDisplayingReservationInfoTab = false;
                $scope.model.isDisplayingPriceInfoTab = true;
            }

            // in iFrame post a message notifying which tab is active
            $scope.pageService.postMessageToIFrameParent("viewSubscriptionTab-" + tab);
        }

        // Reloads the current subscription
        $scope.reloadCurrentSubscription = function () {
            if ($scope.model.selectedVenue) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.subscriptionService.getSubscription($scope.model.tag).then(function (data) {
                    if (data.succeeded != false) {
                        $scope.model.init(data, null, homeModel);
                        $scope.init();
                        $scope.pageService.pageModel.isLoading = false;
                    }
                    else {
                        $scope.pageService.pageModel.isLoading = false;
                        $rootScope.$broadcast('event:ShowAlert', { title: ticketportal.webshop.localization.errorTitle, message: data.errorMessage });
                    }
                });
            }
        }

        // Called by the image map
        $scope.selectImageArea = function (area) {
            if (area.key) {
                if (area.key.indexOf('perspective-') >= 0) {
                    var imageUrl = area.e.currentTarget.getAttribute('imageUrl');
                    var imageName = area.e.currentTarget.getAttribute('imageName');
                    var img = $('<img />', { src: imageUrl, 'class': 'image' });
                    $('#showImageModal .modal-title').html(imageName);
                    $('#showImageModal .modal-body').html(img);
                    $('#showImageModal').modal('show');
                }
                else
                    $scope.selectSection(area.key, true);
            }
        }

        // Shows membership login modal
        $scope.showMembershipLogin = function () {
            $rootScope.$broadcast('patron:displayMembershipLogin');
        }

        $rootScope.$on('event:ReloadCurrentData', function (event, args) {
            $scope.reloadCurrentSubscription();
        });

        $('#seatingMapModal').on('hidden.bs.modal', function () {
            $scope.backToMainSeatingMapView(true);
            $scope.model.isSeatingMapOpen = false;
        })

        $('.modal').on('hidden.bs.modal', function () {
            var area = $('area');
            if (area.length > 0)
                $('area').mapster('deselect');
        })

        // Show seating map booking help
        $scope.showSeatingMapBookingHelp = function () {
            if ($scope.pageService.showSeatingMapHelp())
                $('#seating-map-help').modal('show');
        }

        $scope.buildBestPlaceSelectionModel = function (isAsync) {
            var itemOptions = new Array();
            for (var itemAmount = 0; itemAmount <= $scope.model.basketMaximumItemCount; ++itemAmount) {
                itemOptions.push(itemAmount);
            }

            $scope.model.bestPlaceSelection.priceCategories.length = 0;

            if ($scope.model.selectedSection) {
                $scope.model.bestPlaceSelection.sectionId = $scope.model.selectedSection.id;
                $scope.model.bestPlaceSelection.sectionName = $scope.model.selectedSection.name;
                $scope.model.bestPlaceSelection.sectionCode = $scope.model.selectedSection.code;

                var priceCategoryList = new Array();
                for (var priceCategoryIndex = 0; priceCategoryIndex < $scope.model.selectedSection.priceCategories.length; priceCategoryIndex++) {
                    var priceCategory = $scope.model.selectedSection.priceCategories[priceCategoryIndex];
                    priceCategoryList.push(new $scope.model.bestPlaceSelectionCategoryModel(priceCategory.uniqueId, priceCategory.name, priceCategory.fullPriceName, priceCategory.colorCode, priceCategory.priority, priceCategory.isSoldOut, priceCategory.prices, $scope.model.currencySymbol, itemOptions, priceCategory.ticketAvailabilityText));
                }
                $scope.model.bestPlaceSelection.setPriceCategories(priceCategoryList);

            } else {
                $scope.model.bestPlaceSelection.sectionId = ''
                $scope.model.bestPlaceSelection.sectionName = '';
                $scope.model.bestPlaceSelection.sectionCode = '';

                // best place among all sections
                var priceCategoryList = new Array();
                for (var priceCategoryIndex = 0; priceCategoryIndex < $scope.model.priceCategories.length; priceCategoryIndex++) {
                    var priceCategory = $scope.model.priceCategories[priceCategoryIndex];
                    if (!priceCategory.isSoldOut) {
                        priceCategoryList.push(new $scope.model.bestPlaceSelectionCategoryModel(priceCategory.uniqueId, priceCategory.name, priceCategory.fullPriceName, priceCategory.priority, priceCategory.isSoldOut, priceCategory.prices, $scope.model.currencySymbol, itemOptions, priceCategory.ticketAvailabilityText));
                    }
                }
                $scope.model.bestPlaceSelection.setPriceCategories(priceCategoryList);
            }

            if (isAsync) {
                $scope.$apply(function () {
                    $scope.model.bestPlaceSelection.isValid = true;
                });
            } else {
                $scope.model.bestPlaceSelection.isValid = true;
            }
        }

        $scope.backToMainSeatingMapView = function (isModalClose) {

            // 1. reset any best place being displayed
            $scope.model.bestPlaceSelection.reset();

            // 2. reset current section
            $scope.model.selectedSection = null;
            $scope.model.displayBackToMainViewButton = false;

            // 3. update the current displayed seating map
            if (!isModalClose) {
                $scope.model.selectedSeatingImage = $scope.model.selectedVenue.seatingMapImage;
                $scope.bindImageMap();
            }

            if ($scope.model.isSeatingMapOpen) {
                $scope.setSeatingMap(true);
                $scope.model.sectionNavigationItems.length = 0;
                $('#imgSeatingMapRenderModal').show();
            }
        }

        // Display the current section
        $scope.selectSection = function (sectionCode, isAsync) {
            var selectedSection = $.grep($scope.model.selectedVenue.sections, function (e) { return e.code == sectionCode; })[0];
            if (selectedSection) {
                // check if it has no place
                if (!selectedSection.isGroupingSection && selectedSection.isSoldOut) {
                    $rootScope.$broadcast('event:ShowAlert', { title: selectedSection.name, message: ticketportal.webshop.localization.noTicketsAvailableInThisSector });
                    return;
                }

                $scope.model.selectedSection = selectedSection;
                $scope.model.selectedSeatingImage = selectedSection.seatingMapImage;

                if (selectedSection.isGroupingSection) {
                    $scope.model.bestPlaceSelection.reset();
                    if (!$scope.pageService.isMobile()) {
                        $scope.model.isSeatingMapOpen = true;
                        $('#seatingMapModal').modal({ keyboard: true });
                        $('#imgSeatingMapRenderModal').show();
                    }
                    $scope.bindImageMap();
                    $scope.setSeatingMap(true);

                    $scope.pageService.postMessageToIFrameParent("subscription-showing-section-" + sectionCode);

                } else if (selectedSection.isBestPlace) {
                    $scope.buildBestPlaceSelectionModel(isAsync);
                    if ($scope.model.isSeatingMapOpen) {
                        $('#seatingMapRenderModal').html("");
                        $('#seatingMapLegendModal').html("");
                        $('#seatingMapBestPlaceRenderModal').html($('#seatingMapBestPlace').html());
                        $('#imgSeatingMapRenderModal').hide();
                    }

                    $scope.pageService.postMessageToIFrameParent("subscription-showing-section-" + sectionCode);

                } else {
                    $scope.model.bestPlaceSelection.reset();
                    $scope.pageService.pageModel.isLoading = true;
                    // display seating map booking process
                    $scope.subscriptionService.getSubscriptionSeatingMapHtml($scope.model.tag, $scope.model.selectedSection.code, $scope.model.selectedVenue.id).then(function (data) {
                        if (data.succeeded != false) {
                            if ($scope.model.selectedVenue.showSectionDirectly) {
                                var elements = $(data);
                                var seatMap = $('#seatingMapRenderModal', elements);
                                var legend = jQuery.grep(elements, function (element) { return element.id == 'seatingMapLegendModal'; });
                                $('#seatingMapRender').html(seatMap);
                                $('#seatingMapLegend').html(legend);
                                $scope.pageService.pageModel.isLoading = false;
                                $scope.bindImageMap();
                            }
                            else {
                                $scope.model.isSeatingMapOpen = true;
                                // TODO: use directives to display the seating map modal                    
                                $('#seatingMapModal .modal-body-seatingMapRender').html(data);
                                $scope.pageService.pageModel.isLoading = false;
                                $('#seatingMapModal').modal({ keyboard: true });
                                $('#imgSeatingMapRenderModal').hide();
                                $scope.toggleAddPlaceToBasketButton()
                                $scope.buildSectionNavigation();
                            }
                        }
                        else {
                            $scope.pageService.pageModel.isLoading = false;
                            $rootScope.$broadcast('event:ShowAlert', { title: ticketportal.webshop.localization.errorTitle, message: data.errorMessage });
                        }
                    });
                }

                // Display back button: if is not 'show section directly' and current section has a parent sections or is a grouping section
                var selectSectionHasParent = selectedSection.parentSectionCode != null && selectedSection.parentSectionCode.length > 0;
                var newDisplayBackToMainViewButtonValue = !$scope.model.selectedVenue.showSectionDirectly && (selectedSection.isGroupingSection || selectSectionHasParent);

                if (isAsync) {
                    $scope.$apply(function () {
                        if ($scope.model.isSeatingMapOpen) {
                            $scope.buildSectionNavigation();
                            $scope.toggleAddPlaceToBasketButton();
                        }
                        else
                            $scope.model.displayBackToMainViewButton = newDisplayBackToMainViewButtonValue;
                    });
                } else {
                    if ($scope.model.isSeatingMapOpen) {
                        $scope.buildSectionNavigation();
                        $scope.toggleAddPlaceToBasketButton();
                    }
                    else
                        $scope.model.displayBackToMainViewButton = newDisplayBackToMainViewButtonValue;
                }
            }
        }

        $scope.buildSectionNavigation = function () {
            $scope.model.sectionNavigationItems.length = 0;
            if ($scope.model.selectedSection.parentSectionCode != null && $scope.model.selectedSection.parentSectionCode.length > 0) {
                var sectionCode = $scope.model.selectedSection.parentSectionCode;
                var sectionNavigation = null;
                while (sectionCode != null && sectionCode.length > 0) {
                    var selectedSection = $.grep($scope.model.selectedVenue.sections, function (e) { return e.code == sectionCode; })[0];
                    $scope.model.sectionNavigationItems.push({
                        code: selectedSection.code,
                        name: selectedSection.name,
                        imageUrl: selectedSection.isGroupingSection ? selectedSection.seatingMapImage.imageUrl : '',
                        active: false
                    });
                    sectionCode = selectedSection.parentSectionCode;
                }
            }
            $scope.model.sectionNavigationItems.push({
                code: $scope.model.selectedSection.code,
                name: $scope.model.selectedSection.name,
                imageUrl: $scope.model.selectedSection.isGroupingSection ? $scope.model.selectedSeatingImage.imageUrl : '',
                active: true
            });
        }

        $scope.closeSeatingMap = function () {
            $('#seatingMapModal').modal('hide');
        }

        $scope.toggleAddPlaceToBasketButton = function () {
            if ($scope.model.selectedSection.isBestPlace) {
                $('#seatingMapModal .add-places-to-basket').hide();
                $('#seatingMapModal .add-bestplaces-to-basket').show();
            }
            else {
                $('#seatingMapModal .add-places-to-basket').show();
                $('#seatingMapModal .add-bestplaces-to-basket').hide();
            }
        }

        $scope.setSeatingMap = function (reset) {
            if (reset) {
                $('#seatingMapRenderModal').html("");
                $('#seatingMapBestPlaceRenderModal').html("");
                $('#seatingMapLegendModal').html("");
            }
        }

        $scope.addModalBestPlacesToBasket = function () {
            for (var i = 0; i < $scope.model.bestPlaceSelection.priceCategories.length; i++) {
                var priceCategory = $scope.model.bestPlaceSelection.priceCategories[i];
                var priceCategoryElement = $('#pricecategory-' + priceCategory.id);
                priceCategory.ticketAmount = $('#pricecategory-' + priceCategory.id).val();
            }

            $scope.addBestPlacesToBasket();
        }

        $scope.addBestPlacesToBasket = function () {
            var ticketsToReserve = new Array();

            if ($scope.model.bestPlaceSelection.isValid) {
                for (var i = 0; i < $scope.model.bestPlaceSelection.priceCategories.length; i++) {
                    var priceCategory = $scope.model.bestPlaceSelection.priceCategories[i];
                    if (priceCategory.ticketAmount != '') {
                        ticketsToReserve.push({ ticketId: 'bsp_sec-' + priceCategory.id + '-' + $scope.model.bestPlaceSelection.sectionCode, amount: priceCategory.ticketAmount });
                    }
                }
            }

            if (ticketsToReserve.length > 0) {
                $scope.pageService.pageModel.isLoading = true;

                $scope.subscriptionService.addBestPlacesToBasket($scope.model.tag, ticketsToReserve, $scope.model.selectedVenue.id, $scope.model.bestPlaceSelection.sectionCode).then(function (data) {
                    $scope.handleReservationResult(data);
                });
            }
        }

        $scope.handleReservationResult = function (data) {
            if (data.succeeded) {
                $rootScope.$broadcast('event:BasketChanged', data);
                $scope.pageService.pageModel.isLoading = false;
                if ($scope.model.isSeatingMapOpen) {
                    $scope.closeSeatingMap();
                }

                // go to the basket
                $scope.pageService.setStep(CURRENT_STEP_BASKET);
            } else {
                $scope.pageService.pageModel.isLoading = false;
                $scope.pageService.pageModel.isWorking = false;
                var title = null;
                if ($scope.model.selectedSection)
                    title = $scope.model.selectedSection.name;
                $rootScope.$broadcast('event:ShowAlert', { title: title, message: data.errorMessage });
            }
        }

        $scope.addPlacesToBasket = function () {
            var placesToReserve = $scope.selectedPlacesFinderService.find();

            if (placesToReserve.length > 0) {
                $scope.pageService.pageModel.isLoading = true;
                $scope.pageService.pageModel.isWorking = true;

                var sectionCode = null;
                if ($scope.model.selectedSection && $scope.model.selectedSection.code)
                    sectionCode = $scope.model.selectedSection.code;

                $scope.subscriptionService.addPlacesToBasket($scope.model.tag, placesToReserve, $scope.model.selectedVenue.id, sectionCode).then(function (data) {
                    $scope.ecommerceService.addSubscriptionToCart($scope.model, placesToReserve)
                    $scope.handleReservationResult(data);
                });
            }
        }

        $scope.addPlacesToBasketForSeatMap2 = function () {
            var selectedPlaces = $scope.model.selectedSeats;

            if (selectedPlaces.length > 0) {
                var seats = [];
                for (var i = 0; i < selectedPlaces.length; i++)
                    seats.push({
                        SubscriptionSeatID: selectedPlaces[i].id,
                        RowIdentifier: selectedPlaces[i].rowIdentifier,
                        SeatIdentifier: selectedPlaces[i].seatIdentifier,
                        SectionCode: selectedPlaces[i].sectionCode
                    });
                $scope.subscriptionService.addPlacesToBasketForSeatMap2($scope.model.id, $scope.model.selectedVenue.id, seats).then(function (data) {
                    $scope.handleReservationResult(data);
                });
            }
        }

        // handle 'display subscription if item was clicked in basket'
        $scope.$watch('pageService.pageModel.currentStep', function () {
            if ($scope.pageService.getStep() == CURRENT_STEP_SHOP) {
                if (getQueryStringValue('step') == 'shop') {
                    if (!$scope.model.handledBasketItemClick) {
                        $scope.model.handledBasketItemClick = true;
                        if ($scope.mainModel.basketItemCount > 0) {
                            var section = getQueryStringValue('section');
                            if (section && section.length > 0) {
                                $scope.selectSection(section);
                            }
                        }
                    }
                }
            }
        });

        $scope.$on('event:BasketChanged', function (event, args) {
            if (args.basket && (args.amountOfItemsRemoved || 0) > 0) {
                var seats = args.basket.itemGroups.map(function (g) {
                    return jQuery.map(g.items, function (item) {
                        return { seatIdentifier: item.seatIdentifier, rowIdentifier: item.rowIdentifier, sectionCode: item.sectionCode }
                    })
                }).flat();
                if (seats.length > 0) {
                    var seatsToRemove = $.grep($scope.model.selectedSeats, function (selectedSeat) {
                        var found = $.grep(seats, function (seat) {
                            return seat.seatIdentifier === selectedSeat.seatIdentifier && seat.rowIdentifier === selectedSeat.rowIdentifier && seat.sectionCode === selectedSeat.sectionCode;
                        });

                        if (found.length === 0)
                            return selectedSeat;
                    }); 
                    seatsToRemove.forEach(function (seat) {
                        if ($scope.model.canvasModel && $scope.model.canvasModel.useCanvas) {
                            $scope.seatMapController.removeSeat(seat.id);
                        } else {
                            $scope.seatMapController.removeSeat(seat);
                        }
                    });
                } else {
                    $scope.model.selectedSeats.forEach(function (seat) {
                        if ($scope.model.canvasModel && $scope.model.canvasModel.useCanvas) {
                            $scope.seatMapController.removeSeat(seat.id);
                        } else {
                            $scope.seatMapController.unselectSeat(seat);
                        }
                    });

                    $scope.model.selectedSeats = [];
                }
            }
        });

        $scope.canvasWidthInit = function () {
            var canvasSeatPicker = document.querySelector(".canvas-seatpicker");
            var canvasWidth = canvasSeatPicker.parentElement.offsetWidth;
            canvasSeatPicker.width = canvasWidth;
        };
    }
})();;
(function () {
    'use strict';

    var modelId = 'subscriptionModel';
    angular.module('webShop2App').factory(modelId, ['$sce', 'homeModel', model]);

    function model($sce, homeModel) {
        var model = {
            selectedVenue: null,
            selectedSection: null,
            selectedSeatingImage: null,
            isSeatingMapOpen: false,
            showPriceCategoryDiscounts: false,
            seatingMapReservationHelpAlreadyDisplayed: false,
            displayBackToMainViewButton: false,
            bestPlaceSelection: new bestPlaceSelectionModel(),
            handledBasketItemClick: false,
            allowedCurrencies: new Array(),
            sectionNavigationItems: [],

            // functions
            init: init,
            updateAvailableCurrencies: updateAvailableCurrencies,
            getPrice: getPrice,

            // child models
            bestPlaceSelectionModel: bestPlaceSelectionModel,
            bestPlaceSelectionCategoryModel: bestPlaceSelectionCategoryModel
        };
        model.init(__startupModel, $sce, homeModel);
        return model;

        function init(source, $sce, homeModel) {
            if (source && source != undefined) {
                angular.extend(this, source);
                if (this.venues.length > 0) {
                    this.selectedVenue = this.venues[0];
                    if (this.selectedVenue.locationName.localeCompare(this.selectedVenue.venueName, undefined, { sensitivity: 'accent' }) !== 0)
                        this.selectedVenue.venueAndLocation = this.selectedVenue.locationName + ', ' + this.selectedVenue.venueName;
                    else
                        this.selectedVenue.venueAndLocation = this.selectedVenue.venueName;

                    if ($sce)
                        this.selectedVenue.reservationInformationSafe = $sce.trustAsHtml(this.selectedVenue.reservationInformation);
                }
                if ($sce)
                    this.reservationNotAllowedMessage = $sce.trustAsHtml(this.reservationNotAllowedMessage);

                if (this.selectedVenue.reservationInformation && this.selectedVenue.reservationInformation.length > 0)
                    this.isDisplayingReservationInfoTab = true;
                else
                    this.isDisplayingPriceInfoTab = true;

                // find available currencies
                this.allowedCurrencies.length = 0;
                var mainCurrencies = homeModel.allowedCurrencies;
                if (mainCurrencies && mainCurrencies && mainCurrencies.length > 0) {
                    for (var i = mainCurrencies.length - 1; i >= 0; --i) {
                        var basketAllowedCurrency = mainCurrencies[i];
                        if (this.hasPriceForCurrency(basketAllowedCurrency.symbol)) {
                            this.allowedCurrencies.push(basketAllowedCurrency);
                        }
                    }
                }
            }
        }

        function updateAvailableCurrencies(allowedCurrencies) {
            if (this.selectedPerformance) {
                // 1. update the currency code and the array of allowed currencies
                for (var currencyIndex = 0; currencyIndex < allowedCurrencies.length; ++currencyIndex) {
                    if (allowedCurrencies[currencyIndex].selected) {

                        // update the current selected currency symbol
                        this.currencySymbol = allowedCurrencies[currencyIndex].text;

                        // update the list of available currencies in the subscription definition model
                        for (var j = this.allowedCurrencies.length - 1; j >= 0; --j)
                            this.allowedCurrencies[j].isSelected = this.allowedCurrencies[j].symbol == allowedCurrencies[currencyIndex].text;

                        break;
                    }
                }

                // 2. if there is best place selection active, update with the new currency
                if (this.bestPlaceSelection && this.bestPlaceSelection.isValid) {
                    for (var i = 0; i < this.bestPlaceSelection.priceCategories.length; i++) {
                        this.bestPlaceSelection.priceCategories[i].updateCurrency(this.currencySymbol);
                    }
                }
            }
        }

        function getPrice(priceCategoryID, currencyCode) {
            for (var i = 0; i < this.priceCategories.length; i++) {
                var priceCategory = this.priceCategories[i];
                if (priceCategory.uniqueId == priceCategoryID) {
                    for (var p = 0; p < priceCategory.prices.length; p++) {
                        if (priceCategory.prices[p].currencySymbol == currencyCode) {
                            return priceCategory.prices[p];
                        }
                    }
                    break;
                }
            }
            return '';
        }
    }

})();;
(function () {
    'use strict';

    var serviceId = 'subscriptionService';
    angular.module('webShop2App').factory(serviceId, ['$http', '$sce', 'webShop2AppSettings', service]);

    function service($http, $sce, webShop2AppSettings) {
        return {
            getSubscription: function (tag) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/subscription/' + tag + '?json=1');
                return $http.post(postUrl).then(function (response) {
                    response.data.reservationInformationSafe = $sce.trustAsHtml(response.data.reservationInformation);
                    response.data.reservationNotAllowedMessage = $sce.trustAsHtml(response.data.reservationNotAllowedMessage);
                    return response.data;
                });
            },

            // gets the html with the seating map
            getSubscriptionSeatingMapHtml: function (tag, sectionCode, venueId) {
                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/subscription/' + tag + '/seatingmap?venue=' + venueId + '&section=' + encodeURIComponent(sectionCode));
                return $http.post(postUrl).then(function (response) {
                    return response.data;
                });
            },

            // gets the seats for the region
            getSeats: function (subscriptionDefinitionID, areas) {
                var data = {
                    subscriptionDefinitionID: subscriptionDefinitionID,
                    areas: []
                };

                for (var i = 0; i < areas.length; i++)
                    data.areas.push({
                        X1: areas[i].x1,
                        Y1: areas[i].y1,
                        X2: areas[i].x2,
                        Y2: areas[i].y2
                    });

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/subscription/' + subscriptionDefinitionID + '/seats');
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: data,
                }).then(function (response) {
                    return { data: response.data, areas: areas };
                });
            },

            // adds seating places to the basket
            addPlacesToBasket: function (tag, places, venueId, sectionCode) {
                var data = {
                    json: 1,
                    tag: tag,
                    venue: venueId,
                    section: sectionCode,
                    places: places.join(',')
                };

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/subscription/' + tag + '/reservation');
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // adds best places to the basket
            addBestPlacesToBasket: function (tag, tickets, venueId, sectionCode) {
                var data = {
                    json: 1,
                    tag: tag,
                    venue: venueId,
                    section: sectionCode
                };
                for (var i = 0; i < tickets.length; i++) {
                    data[tickets[i].ticketId] = tickets[i].amount;
                }

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/subscription/' + tag + '/reservation');
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            // adds best places to the basket
            addPlacesToBasketForSeatMap2: function (subscriptionDefinitionID, venueId, seats) {
                var data = {
                    subscriptionDefinitionID: subscriptionDefinitionID,
                    venueID: venueId,
                    seats: seats
                };

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/subscription/sub-' + subscriptionDefinitionID + '/addtobasket');
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: data
                }).then(function (response) {
                    return response.data;
                });
            },

            // Sets the basket currency
            setBasketCurrency: function (currencySymbol) {
                var data = {
                    json: 1,
                    basketAction: 'changeCurrency',
                    basketActionParameter: currencySymbol
                };

                var postUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/cart/');
                return $http({
                    method: 'POST',
                    url: postUrl,
                    data: $.param(data),  // pass in data as strings
                    headers: { 'Content-Type': 'application/x-www-form-urlencoded' }  // set the headers so angular passing info as form data (not request payload)
                }).then(function (response) {
                    return response.data;
                });
            },

            getSeatingPlan: function (tag, seatingPlanId) {
                var getSeatMapUrl = ticketportal.webshop.fixUrl(webShop2AppSettings.folderName, '/subscription/' + tag + '/seatingPlan');
                return $http({ method: 'GET', url: getSeatMapUrl + "?seatingPlanId=" + seatingPlanId, withCredentials: true }).then(function (response) {
                    return response.data;
                });
            }
        }
    }

    var selectedPlacesServiceId = 'selectedSubscriptionPlacesFinderService';
    angular.module('webShop2App').factory(selectedPlacesServiceId, selectedSubscriptionPlacesFinderService);
    function selectedSubscriptionPlacesFinderService() {
        return {
            find: function () {
                var selectedPlaces = [];
                var placeList = $('#placeList');
                if (placeList.length == 1) {
                    selectedPlaces = placeList.val().split(',');
                    if (selectedPlaces.length > 1) {
                        for (var i = selectedPlaces.length - 1; i >= 0 ; --i) {
                            if (selectedPlaces[i].length == 0)
                                selectedPlaces.splice(i, 1);
                            else {
                                var seat = $('#' + selectedPlaces[i]);
                                selectedPlaces[i] = selectedPlaces[i] + '-' + seat.attr('rid') + '-' + seat.attr('sid');
                            };
                        }
                    }
                }
                else if (typeof rasterSeatingMapRender !== "undefined" && rasterSeatingMapRender !== null) {
                    // raster
                    selectedPlaces = rasterSeatingMapRender.getSelectedPlacesId();
                }

                return selectedPlaces;
            }
        };
    }

})();;
(function () {
    'use strict';

    var serviceId = 'ecommerceService';
    angular.module('webShop2App').factory(serviceId, ['$rootScope', '$window', 'homeService', service]);

    function service($rootScope, $window, homeService) {
        $window.dataLayer = $window.dataLayer || [];
        return {
            //call: HomeController line 128
            pushImpressions: function () {
                if (ticketportal.webshop.bookingObjectName) { //check if undefined
                    var bookingObjectId = ticketportal.webshop.bookingObjectId;
                    $window.dataLayer.push({
                        'event': 'impression',
                        'ecommerce': {
                            'currencyCode': 'CHF',
                            'impressions': [
                                this.createAttributes(
                                    ticketportal.webshop.bookingObjectName,
                                    bookingObjectId,
                                    undefined, // price
                                    ticketportal.webshop.bookingObjectPromoterId,
                                    ticketportal.webshop.bookingObjectType,
                                    undefined, // variant
                                    undefined, // quantity
                                    undefined, //  date 
                                    undefined, //  time 
                                    undefined, //  coupon 
                                    undefined, //  artist 
                                    undefined, //  genre 
                                    undefined, //  location 
                                    undefined, //  venue 
                                    undefined, // flyer 
                                    this.getLoggedUserLogin(), //  userLoggin 
                                    undefined //  productList 
                                )
                            ]
                        }
                    });
                }
            },
            //call: AdmissionTicketController 
            pushAdmissionTicketToCart: function (homeModel, admissionTicketModel) {
                for (var i = 0; i < admissionTicketModel.availablePrices.length; i++) {
                    var parsedDate = this.formatDate(admissionTicketModel.selectedDate);
                    let performanceTagId = this.parseIdFromPerformanceTag(admissionTicketModel.tag); //parsing id from model.tag (test-xxx-90)
                    if (admissionTicketModel.availablePrices[i].ticketAmount > 0) {
                        if (admissionTicketModel.availablePrices[i].discountId === 0) {
                            admissionTicketModel.availablePrices[i].discountName = admissionTicketModel.availablePrices[i].fullPriceName; // If an adult ticket is chosen, there is no discount name, so one would have to read the fullPriceName.
                        }

                        var timeDefinitionId = admissionTicketModel.availablePrices[i].timeDefinitionId;
                        var timeDefinition = this.genericFind(admissionTicketModel.availableTimeDefinitions, function (def) { return def.id === timeDefinitionId; });
                        var startTime = timeDefinition ? timeDefinition.startTime : "";


                        $window.dataLayer.push({
                            'event': 'addToCart',
                            'ecommerce': {
                                'currencyCode': homeModel.selectedCurrency && homeModel.selectedCurrency.symbol ? homeModel.selectedCurrency.symbol : '',
                                'add': {
                                    'products': [this.createAttributes( //name, id, price, brand, category, variant, quantity, date, time, coupon, artist, genre, location, venue, flyer) {
                                        ticketportal.webshop.bookingObjectName,
                                        performanceTagId,
                                        admissionTicketModel.availablePrices[i].formattedPrice,
                                        ticketportal.webshop.bookingObjectPromoterId,
                                        "Ticket",
                                        admissionTicketModel.availablePrices[i].discountName,
                                        admissionTicketModel.availablePrices[i].ticketAmount,
                                        parsedDate,
                                        startTime,
                                        undefined,
                                        undefined, //artist
                                        undefined, // genre
                                        window.__admissionTicketModel.locationID,
                                        window.__admissionTicketModel.venueID,
                                        ticketportal.webshop.bookingObjectId,
                                        this.getLoggedUserLogin(),
                                        undefined // product_list
                                    )]
                                }
                            }
                        });
                    }
                }
            },

            /*###################### Method for Perfomances ######################*/
            //call: TicketController line: 720
            pushAddToCartPerformance: function (homeModel, ticketModel, selectedPlaces) {


                let location = ticketModel.selectedPerformance.locationId;
                let venue = ticketModel.selectedPerformance.venueId;
                let eventId = ticketModel.selectedPerformance.eventId;

                // best place selection
                for (var i = 0; i < ticketModel.bestPlaceSelection.priceCategories.length; i++) {
                    if (ticketModel.bestPlaceSelection.priceCategories[i].ticketAmount > 0) {
                        let performanceTagId = this.parseIdFromPerformanceTag(ticketModel.selectedPerformance.tag);
                        $window.dataLayer.push({
                            'event': 'addToCart',
                            'ecommerce': {
                                'currencyCode': homeModel.selectedCurrency && homeModel.selectedCurrency.symbol ? homeModel.selectedCurrency.symbol : '',
                                'add': {
                                    'products':
                                        this.createAttributes( //name, id, price, brand, category, variant, quantity, date, time, coupon
                                            ticketModel.selectedPerformance.name,
                                            performanceTagId,
                                            ticketModel.bestPlaceSelection.priceCategories[i].prices["0"].price,
                                            ticketportal.webshop.bookingObjectPromoterId,
                                            "Ticket",
                                            undefined,
                                            ticketModel.bestPlaceSelection.priceCategories[i].ticketAmount,
                                            ticketModel.selectedDate,
                                            undefined, // date
                                            undefined, // coupon
                                            undefined, // artist
                                            undefined, // genre
                                            location,
                                            venue,
                                            eventId, //flyer aka event
                                            this.getLoggedUserLogin(),
                                            ''
                                        )
                                }
                            }
                        });
                    }
                }

                // seatmap selection
                if (selectedPlaces !== undefined) {
                    let seats = [];

                    let performanceTagId = this.parseIdFromPerformanceTag(ticketModel.selectedPerformance.tag);

                    for (let i = 0; i < selectedPlaces.length; i++) {
                        let place_id = selectedPlaces[i].SeatID;

                        var seatCheckbox = $("#" + place_id);
                        let seat_price = 0;
                        if (seatCheckbox.length > 0) {
                            try {
                                // can't belive I am parsing css class names ... but I see no other way at this point in the process 
                                let cat_style = this.genericFind(seatCheckbox.attr("class").split(" "), function (str) { return str.indexOf("cat_") >= 0; });
                                let price_category_id = cat_style.slice(4);
                                let price_category = this.genericFind(ticketModel.selectedPerformance.priceCategories, function (item) { return item.uniqueId === price_category_id; });
                                seat_price = price_category.prices[0].priceWithFees;
                            } catch (err) { console.log(err); }
                        }
                        else {
                            if (ticketModel.selectedSeats) {
                                let seat = this.genericFind(ticketModel.selectedSeats, function (seat) { return seat.id === place_id });
                                if (seat)
                                    seat_price = seat.price.priceWithFees;
                            }
                        }
                        let ec_prod = this.createAttributes( //name, id, price, brand, category, variant, quantity, date, time, coupon
                            ticketModel.selectedPerformance.name, //name
                            performanceTagId, // id 
                            seat_price, //price
                            ticketportal.webshop.bookingObjectPromoterId,
                            'Ticket', // category
                            undefined, // variant
                            1, // quantity
                            ticketModel.selectedDate, //  date
                            undefined, // time
                            undefined, //coupon
                            undefined, //artist
                            undefined, // genre
                            location,
                            venue,
                            eventId, //flyer
                            this.getLoggedUserLogin(),
                            ''
                        );

                        seats.push(ec_prod);
                    }

                    let currency_symbol = "CHF";
                    try {
                        currency_symbol = ticketModel.selectedPerformance.priceCategories[0].prices[0].currencySymbol;
                    } catch (err) {
                        console.log('performance without price');
                    }

                    $window.dataLayer.push({
                        'event': 'addToCart',
                        'ecommerce': {
                            'currencyCode': currency_symbol,
                            'add': {
                                'products': seats
                            }
                        }
                    });
                }
            },


            //call: CrossSellingController line 31
            pushCrossSellingPerformanceToCart: function (crossSellingModel, performance) {
                let userLogin = this.getLoggedUserLogin();
                let ecommerce_products = performance.bookingCategories.map(function (booking_cat) {
                    return {
                        'name': performance.name,
                        'id': performance.id,
                        'category': 'CrossSelling-Ticket',
                        'quantity': booking_cat.amount,
                        'price': booking_cat.price,
                        'brand': performance.promoterId,
                        'dimension1': "",
                        'dimension2': "",
                        'dimension3': performance.locationId,
                        'dimension4': performance.venueId,
                        'dimension5': performance.eventId,
                        'dimension6': userLogin,
                        'dimension7': ''
                    };
                });

                // filter ecommerce products with quantity/amount = 0
                ecommerce_products = ecommerce_products.filter(function (ele) { return ele.quantity !== ""; });

                $window.dataLayer.push({
                    'event': 'addToCart',
                    'ecommerce': {
                        'currencyCode': 'CHF',
                        'add': {
                            'products': ecommerce_products
                        }
                    }
                });
            },
            //call: BasketController line 113/155
            pushRemoveItemsFromCart: function (item, itemGroups) {
                if (itemGroups.length === 0) {
                    return;
                }

                let ecommerce_products = {
                    'price': item.finalPrice,
                    'quantity': item.quantity,
                    'category': item.typeName,
                    'brand': ticketportal.webshop.bookingObjectPromoterId

                };

                if (item.typeName === "Ticket") {
                    let performance_id = this.parseIdFromPerformanceTag(item.performanceTag);
                    ecommerce_products.id = performance_id;


                    let currentItemGroup = this.genericFind(itemGroups, function (item) { return item.id === performance_id; });
                    let parsedDate = this.formatDate(currentItemGroup.startDate);
                    ecommerce_products.name = currentItemGroup.name;
                    ecommerce_products.date = parsedDate;
                    ecommerce_products.dimension1 = "";
                    ecommerce_products.dimension2 = "";
                    ecommerce_products.dimension3 = item.locationID;
                    ecommerce_products.dimension4 = item.venueID;
                    ecommerce_products.dimension5 = item.eventID;
                    ecommerce_products.dimension6 = this.getLoggedUserLogin();
                    ecommerce_products.dimension7 = "";

                } else if (item.typeName === "Subscription") {
                    if (itemGroups.length > 0) {
                        ecommerce_products.id = item.articleDefinitionId;
                        ecommerce_products.name = itemGroups[0].name;
                        ecommerce_products.dimension1 = "";
                        ecommerce_products.dimension2 = "";
                        ecommerce_products.dimension3 = "";
                        ecommerce_products.dimension4 = "";
                        ecommerce_products.dimension5 = "";
                        ecommerce_products.dimension6 = this.getLoggedUserLogin();
                        ecommerce_products.dimension7 = "";
                    }
                } else if (item.typeName === "Package") {
                    let packageId = $rootScope.packageIdMap !== undefined ? $rootScope.packageIdMap[item.description] : "";
                    ecommerce_products.id = packageId;
                    ecommerce_products.name = item.description;
                    ecommerce_products.price = item.basePrice;
                    ecommerce_products.dimension1 = "";
                    ecommerce_products.dimension2 = "";
                    ecommerce_products.dimension3 = "";
                    ecommerce_products.dimension4 = "";
                    ecommerce_products.dimension5 = "";
                    ecommerce_products.dimension6 = this.getLoggedUserLogin();
                    ecommerce_products.dimension7 = "";
                } else {
                    ecommerce_products.name = item.description;
                    ecommerce_products.id = item.articleDefinitionId;
                    ecommerce_products.dimension1 = "";
                    ecommerce_products.dimension2 = "";
                    ecommerce_products.dimension3 = "";
                    ecommerce_products.dimension4 = "";
                    ecommerce_products.dimension5 = "";
                    ecommerce_products.dimension6 = this.getLoggedUserLogin();
                    ecommerce_products.dimension7 = "";
                }

                $window.dataLayer.push({
                    'event': 'removeFromCart',
                    'ecommerce': {
                        'remove': {
                            'products': [ecommerce_products]
                        }
                    }
                });

            },

            checkoutV1: function (basketModel) {

                var userLogin = this.getLoggedUserLogin();
                let productsArray = [];
                for (var i = 0; i < basketModel.itemGroups.length; i++) {
                    let b_item_grp = basketModel.itemGroups[i];
                    // to do : refactor
                    if (b_item_grp.typeName === "Ticket") {

                        var parsedDate = this.formatDate(b_item_grp.startDate);
                        let ecommmerce_products = b_item_grp.items.map(function (b_item) {
                            return {

                                'id': b_item_grp.id, //  b_item.id is not the performance id!
                                'price': b_item.finalPrice,
                                'name': b_item_grp.name,
                                'quantity': b_item.quantity,
                                'category': b_item.typeName,
                                'brand': ticketportal.webshop.bookingObjectPromoterId,
                                'date': parsedDate,
                                'dimension1': "",
                                'dimension2': "",
                                'dimension3': b_item.locationID,
                                'dimension4': b_item.venueID,
                                'dimension5': b_item.eventID,
                                'dimension6': userLogin,
                                'dimension7': ''

                                // todo: promocode
                            };
                        });

                        productsArray = productsArray.concat(ecommmerce_products);
                    } else if (b_item_grp.typeName === "GiftCertificate") {

                        let ecommmerce_products = b_item_grp.items.map(function (b_item) {
                            return {

                                // 'id' : b_item.id, 
                                // no ids in this call. ... 
                                // in the basketModel there is no GiftCertificateDefinitionId present
                                // the b_item.id is the unique (!) GiftCertificate Id
                                // is Id is generated after the checkout is started!
                                'id': b_item.articleDefinitionId,
                                'price': b_item.finalPrice,
                                'name': b_item.description,
                                'quantity': b_item.quantity,
                                'category': b_item.typeName,
                                'brand': ticketportal.webshop.bookingObjectPromoterId,
                                'dimension1': '',
                                'dimension2': '',
                                'dimension3': '',
                                'dimension4': '',
                                'dimension5': '',
                                'dimension6': userLogin,
                                'dimension7': ''
                            };
                        });

                        productsArray = productsArray.concat(ecommmerce_products);
                    } else if (b_item_grp.typeName === "Product") {

                        let ecommmerce_products = b_item_grp.items.map(function (b_item) {
                            return {
                                'id': b_item.articleDefinitionId,
                                'price': b_item.finalPrice,
                                'name': b_item.description,
                                'quantity': b_item.quantity,
                                'category': b_item.typeName,
                                'brand': ticketportal.webshop.bookingObjectPromoterId,
                                'dimension1': '',
                                'dimension2': '',
                                'dimension3': '',
                                'dimension4': '',
                                'dimension5': '',
                                'dimension6': userLogin,
                                'dimension7': ''
                            };
                        });

                        productsArray = productsArray.concat(ecommmerce_products);
                    } else if (b_item_grp.typeName === "Subscription") {

                        let ecommmerce_products = b_item_grp.items.map(function (b_item) {
                            let subscriptionId = b_item.subscriptionTag.split("-");
                            subscriptionId = subscriptionId[subscriptionId.length - 1];
                            return {
                                'id': subscriptionId,
                                'price': b_item.finalPrice,
                                'name': b_item_grp.name,
                                'quantity': b_item.quantity,
                                'category': b_item.typeName,
                                'brand': ticketportal.webshop.bookingObjectPromoterId,
                                'dimension1': '',
                                'dimension2': '',
                                'dimension3': '',
                                'dimension4': '',
                                'dimension5': '',
                                'dimension6': userLogin,
                                'dimension7': ''
                                // todo: promocode
                            };
                        });
                        productsArray = productsArray.concat(ecommmerce_products);
                    } else if (b_item_grp.typeName === "Package") {

                        let ecommmerce_products = b_item_grp.items.map(function (b_item) {
                            let packageId = $rootScope.packageIdMap !== undefined ? $rootScope.packageIdMap[b_item.description] : "";

                            return {
                                'id': packageId,
                                'price': b_item.total,
                                'name': b_item.description,
                                'quantity': b_item.quantity,
                                'category': b_item.typeName,
                                'brand': ticketportal.webshop.bookingObjectPromoterId,
                                'dimension1': '',
                                'dimension2': '',
                                'dimension3': '',
                                'dimension4': '',
                                'dimension5': '',
                                'dimension6': userLogin,
                                'dimension7': ''
                                // todo: promocode
                            };
                        });
                        productsArray = productsArray.concat(ecommmerce_products);

                    } else {

                        let ecommmerce_products = b_item_grp.items.map(function (b_item) {
                            return {

                                'id': b_item.id,
                                'price': b_item.finalPrice,
                                'name': b_item.description,
                                'quantity': b_item.quantity,
                                'category': b_item.typeName,
                                'brand': ticketportal.webshop.bookingObjectPromoterId,
                                'dimension1': '',
                                'dimension2': '',
                                'dimension3': '',
                                'dimension4': '',
                                'dimension5': '',
                                'dimension6': userLogin,
                                'dimension7': ''
                                // todo: promocode
                            };
                        });
                        productsArray = productsArray.concat(ecommmerce_products);

                    }
                }
                return productsArray;
            },


            //call: basketController line 236 / 246
            pushStartCheckout: function (model) {

                let products_items = this.checkoutV1(model);
                this.productsArray = products_items;

                $window.dataLayer.push({
                    'event': 'checkout',
                    'dimension8': ticketportal.webshop.bookingObjectPromoterId,
                    'ecommerce': {
                        'checkout': {
                            'actionField': {
                                'step': 1,
                                'currencyCode': model.itemGroups[0].currencySymbol
                                //'delivery': 'print at home',
                                //'payment': 'Visa'
                            },
                            'products': this.productsArray
                        }
                    }
                });
                $rootScope.checkout = this.productsArray;
            },
            //call: CheckoutController line 205
            pushCheckout: function (model) {
                // delivery options are of the form '[mediaID]_[deliverID]'
                let delivery_combined = model.selectedDeliveryOption.id.split("_")[1];
                // in case of "free" tickets the ordertotal might be 0.00 and therefore no paymentmethod needed
                let selectedPaymentOptionSafe = (model.selectedPaymentOption != null && model.selectedPaymentOption.id != null)
                    ? model.selectedPaymentOption.id
                    : "";

                $window.dataLayer.push({
                    'event': 'checkout',
                    'dimension8': ticketportal.webshop.bookingObjectPromoterId,
                    'ecommerce': {
                        'checkout': {
                            'actionField': {
                                'step': 2,
                                'currencyCode': model.currencySymbol,
                                'delivery': delivery_combined,
                                'payment': selectedPaymentOptionSafe
                            },
                            'products': $rootScope.checkout
                        }
                    }
                });
            },

            /*##### ################# Method for gift certificates ######################*/
            pushAddToCartGiftCertificates: function (giftCertificateModel, giftCertificates) {
                var userLogin = this.getLoggedUserLogin();
                var giftcertificates = giftCertificates.map(function (gf) {
                    return {
                        "name": gf.name,
                        "id": gf.id,
                        "quantity": gf.amount,
                        "price": gf.value,
                        "brand": ticketportal.webshop.bookingObjectPromoterId,
                        'dimension1': '',
                        'dimension2': '',
                        'dimension3': '',
                        'dimension4': '',
                        'dimension5': '',
                        'dimension6': userLogin,
                        'dimension7': ''
                    };
                });

                // add category
                giftcertificates = giftcertificates.map(function (item) { item.category = "GiftCertificate"; return item; });

                $window.dataLayer.push({
                    'event': 'addToCart',
                    'ecommerce': {
                        'currencyCode': 'CHF',
                        'add': {
                            'products': giftcertificates
                        }
                    }
                });
            },


            pushCrossSellingGiftCertificateToCart: function (giftCertificateItem) {
                let certificate_item = {
                    'id': giftCertificateItem.id,
                    'name': giftCertificateItem.name,
                    'quantity': giftCertificateItem.amount,
                    'category': 'CrossSelling-GiftCertificate',
                    'brand': ticketportal.webshop.bookingObjectPromoterId,
                    'dimension1': '',
                    'dimension2': '',
                    'dimension3': '',
                    'dimension4': '',
                    'dimension5': '',
                    'dimension6': this.getLoggedUserLogin(),
                    'dimension7': ''
                };

                if (giftCertificateItem.isFreeValue === true) {
                    certificate_item.price = giftCertificateItem.value;
                } else {
                    certificate_item.price = giftCertificateItem.value.value;
                }


                let currency_symbol = giftCertificateItem.currencySymbol;

                $window.dataLayer.push({
                    'event': 'addToCart',
                    'ecommerce': {
                        'currencyCode': currency_symbol,
                        'add': {
                            'products': [certificate_item]
                        }
                    }
                });


            },

            /*##### ################# Method for products ######################*/
            pushAddToCartProduct: function (productsModel, selected_products) {

                // for convenience
                let product_specifications = productsModel.bookingObject.products;

                // extract ecommerce fields
                var userLogin = this.getLoggedUserLogin();
                let self = this;
                let ecommerce_products = selected_products.map(function (prod) {
                    // search for select product informations
                    let product_spec = self.genericFind(product_specifications, function (product_def) { return product_def.id === prod.id; });

                    return {
                        'id': product_spec.id,
                        'price': product_spec.price,
                        'name': product_spec.name,
                        'category': "Product",
                        'quantity': prod.amount,
                        'brand': ticketportal.webshop.bookingObjectPromoterId,
                        'dimension1': '',
                        'dimension2': '',
                        'dimension3': '',
                        'dimension4': '',
                        'dimension5': '',
                        'dimension6': userLogin,
                        'dimension7': ''
                    };

                });

                let currency_symbol = product_specifications[0].currencySymbol;

                $window.dataLayer.push({
                    'event': 'addToCart',
                    'ecommerce': {
                        'currencyCode': currency_symbol,
                        'add': {
                            'products': ecommerce_products
                        }
                    }
                });
            },

            pushCrossSellingProductsToCart: function (crossSellingModel, productDefinition) {
                var userLogin = this.getLoggedUserLogin();
                let ecommerce_products = productDefinition.products.map(function (product) {
                    return {
                        'name': productDefinition.name,
                        'id': productDefinition.id,
                        'category': 'CrossSelling-Product',
                        'quantity': product.amount,
                        'price': product.price,
                        'brand': ticketportal.webshop.bookingObjectPromoterId,
                        'dimension1': '',
                        'dimension2': '',
                        'dimension3': '',
                        'dimension4': '',
                        'dimension5': '',
                        'dimension6': userLogin,
                        'dimension7': ''
                    };
                });

                $window.dataLayer.push({
                    'event': 'addToCart',
                    'ecommerce': {
                        'currencyCode': 'CHF',
                        'add': {
                            'products': ecommerce_products
                        }
                    }
                });
            },

            /* packages */

            pushAddPackageTicketToCart: function (packageModel, data) {

                let data_layer_product = {
                    "name": packageModel.name,
                    "id": packageModel.id,
                    "price": data.total,
                    "quantity": data.totalItems,
                    "brand": packageModel.promoterID,
                    "category": "Package",
                    'dimension1': '',
                    'dimension2': '',
                    'dimension3': '',
                    'dimension4': '',
                    'dimension5': '',
                    'dimension6': this.getLoggedUserLogin(),
                    'dimension7': ''
                };

                var data_layer_element = {
                    "event": "addToCart",
                    "ecommerce": {
                        "currencyCode": "CHF",
                        "add": {
                            "products": data_layer_product
                        }
                    }
                };
                if (!('packageIdMap' in $rootScope)) {
                    $rootScope.packageIdMap = {};
                }

                $rootScope.packageIdMap[packageModel.name] = packageModel.id;

                $window.dataLayer.push(data_layer_element);
            },


            /* subscription */
            addSubscriptionToCart: function (subscriptionModel, placesToReserve) {
                let products = {};
                for (let i = 0; i < placesToReserve.length; i++) {

                    // TODO: refactor css paring -> move to custom function (see ticket parsing)
                    // can't belive I am parsing css class names ... but I see no other way at this point in the process
                    let place_id = placesToReserve[i].split("-")[0];
                    let cat_style = this.genericFind($("#" + place_id).attr("class").split(" "), function (str) { return str.indexOf("cat_") >= 0; });
                    let price_category_id = cat_style.slice(4);
                    let price = subscriptionModel.getPrice(price_category_id, subscriptionModel.currencySymbol);

                    let data_layer_product = {
                        "name": subscriptionModel.name,
                        "id": subscriptionModel.id,
                        "price": price.price,
                        "quantity": 1,
                        "brand": ticketportal.webshop.bookingObjectPromoterId,
                        "category": "Subscription",
                        'dimension1': '',
                        'dimension2': '',
                        'dimension3': '',
                        'dimension4': '',
                        'dimension5': '',
                        'dimension6': this.getLoggedUserLogin(),
                        'dimension7': ''
                    };

                    if (products.hasOwnProperty(price_category_id)) {
                        products[price_category_id].quantity += 1;
                    } else {
                        products[price_category_id] = data_layer_product;
                    }



                }

                // group products




                var data_layer_element = {
                    "event": "addToCart",
                    "ecommerce": {
                        "currencyCode": "CHF",
                        "add": {
                            "products": Object.values(products)
                        }
                    }
                };


                $window.dataLayer.push(data_layer_element);


            },

            /*###################### Helping methods ######################*/

            parseIdFromPerformanceTag: function (performanceTag) { // get id from performanceTag
                if (performanceTag !== undefined) {
                    let id = performanceTag.split('-');
                    id = id[id.length - 1];
                    if (isFinite(String(id))) {         //check if returned value is actually a number
                        return id;
                    }
                }
                return null;
            },

            createAttributes: function (name, id, price, brand, category, variant, quantity, date, time, coupon, artist, genre, location, venue, flyer, userLogin, productList) {

                let newProduct = {};
                if (name !== undefined) {
                    newProduct.name = name;
                }
                if (id !== undefined) {
                    newProduct.id = id;
                }
                if (price !== undefined) {
                    newProduct.price = price;
                }
                if (brand !== undefined) {
                    newProduct.brand = brand;
                }
                if (category !== undefined) {
                    newProduct.category = category;
                }
                if (variant !== undefined) {
                    newProduct.variant = variant;
                }
                if (quantity !== undefined) {
                    newProduct.quantity = quantity;
                }
                if (date !== undefined) {
                    newProduct.date = date;
                }
                if (time !== undefined) {
                    newProduct.time = time;
                }
                if (coupon !== undefined) {
                    newProduct.coupon = coupon;
                }

                newProduct.dimension1 = artist ? artist : "";
                newProduct.dimension2 = genre ? genre : "";
                newProduct.dimension3 = location ? location : "";
                newProduct.dimension4 = venue ? venue : "";
                newProduct.dimension5 = flyer ? flyer : "";
                newProduct.dimension6 = userLogin ? userLogin : "";
                newProduct.dimension7 = productList ? productList : "";


                return newProduct;
            },

            getLoggedUserLogin: function () {
                let loginName = homeService.getLoginName();
                return loginName ? loginName : "";
            },

            formatDate: function (date) {
                try {
                    // If date is a string, the expected format is ISO 2019-06-14T19:30:00
                    // Returned format: 2019-06-14
                    if (typeof date === 'string' && date) {
                        return date.slice(0, 10);
                    }
                    // Check if the date is a date object
                    else if (Object.prototype.toString.call(date) === '[object Date]') {
                        return date.toISOString().slice(0, 10);
                    }
                } catch (err) {
                    console.log(err);
                }
                return "";
            },

            genericFind: function (arr, callback) {
                for (var i = 0; i < arr.length; i++) {
                    var match = callback(arr[i]);
                    if (match) {
                        return arr[i];
                    }
                }
            }
        };
    }
})();
;
