diff --git a/1.4.10/angular-1.4.10.zip b/1.4.10/angular-1.4.10.zip new file mode 100644 index 0000000000..bce5f79833 Binary files /dev/null and b/1.4.10/angular-1.4.10.zip differ diff --git a/1.4.10/angular-animate.js b/1.4.10/angular-animate.js new file mode 100644 index 0000000000..7c0677e261 --- /dev/null +++ b/1.4.10/angular-animate.js @@ -0,0 +1,4010 @@ +/** + * @license AngularJS v1.4.10 + * (c) 2010-2015 Google, Inc. http://angularjs.org + * License: MIT + */ +(function(window, angular, undefined) {'use strict'; + +/* jshint ignore:start */ +var noop = angular.noop; +var copy = angular.copy; +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 PREPARE_CLASS_SUFFIX = '-prepare'; + +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; +}; + +var ngMinErr = angular.$$minErr('ng'); +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 mergeAnimationDetails(element, oldAnimation, newAnimation) { + var target = oldAnimation.options || {}; + var newOptions = newAnimation.options || {}; + + 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; + } + + oldAnimation.addClass = target.addClass; + oldAnimation.removeClass = target.removeClass; + + 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(); + }); + } + } +}]; + +/** + * @ngdoc directive + * @name ngAnimateChildren + * @restrict AE + * @element ANY + * + * @description + * + * ngAnimateChildren allows you to specify that children of this element should animate even if any + * of the children's parents are currently animating. By default, when an element has an active `enter`, `leave`, or `move` + * (structural) animation, child elements that also have an active structural animation are not animated. + * + * Note that even if `ngAnimteChildren` is set, no child animations will run when the parent element is removed from the DOM (`leave` animation). + * + * + * @param {string} ngAnimateChildren If the value is empty, `true` or `on`, + * then child animations are allowed. If the value is `false`, child animations are not allowed. + * + * @example + * + +
+ + +
+
+
+ List of items: +
Item {{item}}
+
+
+
+
+ + + .container.ng-enter, + .container.ng-leave { + transition: all ease 1.5s; + } + + .container.ng-enter, + .container.ng-leave-active { + opacity: 0; + } + + .container.ng-leave, + .container.ng-enter-active { + opacity: 1; + } + + .item { + background: firebrick; + color: #FFF; + margin-bottom: 10px; + } + + .item.ng-enter, + .item.ng-leave { + transition: transform 1.5s ease; + } + + .item.ng-enter { + transform: translateX(50px); + } + + .item.ng-enter-active { + transform: translateX(0); + } + + + angular.module('ngAnimateChildren', ['ngAnimate']) + .controller('mainController', function() { + this.animateChildren = false; + this.enterElement = false; + }); + +
+ */ +var $$AnimateChildrenDirective = ['$interpolate', function($interpolate) { + return { + link: 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 { + // Interpolate and set the value, so that it is available to + // animations that run right after compilation + setData($interpolate(val)(scope)); + attrs.$observe('ngAnimateChildren', setData); + } + + function setData(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 + * + *
+ * This element will go BOOM + *
+ * + * ``` + * + * 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 + * /* 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 */ + * .red { background:red; } + * .large-text { font-size:20px; } + * + * /* we can also use a keyframe animation and $animateCss will make it work alongside the transition */ + * .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', '$$animateQueue', + function($window, $$jqLite, $$AnimateRunner, $timeout, + $$forceReflow, $sniffer, $$rAFScheduler, $$animateQueue) { + + 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, initialOptions) { + // all of the animation functions should create + // a copy of the options data, however, if a + // parent service has already created a copy then + // we should stick to using that + var options = initialOptions || {}; + if (!options.$$prepared) { + options = prepareAnimationOptions(copy(options)); + } + + var restoreStyles = {}; + var node = getDomNode(element); + if (!node + || !node.parentNode + || !$$animateQueue.enabled()) { + return closeAndReturnNoopAnimator(); + } + + 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; + var startTime; + var events = []; + + 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; + if (typeof options.delay !== "boolean") { + delayStyle = parseFloat(options.delay); + // number in options.delay means we have to recalculate the delay for the closing timeout + maxDelay = Math.max(delayStyle, 0); + } + + 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 (events && events.length) { + // Remove the transitionend / animationend listener(s) + element.off(events.join(' '), onAnimationProgress); + } + + //Cancel the fallback closing timeout and remove the timer data + var animationTimerData = element.data(ANIMATE_TIMER_KEY); + if (animationTimerData) { + $timeout.cancel(animationTimerData[0].timer); + element.removeData(ANIMATE_TIMER_KEY); + } + + // 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 onAnimationProgress(event) { + event.stopPropagation(); + var ev = event.originalEvent || event; + + // we now always use `Date.now()` due to the recent changes with + // event.timeStamp in Firefox, Webkit and Chrome (see #13494 for more info) + var timeStamp = ev.$manualTimeStamp || 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 animationPauseds 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(); + } + } + + function start() { + if (animationClosed) return; + if (!node.parentNode) { + close(); + return; + } + + // 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); + } + + if (events.length) { + 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); + } + } + } + }; + }]; +}]; + +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) { + var animationClosed = false; + + // 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); + } + + function close() { + animationClosed = true; + applyOptions(); + applyAnimationStyles(element, options); + } + + var runner; + + return { + $$willAnimate: true, + end: function() { + if (runner) { + runner.end(); + } else { + close(); + runner = new $$AnimateRunner(); + runner.complete(true); + } + return runner; + }, + start: function() { + if (runner) { + return runner; + } + + runner = new $$AnimateRunner(); + 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); + }); + } + + runner.setHost({ + end: function() { + endAnimations(); + }, + cancel: function() { + endAnimations(true); + } + }); + + $$AnimateRunner.chain(chain, onComplete); + return runner; + + function onComplete(success) { + close(success); + 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 ONE_SPACE = ' '; + + var rules = this.rules = { + skip: [], + cancel: [], + join: [] + }; + + function makeTruthyCssClassMap(classString) { + if (!classString) { + return null; + } + + var keys = classString.split(ONE_SPACE); + var map = Object.create(null); + + forEach(keys, function(key) { + map[key] = true; + }); + return map; + } + + function hasMatchingClasses(newClassString, currentClassString) { + if (newClassString && currentClassString) { + var currentClassMap = makeTruthyCssClassMap(currentClassString); + return newClassString.split(ONE_SPACE).some(function(className) { + return currentClassMap[className]; + }); + } + } + + function isAllowed(ruleType, element, currentAnimation, previousAnimation) { + return rules[ruleType].some(function(fn) { + return fn(element, currentAnimation, previousAnimation); + }); + } + + function hasAnimationClasses(animation, and) { + var a = (animation.addClass || '').length > 0; + var b = (animation.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); + }); + + 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); + }); + + 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 nA = newAnimation.addClass; + var nR = newAnimation.removeClass; + var cA = currentAnimation.addClass; + var cR = currentAnimation.removeClass; + + // early detection to save the global CPU shortage :) + if ((isUndefined(nA) && isUndefined(nR)) || (isUndefined(cA) && isUndefined(cR))) { + return false; + } + + return hasMatchingClasses(nA, cR) || hasMatchingClasses(nR, cA); + }); + + 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 normalizeAnimationDetails(element, animation) { + return mergeAnimationDetails(element, animation, {}); + } + + // IE9-11 has no method "contains" in SVG element and in Node.prototype. Bug #10259. + var contains = Node.prototype.contains || function(arg) { + // jshint bitwise: false + return this === arg || !!(this.compareDocumentPosition(arg) & 16); + // jshint bitwise: true + }; + + 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 (contains.call(entry.node, targetNode)) { + matches.push(entry.callback); + } else if (event === 'leave' && contains.call(entry.node, 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 + 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 + disabledElementsLookup.put(node, !bool); + } + } + } + + return bool; + } + }; + + function queueAnimation(element, event, initialOptions) { + // we always make a copy of the options since + // there should never be any side effects on + // the input data when running `$animateCss`. + var options = copy(initialOptions); + + 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 + // Animations are also disabled if the document is currently hidden (page is not visible + // to the user), because browsers slow down or do not flush calls to requestAnimationFrame + var skipAnimations = !animationsEnabled || $document[0].hidden || 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, + addClass: options.addClass, + removeClass: options.removeClass, + 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 { + mergeAnimationDetails(element, existingAnimation, newAnimation); + 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 + mergeAnimationDetails(element, existingAnimation, newAnimation); + + 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) { + normalizeAnimationDetails(element, newAnimation); + } else { + applyGeneratedPreparationClasses(element, isStructural ? event : null, options); + + event = newAnimation.event = existingAnimation.event; + options = mergeAnimationDetails(element, existingAnimation, newAnimation); + + //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 + normalizeAnimationDetails(element, newAnimation); + } + + // 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); + } + + 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)); + + // 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, 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); + if (animationDetails) { + switch (state) { + case RUNNING_STATE: + animationDetails.runner.end(); + /* falls through */ + case PRE_DIGEST_STATE: + 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); + } + + /** + * This fn returns false if any of the following is true: + * a) animations on any parent element are disabled, and animations on the element aren't explicitly allowed + * b) a parent element has an ongoing structural animation, and animateChildren is false + * c) the element is not a child of the body + * d) the element is not a child of the $rootElement + */ + 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 elementDisabled = disabledElementsLookup.get(getDomNode(element)); + + var parentHost = jqLite.data(element[0], NG_ANIMATE_PIN_DATA); + if (parentHost) { + parentElement = parentHost; + } + + parentElement = getDomNode(parentElement); + + while (parentElement) { + 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 (parentElement.nodeType !== ELEMENT_NODE) { + // no point in inspecting the #document element + break; + } + + var details = activeAnimationsLookup.get(parentElement) || {}; + // 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) { + var parentElementDisabled = disabledElementsLookup.get(parentElement); + + if (parentElementDisabled === true && elementDisabled !== false) { + // disable animations if the user hasn't explicitly enabled animations on the + // current element + elementDisabled = true; + // element is disabled via parent element, no need to check anything else + break; + } else if (parentElementDisabled === false) { + elementDisabled = false; + } + parentAnimationDetected = details.structural; + } + + if (isUndefined(animateChildren) || animateChildren === true) { + var value = jqLite.data(parentElement, 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 (!bodyElementDetected) { + // we also need to ensure that the element is or will be a part of the body element + // otherwise it is pointless to even issue an animation to be rendered + bodyElementDetected = isMatchingElement(parentElement, bodyElement); + } + + if (bodyElementDetected && rootElementDetected) { + // If both body and root have been found, any other checks are pointless, + // as no animation data should live outside the application + break; + } + + if (!rootElementDetected) { + // If no rootElement is detected, check if the parentElement is pinned to another element + parentHost = jqLite.data(parentElement, NG_ANIMATE_PIN_DATA); + if (parentHost) { + // The pin target element becomes the next parent element + parentElement = getDomNode(parentHost); + continue; + } + } + + parentElement = parentElement.parentNode; + } + + var allowAnimation = (!parentAnimationDetected || animateChildren) && elementDisabled !== true; + 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 $$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; + } + + var prepareClassName; + if (isStructural) { + prepareClassName = 'ng-' + event + PREPARE_CLASS_SUFFIX; + $$jqLite.addClass(element, prepareClassName); + } + + 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); + } + if (prepareClassName) { + $$jqLite.removeClass(element, prepareClassName); + prepareClassName = null; + } + } + + 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, + $$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. + * + *
+ * + * # 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 + *
+ * Fade me in out + *
+ * + * + * ``` + * + * Notice the CSS class **fade**? We can now create the CSS transition code that references this class: + * + * ```css + * /* The starting CSS styles for the enter animation */ + * .fade.ng-enter { + * transition:0.5s linear all; + * opacity:0; + * } + * + * /* The finishing CSS styles for the enter animation */ + * .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 + * /* now the element will fade out before it is removed from the DOM */ + * .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 + * /* there is no need to define anything inside of the destination + * CSS class since the keyframe will take charge of the animation */ + * .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 + *
+ * Show and hide me + *
+ * + * + * + * ``` + * + * 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 + *
+ * Highlight this box + *
+ * + * + * + * ``` + * + * 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 { + * /* standard transition code */ + * transition: 1s linear all; + * opacity:0; + * } + * .my-animation.ng-enter-stagger { + * /* this will have a 100ms delay between each successive leave animation */ + * transition-delay: 0.1s; + * + * /* As of 1.4.4, this must always be set: it signals ngAnimate + * to not accidentally inherit a delay property from another CSS class */ + * transition-duration: 0s; + * } + * .my-animation.ng-enter.ng-enter-active { + * /* standard transition styles */ + * 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.) + * + * + * ### The `ng-[event]-prepare` class + * + * This is a special class that can be used to prevent unwanted flickering / flash of content before + * the actual animation starts. The class is added as soon as an animation is initialized, but removed + * before the actual animation starts (after waiting for a $digest). + * It is also only added for *structural* animations (`enter`, `move`, and `leave`). + * + * In practice, flickering can appear when nesting elements with structural animations such as `ngIf` + * into elements that have class-based animations such as `ngClass`. + * + * ```html + *
+ *
+ *
+ *
+ *
+ * ``` + * + * It is possible that during the `enter` animation, the `.message` div will be briefly visible before it starts animating. + * In that case, you can add styles to the CSS that make sure the element stays hidden before the animation starts: + * + * ```css + * .message.ng-enter-prepare { + * opacity: 0; + * } + * + * ``` + * + * ## 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 + *
+ * {{ item }} + *
+ * ``` + * + * 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 + *
+ * this box is moody + *
+ * + * + * + * ``` + * + * ```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 + *
+ * Slide in and out + *
+ * ``` + * + * ```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 + * /* the transition tells ngAnimate to make the animation happen */ + * .slide.ng-enter { transition:0.5s linear all; } + * + * /* this extra CSS class will be absorbed into the transition + * since the $animateCss code is adding the class */ + * .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 + * + *
+ *
+ * + * + * + * + * + * + * + * + * ``` + * + * 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 { + * /* this animation will last for 1 second since there are + * two phases to the animation (an `in` and an `out` phase) */ + * 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; + * + * /* the scale will be applied during the out animation, + * but will be animated away when the in animation runs */ + * transform: scale(1.2); + * } + * + * .banner.ng-anchor-in { + * transition: 1s linear all; + * } + * ``` + * + * + * + * + * ### Anchoring Demo + * + + + Home +
+
+
+
+
+ + 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; + }]); + + +

Welcome to the home page

+

Please click on an element

+ + {{ record.title }} + +
+ +
+ {{ profile.title }} +
+
+ + .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; + } + +
+ * + * ### 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 `` 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 + * Hi there + * ``` + * + * ```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 + * /* normally we would create a CSS class to reference on the element */ + * 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 + * /* prefixed with animate- */ + * .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) + + .provider('$$animateQueue', $$AnimateQueueProvider) + .provider('$$animation', $$AnimationProvider) + + .provider('$animateCss', $AnimateCssProvider) + .provider('$$animateCssDriver', $$AnimateCssDriverProvider) + + .provider('$$animateJs', $$AnimateJsProvider) + .provider('$$animateJsDriver', $$AnimateJsDriverProvider); + + +})(window, window.angular); diff --git a/1.4.10/angular-animate.min.js b/1.4.10/angular-animate.min.js new file mode 100644 index 0000000000..102df66294 --- /dev/null +++ b/1.4.10/angular-animate.min.js @@ -0,0 +1,55 @@ +/* + AngularJS v1.4.10 + (c) 2010-2015 Google, Inc. http://angularjs.org + License: MIT +*/ +(function(B,r,Ua){'use strict';function xa(a,b,c){if(!a)throw Ja("areq",b||"?",c||"required");return a}function ya(a,b){if(!a&&!b)return"";if(!a)return b;if(!b)return a;ba(a)&&(a=a.join(" "));ba(b)&&(b=b.join(" "));return a+" "+b}function Ka(a){var b={};a&&(a.to||a.from)&&(b.to=a.to,b.from=a.from);return b}function X(a,b,c){var d="";a=ba(a)?a:a&&Q(a)&&a.length?a.split(/\s+/):[];s(a,function(a,g){a&&0=a&&(a=e,e=0,b.push(u),u=[]);u.push(t.fn);t.children.forEach(function(a){e++;c.push(a)});a--}u.length&&b.push(u);return b}(c)}var M=[],r=U(a);return function(v,G,w){function C(a){a=a.hasAttribute("ng-animate-ref")?[a]:a.querySelectorAll("[ng-animate-ref]");var b=[];s(a,function(a){var c=a.getAttribute("ng-animate-ref");c&&c.length&&b.push(a)});return b}function K(a){var b=[], +c={};s(a,function(a,f){var d=A(a.element),h=0<=["enter","move"].indexOf(a.event),d=a.structural?C(d):[];if(d.length){var e=h?"to":"from";s(d,function(a){var b=a.getAttribute("ng-animate-ref");c[b]=c[b]||{};c[b][e]={animationID:f,element:F(a)}})}else b.push(a)});var d={},h={};s(c,function(c,e){var l=c.from,u=c.to;if(l&&u){var D=a[l.animationID],t=a[u.animationID],k=l.animationID.toString();if(!h[k]){var g=h[k]={structural:!0,beforeStart:function(){D.beforeStart();t.beforeStart()},close:function(){D.close(); +t.close()},classes:H(D.classes,t.classes),from:D,to:t,anchors:[]};g.classes.length?b.push(g):(b.push(D),b.push(t))}h[k].anchors.push({out:l.element,"in":u.element})}else l=l?l.animationID:u.animationID,u=l.toString(),d[u]||(d[u]=!0,b.push(a[l]))});return b}function H(a,b){a=a.split(" ");b=b.split(" ");for(var c=[],d=0;d=Q&&b>=O&&(F=!0,q())}function L(){function b(){if(!G){u(!1);s(m,function(a){l.style[a[0]]=a[1]});C(a,f);e.addClass(a,da);if(p.recalculateTimingStyles){ka=l.className+" "+ea;ha=r(l,ka);E=w(l,ka,ha);$=E.maxDelay;n=Math.max($,0);O=E.maxDuration;if(0===O){q();return}p.hasTransitions=0L.expectedEndTime)? +I.cancel(L.timer):g.push(q)}k&&(t=I(c,t,!1),g[0]={timer:t,expectedEndTime:d},g.push(q),a.data("$$animateCss",g));if(fa.length)a.on(fa.join(" "),D);f.to&&(f.cleanupStyles&&Fa(z,l,Object.keys(f.to)),Aa(a,f))}}function c(){var b=a.data("$$animateCss");if(b){for(var d=1;dARIA](http://www.w3.org/TR/wai-aria/) + * attributes that convey state or semantic information about the application for users + * of assistive technologies, such as screen readers. + * + *
+ * + * ## Usage + * + * For ngAria to do its magic, simply include the module `ngAria` as a dependency. The following + * directives are supported: + * `ngModel`, `ngDisabled`, `ngShow`, `ngHide`, `ngClick`, `ngDblClick`, and `ngMessages`. + * + * Below is a more detailed breakdown of the attributes handled by ngAria: + * + * | Directive | Supported Attributes | + * |---------------------------------------------|----------------------------------------------------------------------------------------| + * | {@link ng.directive:ngDisabled ngDisabled} | aria-disabled | + * | {@link ng.directive:ngShow ngShow} | aria-hidden | + * | {@link ng.directive:ngHide ngHide} | aria-hidden | + * | {@link ng.directive:ngDblclick ngDblclick} | tabindex | + * | {@link module:ngMessages ngMessages} | aria-live | + * | {@link ng.directive:ngModel ngModel} | aria-checked, aria-valuemin, aria-valuemax, aria-valuenow, aria-invalid, aria-required, input roles | + * | {@link ng.directive:ngClick ngClick} | tabindex, keypress event, button role | + * + * Find out more information about each directive by reading the + * {@link guide/accessibility ngAria Developer Guide}. + * + * ##Example + * Using ngDisabled with ngAria: + * ```html + * + * ``` + * Becomes: + * ```html + * + * ``` + * + * ##Disabling Attributes + * It's possible to disable individual attributes added by ngAria with the + * {@link ngAria.$ariaProvider#config config} method. For more details, see the + * {@link guide/accessibility Developer Guide}. + */ + /* global -ngAriaModule */ +var ngAriaModule = angular.module('ngAria', ['ng']). + provider('$aria', $AriaProvider); + +/** +* Internal Utilities +*/ +var nodeBlackList = ['BUTTON', 'A', 'INPUT', 'TEXTAREA', 'SELECT', 'DETAILS', 'SUMMARY']; + +var isNodeOneOf = function(elem, nodeTypeArray) { + if (nodeTypeArray.indexOf(elem[0].nodeName) !== -1) { + return true; + } +}; +/** + * @ngdoc provider + * @name $ariaProvider + * + * @description + * + * Used for configuring the ARIA attributes injected and managed by ngAria. + * + * ```js + * angular.module('myApp', ['ngAria'], function config($ariaProvider) { + * $ariaProvider.config({ + * ariaValue: true, + * tabindex: false + * }); + * }); + *``` + * + * ## Dependencies + * Requires the {@link ngAria} module to be installed. + * + */ +function $AriaProvider() { + var config = { + ariaHidden: true, + ariaChecked: true, + ariaDisabled: true, + ariaRequired: true, + ariaInvalid: true, + ariaMultiline: true, + ariaValue: true, + tabindex: true, + bindKeypress: true, + bindRoleForClick: true + }; + + /** + * @ngdoc method + * @name $ariaProvider#config + * + * @param {object} config object to enable/disable specific ARIA attributes + * + * - **ariaHidden** – `{boolean}` – Enables/disables aria-hidden tags + * - **ariaChecked** – `{boolean}` – Enables/disables aria-checked tags + * - **ariaDisabled** – `{boolean}` – Enables/disables aria-disabled tags + * - **ariaRequired** – `{boolean}` – Enables/disables aria-required tags + * - **ariaInvalid** – `{boolean}` – Enables/disables aria-invalid tags + * - **ariaMultiline** – `{boolean}` – Enables/disables aria-multiline tags + * - **ariaValue** – `{boolean}` – Enables/disables aria-valuemin, aria-valuemax and aria-valuenow tags + * - **tabindex** – `{boolean}` – Enables/disables tabindex tags + * - **bindKeypress** – `{boolean}` – Enables/disables keypress event binding on `<div>` and + * `<li>` elements with ng-click + * - **bindRoleForClick** – `{boolean}` – Adds role=button to non-interactive elements like `div` + * using ng-click, making them more accessible to users of assistive technologies + * + * @description + * Enables/disables various ARIA attributes + */ + this.config = function(newConfig) { + config = angular.extend(config, newConfig); + }; + + function watchExpr(attrName, ariaAttr, nodeBlackList, negate) { + return function(scope, elem, attr) { + var ariaCamelName = attr.$normalize(ariaAttr); + if (config[ariaCamelName] && !isNodeOneOf(elem, nodeBlackList) && !attr[ariaCamelName]) { + scope.$watch(attr[attrName], function(boolVal) { + // ensure boolean value + boolVal = negate ? !boolVal : !!boolVal; + elem.attr(ariaAttr, boolVal); + }); + } + }; + } + /** + * @ngdoc service + * @name $aria + * + * @description + * @priority 200 + * + * The $aria service contains helper methods for applying common + * [ARIA](http://www.w3.org/TR/wai-aria/) attributes to HTML directives. + * + * ngAria injects common accessibility attributes that tell assistive technologies when HTML + * elements are enabled, selected, hidden, and more. To see how this is performed with ngAria, + * let's review a code snippet from ngAria itself: + * + *```js + * ngAriaModule.directive('ngDisabled', ['$aria', function($aria) { + * return $aria.$$watchExpr('ngDisabled', 'aria-disabled'); + * }]) + *``` + * Shown above, the ngAria module creates a directive with the same signature as the + * traditional `ng-disabled` directive. But this ngAria version is dedicated to + * solely managing accessibility attributes. The internal `$aria` service is used to watch the + * boolean attribute `ngDisabled`. If it has not been explicitly set by the developer, + * `aria-disabled` is injected as an attribute with its value synchronized to the value in + * `ngDisabled`. + * + * Because ngAria hooks into the `ng-disabled` directive, developers do not have to do + * anything to enable this feature. The `aria-disabled` attribute is automatically managed + * simply as a silent side-effect of using `ng-disabled` with the ngAria module. + * + * The full list of directives that interface with ngAria: + * * **ngModel** + * * **ngShow** + * * **ngHide** + * * **ngClick** + * * **ngDblclick** + * * **ngMessages** + * * **ngDisabled** + * + * Read the {@link guide/accessibility ngAria Developer Guide} for a thorough explanation of each + * directive. + * + * + * ## Dependencies + * Requires the {@link ngAria} module to be installed. + */ + this.$get = function() { + return { + config: function(key) { + return config[key]; + }, + $$watchExpr: watchExpr + }; + }; +} + + +ngAriaModule.directive('ngShow', ['$aria', function($aria) { + return $aria.$$watchExpr('ngShow', 'aria-hidden', [], true); +}]) +.directive('ngHide', ['$aria', function($aria) { + return $aria.$$watchExpr('ngHide', 'aria-hidden', [], false); +}]) +.directive('ngModel', ['$aria', function($aria) { + + function shouldAttachAttr(attr, normalizedAttr, elem) { + return $aria.config(normalizedAttr) && !elem.attr(attr); + } + + function shouldAttachRole(role, elem) { + return !elem.attr('role') && (elem.attr('type') === role) && (elem[0].nodeName !== 'INPUT'); + } + + function getShape(attr, elem) { + var type = attr.type, + role = attr.role; + + return ((type || role) === 'checkbox' || role === 'menuitemcheckbox') ? 'checkbox' : + ((type || role) === 'radio' || role === 'menuitemradio') ? 'radio' : + (type === 'range' || role === 'progressbar' || role === 'slider') ? 'range' : + (type || role) === 'textbox' || elem[0].nodeName === 'TEXTAREA' ? 'multiline' : ''; + } + + return { + restrict: 'A', + require: '?ngModel', + priority: 200, //Make sure watches are fired after any other directives that affect the ngModel value + compile: function(elem, attr) { + var shape = getShape(attr, elem); + + return { + pre: function(scope, elem, attr, ngModel) { + if (shape === 'checkbox' && attr.type !== 'checkbox') { + //Use the input[checkbox] $isEmpty implementation for elements with checkbox roles + ngModel.$isEmpty = function(value) { + return value === false; + }; + } + }, + post: function(scope, elem, attr, ngModel) { + var needsTabIndex = shouldAttachAttr('tabindex', 'tabindex', elem) + && !isNodeOneOf(elem, nodeBlackList); + + function ngAriaWatchModelValue() { + return ngModel.$modelValue; + } + + function getRadioReaction() { + if (needsTabIndex) { + needsTabIndex = false; + return function ngAriaRadioReaction(newVal) { + var boolVal = (attr.value == ngModel.$viewValue); + elem.attr('aria-checked', boolVal); + elem.attr('tabindex', 0 - !boolVal); + }; + } else { + return function ngAriaRadioReaction(newVal) { + elem.attr('aria-checked', (attr.value == ngModel.$viewValue)); + }; + } + } + + function ngAriaCheckboxReaction() { + elem.attr('aria-checked', !ngModel.$isEmpty(ngModel.$viewValue)); + } + + switch (shape) { + case 'radio': + case 'checkbox': + if (shouldAttachRole(shape, elem)) { + elem.attr('role', shape); + } + if (shouldAttachAttr('aria-checked', 'ariaChecked', elem)) { + scope.$watch(ngAriaWatchModelValue, shape === 'radio' ? + getRadioReaction() : ngAriaCheckboxReaction); + } + if (needsTabIndex) { + elem.attr('tabindex', 0); + } + break; + case 'range': + if (shouldAttachRole(shape, elem)) { + elem.attr('role', 'slider'); + } + if ($aria.config('ariaValue')) { + var needsAriaValuemin = !elem.attr('aria-valuemin') && + (attr.hasOwnProperty('min') || attr.hasOwnProperty('ngMin')); + var needsAriaValuemax = !elem.attr('aria-valuemax') && + (attr.hasOwnProperty('max') || attr.hasOwnProperty('ngMax')); + var needsAriaValuenow = !elem.attr('aria-valuenow'); + + if (needsAriaValuemin) { + attr.$observe('min', function ngAriaValueMinReaction(newVal) { + elem.attr('aria-valuemin', newVal); + }); + } + if (needsAriaValuemax) { + attr.$observe('max', function ngAriaValueMinReaction(newVal) { + elem.attr('aria-valuemax', newVal); + }); + } + if (needsAriaValuenow) { + scope.$watch(ngAriaWatchModelValue, function ngAriaValueNowReaction(newVal) { + elem.attr('aria-valuenow', newVal); + }); + } + } + if (needsTabIndex) { + elem.attr('tabindex', 0); + } + break; + case 'multiline': + if (shouldAttachAttr('aria-multiline', 'ariaMultiline', elem)) { + elem.attr('aria-multiline', true); + } + break; + } + + if (ngModel.$validators.required && shouldAttachAttr('aria-required', 'ariaRequired', elem)) { + scope.$watch(function ngAriaRequiredWatch() { + return ngModel.$error.required; + }, function ngAriaRequiredReaction(newVal) { + elem.attr('aria-required', !!newVal); + }); + } + + if (shouldAttachAttr('aria-invalid', 'ariaInvalid', elem)) { + scope.$watch(function ngAriaInvalidWatch() { + return ngModel.$invalid; + }, function ngAriaInvalidReaction(newVal) { + elem.attr('aria-invalid', !!newVal); + }); + } + } + }; + } + }; +}]) +.directive('ngDisabled', ['$aria', function($aria) { + return $aria.$$watchExpr('ngDisabled', 'aria-disabled', []); +}]) +.directive('ngMessages', function() { + return { + restrict: 'A', + require: '?ngMessages', + link: function(scope, elem, attr, ngMessages) { + if (!elem.attr('aria-live')) { + elem.attr('aria-live', 'assertive'); + } + } + }; +}) +.directive('ngClick',['$aria', '$parse', function($aria, $parse) { + return { + restrict: 'A', + compile: function(elem, attr) { + var fn = $parse(attr.ngClick, /* interceptorFn */ null, /* expensiveChecks */ true); + return function(scope, elem, attr) { + + if (!isNodeOneOf(elem, nodeBlackList)) { + + if ($aria.config('bindRoleForClick') && !elem.attr('role')) { + elem.attr('role', 'button'); + } + + if ($aria.config('tabindex') && !elem.attr('tabindex')) { + elem.attr('tabindex', 0); + } + + if ($aria.config('bindKeypress') && !attr.ngKeypress) { + elem.on('keypress', function(event) { + var keyCode = event.which || event.keyCode; + if (keyCode === 32 || keyCode === 13) { + scope.$apply(callback); + } + + function callback() { + fn(scope, { $event: event }); + } + }); + } + } + }; + } + }; +}]) +.directive('ngDblclick', ['$aria', function($aria) { + return function(scope, elem, attr) { + if ($aria.config('tabindex') && !elem.attr('tabindex') && !isNodeOneOf(elem, nodeBlackList)) { + elem.attr('tabindex', 0); + } + }; +}]); + + +})(window, window.angular); diff --git a/1.4.10/angular-aria.min.js b/1.4.10/angular-aria.min.js new file mode 100644 index 0000000000..55f6975cb3 --- /dev/null +++ b/1.4.10/angular-aria.min.js @@ -0,0 +1,14 @@ +/* + AngularJS v1.4.10 + (c) 2010-2015 Google, Inc. http://angularjs.org + License: MIT +*/ +(function(u,n,v){'use strict';var r="BUTTON A INPUT TEXTAREA SELECT DETAILS SUMMARY".split(" "),p=function(a,c){if(-1!==c.indexOf(a[0].nodeName))return!0};n.module("ngAria",["ng"]).provider("$aria",function(){function a(a,f,l,m){return function(d,e,b){var g=b.$normalize(f);!c[g]||p(e,l)||b[g]||d.$watch(b[a],function(b){b=m?!b:!!b;e.attr(f,b)})}}var c={ariaHidden:!0,ariaChecked:!0,ariaDisabled:!0,ariaRequired:!0,ariaInvalid:!0,ariaMultiline:!0,ariaValue:!0,tabindex:!0,bindKeypress:!0,bindRoleForClick:!0}; +this.config=function(a){c=n.extend(c,a)};this.$get=function(){return{config:function(a){return c[a]},$$watchExpr:a}}}).directive("ngShow",["$aria",function(a){return a.$$watchExpr("ngShow","aria-hidden",[],!0)}]).directive("ngHide",["$aria",function(a){return a.$$watchExpr("ngHide","aria-hidden",[],!1)}]).directive("ngModel",["$aria",function(a){function c(c,m,d){return a.config(m)&&!d.attr(c)}function k(a,c){return!c.attr("role")&&c.attr("type")===a&&"INPUT"!==c[0].nodeName}function f(a,c){var d= +a.type,e=a.role;return"checkbox"===(d||e)||"menuitemcheckbox"===e?"checkbox":"radio"===(d||e)||"menuitemradio"===e?"radio":"range"===d||"progressbar"===e||"slider"===e?"range":"textbox"===(d||e)||"TEXTAREA"===c[0].nodeName?"multiline":""}return{restrict:"A",require:"?ngModel",priority:200,compile:function(l,m){var d=f(m,l);return{pre:function(a,b,c,h){"checkbox"===d&&"checkbox"!==c.type&&(h.$isEmpty=function(b){return!1===b})},post:function(e,b,g,h){function f(){return h.$modelValue}function m(){return q? +(q=!1,function(a){a=g.value==h.$viewValue;b.attr("aria-checked",a);b.attr("tabindex",0-!a)}):function(a){b.attr("aria-checked",g.value==h.$viewValue)}}function l(){b.attr("aria-checked",!h.$isEmpty(h.$viewValue))}var q=c("tabindex","tabindex",b)&&!p(b,r);switch(d){case "radio":case "checkbox":k(d,b)&&b.attr("role",d);c("aria-checked","ariaChecked",b)&&e.$watch(f,"radio"===d?m():l);q&&b.attr("tabindex",0);break;case "range":k(d,b)&&b.attr("role","slider");if(a.config("ariaValue")){var n=!b.attr("aria-valuemin")&& +(g.hasOwnProperty("min")||g.hasOwnProperty("ngMin")),s=!b.attr("aria-valuemax")&&(g.hasOwnProperty("max")||g.hasOwnProperty("ngMax")),t=!b.attr("aria-valuenow");n&&g.$observe("min",function(a){b.attr("aria-valuemin",a)});s&&g.$observe("max",function(a){b.attr("aria-valuemax",a)});t&&e.$watch(f,function(a){b.attr("aria-valuenow",a)})}q&&b.attr("tabindex",0);break;case "multiline":c("aria-multiline","ariaMultiline",b)&&b.attr("aria-multiline",!0)}h.$validators.required&&c("aria-required","ariaRequired", +b)&&e.$watch(function(){return h.$error.required},function(a){b.attr("aria-required",!!a)});c("aria-invalid","ariaInvalid",b)&&e.$watch(function(){return h.$invalid},function(a){b.attr("aria-invalid",!!a)})}}}}}]).directive("ngDisabled",["$aria",function(a){return a.$$watchExpr("ngDisabled","aria-disabled",[])}]).directive("ngMessages",function(){return{restrict:"A",require:"?ngMessages",link:function(a,c,k,f){c.attr("aria-live")||c.attr("aria-live","assertive")}}}).directive("ngClick",["$aria","$parse", +function(a,c){return{restrict:"A",compile:function(k,f){var l=c(f.ngClick,null,!0);return function(c,d,e){if(!p(d,r)&&(a.config("bindRoleForClick")&&!d.attr("role")&&d.attr("role","button"),a.config("tabindex")&&!d.attr("tabindex")&&d.attr("tabindex",0),a.config("bindKeypress")&&!e.ngKeypress))d.on("keypress",function(a){function d(){l(c,{$event:a})}var e=a.which||a.keyCode;32!==e&&13!==e||c.$apply(d)})}}}}]).directive("ngDblclick",["$aria",function(a){return function(c,k,f){!a.config("tabindex")|| +k.attr("tabindex")||p(k,r)||k.attr("tabindex",0)}}])})(window,window.angular); +//# sourceMappingURL=angular-aria.min.js.map diff --git a/1.4.10/angular-aria.min.js.map b/1.4.10/angular-aria.min.js.map new file mode 100644 index 0000000000..13000e15d3 --- /dev/null +++ b/1.4.10/angular-aria.min.js.map @@ -0,0 +1,8 @@ +{ +"version":3, +"file":"angular-aria.min.js", +"lineCount":13, +"mappings":"A;;;;;aAKC,SAAQ,CAACA,CAAD,CAASC,CAAT,CAAkBC,CAAlB,CAA6B,CAyDtC,IAAIC,EAAgB,gDAAA,MAAA,CAAA,GAAA,CAApB,CAEIC,EAAcA,QAAQ,CAACC,CAAD,CAAOC,CAAP,CAAsB,CAC9C,GAAiD,EAAjD,GAAIA,CAAAC,QAAA,CAAsBF,CAAA,CAAK,CAAL,CAAAG,SAAtB,CAAJ,CACE,MAAO,CAAA,CAFqC,CAR7BP,EAAAQ,OAAA,CAAe,QAAf,CAAyB,CAAC,IAAD,CAAzB,CAAAC,SAAAC,CACc,OADdA,CAkCnBC,QAAsB,EAAG,CAwCvBC,QAASA,EAAS,CAACC,CAAD,CAAWC,CAAX,CAAqBZ,CAArB,CAAoCa,CAApC,CAA4C,CAC5D,MAAO,SAAQ,CAACC,CAAD,CAAQZ,CAAR,CAAca,CAAd,CAAoB,CACjC,IAAIC,EAAgBD,CAAAE,WAAA,CAAgBL,CAAhB,CAChB,EAAAM,CAAA,CAAOF,CAAP,CAAJ,EAA8Bf,CAAA,CAAYC,CAAZ,CAAkBF,CAAlB,CAA9B,EAAmEe,CAAA,CAAKC,CAAL,CAAnE,EACEF,CAAAK,OAAA,CAAaJ,CAAA,CAAKJ,CAAL,CAAb,CAA6B,QAAQ,CAACS,CAAD,CAAU,CAE7CA,CAAA,CAAUP,CAAA,CAAS,CAACO,CAAV,CAAoB,CAAEA,CAAAA,CAChClB,EAAAa,KAAA,CAAUH,CAAV,CAAoBQ,CAApB,CAH6C,CAA/C,CAH+B,CADyB,CAvC9D,IAAIF,EAAS,CACXG,WAAY,CAAA,CADD,CAEXC,YAAa,CAAA,CAFF,CAGXC,aAAc,CAAA,CAHH,CAIXC,aAAc,CAAA,CAJH,CAKXC,YAAa,CAAA,CALF,CAMXC,cAAe,CAAA,CANJ,CAOXC,UAAW,CAAA,CAPA,CAQXC,SAAU,CAAA,CARC,CASXC,aAAc,CAAA,CATH,CAUXC,iBAAkB,CAAA,CAVP,CAmCb;IAAAZ,OAAA,CAAca,QAAQ,CAACC,CAAD,CAAY,CAChCd,CAAA,CAASpB,CAAAmC,OAAA,CAAef,CAAf,CAAuBc,CAAvB,CADuB,CA8DlC,KAAAE,KAAA,CAAYC,QAAQ,EAAG,CACrB,MAAO,CACLjB,OAAQA,QAAQ,CAACkB,CAAD,CAAM,CACpB,MAAOlB,EAAA,CAAOkB,CAAP,CADa,CADjB,CAILC,YAAa3B,CAJR,CADc,CAlGA,CAlCNF,CA+InB8B,UAAA,CAAuB,QAAvB,CAAiC,CAAC,OAAD,CAAU,QAAQ,CAACC,CAAD,CAAQ,CACzD,MAAOA,EAAAF,YAAA,CAAkB,QAAlB,CAA4B,aAA5B,CAA2C,EAA3C,CAA+C,CAAA,CAA/C,CADkD,CAA1B,CAAjC,CAAAC,UAAA,CAGW,QAHX,CAGqB,CAAC,OAAD,CAAU,QAAQ,CAACC,CAAD,CAAQ,CAC7C,MAAOA,EAAAF,YAAA,CAAkB,QAAlB,CAA4B,aAA5B,CAA2C,EAA3C,CAA+C,CAAA,CAA/C,CADsC,CAA1B,CAHrB,CAAAC,UAAA,CAMW,SANX,CAMsB,CAAC,OAAD,CAAU,QAAQ,CAACC,CAAD,CAAQ,CAE9CC,QAASA,EAAgB,CAACzB,CAAD,CAAO0B,CAAP,CAAuBvC,CAAvB,CAA6B,CACpD,MAAOqC,EAAArB,OAAA,CAAauB,CAAb,CAAP,EAAuC,CAACvC,CAAAa,KAAA,CAAUA,CAAV,CADY,CAItD2B,QAASA,EAAgB,CAACC,CAAD,CAAOzC,CAAP,CAAa,CACpC,MAAO,CAACA,CAAAa,KAAA,CAAU,MAAV,CAAR,EAA8Bb,CAAAa,KAAA,CAAU,MAAV,CAA9B,GAAoD4B,CAApD,EAAmF,OAAnF,GAA8DzC,CAAA,CAAK,CAAL,CAAAG,SAD1B,CAItCuC,QAASA,EAAQ,CAAC7B,CAAD,CAAOb,CAAP,CAAa,CAAA,IACxB2C;AAAO9B,CAAA8B,KADiB,CAExBF,EAAO5B,CAAA4B,KAEX,OAA2B,UAApB,IAAEE,CAAF,EAAUF,CAAV,GAA2C,kBAA3C,GAAkCA,CAAlC,CAAiE,UAAjE,CACoB,OAApB,IAAEE,CAAF,EAAUF,CAAV,GAA2C,eAA3C,GAAkCA,CAAlC,CAA8D,OAA9D,CACU,OAAV,GAACE,CAAD,EAA2C,aAA3C,GAAkCF,CAAlC,EAAqE,QAArE,GAA4DA,CAA5D,CAAiF,OAAjF,CACmB,SAAnB,IAACE,CAAD,EAASF,CAAT,GAAuD,UAAvD,GAAkCzC,CAAA,CAAK,CAAL,CAAAG,SAAlC,CAAoE,WAApE,CAAkF,EAP7D,CAU9B,MAAO,CACLyC,SAAU,GADL,CAELC,QAAS,UAFJ,CAGLC,SAAU,GAHL,CAILC,QAASA,QAAQ,CAAC/C,CAAD,CAAOa,CAAP,CAAa,CAC5B,IAAImC,EAAQN,CAAA,CAAS7B,CAAT,CAAeb,CAAf,CAEZ,OAAO,CACLiD,IAAKA,QAAQ,CAACrC,CAAD,CAAQZ,CAAR,CAAca,CAAd,CAAoBqC,CAApB,CAA6B,CAC1B,UAAd,GAAIF,CAAJ,EAA0C,UAA1C,GAA4BnC,CAAA8B,KAA5B,GAEEO,CAAAC,SAFF,CAEqBC,QAAQ,CAACC,CAAD,CAAQ,CACjC,MAAiB,CAAA,CAAjB,GAAOA,CAD0B,CAFrC,CADwC,CADrC,CASLC,KAAMA,QAAQ,CAAC1C,CAAD,CAAQZ,CAAR,CAAca,CAAd,CAAoBqC,CAApB,CAA6B,CAIzCK,QAASA,EAAqB,EAAG,CAC/B,MAAOL,EAAAM,YADwB,CAIjCC,QAASA,EAAgB,EAAG,CAC1B,MAAIC,EAAJ;CACEA,CACOC,CADS,CAAA,CACTA,CAAAA,QAA4B,CAACC,CAAD,CAAS,CACtC1C,CAAAA,CAAWL,CAAAwC,MAAXnC,EAAyBgC,CAAAW,WAC7B7D,EAAAa,KAAA,CAAU,cAAV,CAA0BK,CAA1B,CACAlB,EAAAa,KAAA,CAAU,UAAV,CAAsB,CAAtB,CAA0B,CAACK,CAA3B,CAH0C,CAF9C,EAQSyC,QAA4B,CAACC,CAAD,CAAS,CAC1C5D,CAAAa,KAAA,CAAU,cAAV,CAA2BA,CAAAwC,MAA3B,EAAyCH,CAAAW,WAAzC,CAD0C,CATpB,CAe5BC,QAASA,EAAsB,EAAG,CAChC9D,CAAAa,KAAA,CAAU,cAAV,CAA0B,CAACqC,CAAAC,SAAA,CAAiBD,CAAAW,WAAjB,CAA3B,CADgC,CAtBlC,IAAIH,EAAgBpB,CAAA,CAAiB,UAAjB,CAA6B,UAA7B,CAAyCtC,CAAzC,CAAhB0D,EACqB,CAAC3D,CAAA,CAAYC,CAAZ,CAAkBF,CAAlB,CAyB1B,QAAQkD,CAAR,EACE,KAAK,OAAL,CACA,KAAK,UAAL,CACMR,CAAA,CAAiBQ,CAAjB,CAAwBhD,CAAxB,CAAJ,EACEA,CAAAa,KAAA,CAAU,MAAV,CAAkBmC,CAAlB,CAEEV,EAAA,CAAiB,cAAjB,CAAiC,aAAjC,CAAgDtC,CAAhD,CAAJ,EACEY,CAAAK,OAAA,CAAasC,CAAb,CAA8C,OAAV,GAAAP,CAAA,CAChCS,CAAA,EADgC,CACXK,CADzB,CAGEJ,EAAJ,EACE1D,CAAAa,KAAA,CAAU,UAAV,CAAsB,CAAtB,CAEF,MACF,MAAK,OAAL,CACM2B,CAAA,CAAiBQ,CAAjB,CAAwBhD,CAAxB,CAAJ,EACEA,CAAAa,KAAA,CAAU,MAAV,CAAkB,QAAlB,CAEF,IAAIwB,CAAArB,OAAA,CAAa,WAAb,CAAJ,CAA+B,CAC7B,IAAI+C,EAAoB,CAAC/D,CAAAa,KAAA,CAAU,eAAV,CAArBkD;CACClD,CAAAmD,eAAA,CAAoB,KAApB,CADDD,EAC+BlD,CAAAmD,eAAA,CAAoB,OAApB,CAD/BD,CAAJ,CAEIE,EAAoB,CAACjE,CAAAa,KAAA,CAAU,eAAV,CAArBoD,GACCpD,CAAAmD,eAAA,CAAoB,KAApB,CADDC,EAC+BpD,CAAAmD,eAAA,CAAoB,OAApB,CAD/BC,CAFJ,CAIIC,EAAoB,CAAClE,CAAAa,KAAA,CAAU,eAAV,CAErBkD,EAAJ,EACElD,CAAAsD,SAAA,CAAc,KAAd,CAAqBC,QAA+B,CAACR,CAAD,CAAS,CAC3D5D,CAAAa,KAAA,CAAU,eAAV,CAA2B+C,CAA3B,CAD2D,CAA7D,CAIEK,EAAJ,EACEpD,CAAAsD,SAAA,CAAc,KAAd,CAAqBC,QAA+B,CAACR,CAAD,CAAS,CAC3D5D,CAAAa,KAAA,CAAU,eAAV,CAA2B+C,CAA3B,CAD2D,CAA7D,CAIEM,EAAJ,EACEtD,CAAAK,OAAA,CAAasC,CAAb,CAAoCc,QAA+B,CAACT,CAAD,CAAS,CAC1E5D,CAAAa,KAAA,CAAU,eAAV,CAA2B+C,CAA3B,CAD0E,CAA5E,CAlB2B,CAuB3BF,CAAJ,EACE1D,CAAAa,KAAA,CAAU,UAAV,CAAsB,CAAtB,CAEF,MACF,MAAK,WAAL,CACMyB,CAAA,CAAiB,gBAAjB,CAAmC,eAAnC,CAAoDtC,CAApD,CAAJ,EACEA,CAAAa,KAAA,CAAU,gBAAV,CAA4B,CAAA,CAA5B,CA/CN,CAoDIqC,CAAAoB,YAAAC,SAAJ,EAAoCjC,CAAA,CAAiB,eAAjB,CAAkC,cAAlC;AAAkDtC,CAAlD,CAApC,EACEY,CAAAK,OAAA,CAAauD,QAA4B,EAAG,CAC1C,MAAOtB,EAAAuB,OAAAF,SADmC,CAA5C,CAEGG,QAA+B,CAACd,CAAD,CAAS,CACzC5D,CAAAa,KAAA,CAAU,eAAV,CAA2B,CAAE+C,CAAAA,CAA7B,CADyC,CAF3C,CAOEtB,EAAA,CAAiB,cAAjB,CAAiC,aAAjC,CAAgDtC,CAAhD,CAAJ,EACEY,CAAAK,OAAA,CAAa0D,QAA2B,EAAG,CACzC,MAAOzB,EAAA0B,SADkC,CAA3C,CAEGC,QAA8B,CAACjB,CAAD,CAAS,CACxC5D,CAAAa,KAAA,CAAU,cAAV,CAA0B,CAAE+C,CAAAA,CAA5B,CADwC,CAF1C,CAxFuC,CATtC,CAHqB,CAJzB,CApBuC,CAA1B,CANtB,CAAAxB,UAAA,CA6IW,YA7IX,CA6IyB,CAAC,OAAD,CAAU,QAAQ,CAACC,CAAD,CAAQ,CACjD,MAAOA,EAAAF,YAAA,CAAkB,YAAlB,CAAgC,eAAhC,CAAiD,EAAjD,CAD0C,CAA1B,CA7IzB,CAAAC,UAAA,CAgJW,YAhJX,CAgJyB,QAAQ,EAAG,CAClC,MAAO,CACLQ,SAAU,GADL,CAELC,QAAS,aAFJ,CAGLiC,KAAMA,QAAQ,CAAClE,CAAD,CAAQZ,CAAR,CAAca,CAAd,CAAoBkE,CAApB,CAAgC,CACvC/E,CAAAa,KAAA,CAAU,WAAV,CAAL,EACEb,CAAAa,KAAA,CAAU,WAAV,CAAuB,WAAvB,CAF0C,CAHzC,CAD2B,CAhJpC,CAAAuB,UAAA,CA2JW,SA3JX,CA2JqB,CAAC,OAAD,CAAU,QAAV;AAAoB,QAAQ,CAACC,CAAD,CAAQ2C,CAAR,CAAgB,CAC/D,MAAO,CACLpC,SAAU,GADL,CAELG,QAASA,QAAQ,CAAC/C,CAAD,CAAOa,CAAP,CAAa,CAC5B,IAAIoE,EAAKD,CAAA,CAAOnE,CAAAqE,QAAP,CAAyC,IAAzC,CAAqE,CAAA,CAArE,CACT,OAAO,SAAQ,CAACtE,CAAD,CAAQZ,CAAR,CAAca,CAAd,CAAoB,CAEjC,GAAK,CAAAd,CAAA,CAAYC,CAAZ,CAAkBF,CAAlB,CAAL,GAEMuC,CAAArB,OAAA,CAAa,kBAAb,CAQA,EARqC,CAAAhB,CAAAa,KAAA,CAAU,MAAV,CAQrC,EAPFb,CAAAa,KAAA,CAAU,MAAV,CAAkB,QAAlB,CAOE,CAJAwB,CAAArB,OAAA,CAAa,UAAb,CAIA,EAJ6B,CAAAhB,CAAAa,KAAA,CAAU,UAAV,CAI7B,EAHFb,CAAAa,KAAA,CAAU,UAAV,CAAsB,CAAtB,CAGE,CAAAwB,CAAArB,OAAA,CAAa,cAAb,CAAA,EAAiCmE,CAAAtE,CAAAsE,WAVvC,EAWInF,CAAAoF,GAAA,CAAQ,UAAR,CAAoB,QAAQ,CAACC,CAAD,CAAQ,CAMlCC,QAASA,EAAQ,EAAG,CAClBL,CAAA,CAAGrE,CAAH,CAAU,CAAE2E,OAAQF,CAAV,CAAV,CADkB,CALpB,IAAIG,EAAUH,CAAAI,MAAVD,EAAyBH,CAAAG,QACb,GAAhB,GAAIA,CAAJ,EAAkC,EAAlC,GAAsBA,CAAtB,EACE5E,CAAA8E,OAAA,CAAaJ,CAAb,CAHgC,CAApC,CAb6B,CAFP,CAFzB,CADwD,CAA5C,CA3JrB,CAAAlD,UAAA,CA6LW,YA7LX,CA6LyB,CAAC,OAAD,CAAU,QAAQ,CAACC,CAAD,CAAQ,CACjD,MAAO,SAAQ,CAACzB,CAAD,CAAQZ,CAAR,CAAca,CAAd,CAAoB,CAC7B,CAAAwB,CAAArB,OAAA,CAAa,UAAb,CAAJ;AAAiChB,CAAAa,KAAA,CAAU,UAAV,CAAjC,EAA2Dd,CAAA,CAAYC,CAAZ,CAAkBF,CAAlB,CAA3D,EACEE,CAAAa,KAAA,CAAU,UAAV,CAAsB,CAAtB,CAF+B,CADc,CAA1B,CA7LzB,CAlMsC,CAArC,CAAD,CAwYGlB,MAxYH,CAwYWA,MAAAC,QAxYX;", +"sources":["angular-aria.js"], +"names":["window","angular","undefined","nodeBlackList","isNodeOneOf","elem","nodeTypeArray","indexOf","nodeName","module","provider","ngAriaModule","$AriaProvider","watchExpr","attrName","ariaAttr","negate","scope","attr","ariaCamelName","$normalize","config","$watch","boolVal","ariaHidden","ariaChecked","ariaDisabled","ariaRequired","ariaInvalid","ariaMultiline","ariaValue","tabindex","bindKeypress","bindRoleForClick","this.config","newConfig","extend","$get","this.$get","key","$$watchExpr","directive","$aria","shouldAttachAttr","normalizedAttr","shouldAttachRole","role","getShape","type","restrict","require","priority","compile","shape","pre","ngModel","$isEmpty","ngModel.$isEmpty","value","post","ngAriaWatchModelValue","$modelValue","getRadioReaction","needsTabIndex","ngAriaRadioReaction","newVal","$viewValue","ngAriaCheckboxReaction","needsAriaValuemin","hasOwnProperty","needsAriaValuemax","needsAriaValuenow","$observe","ngAriaValueMinReaction","ngAriaValueNowReaction","$validators","required","ngAriaRequiredWatch","$error","ngAriaRequiredReaction","ngAriaInvalidWatch","$invalid","ngAriaInvalidReaction","link","ngMessages","$parse","fn","ngClick","ngKeypress","on","event","callback","$event","keyCode","which","$apply"] +} diff --git a/1.4.10/angular-cookies.js b/1.4.10/angular-cookies.js new file mode 100644 index 0000000000..beb8b62055 --- /dev/null +++ b/1.4.10/angular-cookies.js @@ -0,0 +1,322 @@ +/** + * @license AngularJS v1.4.10 + * (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. + * + * + *
+ * + * 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 is the URL that appears in your `` tag. + * - **domain** - `{string}` - The cookie will be available only for this domain and + * its sub-domains. For security reasons the user agent will not accept the cookie + * if the current domain is not a sub-domain of this domain or equal to it. + * - **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}` - If `true`, then the cookie will only be available through a + * secured connection. + * + * Note: By default, the address that appears in your `` tag will be used as the path. + * This is important so that cookies will be visible for all routes when 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. + * + *
+ * 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. + *
+ * + * 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. + * + *
+ * **Note:** The $cookieStore service is **deprecated**. + * Please use the {@link ngCookies.$cookies `$cookies`} service instead. + *
+ * + * @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); diff --git a/1.4.10/angular-cookies.min.js b/1.4.10/angular-cookies.min.js new file mode 100644 index 0000000000..b96469701b --- /dev/null +++ b/1.4.10/angular-cookies.min.js @@ -0,0 +1,9 @@ +/* + AngularJS v1.4.10 + (c) 2010-2015 Google, Inc. http://angularjs.org + License: MIT +*/ +(function(p,c,n){'use strict';function l(b,a,g){var d=g.baseHref(),k=b[0];return function(b,e,f){var g,h;f=f||{};h=f.expires;g=c.isDefined(f.path)?f.path:d;c.isUndefined(e)&&(h="Thu, 01 Jan 1970 00:00:00 GMT",e="");c.isString(h)&&(h=new Date(h));e=encodeURIComponent(b)+"="+encodeURIComponent(e);e=e+(g?";path="+g:"")+(f.domain?";domain="+f.domain:"");e+=h?";expires="+h.toUTCString():"";e+=f.secure?";secure":"";f=e.length+1;4096 4096 bytes)!");k.cookie=e}}c.module("ngCookies",["ng"]).provider("$cookies",[function(){var b=this.defaults={};this.$get=["$$cookieReader","$$cookieWriter",function(a,g){return{get:function(d){return a()[d]},getObject:function(d){return(d=this.get(d))?c.fromJson(d):d},getAll:function(){return a()},put:function(d,a,m){g(d,a,m?c.extend({},b,m):b)},putObject:function(d,b,a){this.put(d,c.toJson(b),a)},remove:function(a,k){g(a,n,k?c.extend({},b,k):b)}}}]}]);c.module("ngCookies").factory("$cookieStore", +["$cookies",function(b){return{get:function(a){return b.getObject(a)},put:function(a,c){b.putObject(a,c)},remove:function(a){b.remove(a)}}}]);l.$inject=["$document","$log","$browser"];c.module("ngCookies").provider("$$cookieWriter",function(){this.$get=l})})(window,window.angular); +//# sourceMappingURL=angular-cookies.min.js.map diff --git a/1.4.10/angular-cookies.min.js.map b/1.4.10/angular-cookies.min.js.map new file mode 100644 index 0000000000..555b5103b7 --- /dev/null +++ b/1.4.10/angular-cookies.min.js.map @@ -0,0 +1,8 @@ +{ +"version":3, +"file":"angular-cookies.min.js", +"lineCount":8, +"mappings":"A;;;;;aAKC,SAAQ,CAACA,CAAD,CAASC,CAAT,CAAkBC,CAAlB,CAA6B,CA2QtCC,QAASA,EAAc,CAACC,CAAD,CAAYC,CAAZ,CAAkBC,CAAlB,CAA4B,CACjD,IAAIC,EAAaD,CAAAE,SAAA,EAAjB,CACIC,EAAcL,CAAA,CAAU,CAAV,CAmClB,OAAO,SAAQ,CAACM,CAAD,CAAOC,CAAP,CAAcC,CAAd,CAAuB,CAjCW,IAC3CC,CAD2C,CACrCC,CACVF,EAAA,CAgCoDA,CAhCpD,EAAqB,EACrBE,EAAA,CAAUF,CAAAE,QACVD,EAAA,CAAOZ,CAAAc,UAAA,CAAkBH,CAAAC,KAAlB,CAAA,CAAkCD,CAAAC,KAAlC,CAAiDN,CACpDN,EAAAe,YAAA,CAAoBL,CAApB,CAAJ,GACEG,CACA,CADU,+BACV,CAAAH,CAAA,CAAQ,EAFV,CAIIV,EAAAgB,SAAA,CAAiBH,CAAjB,CAAJ,GACEA,CADF,CACY,IAAII,IAAJ,CAASJ,CAAT,CADZ,CAIIK,EAAAA,CAAMC,kBAAA,CAqB6BV,CArB7B,CAANS,CAAiC,GAAjCA,CAAuCC,kBAAA,CAAmBT,CAAnB,CAE3CQ,EAAA,CADAA,CACA,EADON,CAAA,CAAO,QAAP,CAAkBA,CAAlB,CAAyB,EAChC,GAAOD,CAAAS,OAAA,CAAiB,UAAjB,CAA8BT,CAAAS,OAA9B,CAA+C,EAAtD,CACAF,EAAA,EAAOL,CAAA,CAAU,WAAV,CAAwBA,CAAAQ,YAAA,EAAxB,CAAgD,EACvDH,EAAA,EAAOP,CAAAW,OAAA,CAAiB,SAAjB,CAA6B,EAMhCC,EAAAA,CAAeL,CAAAM,OAAfD,CAA4B,CACb,KAAnB,CAAIA,CAAJ,EACEnB,CAAAqB,KAAA,CAAU,UAAV,CASqChB,CATrC,CACE,6DADF;AAEEc,CAFF,CAEiB,iBAFjB,CASFf,EAAAkB,OAAA,CAJOR,CAG6B,CArCW,CAzPnDlB,CAAA2B,OAAA,CAAe,WAAf,CAA4B,CAAC,IAAD,CAA5B,CAAAC,SAAA,CAOY,UAPZ,CAOwB,CAACC,QAAyB,EAAG,CAwBjD,IAAIC,EAAW,IAAAA,SAAXA,CAA2B,EAiC/B,KAAAC,KAAA,CAAY,CAAC,gBAAD,CAAmB,gBAAnB,CAAqC,QAAQ,CAACC,CAAD,CAAiBC,CAAjB,CAAiC,CACxF,MAAO,CAWLC,IAAKA,QAAQ,CAACC,CAAD,CAAM,CACjB,MAAOH,EAAA,EAAA,CAAiBG,CAAjB,CADU,CAXd,CAyBLC,UAAWA,QAAQ,CAACD,CAAD,CAAM,CAEvB,MAAO,CADHzB,CACG,CADK,IAAAwB,IAAA,CAASC,CAAT,CACL,EAAQnC,CAAAqC,SAAA,CAAiB3B,CAAjB,CAAR,CAAkCA,CAFlB,CAzBpB,CAuCL4B,OAAQA,QAAQ,EAAG,CACjB,MAAON,EAAA,EADU,CAvCd,CAuDLO,IAAKA,QAAQ,CAACJ,CAAD,CAAMzB,CAAN,CAAaC,CAAb,CAAsB,CACjCsB,CAAA,CAAeE,CAAf,CAAoBzB,CAApB,CAAuCC,CAvFpC,CAAUX,CAAAwC,OAAA,CAAe,EAAf,CAAmBV,CAAnB,CAuF0BnB,CAvF1B,CAAV,CAAkDmB,CAuFrD,CADiC,CAvD9B,CAuELW,UAAWA,QAAQ,CAACN,CAAD,CAAMzB,CAAN,CAAaC,CAAb,CAAsB,CACvC,IAAA4B,IAAA,CAASJ,CAAT,CAAcnC,CAAA0C,OAAA,CAAehC,CAAf,CAAd,CAAqCC,CAArC,CADuC,CAvEpC,CAsFLgC,OAAQA,QAAQ,CAACR,CAAD,CAAMxB,CAAN,CAAe,CAC7BsB,CAAA,CAAeE,CAAf,CAAoBlC,CAApB,CAA2CU,CAtHxC,CAAUX,CAAAwC,OAAA,CAAe,EAAf,CAAmBV,CAAnB,CAsH8BnB,CAtH9B,CAAV,CAAkDmB,CAsHrD,CAD6B,CAtF1B,CADiF,CAA9E,CAzDqC,CAA7B,CAPxB,CA8JA9B,EAAA2B,OAAA,CAAe,WAAf,CAAAiB,QAAA,CAiCS,cAjCT;AAiCyB,CAAC,UAAD,CAAa,QAAQ,CAACC,CAAD,CAAW,CAErD,MAAO,CAWLX,IAAKA,QAAQ,CAACC,CAAD,CAAM,CACjB,MAAOU,EAAAT,UAAA,CAAmBD,CAAnB,CADU,CAXd,CAyBLI,IAAKA,QAAQ,CAACJ,CAAD,CAAMzB,CAAN,CAAa,CACxBmC,CAAAJ,UAAA,CAAmBN,CAAnB,CAAwBzB,CAAxB,CADwB,CAzBrB,CAsCLiC,OAAQA,QAAQ,CAACR,CAAD,CAAM,CACpBU,CAAAF,OAAA,CAAgBR,CAAhB,CADoB,CAtCjB,CAF8C,CAAhC,CAjCzB,CAqIAjC,EAAA4C,QAAA,CAAyB,CAAC,WAAD,CAAc,MAAd,CAAsB,UAAtB,CAEzB9C,EAAA2B,OAAA,CAAe,WAAf,CAAAC,SAAA,CAAqC,gBAArC,CAAuDmB,QAA+B,EAAG,CACvF,IAAAhB,KAAA,CAAY7B,CAD2E,CAAzF,CAvTsC,CAArC,CAAD,CA4TGH,MA5TH,CA4TWA,MAAAC,QA5TX;", +"sources":["angular-cookies.js"], +"names":["window","angular","undefined","$$CookieWriter","$document","$log","$browser","cookiePath","baseHref","rawDocument","name","value","options","path","expires","isDefined","isUndefined","isString","Date","str","encodeURIComponent","domain","toUTCString","secure","cookieLength","length","warn","cookie","module","provider","$CookiesProvider","defaults","$get","$$cookieReader","$$cookieWriter","get","key","getObject","fromJson","getAll","put","extend","putObject","toJson","remove","factory","$cookies","$inject","$$CookieWriterProvider"] +} diff --git a/1.4.10/angular-csp.css b/1.4.10/angular-csp.css new file mode 100644 index 0000000000..f3cd926cb3 --- /dev/null +++ b/1.4.10/angular-csp.css @@ -0,0 +1,21 @@ +/* Include this file in your html if you are using the CSP mode. */ + +@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; +} diff --git a/1.4.10/angular-loader.js b/1.4.10/angular-loader.js new file mode 100644 index 0000000000..f90a5c7d60 --- /dev/null +++ b/1.4.10/angular-loader.js @@ -0,0 +1,471 @@ +/** + * @license AngularJS v1.4.10 + * (c) 2010-2015 Google, Inc. http://angularjs.org + * License: MIT + */ + +(function() {'use strict'; + function isFunction(value) {return typeof value === 'function';}; + +/* 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; +} + +/** + * @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.10/' + + (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); + }; +} + +/** + * @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.} */ + 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.=} 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 {angular.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.>} */ + var invokeQueue = []; + + /** @type {!Array.} */ + var configBlocks = []; + + /** @type {!Array.} */ + 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()}. + * + *
+ * **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`). + *
+ */ + 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; + }; + } + }); + }; + }); + +} + +setupModuleLoader(window); +})(window); + +/** + * Closure compiler type information + * + * @typedef { { + * requires: !Array., + * invokeQueue: !Array.>, + * + * service: function(string, Function):angular.Module, + * factory: function(string, Function):angular.Module, + * value: function(string, *):angular.Module, + * + * filter: function(string, Function):angular.Module, + * + * init: function(Function):angular.Module + * } } + */ +angular.Module; + diff --git a/1.4.10/angular-loader.min.js b/1.4.10/angular-loader.min.js new file mode 100644 index 0000000000..e6557b7323 --- /dev/null +++ b/1.4.10/angular-loader.min.js @@ -0,0 +1,10 @@ +/* + AngularJS v1.4.10 + (c) 2010-2015 Google, Inc. http://angularjs.org + License: MIT +*/ +(function(){'use strict';function d(b){return function(){var a=arguments[0],e;e="["+(b?b+":":"")+a+"] http://errors.angularjs.org/1.4.10/"+(b?b+"/":"")+a;for(a=1;a= line.length) { + index -= line.length; + } else { + return { line: i + 1, column: index + 1 }; + } + } +} +var PARSE_CACHE_FOR_TEXT_LITERALS = Object.create(null); + +function parseTextLiteral(text) { + var cachedFn = PARSE_CACHE_FOR_TEXT_LITERALS[text]; + if (cachedFn != null) { + return cachedFn; + } + function parsedFn(context) { return text; } + parsedFn['$$watchDelegate'] = function watchDelegate(scope, listener, objectEquality) { + var unwatch = scope['$watch'](noop, + function textLiteralWatcher() { + if (isFunction(listener)) { listener.call(null, text, text, scope); } + unwatch(); + }, + objectEquality); + return unwatch; + }; + PARSE_CACHE_FOR_TEXT_LITERALS[text] = parsedFn; + parsedFn['exp'] = text; // Needed to pretend to be $interpolate for tests copied from interpolateSpec.js + parsedFn['expressions'] = []; // Require this to call $compile.$$addBindingInfo() which allows Protractor to find elements by binding. + return parsedFn; +} + +function subtractOffset(expressionFn, offset) { + if (offset === 0) { + return expressionFn; + } + function minusOffset(value) { + return (value == void 0) ? value : value - offset; + } + function parsedFn(context) { return minusOffset(expressionFn(context)); } + var unwatch; + parsedFn['$$watchDelegate'] = function watchDelegate(scope, listener, objectEquality) { + unwatch = scope['$watch'](expressionFn, + function pluralExpressionWatchListener(newValue, oldValue) { + if (isFunction(listener)) { listener.call(null, minusOffset(newValue), minusOffset(oldValue), scope); } + }, + objectEquality); + return unwatch; + }; + return parsedFn; +} + +// NOTE: ADVANCED_OPTIMIZATIONS mode. +// +// This file is compiled with Closure compiler's ADVANCED_OPTIMIZATIONS flag! Be wary of using +// constructs incompatible with that mode. + +/* global $interpolateMinErr: false */ +/* global isFunction: false */ +/* global noop: false */ + +/** + * @constructor + * @private + */ +function MessageSelectorBase(expressionFn, choices) { + var self = this; + this.expressionFn = expressionFn; + this.choices = choices; + if (choices["other"] === void 0) { + throw $interpolateMinErr('reqother', '“other” is a required option.'); + } + this.parsedFn = function(context) { return self.getResult(context); }; + this.parsedFn['$$watchDelegate'] = function $$watchDelegate(scope, listener, objectEquality) { + return self.watchDelegate(scope, listener, objectEquality); + }; + this.parsedFn['exp'] = expressionFn['exp']; + this.parsedFn['expressions'] = expressionFn['expressions']; +} + +MessageSelectorBase.prototype.getMessageFn = function getMessageFn(value) { + return this.choices[this.categorizeValue(value)]; +}; + +MessageSelectorBase.prototype.getResult = function getResult(context) { + return this.getMessageFn(this.expressionFn(context))(context); +}; + +MessageSelectorBase.prototype.watchDelegate = function watchDelegate(scope, listener, objectEquality) { + var watchers = new MessageSelectorWatchers(this, scope, listener, objectEquality); + return function() { watchers.cancelWatch(); }; +}; + +/** + * @constructor + * @private + */ +function MessageSelectorWatchers(msgSelector, scope, listener, objectEquality) { + var self = this; + this.scope = scope; + this.msgSelector = msgSelector; + this.listener = listener; + this.objectEquality = objectEquality; + this.lastMessage = void 0; + this.messageFnWatcher = noop; + var expressionFnListener = function(newValue, oldValue) { return self.expressionFnListener(newValue, oldValue); }; + this.expressionFnWatcher = scope['$watch'](msgSelector.expressionFn, expressionFnListener, objectEquality); +} + +MessageSelectorWatchers.prototype.expressionFnListener = function expressionFnListener(newValue, oldValue) { + var self = this; + this.messageFnWatcher(); + var messageFnListener = function(newMessage, oldMessage) { return self.messageFnListener(newMessage, oldMessage); }; + var messageFn = this.msgSelector.getMessageFn(newValue); + this.messageFnWatcher = this.scope['$watch'](messageFn, messageFnListener, this.objectEquality); +}; + +MessageSelectorWatchers.prototype.messageFnListener = function messageFnListener(newMessage, oldMessage) { + if (isFunction(this.listener)) { + this.listener.call(null, newMessage, newMessage === oldMessage ? newMessage : this.lastMessage, this.scope); + } + this.lastMessage = newMessage; +}; + +MessageSelectorWatchers.prototype.cancelWatch = function cancelWatch() { + this.expressionFnWatcher(); + this.messageFnWatcher(); +}; + +/** + * @constructor + * @extends MessageSelectorBase + * @private + */ +function SelectMessage(expressionFn, choices) { + MessageSelectorBase.call(this, expressionFn, choices); +} + +function SelectMessageProto() {} +SelectMessageProto.prototype = MessageSelectorBase.prototype; + +SelectMessage.prototype = new SelectMessageProto(); +SelectMessage.prototype.categorizeValue = function categorizeSelectValue(value) { + return (this.choices[value] !== void 0) ? value : "other"; +}; + +/** + * @constructor + * @extends MessageSelectorBase + * @private + */ +function PluralMessage(expressionFn, choices, offset, pluralCat) { + MessageSelectorBase.call(this, expressionFn, choices); + this.offset = offset; + this.pluralCat = pluralCat; +} + +function PluralMessageProto() {} +PluralMessageProto.prototype = MessageSelectorBase.prototype; + +PluralMessage.prototype = new PluralMessageProto(); +PluralMessage.prototype.categorizeValue = function categorizePluralValue(value) { + if (isNaN(value)) { + return "other"; + } else if (this.choices[value] !== void 0) { + return value; + } else { + var category = this.pluralCat(value - this.offset); + return (this.choices[category] !== void 0) ? category : "other"; + } +}; + +// NOTE: ADVANCED_OPTIMIZATIONS mode. +// +// This file is compiled with Closure compiler's ADVANCED_OPTIMIZATIONS flag! Be wary of using +// constructs incompatible with that mode. + +/* global $interpolateMinErr: false */ +/* global isFunction: false */ +/* global parseTextLiteral: false */ + +/** + * @constructor + * @private + */ +function InterpolationParts(trustedContext, allOrNothing) { + this.trustedContext = trustedContext; + this.allOrNothing = allOrNothing; + this.textParts = []; + this.expressionFns = []; + this.expressionIndices = []; + this.partialText = ''; + this.concatParts = null; +} + +InterpolationParts.prototype.flushPartialText = function flushPartialText() { + if (this.partialText) { + if (this.concatParts == null) { + this.textParts.push(this.partialText); + } else { + this.textParts.push(this.concatParts.join('')); + this.concatParts = null; + } + this.partialText = ''; + } +}; + +InterpolationParts.prototype.addText = function addText(text) { + if (text.length) { + if (!this.partialText) { + this.partialText = text; + } else if (this.concatParts) { + this.concatParts.push(text); + } else { + this.concatParts = [this.partialText, text]; + } + } +}; + +InterpolationParts.prototype.addExpressionFn = function addExpressionFn(expressionFn) { + this.flushPartialText(); + this.expressionIndices.push(this.textParts.length); + this.expressionFns.push(expressionFn); + this.textParts.push(''); +}; + +InterpolationParts.prototype.getExpressionValues = function getExpressionValues(context) { + var expressionValues = new Array(this.expressionFns.length); + for (var i = 0; i < this.expressionFns.length; i++) { + expressionValues[i] = this.expressionFns[i](context); + } + return expressionValues; +}; + +InterpolationParts.prototype.getResult = function getResult(expressionValues) { + for (var i = 0; i < this.expressionIndices.length; i++) { + var expressionValue = expressionValues[i]; + if (this.allOrNothing && expressionValue === void 0) return; + this.textParts[this.expressionIndices[i]] = expressionValue; + } + return this.textParts.join(''); +}; + + +InterpolationParts.prototype.toParsedFn = function toParsedFn(mustHaveExpression, originalText) { + var self = this; + this.flushPartialText(); + if (mustHaveExpression && this.expressionFns.length === 0) { + return void 0; + } + if (this.textParts.length === 0) { + return parseTextLiteral(''); + } + if (this.trustedContext && this.textParts.length > 1) { + $interpolateMinErr['throwNoconcat'](originalText); + } + if (this.expressionFns.length === 0) { + if (this.textParts.length != 1) { this.errorInParseLogic(); } + return parseTextLiteral(this.textParts[0]); + } + var parsedFn = function(context) { + return self.getResult(self.getExpressionValues(context)); + }; + parsedFn['$$watchDelegate'] = function $$watchDelegate(scope, listener, objectEquality) { + return self.watchDelegate(scope, listener, objectEquality); + }; + + parsedFn['exp'] = originalText; // Needed to pretend to be $interpolate for tests copied from interpolateSpec.js + parsedFn['expressions'] = new Array(this.expressionFns.length); // Require this to call $compile.$$addBindingInfo() which allows Protractor to find elements by binding. + for (var i = 0; i < this.expressionFns.length; i++) { + parsedFn['expressions'][i] = this.expressionFns[i]['exp']; + } + + return parsedFn; +}; + +InterpolationParts.prototype.watchDelegate = function watchDelegate(scope, listener, objectEquality) { + var watcher = new InterpolationPartsWatcher(this, scope, listener, objectEquality); + return function() { watcher.cancelWatch(); }; +}; + +function InterpolationPartsWatcher(interpolationParts, scope, listener, objectEquality) { + this.interpolationParts = interpolationParts; + this.scope = scope; + this.previousResult = (void 0); + this.listener = listener; + var self = this; + this.expressionFnsWatcher = scope['$watchGroup'](interpolationParts.expressionFns, function(newExpressionValues, oldExpressionValues) { + self.watchListener(newExpressionValues, oldExpressionValues); + }); +} + +InterpolationPartsWatcher.prototype.watchListener = function watchListener(newExpressionValues, oldExpressionValues) { + var result = this.interpolationParts.getResult(newExpressionValues); + if (isFunction(this.listener)) { + this.listener.call(null, result, newExpressionValues === oldExpressionValues ? result : this.previousResult, this.scope); + } + this.previousResult = result; +}; + +InterpolationPartsWatcher.prototype.cancelWatch = function cancelWatch() { + this.expressionFnsWatcher(); +}; + +// NOTE: ADVANCED_OPTIMIZATIONS mode. +// +// This file is compiled with Closure compiler's ADVANCED_OPTIMIZATIONS flag! Be wary of using +// constructs incompatible with that mode. + +/* global $interpolateMinErr: false */ +/* global indexToLineAndColumn: false */ +/* global InterpolationParts: false */ +/* global PluralMessage: false */ +/* global SelectMessage: false */ +/* global subtractOffset: false */ + +// The params src and dst are exactly one of two types: NestedParserState or MessageFormatParser. +// This function is fully optimized by V8. (inspect via IRHydra or --trace-deopt.) +// The idea behind writing it this way is to avoid repeating oneself. This is the ONE place where +// the parser state that is saved/restored when parsing nested mustaches is specified. +function copyNestedParserState(src, dst) { + dst.expressionFn = src.expressionFn; + dst.expressionMinusOffsetFn = src.expressionMinusOffsetFn; + dst.pluralOffset = src.pluralOffset; + dst.choices = src.choices; + dst.choiceKey = src.choiceKey; + dst.interpolationParts = src.interpolationParts; + dst.ruleChoiceKeyword = src.ruleChoiceKeyword; + dst.msgStartIndex = src.msgStartIndex; + dst.expressionStartIndex = src.expressionStartIndex; +} + +function NestedParserState(parser) { + copyNestedParserState(parser, this); +} + +/** + * @constructor + * @private + */ +function MessageFormatParser(text, startIndex, $parse, pluralCat, stringifier, + mustHaveExpression, trustedContext, allOrNothing) { + this.text = text; + this.index = startIndex || 0; + this.$parse = $parse; + this.pluralCat = pluralCat; + this.stringifier = stringifier; + this.mustHaveExpression = !!mustHaveExpression; + this.trustedContext = trustedContext; + this.allOrNothing = !!allOrNothing; + this.expressionFn = null; + this.expressionMinusOffsetFn = null; + this.pluralOffset = null; + this.choices = null; + this.choiceKey = null; + this.interpolationParts = null; + this.msgStartIndex = null; + this.nestedStateStack = []; + this.parsedFn = null; + this.rule = null; + this.ruleStack = null; + this.ruleChoiceKeyword = null; + this.interpNestLevel = null; + this.expressionStartIndex = null; + this.stringStartIndex = null; + this.stringQuote = null; + this.stringInterestsRe = null; + this.angularOperatorStack = null; + this.textPart = null; +} + +// preserve v8 optimization. +var EMPTY_STATE = new NestedParserState(new MessageFormatParser( + /* text= */ '', /* startIndex= */ 0, /* $parse= */ null, /* pluralCat= */ null, /* stringifier= */ null, + /* mustHaveExpression= */ false, /* trustedContext= */ null, /* allOrNothing */ false)); + +MessageFormatParser.prototype.pushState = function pushState() { + this.nestedStateStack.push(new NestedParserState(this)); + copyNestedParserState(EMPTY_STATE, this); +}; + +MessageFormatParser.prototype.popState = function popState() { + if (this.nestedStateStack.length === 0) { + this.errorInParseLogic(); + } + var previousState = this.nestedStateStack.pop(); + copyNestedParserState(previousState, this); +}; + +// Oh my JavaScript! Who knew you couldn't match a regex at a specific +// location in a string but will always search forward?! +// Apparently you'll be growing this ability via the sticky flag (y) in +// ES6. I'll just to work around you for now. +MessageFormatParser.prototype.matchRe = function matchRe(re, search) { + re.lastIndex = this.index; + var match = re.exec(this.text); + if (match != null && (search === true || (match.index == this.index))) { + this.index = re.lastIndex; + return match; + } + return null; +}; + +MessageFormatParser.prototype.searchRe = function searchRe(re) { + return this.matchRe(re, true); +}; + + +MessageFormatParser.prototype.consumeRe = function consumeRe(re) { + // Without the sticky flag, we can't use the .test() method to consume a + // match at the current index. Instead, we'll use the slower .exec() method + // and verify match.index. + return !!this.matchRe(re); +}; + +// Run through our grammar avoiding deeply nested function call chains. +MessageFormatParser.prototype.run = function run(initialRule) { + this.ruleStack = [initialRule]; + do { + this.rule = this.ruleStack.pop(); + while (this.rule) { + this.rule(); + } + this.assertRuleOrNull(this.rule); + } while (this.ruleStack.length > 0); +}; + +MessageFormatParser.prototype.errorInParseLogic = function errorInParseLogic() { + throw $interpolateMinErr('logicbug', + 'The messageformat parser has encountered an internal error. Please file a github issue against the AngularJS project and provide this message text that triggers the bug. Text: “{0}”', + this.text); +}; + +MessageFormatParser.prototype.assertRuleOrNull = function assertRuleOrNull(rule) { + if (rule === void 0) { + this.errorInParseLogic(); + } +}; + +var NEXT_WORD_RE = /\s*(\w+)\s*/g; +MessageFormatParser.prototype.errorExpecting = function errorExpecting() { + // What was wrong with the syntax? Unsupported type, missing comma, or something else? + var match = this.matchRe(NEXT_WORD_RE), position; + if (match == null) { + position = indexToLineAndColumn(this.text, this.index); + throw $interpolateMinErr('reqarg', + 'Expected one of “plural” or “select” at line {0}, column {1} of text “{2}”', + position.line, position.column, this.text); + } + var word = match[1]; + if (word == "select" || word == "plural") { + position = indexToLineAndColumn(this.text, this.index); + throw $interpolateMinErr('reqcomma', + 'Expected a comma after the keyword “{0}” at line {1}, column {2} of text “{3}”', + word, position.line, position.column, this.text); + } else { + position = indexToLineAndColumn(this.text, this.index); + throw $interpolateMinErr('unknarg', + 'Unsupported keyword “{0}” at line {0}, column {1}. Only “plural” and “select” are currently supported. Text: “{3}”', + word, position.line, position.column, this.text); + } +}; + +var STRING_START_RE = /['"]/g; +MessageFormatParser.prototype.ruleString = function ruleString() { + var match = this.matchRe(STRING_START_RE); + if (match == null) { + var position = indexToLineAndColumn(this.text, this.index); + throw $interpolateMinErr('wantstring', + 'Expected the beginning of a string at line {0}, column {1} in text “{2}”', + position.line, position.column, this.text); + } + this.startStringAtMatch(match); +}; + +MessageFormatParser.prototype.startStringAtMatch = function startStringAtMatch(match) { + this.stringStartIndex = match.index; + this.stringQuote = match[0]; + this.stringInterestsRe = this.stringQuote == "'" ? SQUOTED_STRING_INTEREST_RE : DQUOTED_STRING_INTEREST_RE; + this.rule = this.ruleInsideString; +}; + +var SQUOTED_STRING_INTEREST_RE = /\\(?:\\|'|u[0-9A-Fa-f]{4}|x[0-9A-Fa-f]{2}|[0-7]{3}|\r\n|\n|[\s\S])|'/g; +var DQUOTED_STRING_INTEREST_RE = /\\(?:\\|"|u[0-9A-Fa-f]{4}|x[0-9A-Fa-f]{2}|[0-7]{3}|\r\n|\n|[\s\S])|"/g; +MessageFormatParser.prototype.ruleInsideString = function ruleInsideString() { + var match = this.searchRe(this.stringInterestsRe); + if (match == null) { + var position = indexToLineAndColumn(this.text, this.stringStartIndex); + throw $interpolateMinErr('untermstr', + 'The string beginning at line {0}, column {1} is unterminated in text “{2}”', + position.line, position.column, this.text); + } + var chars = match[0]; + if (match == this.stringQuote) { + this.rule = null; + } +}; + +var PLURAL_OR_SELECT_ARG_TYPE_RE = /\s*(plural|select)\s*,\s*/g; +MessageFormatParser.prototype.rulePluralOrSelect = function rulePluralOrSelect() { + var match = this.searchRe(PLURAL_OR_SELECT_ARG_TYPE_RE); + if (match == null) { + this.errorExpecting(); + } + var argType = match[1]; + switch (argType) { + case "plural": this.rule = this.rulePluralStyle; break; + case "select": this.rule = this.ruleSelectStyle; break; + default: this.errorInParseLogic(); + } +}; + +MessageFormatParser.prototype.rulePluralStyle = function rulePluralStyle() { + this.choices = Object.create(null); + this.ruleChoiceKeyword = this.rulePluralValueOrKeyword; + this.rule = this.rulePluralOffset; +}; + +MessageFormatParser.prototype.ruleSelectStyle = function ruleSelectStyle() { + this.choices = Object.create(null); + this.ruleChoiceKeyword = this.ruleSelectKeyword; + this.rule = this.ruleSelectKeyword; +}; + +var NUMBER_RE = /[0]|(?:[1-9][0-9]*)/g; +var PLURAL_OFFSET_RE = new RegExp("\\s*offset\\s*:\\s*(" + NUMBER_RE.source + ")", "g"); + +MessageFormatParser.prototype.rulePluralOffset = function rulePluralOffset() { + var match = this.matchRe(PLURAL_OFFSET_RE); + this.pluralOffset = (match == null) ? 0 : parseInt(match[1], 10); + this.expressionMinusOffsetFn = subtractOffset(this.expressionFn, this.pluralOffset); + this.rule = this.rulePluralValueOrKeyword; +}; + +MessageFormatParser.prototype.assertChoiceKeyIsNew = function assertChoiceKeyIsNew(choiceKey, index) { + if (this.choices[choiceKey] !== void 0) { + var position = indexToLineAndColumn(this.text, index); + throw $interpolateMinErr('dupvalue', + 'The choice “{0}” is specified more than once. Duplicate key is at line {1}, column {2} in text “{3}”', + choiceKey, position.line, position.column, this.text); + } +}; + +var SELECT_KEYWORD = /\s*(\w+)/g; +MessageFormatParser.prototype.ruleSelectKeyword = function ruleSelectKeyword() { + var match = this.matchRe(SELECT_KEYWORD); + if (match == null) { + this.parsedFn = new SelectMessage(this.expressionFn, this.choices).parsedFn; + this.rule = null; + return; + } + this.choiceKey = match[1]; + this.assertChoiceKeyIsNew(this.choiceKey, match.index); + this.rule = this.ruleMessageText; +}; + +var EXPLICIT_VALUE_OR_KEYWORD_RE = new RegExp("\\s*(?:(?:=(" + NUMBER_RE.source + "))|(\\w+))", "g"); +MessageFormatParser.prototype.rulePluralValueOrKeyword = function rulePluralValueOrKeyword() { + var match = this.matchRe(EXPLICIT_VALUE_OR_KEYWORD_RE); + if (match == null) { + this.parsedFn = new PluralMessage(this.expressionFn, this.choices, this.pluralOffset, this.pluralCat).parsedFn; + this.rule = null; + return; + } + if (match[1] != null) { + this.choiceKey = parseInt(match[1], 10); + } else { + this.choiceKey = match[2]; + } + this.assertChoiceKeyIsNew(this.choiceKey, match.index); + this.rule = this.ruleMessageText; +}; + +var BRACE_OPEN_RE = /\s*{/g; +var BRACE_CLOSE_RE = /}/g; +MessageFormatParser.prototype.ruleMessageText = function ruleMessageText() { + if (!this.consumeRe(BRACE_OPEN_RE)) { + var position = indexToLineAndColumn(this.text, this.index); + throw $interpolateMinErr('reqopenbrace', + 'The plural choice “{0}” must be followed by a message in braces at line {1}, column {2} in text “{3}”', + this.choiceKey, position.line, position.column, this.text); + } + this.msgStartIndex = this.index; + this.interpolationParts = new InterpolationParts(this.trustedContext, this.allOrNothing); + this.rule = this.ruleInInterpolationOrMessageText; +}; + +// Note: Since "\" is used as an escape character, don't allow it to be part of the +// startSymbol/endSymbol when I add the feature to allow them to be redefined. +var INTERP_OR_END_MESSAGE_RE = /\\.|{{|}/g; +var INTERP_OR_PLURALVALUE_OR_END_MESSAGE_RE = /\\.|{{|#|}/g; +var ESCAPE_OR_MUSTACHE_BEGIN_RE = /\\.|{{/g; +MessageFormatParser.prototype.advanceInInterpolationOrMessageText = function advanceInInterpolationOrMessageText() { + var currentIndex = this.index, match, re; + if (this.ruleChoiceKeyword == null) { // interpolation + match = this.searchRe(ESCAPE_OR_MUSTACHE_BEGIN_RE); + if (match == null) { // End of interpolation text. Nothing more to process. + this.textPart = this.text.substring(currentIndex); + this.index = this.text.length; + return null; + } + } else { + match = this.searchRe(this.ruleChoiceKeyword == this.rulePluralValueOrKeyword ? + INTERP_OR_PLURALVALUE_OR_END_MESSAGE_RE : INTERP_OR_END_MESSAGE_RE); + if (match == null) { + var position = indexToLineAndColumn(this.text, this.msgStartIndex); + throw $interpolateMinErr('reqendbrace', + 'The plural/select choice “{0}” message starting at line {1}, column {2} does not have an ending closing brace. Text “{3}”', + this.choiceKey, position.line, position.column, this.text); + } + } + // match is non-null. + var token = match[0]; + this.textPart = this.text.substring(currentIndex, match.index); + return token; +}; + +MessageFormatParser.prototype.ruleInInterpolationOrMessageText = function ruleInInterpolationOrMessageText() { + var currentIndex = this.index; + var token = this.advanceInInterpolationOrMessageText(); + if (token == null) { + // End of interpolation text. Nothing more to process. + this.index = this.text.length; + this.interpolationParts.addText(this.text.substring(currentIndex)); + this.rule = null; + return; + } + if (token[0] == "\\") { + // unescape next character and continue + this.interpolationParts.addText(this.textPart + token[1]); + return; + } + this.interpolationParts.addText(this.textPart); + if (token == "{{") { + this.pushState(); + this.ruleStack.push(this.ruleEndMustacheInInterpolationOrMessage); + this.rule = this.ruleEnteredMustache; + } else if (token == "}") { + this.choices[this.choiceKey] = this.interpolationParts.toParsedFn(/*mustHaveExpression=*/false, this.text); + this.rule = this.ruleChoiceKeyword; + } else if (token == "#") { + this.interpolationParts.addExpressionFn(this.expressionMinusOffsetFn); + } else { + this.errorInParseLogic(); + } +}; + +MessageFormatParser.prototype.ruleInterpolate = function ruleInterpolate() { + this.interpolationParts = new InterpolationParts(this.trustedContext, this.allOrNothing); + this.rule = this.ruleInInterpolation; +}; + +MessageFormatParser.prototype.ruleInInterpolation = function ruleInInterpolation() { + var currentIndex = this.index; + var match = this.searchRe(ESCAPE_OR_MUSTACHE_BEGIN_RE); + if (match == null) { + // End of interpolation text. Nothing more to process. + this.index = this.text.length; + this.interpolationParts.addText(this.text.substring(currentIndex)); + this.parsedFn = this.interpolationParts.toParsedFn(this.mustHaveExpression, this.text); + this.rule = null; + return; + } + var token = match[0]; + if (token[0] == "\\") { + // unescape next character and continue + this.interpolationParts.addText(this.text.substring(currentIndex, match.index) + token[1]); + return; + } + this.interpolationParts.addText(this.text.substring(currentIndex, match.index)); + this.pushState(); + this.ruleStack.push(this.ruleInterpolationEndMustache); + this.rule = this.ruleEnteredMustache; +}; + +MessageFormatParser.prototype.ruleInterpolationEndMustache = function ruleInterpolationEndMustache() { + var expressionFn = this.parsedFn; + this.popState(); + this.interpolationParts.addExpressionFn(expressionFn); + this.rule = this.ruleInInterpolation; +}; + +MessageFormatParser.prototype.ruleEnteredMustache = function ruleEnteredMustache() { + this.parsedFn = null; + this.ruleStack.push(this.ruleEndMustache); + this.rule = this.ruleAngularExpression; +}; + +MessageFormatParser.prototype.ruleEndMustacheInInterpolationOrMessage = function ruleEndMustacheInInterpolationOrMessage() { + var expressionFn = this.parsedFn; + this.popState(); + this.interpolationParts.addExpressionFn(expressionFn); + this.rule = this.ruleInInterpolationOrMessageText; +}; + + + +var INTERP_END_RE = /\s*}}/g; +MessageFormatParser.prototype.ruleEndMustache = function ruleEndMustache() { + var match = this.matchRe(INTERP_END_RE); + if (match == null) { + var position = indexToLineAndColumn(this.text, this.index); + throw $interpolateMinErr('reqendinterp', + 'Expecting end of interpolation symbol, “{0}”, at line {1}, column {2} in text “{3}”', + '}}', position.line, position.column, this.text); + } + if (this.parsedFn == null) { + // If we parsed a MessageFormat extension, (e.g. select/plural today, maybe more some other + // day), then the result *has* to be a string and those rules would have already set + // this.parsedFn. If there was no MessageFormat extension, then there is no requirement to + // stringify the result and parsedFn isn't set. We set it here. While we could have set it + // unconditionally when exiting the Angular expression, I intend for us to not just replace + // $interpolate, but also to replace $parse in a future version (so ng-bind can work), and in + // such a case we do not want to unnecessarily stringify something if it's not going to be used + // in a string context. + this.parsedFn = this.$parse(this.expressionFn, this.stringifier); + this.parsedFn['exp'] = this.expressionFn['exp']; // Needed to pretend to be $interpolate for tests copied from interpolateSpec.js + this.parsedFn['expressions'] = this.expressionFn['expressions']; // Require this to call $compile.$$addBindingInfo() which allows Protractor to find elements by binding. + } + this.rule = null; +}; + +MessageFormatParser.prototype.ruleAngularExpression = function ruleAngularExpression() { + this.angularOperatorStack = []; + this.expressionStartIndex = this.index; + this.rule = this.ruleInAngularExpression; +}; + +function getEndOperator(opBegin) { + switch (opBegin) { + case "{": return "}"; + case "[": return "]"; + case "(": return ")"; + default: return null; + } +} + +function getBeginOperator(opEnd) { + switch (opEnd) { + case "}": return "{"; + case "]": return "["; + case ")": return "("; + default: return null; + } +} + +// TODO(chirayu): The interpolation endSymbol must also be accounted for. It +// just so happens that "}" is an operator so it's in the list below. But we +// should support any other type of start/end interpolation symbol. +var INTERESTING_OPERATORS_RE = /[[\]{}()'",]/g; +MessageFormatParser.prototype.ruleInAngularExpression = function ruleInAngularExpression() { + var startIndex = this.index; + var match = this.searchRe(INTERESTING_OPERATORS_RE); + var position; + if (match == null) { + if (this.angularOperatorStack.length === 0) { + // This is the end of the Angular expression so this is actually a + // success. Note that when inside an interpolation, this means we even + // consumed the closing interpolation symbols if they were curlies. This + // is NOT an error at this point but will become an error further up the + // stack when the part that saw the opening curlies is unable to find the + // closing ones. + this.index = this.text.length; + this.expressionFn = this.$parse(this.text.substring(this.expressionStartIndex, this.index)); + // Needed to pretend to be $interpolate for tests copied from interpolateSpec.js + this.expressionFn['exp'] = this.text.substring(this.expressionStartIndex, this.index); + this.expressionFn['expressions'] = this.expressionFn['expressions']; + this.rule = null; + return; + } + var innermostOperator = this.angularOperatorStack[0]; + throw $interpolateMinErr('badexpr', + 'Unexpected end of Angular expression. Expecting operator “{0}” at the end of the text “{1}”', + this.getEndOperator(innermostOperator), this.text); + } + var operator = match[0]; + if (operator == "'" || operator == '"') { + this.ruleStack.push(this.ruleInAngularExpression); + this.startStringAtMatch(match); + return; + } + if (operator == ",") { + if (this.trustedContext) { + position = indexToLineAndColumn(this.text, this.index); + throw $interpolateMinErr('unsafe', + 'Use of select/plural MessageFormat syntax is currently disallowed in a secure context ({0}). At line {1}, column {2} of text “{3}”', + this.trustedContext, position.line, position.column, this.text); + } + // only the top level comma has relevance. + if (this.angularOperatorStack.length === 0) { + // todo: does this need to be trimmed? + this.expressionFn = this.$parse(this.text.substring(this.expressionStartIndex, match.index)); + // Needed to pretend to be $interpolate for tests copied from interpolateSpec.js + this.expressionFn['exp'] = this.text.substring(this.expressionStartIndex, match.index); + this.expressionFn['expressions'] = this.expressionFn['expressions']; + this.rule = null; + this.rule = this.rulePluralOrSelect; + } + return; + } + if (getEndOperator(operator) != null) { + this.angularOperatorStack.unshift(operator); + return; + } + var beginOperator = getBeginOperator(operator); + if (beginOperator == null) { + this.errorInParseLogic(); + } + if (this.angularOperatorStack.length > 0) { + if (beginOperator == this.angularOperatorStack[0]) { + this.angularOperatorStack.shift(); + return; + } + position = indexToLineAndColumn(this.text, this.index); + throw $interpolateMinErr('badexpr', + 'Unexpected operator “{0}” at line {1}, column {2} in text. Was expecting “{3}”. Text: “{4}”', + operator, position.line, position.column, getEndOperator(this.angularOperatorStack[0]), this.text); + } + // We are trying to pop off the operator stack but there really isn't anything to pop off. + this.index = match.index; + this.expressionFn = this.$parse(this.text.substring(this.expressionStartIndex, this.index)); + // Needed to pretend to be $interpolate for tests copied from interpolateSpec.js + this.expressionFn['exp'] = this.text.substring(this.expressionStartIndex, this.index); + this.expressionFn['expressions'] = this.expressionFn['expressions']; + this.rule = null; +}; + +// NOTE: ADVANCED_OPTIMIZATIONS mode. +// +// This file is compiled with Closure compiler's ADVANCED_OPTIMIZATIONS flag! Be wary of using +// constructs incompatible with that mode. + +/* global $interpolateMinErr: false */ +/* global MessageFormatParser: false */ +/* global stringify: false */ + +/** + * @ngdoc service + * @name $$messageFormat + * + * @description + * Angular internal service to recognize MessageFormat extensions in interpolation expressions. + * For more information, see: + * https://docs.google.com/a/google.com/document/d/1pbtW2yvtmFBikfRrJd8VAsabiFkKezmYZ_PbgdjQOVU/edit + * + * ## Example + * + * + * + *
+ *
+ * {{recipients.length, plural, offset:1 + * =0 {{{sender.name}} gave no gifts (\#=#)} + * =1 {{{sender.name}} gave one gift to {{recipients[0].name}} (\#=#)} + * one {{{sender.name}} gave {{recipients[0].name}} and one other person a gift (\#=#)} + * other {{{sender.name}} gave {{recipients[0].name}} and # other people a gift (\#=#)} + * }} + *
+ *
+ * + * + * function Person(name, gender) { + * this.name = name; + * this.gender = gender; + * } + * + * var alice = new Person("Alice", "female"), + * bob = new Person("Bob", "male"), + * charlie = new Person("Charlie", "male"), + * harry = new Person("Harry Potter", "male"); + * + * angular.module('msgFmtExample', ['ngMessageFormat']) + * .controller('AppController', ['$scope', function($scope) { + * $scope.recipients = [alice, bob, charlie]; + * $scope.sender = harry; + * $scope.decreaseRecipients = function() { + * --$scope.recipients.length; + * }; + * }]); + * + * + * + * describe('MessageFormat plural', function() { + * it('should pluralize initial values', function() { + * var messageElem = element(by.binding('recipients.length')), decreaseRecipientsBtn = element(by.id('decreaseRecipients')); + * expect(messageElem.getText()).toEqual('Harry Potter gave Alice and 2 other people a gift (#=2)'); + * decreaseRecipientsBtn.click(); + * expect(messageElem.getText()).toEqual('Harry Potter gave Alice and one other person a gift (#=1)'); + * decreaseRecipientsBtn.click(); + * expect(messageElem.getText()).toEqual('Harry Potter gave one gift to Alice (#=0)'); + * decreaseRecipientsBtn.click(); + * expect(messageElem.getText()).toEqual('Harry Potter gave no gifts (#=-1)'); + * }); + * }); + * + *
+ */ +var $$MessageFormatFactory = ['$parse', '$locale', '$sce', '$exceptionHandler', function $$messageFormat( + $parse, $locale, $sce, $exceptionHandler) { + + function getStringifier(trustedContext, allOrNothing, text) { + return function stringifier(value) { + try { + value = trustedContext ? $sce['getTrusted'](trustedContext, value) : $sce['valueOf'](value); + return allOrNothing && (value === void 0) ? value : stringify(value); + } catch (err) { + $exceptionHandler($interpolateMinErr['interr'](text, err)); + } + }; + } + + function interpolate(text, mustHaveExpression, trustedContext, allOrNothing) { + var stringifier = getStringifier(trustedContext, allOrNothing, text); + var parser = new MessageFormatParser(text, 0, $parse, $locale['pluralCat'], stringifier, + mustHaveExpression, trustedContext, allOrNothing); + parser.run(parser.ruleInterpolate); + return parser.parsedFn; + } + + return { + 'interpolate': interpolate + }; +}]; + +var $$interpolateDecorator = ['$$messageFormat', '$delegate', function $$interpolateDecorator($$messageFormat, $interpolate) { + if ($interpolate['startSymbol']() != "{{" || $interpolate['endSymbol']() != "}}") { + throw $interpolateMinErr('nochgmustache', 'angular-message-format.js currently does not allow you to use custom start and end symbols for interpolation.'); + } + var interpolate = $$messageFormat['interpolate']; + interpolate['startSymbol'] = $interpolate['startSymbol']; + interpolate['endSymbol'] = $interpolate['endSymbol']; + return interpolate; +}]; + + +/** + * @ngdoc module + * @name ngMessageFormat + * @packageName angular-message-format + * @description + */ +var module = window['angular']['module']('ngMessageFormat', ['ng']); +module['factory']('$$messageFormat', $$MessageFormatFactory); +module['config'](['$provide', function($provide) { + $provide['decorator']('$interpolate', $$interpolateDecorator); +}]); + + +})(window, window.angular); diff --git a/1.4.10/angular-message-format.min.js b/1.4.10/angular-message-format.min.js new file mode 100644 index 0000000000..e71cbf271f --- /dev/null +++ b/1.4.10/angular-message-format.min.js @@ -0,0 +1,26 @@ +/* + AngularJS v1.4.10 + (c) 2010-2015 Google, Inc. http://angularjs.org + License: MIT +*/ +(function(h){'use strict';function C(a){if(null==a)return"";switch(typeof a){case "string":return a;case "number":return""+a;default:return D(a)}}function f(a,b){for(var d=a.split(/\n/g),k=0;k=c.length)b-=c.length;else return{h:k+1,f:b+1}}}function t(a){function b(){return a}var d=u[a];if(null!=d)return d;b.$$watchDelegate=function(b,d,c){var e=b.$watch(v,function(){m(d)&&d.call(null,a,a,b);e()},c);return e};u[a]=b;b.exp=a;b.expressions=[];return b}function F(a,b){function d(a){return void 0== +a?a:a-b}function c(b){return d(a(b))}if(0===b)return a;var e;c.$$watchDelegate=function(b,c,k){return e=b.$watch(a,function(a,k){m(c)&&c.call(null,d(a),d(k),b)},k)};return c}function l(a,b){var d=this;this.b=a;this.e=b;if(void 0===b.other)throw e("reqother");this.d=function(a){return d.D(a)};this.d.$$watchDelegate=function(a,b,c){return d.P(a,b,c)};this.d.exp=a.exp;this.d.expressions=a.expressions}function n(a,b,d,c){var e=this;this.scope=b;this.oa=a;this.v=d;this.qa=c;this.U=void 0;this.K=v;this.ka= +b.$watch(a.b,function(a){return e.ja(a)},c)}function p(a,b){l.call(this,a,b)}function w(){}function q(a,b,d,c){l.call(this,a,b);this.offset=d;this.M=c}function x(){}function g(a,b){this.u=a;this.B=b;this.i=[];this.g=[];this.J=[];this.s="";this.q=null}function r(a,b,d){this.c=a;this.scope=b;this.W=void 0;this.v=d;var c=this;this.la=b.$watchGroup(a.g,function(a,b){c.Ea(a,b)})}function s(a,b){b.b=a.b;b.C=a.C;b.w=a.w;b.e=a.e;b.k=a.k;b.c=a.c;b.n=a.n;b.F=a.F;b.l=a.l}function y(a){s(a,this)}function c(a, +b,d,c,e,E,f,g){this.text=a;this.index=b||0;this.A=d;this.M=c;this.Da=e;this.pa=!!E;this.u=f;this.B=!!g;this.F=this.c=this.k=this.e=this.w=this.C=this.b=null;this.L=[];this.G=this.j=this.ca=this.O=this.da=this.l=this.n=this.o=this.a=this.d=null}function z(a){switch(a){case "{":return"}";case "[":return"]";case "(":return")";default:return null}}function G(a){switch(a){case "}":return"{";case "]":return"[";case ")":return"(";default:return null}}var e=h.angular.$interpolateMinErr,v=h.angular.noop,m= +h.angular.isFunction,D=h.angular.toJson,u=Object.create(null);l.prototype.T=function(a){return this.e[this.R(a)]};l.prototype.D=function(a){return this.T(this.b(a))(a)};l.prototype.P=function(a,b,d){var c=new n(this,a,b,d);return function(){c.I()}};n.prototype.ja=function(a){var b=this;this.K();a=this.oa.T(a);this.K=this.scope.$watch(a,function(a,c){return b.na(a,c)},this.qa)};n.prototype.na=function(a,b){m(this.v)&&this.v.call(null,a,a===b?a:this.U,this.scope);this.U=a};n.prototype.I=function(){this.ka(); +this.K()};w.prototype=l.prototype;p.prototype=new w;p.prototype.R=function(a){return void 0!==this.e[a]?a:"other"};x.prototype=l.prototype;q.prototype=new x;q.prototype.R=function(a){if(isNaN(a))return"other";if(void 0!==this.e[a])return a;a=this.M(a-this.offset);return void 0!==this.e[a]?a:"other"};g.prototype.S=function(){this.s&&(null==this.q?this.i.push(this.s):(this.i.push(this.q.join("")),this.q=null),this.s="")};g.prototype.p=function(a){a.length&&(this.s?this.q?this.q.push(a):this.q=[this.s, +a]:this.s=a)};g.prototype.H=function(a){this.S();this.J.push(this.i.length);this.g.push(a);this.i.push("")};g.prototype.ma=function(a){for(var b=Array(this.g.length),d=0;d + * + *
+ *
Please enter a value for this field.
+ *
This field must be a valid email address.
+ *
This field can be at most 15 characters long.
+ *
+ * + * ``` + * + * In order to show error messages corresponding to `myField` we first create an element with an `ngMessages` attribute + * set to the `$error` object owned by the `myField` input in our `myForm` form. + * + * Within this element we then create separate elements for each of the possible errors that `myField` could have. + * The `ngMessage` attribute is used to declare which element(s) will appear for which error - for example, + * setting `ng-message="required"` specifies that this particular element should be displayed when there + * is no value present for the required field `myField` (because the key `required` will be `true` in the object + * `myForm.myField.$error`). + * + * ### Message order + * + * By default, `ngMessages` will only display one message for a particular key/value collection at any time. If more + * than one message (or error) key is currently true, then which message is shown is determined by the order of messages + * in the HTML template code (messages declared first are prioritised). This mechanism means the developer does not have + * to prioritise messages using custom JavaScript code. + * + * Given the following error object for our example (which informs us that the field `myField` currently has both the + * `required` and `email` errors): + * + * ```javascript + * + * myField.$error = { required : true, email: true, maxlength: false }; + * ``` + * The `required` message will be displayed to the user since it appears before the `email` message in the DOM. + * Once the user types a single character, the `required` message will disappear (since the field now has a value) + * but the `email` message will be visible because it is still applicable. + * + * ### Displaying multiple messages at the same time + * + * While `ngMessages` will by default only display one error element at a time, the `ng-messages-multiple` attribute can + * be applied to the `ngMessages` container element to cause it to display all applicable error messages at once: + * + * ```html + * + *
...
+ * + * + * ... + * ``` + * + * ## Reusing and Overriding Messages + * In addition to prioritization, ngMessages also allows for including messages from a remote or an inline + * template. This allows for generic collection of messages to be reused across multiple parts of an + * application. + * + * ```html + * + * + *
+ *
+ *
+ * ``` + * + * However, including generic messages may not be useful enough to match all input fields, therefore, + * `ngMessages` provides the ability to override messages defined in the remote template by redefining + * them within the directive container. + * + * ```html + * + * + * + *
+ * + * + *
+ * + *
You did not enter your email address
+ * + * + *
Your email address is invalid
+ * + * + *
+ *
+ *
+ * ``` + * + * In the example HTML code above the message that is set on required will override the corresponding + * required message defined within the remote template. Therefore, with particular input fields (such + * email addresses, date fields, autocomplete inputs, etc...), specialized error messages can be applied + * while more generic messages can be used to handle other, more general input errors. + * + * ## Dynamic Messaging + * ngMessages also supports using expressions to dynamically change key values. Using arrays and + * repeaters to list messages is also supported. This means that the code below will be able to + * fully adapt itself and display the appropriate message when any of the expression data changes: + * + * ```html + *
+ * + *
+ *
You did not enter your email address
+ *
+ * + *
{{ errorMessage.text }}
+ *
+ *
+ *
+ * ``` + * + * The `errorMessage.type` expression can be a string value or it can be an array so + * that multiple errors can be associated with a single error message: + * + * ```html + * + *
+ *
You did not enter your email address
+ *
+ * Your email must be between 5 and 100 characters long + *
+ *
+ * ``` + * + * Feel free to use other structural directives such as ng-if and ng-switch to further control + * what messages are active and when. Be careful, if you place ng-message on the same element + * as these structural directives, Angular may not be able to determine if a message is active + * or not. Therefore it is best to place the ng-message on a child element of the structural + * directive. + * + * ```html + *
+ *
+ *
Please enter something
+ *
+ *
+ * ``` + * + * ## Animations + * If the `ngAnimate` module is active within the application then the `ngMessages`, `ngMessage` and + * `ngMessageExp` directives will trigger animations whenever any messages are added and removed from + * the DOM by the `ngMessages` directive. + * + * Whenever the `ngMessages` directive contains one or more visible messages then the `.ng-active` CSS + * class will be added to the element. The `.ng-inactive` CSS class will be applied when there are no + * messages present. Therefore, CSS transitions and keyframes as well as JavaScript animations can + * hook into the animations whenever these classes are added/removed. + * + * Let's say that our HTML code for our messages container looks like so: + * + * ```html + * + * ``` + * + * Then the CSS animation code for the message container looks like so: + * + * ```css + * .my-messages { + * transition:1s linear all; + * } + * .my-messages.ng-active { + * // messages are visible + * } + * .my-messages.ng-inactive { + * // messages are hidden + * } + * ``` + * + * Whenever an inner message is attached (becomes visible) or removed (becomes hidden) then the enter + * and leave animation is triggered for each particular element bound to the `ngMessage` directive. + * + * Therefore, the CSS code for the inner messages looks like so: + * + * ```css + * .some-message { + * transition:1s linear all; + * } + * + * .some-message.ng-enter {} + * .some-message.ng-enter.ng-enter-active {} + * + * .some-message.ng-leave {} + * .some-message.ng-leave.ng-leave-active {} + * ``` + * + * {@link ngAnimate Click here} to learn how to use JavaScript animations or to learn more about ngAnimate. + */ +angular.module('ngMessages', []) + + /** + * @ngdoc directive + * @module ngMessages + * @name ngMessages + * @restrict AE + * + * @description + * `ngMessages` is a directive that is designed to show and hide messages based on the state + * of a key/value object that it listens on. The directive itself complements error message + * reporting with the `ngModel` $error object (which stores a key/value state of validation errors). + * + * `ngMessages` manages the state of internal messages within its container element. The internal + * messages use the `ngMessage` directive and will be inserted/removed from the page depending + * on if they're present within the key/value object. By default, only one message will be displayed + * at a time and this depends on the prioritization of the messages within the template. (This can + * be changed by using the `ng-messages-multiple` or `multiple` attribute on the directive container.) + * + * A remote template can also be used to promote message reusability and messages can also be + * overridden. + * + * {@link module:ngMessages Click here} to learn more about `ngMessages` and `ngMessage`. + * + * @usage + * ```html + * + * + * ... + * ... + * ... + * + * + * + * + * ... + * ... + * ... + * + * ``` + * + * @param {string} ngMessages an angular expression evaluating to a key/value object + * (this is typically the $error object on an ngModel instance). + * @param {string=} ngMessagesMultiple|multiple when set, all messages will be displayed with true + * + * @example + * + * + *
+ * + *
myForm.myName.$error = {{ myForm.myName.$error | json }}
+ * + *
+ *
You did not enter a field
+ *
Your field is too short
+ *
Your field is too long
+ *
+ *
+ *
+ * + * angular.module('ngMessagesExample', ['ngMessages']); + * + *
+ */ + .directive('ngMessages', ['$animate', function($animate) { + var ACTIVE_CLASS = 'ng-active'; + var INACTIVE_CLASS = 'ng-inactive'; + + return { + require: 'ngMessages', + restrict: 'AE', + controller: ['$element', '$scope', '$attrs', function($element, $scope, $attrs) { + var ctrl = this; + var latestKey = 0; + var nextAttachId = 0; + + this.getAttachId = function getAttachId() { return nextAttachId++; }; + + var messages = this.messages = {}; + var renderLater, cachedCollection; + + this.render = function(collection) { + collection = collection || {}; + + renderLater = false; + cachedCollection = collection; + + // this is true if the attribute is empty or if the attribute value is truthy + var multiple = isAttrTruthy($scope, $attrs.ngMessagesMultiple) || + isAttrTruthy($scope, $attrs.multiple); + + var unmatchedMessages = []; + var matchedKeys = {}; + var messageItem = ctrl.head; + var messageFound = false; + var totalMessages = 0; + + // we use != instead of !== to allow for both undefined and null values + while (messageItem != null) { + totalMessages++; + var messageCtrl = messageItem.message; + + var messageUsed = false; + if (!messageFound) { + forEach(collection, function(value, key) { + if (!messageUsed && truthy(value) && messageCtrl.test(key)) { + // this is to prevent the same error name from showing up twice + if (matchedKeys[key]) return; + matchedKeys[key] = true; + + messageUsed = true; + messageCtrl.attach(); + } + }); + } + + if (messageUsed) { + // unless we want to display multiple messages then we should + // set a flag here to avoid displaying the next message in the list + messageFound = !multiple; + } else { + unmatchedMessages.push(messageCtrl); + } + + messageItem = messageItem.next; + } + + forEach(unmatchedMessages, function(messageCtrl) { + messageCtrl.detach(); + }); + + unmatchedMessages.length !== totalMessages + ? $animate.setClass($element, ACTIVE_CLASS, INACTIVE_CLASS) + : $animate.setClass($element, INACTIVE_CLASS, ACTIVE_CLASS); + }; + + $scope.$watchCollection($attrs.ngMessages || $attrs['for'], ctrl.render); + + this.reRender = function() { + if (!renderLater) { + renderLater = true; + $scope.$evalAsync(function() { + if (renderLater) { + cachedCollection && ctrl.render(cachedCollection); + } + }); + } + }; + + this.register = function(comment, messageCtrl) { + var nextKey = latestKey.toString(); + messages[nextKey] = { + message: messageCtrl + }; + insertMessageNode($element[0], comment, nextKey); + comment.$$ngMessageNode = nextKey; + latestKey++; + + ctrl.reRender(); + }; + + this.deregister = function(comment) { + var key = comment.$$ngMessageNode; + delete comment.$$ngMessageNode; + removeMessageNode($element[0], comment, key); + delete messages[key]; + ctrl.reRender(); + }; + + function findPreviousMessage(parent, comment) { + var prevNode = comment; + var parentLookup = []; + while (prevNode && prevNode !== parent) { + var prevKey = prevNode.$$ngMessageNode; + if (prevKey && prevKey.length) { + return messages[prevKey]; + } + + // dive deeper into the DOM and examine its children for any ngMessage + // comments that may be in an element that appears deeper in the list + if (prevNode.childNodes.length && parentLookup.indexOf(prevNode) == -1) { + parentLookup.push(prevNode); + prevNode = prevNode.childNodes[prevNode.childNodes.length - 1]; + } else { + prevNode = prevNode.previousSibling || prevNode.parentNode; + } + } + } + + function insertMessageNode(parent, comment, key) { + var messageNode = messages[key]; + if (!ctrl.head) { + ctrl.head = messageNode; + } else { + var match = findPreviousMessage(parent, comment); + if (match) { + messageNode.next = match.next; + match.next = messageNode; + } else { + messageNode.next = ctrl.head; + ctrl.head = messageNode; + } + } + } + + function removeMessageNode(parent, comment, key) { + var messageNode = messages[key]; + + var match = findPreviousMessage(parent, comment); + if (match) { + match.next = messageNode.next; + } else { + ctrl.head = messageNode.next; + } + } + }] + }; + + function isAttrTruthy(scope, attr) { + return (isString(attr) && attr.length === 0) || //empty attribute + truthy(scope.$eval(attr)); + } + + function truthy(val) { + return isString(val) ? val.length : !!val; + } + }]) + + /** + * @ngdoc directive + * @name ngMessagesInclude + * @restrict AE + * @scope + * + * @description + * `ngMessagesInclude` is a directive with the purpose to import existing ngMessage template + * code from a remote template and place the downloaded template code into the exact spot + * that the ngMessagesInclude directive is placed within the ngMessages container. This allows + * for a series of pre-defined messages to be reused and also allows for the developer to + * determine what messages are overridden due to the placement of the ngMessagesInclude directive. + * + * @usage + * ```html + * + * + * ... + * + * + * + * + * ... + * + * ``` + * + * {@link module:ngMessages Click here} to learn more about `ngMessages` and `ngMessage`. + * + * @param {string} ngMessagesInclude|src a string value corresponding to the remote template. + */ + .directive('ngMessagesInclude', + ['$templateRequest', '$document', '$compile', function($templateRequest, $document, $compile) { + + return { + restrict: 'AE', + require: '^^ngMessages', // we only require this for validation sake + link: function($scope, element, attrs) { + var src = attrs.ngMessagesInclude || attrs.src; + $templateRequest(src).then(function(html) { + $compile(html)($scope, function(contents) { + element.after(contents); + + // the anchor is placed for debugging purposes + var anchor = jqLite($document[0].createComment(' ngMessagesInclude: ' + src + ' ')); + element.after(anchor); + + // we don't want to pollute the DOM anymore by keeping an empty directive element + element.remove(); + }); + }); + } + }; + }]) + + /** + * @ngdoc directive + * @name ngMessage + * @restrict AE + * @scope + * + * @description + * `ngMessage` is a directive with the purpose to show and hide a particular message. + * For `ngMessage` to operate, a parent `ngMessages` directive on a parent DOM element + * must be situated since it determines which messages are visible based on the state + * of the provided key/value map that `ngMessages` listens on. + * + * More information about using `ngMessage` can be found in the + * {@link module:ngMessages `ngMessages` module documentation}. + * + * @usage + * ```html + * + * + * ... + * ... + * + * + * + * + * ... + * ... + * + * ``` + * + * @param {expression} ngMessage|when a string value corresponding to the message key. + */ + .directive('ngMessage', ngMessageDirectiveFactory('AE')) + + + /** + * @ngdoc directive + * @name ngMessageExp + * @restrict AE + * @scope + * + * @description + * `ngMessageExp` is a directive with the purpose to show and hide a particular message. + * For `ngMessageExp` to operate, a parent `ngMessages` directive on a parent DOM element + * must be situated since it determines which messages are visible based on the state + * of the provided key/value map that `ngMessages` listens on. + * + * @usage + * ```html + * + * + * ... + * + * + * + * + * ... + * + * ``` + * + * {@link module:ngMessages Click here} to learn more about `ngMessages` and `ngMessage`. + * + * @param {expression} ngMessageExp|whenExp an expression value corresponding to the message key. + */ + .directive('ngMessageExp', ngMessageDirectiveFactory('A')); + +function ngMessageDirectiveFactory(restrict) { + return ['$animate', function($animate) { + return { + restrict: 'AE', + transclude: 'element', + terminal: true, + require: '^^ngMessages', + link: function(scope, element, attrs, ngMessagesCtrl, $transclude) { + var commentNode = element[0]; + + var records; + var staticExp = attrs.ngMessage || attrs.when; + var dynamicExp = attrs.ngMessageExp || attrs.whenExp; + var assignRecords = function(items) { + records = items + ? (isArray(items) + ? items + : items.split(/[\s,]+/)) + : null; + ngMessagesCtrl.reRender(); + }; + + if (dynamicExp) { + assignRecords(scope.$eval(dynamicExp)); + scope.$watchCollection(dynamicExp, assignRecords); + } else { + assignRecords(staticExp); + } + + var currentElement, messageCtrl; + ngMessagesCtrl.register(commentNode, messageCtrl = { + test: function(name) { + return contains(records, name); + }, + attach: function() { + if (!currentElement) { + $transclude(scope, function(elm) { + $animate.enter(elm, null, element); + currentElement = elm; + + // Each time we attach this node to a message we get a new id that we can match + // when we are destroying the node later. + var $$attachId = currentElement.$$attachId = ngMessagesCtrl.getAttachId(); + + // in the event that the parent element is destroyed + // by any other structural directive then it's time + // to deregister the message from the controller + currentElement.on('$destroy', function() { + if (currentElement && currentElement.$$attachId === $$attachId) { + ngMessagesCtrl.deregister(commentNode); + messageCtrl.detach(); + } + }); + }); + } + }, + detach: function() { + if (currentElement) { + var elm = currentElement; + currentElement = null; + $animate.leave(elm); + } + } + }); + } + }; + }]; + + function contains(collection, key) { + if (collection) { + return isArray(collection) + ? collection.indexOf(key) >= 0 + : collection.hasOwnProperty(key); + } + } +} + + +})(window, window.angular); diff --git a/1.4.10/angular-messages.min.js b/1.4.10/angular-messages.min.js new file mode 100644 index 0000000000..bb5be59f70 --- /dev/null +++ b/1.4.10/angular-messages.min.js @@ -0,0 +1,12 @@ +/* + AngularJS v1.4.10 + (c) 2010-2015 Google, Inc. http://angularjs.org + License: MIT +*/ +(function(z,h,A){'use strict';function m(h){return["$animate",function(r){return{restrict:"AE",transclude:"element",terminal:!0,require:"^^ngMessages",link:function(n,f,a,g,l){var c=f[0],p,h=a.ngMessage||a.when;a=a.ngMessageExp||a.whenExp;var k=function(b){p=b?v(b)?b:b.split(/[\s,]+/):null;g.reRender()};a?(k(n.$eval(a)),n.$watchCollection(a,k)):k(h);var e,q;g.register(c,q={test:function(b){var a=p;b=a?v(a)?0<=a.indexOf(b):a.hasOwnProperty(b):void 0;return b},attach:function(){e||l(n,function(b){r.enter(b, +null,f);e=b;var a=e.$$attachId=g.getAttachId();e.on("$destroy",function(){e&&e.$$attachId===a&&(g.deregister(c),q.detach())})})},detach:function(){if(e){var b=e;e=null;r.leave(b)}}})}}}]}var v=h.isArray,w=h.forEach,x=h.isString,y=h.element;h.module("ngMessages",[]).directive("ngMessages",["$animate",function(h){function r(f,a){return x(a)&&0===a.length||n(f.$eval(a))}function n(f){return x(f)?f.length:!!f}return{require:"ngMessages",restrict:"AE",controller:["$element","$scope","$attrs",function(f, +a,g){function l(b,a){for(var d=a,f=[];d&&d!==b;){var c=d.$$ngMessageNode;if(c&&c.length)return k[c];d.childNodes.length&&-1==f.indexOf(d)?(f.push(d),d=d.childNodes[d.childNodes.length-1]):d=d.previousSibling||d.parentNode}}var c=this,p=0,m=0;this.getAttachId=function(){return m++};var k=this.messages={},e,q;this.render=function(b){b=b||{};e=!1;q=b;for(var p=r(a,g.ngMessagesMultiple)||r(a,g.multiple),d=[],k={},s=c.head,l=!1,m=0;null!=s;){m++;var t=s.message,u=!1;l||w(b,function(a,b){!u&&n(a)&&t.test(b)&& +!k[b]&&(u=k[b]=!0,t.attach())});u?l=!p:d.push(t);s=s.next}w(d,function(b){b.detach()});d.length!==m?h.setClass(f,"ng-active","ng-inactive"):h.setClass(f,"ng-inactive","ng-active")};a.$watchCollection(g.ngMessages||g["for"],c.render);this.reRender=function(){e||(e=!0,a.$evalAsync(function(){e&&q&&c.render(q)}))};this.register=function(b,a){var d=p.toString();k[d]={message:a};var e=f[0],g=k[d];c.head?(e=l(e,b))?(g.next=e.next,e.next=g):(g.next=c.head,c.head=g):c.head=g;b.$$ngMessageNode=d;p++;c.reRender()}; +this.deregister=function(b){var a=b.$$ngMessageNode;delete b.$$ngMessageNode;var d=k[a];(b=l(f[0],b))?b.next=d.next:c.head=d.next;delete k[a];c.reRender()}}]}}]).directive("ngMessagesInclude",["$templateRequest","$document","$compile",function(h,m,n){return{restrict:"AE",require:"^^ngMessages",link:function(f,a,g){var l=g.ngMessagesInclude||g.src;h(l).then(function(c){n(c)(f,function(c){a.after(c);c=y(m[0].createComment(" ngMessagesInclude: "+l+" "));a.after(c);a.remove()})})}}}]).directive("ngMessage", +m("AE")).directive("ngMessageExp",m("A"))})(window,window.angular); +//# sourceMappingURL=angular-messages.min.js.map diff --git a/1.4.10/angular-messages.min.js.map b/1.4.10/angular-messages.min.js.map new file mode 100644 index 0000000000..a4afbe5514 --- /dev/null +++ b/1.4.10/angular-messages.min.js.map @@ -0,0 +1,8 @@ +{ +"version":3, +"file":"angular-messages.min.js", +"lineCount":11, +"mappings":"A;;;;;aAKC,SAAQ,CAACA,CAAD,CAASC,CAAT,CAAkBC,CAAlB,CAA6B,CA8mBtCC,QAASA,EAAyB,CAACC,CAAD,CAAW,CAC3C,MAAO,CAAC,UAAD,CAAa,QAAQ,CAACC,CAAD,CAAW,CACrC,MAAO,CACLD,SAAU,IADL,CAELE,WAAY,SAFP,CAGLC,SAAU,CAAA,CAHL,CAILC,QAAS,cAJJ,CAKLC,KAAMA,QAAQ,CAACC,CAAD,CAAQC,CAAR,CAAiBC,CAAjB,CAAwBC,CAAxB,CAAwCC,CAAxC,CAAqD,CACjE,IAAIC,EAAcJ,CAAA,CAAQ,CAAR,CAAlB,CAEIK,CAFJ,CAGIC,EAAYL,CAAAM,UAAZD,EAA+BL,CAAAO,KAC/BC,EAAAA,CAAaR,CAAAS,aAAbD,EAAmCR,CAAAU,QACvC,KAAIC,EAAgBA,QAAQ,CAACC,CAAD,CAAQ,CAClCR,CAAA,CAAUQ,CAAA,CACHC,CAAA,CAAQD,CAAR,CAAA,CACKA,CADL,CAEKA,CAAAE,MAAA,CAAY,QAAZ,CAHF,CAIJ,IACNb,EAAAc,SAAA,EANkC,CAShCP,EAAJ,EACEG,CAAA,CAAcb,CAAAkB,MAAA,CAAYR,CAAZ,CAAd,CACA,CAAAV,CAAAmB,iBAAA,CAAuBT,CAAvB,CAAmCG,CAAnC,CAFF,EAIEA,CAAA,CAAcN,CAAd,CAnB+D,KAsB7Da,CAtB6D,CAsB7CC,CACpBlB,EAAAmB,SAAA,CAAwBjB,CAAxB,CAAqCgB,CAArC,CAAmD,CACjDE,KAAMA,QAAQ,CAACC,CAAD,CAAO,CACHlB,IAAAA,EAAAA,CAsCtB,EAAA,CADEmB,CAAJ,CACSV,CAAA,CAAQU,CAAR,CAAA,CAC0B,CAD1B,EACDA,CAAAC,QAAA,CAvCyBF,CAuCzB,CADC,CAEDC,CAAAE,eAAA,CAxCyBH,CAwCzB,CAHR,CADiC,IAAA,EApCzB,OAAO,EADY,CAD4B,CAIjDI,OAAQA,QAAQ,EAAG,CACZR,CAAL,EACEhB,CAAA,CAAYJ,CAAZ,CAAmB,QAAQ,CAAC6B,CAAD,CAAM,CAC/BlC,CAAAmC,MAAA,CAAeD,CAAf;AAAoB,IAApB,CAA0B5B,CAA1B,CACAmB,EAAA,CAAiBS,CAIjB,KAAIE,EAAaX,CAAAW,WAAbA,CAAyC5B,CAAA6B,YAAA,EAK7CZ,EAAAa,GAAA,CAAkB,UAAlB,CAA8B,QAAQ,EAAG,CACnCb,CAAJ,EAAsBA,CAAAW,WAAtB,GAAoDA,CAApD,GACE5B,CAAA+B,WAAA,CAA0B7B,CAA1B,CACA,CAAAgB,CAAAc,OAAA,EAFF,CADuC,CAAzC,CAX+B,CAAjC,CAFe,CAJ8B,CA0BjDA,OAAQA,QAAQ,EAAG,CACjB,GAAIf,CAAJ,CAAoB,CAClB,IAAIS,EAAMT,CACVA,EAAA,CAAiB,IACjBzB,EAAAyC,MAAA,CAAeP,CAAf,CAHkB,CADH,CA1B8B,CAAnD,CAvBiE,CAL9D,CAD8B,CAAhC,CADoC,CA1mB7C,IAAId,EAAUxB,CAAAwB,QAAd,CACIsB,EAAU9C,CAAA8C,QADd,CAEIC,EAAW/C,CAAA+C,SAFf,CAGIC,EAAShD,CAAAU,QAiQbV,EAAAiD,OAAA,CAAe,YAAf,CAA6B,EAA7B,CAAAC,UAAA,CA0Ec,YA1Ed,CA0E4B,CAAC,UAAD,CAAa,QAAQ,CAAC9C,CAAD,CAAW,CA0JvD+C,QAASA,EAAY,CAAC1C,CAAD,CAAQ2C,CAAR,CAAc,CAClC,MAAQL,EAAA,CAASK,CAAT,CAAR,EAA0C,CAA1C,GAA0BA,CAAAC,OAA1B,EACOC,CAAA,CAAO7C,CAAAkB,MAAA,CAAYyB,CAAZ,CAAP,CAF2B,CAKnCE,QAASA,EAAM,CAACC,CAAD,CAAM,CACnB,MAAOR,EAAA,CAASQ,CAAT,CAAA,CAAgBA,CAAAF,OAAhB,CAA6B,CAAEE,CAAAA,CADnB,CA3JrB,MAAO,CACLhD,QAAS,YADJ,CAELJ,SAAU,IAFL,CAGLqD,WAAY,CAAC,UAAD,CAAa,QAAb,CAAuB,QAAvB,CAAiC,QAAQ,CAACC,CAAD;AAAWC,CAAX,CAAmBC,CAAnB,CAA2B,CAkG9EC,QAASA,EAAmB,CAACC,CAAD,CAASC,CAAT,CAAkB,CAG5C,IAFA,IAAIC,EAAWD,CAAf,CACIE,EAAe,EACnB,CAAOD,CAAP,EAAmBA,CAAnB,GAAgCF,CAAhC,CAAA,CAAwC,CACtC,IAAII,EAAUF,CAAAG,gBACd,IAAID,CAAJ,EAAeA,CAAAZ,OAAf,CACE,MAAOc,EAAA,CAASF,CAAT,CAKLF,EAAAK,WAAAf,OAAJ,EAAqE,EAArE,EAAkCW,CAAA7B,QAAA,CAAqB4B,CAArB,CAAlC,EACEC,CAAAK,KAAA,CAAkBN,CAAlB,CACA,CAAAA,CAAA,CAAWA,CAAAK,WAAA,CAAoBL,CAAAK,WAAAf,OAApB,CAAiD,CAAjD,CAFb,EAIEU,CAJF,CAIaA,CAAAO,gBAJb,EAIyCP,CAAAQ,WAZH,CAHI,CAjG9C,IAAIC,EAAO,IAAX,CACIC,EAAY,CADhB,CAEIC,EAAe,CAEnB,KAAAjC,YAAA,CAAmBkC,QAAoB,EAAG,CAAE,MAAOD,EAAA,EAAT,CAE1C,KAAIP,EAAW,IAAAA,SAAXA,CAA2B,EAA/B,CACIS,CADJ,CACiBC,CAEjB,KAAAC,OAAA,CAAcC,QAAQ,CAAC7C,CAAD,CAAa,CACjCA,CAAA,CAAaA,CAAb,EAA2B,EAE3B0C,EAAA,CAAc,CAAA,CACdC,EAAA,CAAmB3C,CAanB,KAVA,IAAI8C,EAAW7B,CAAA,CAAaO,CAAb,CAAqBC,CAAAsB,mBAArB,CAAXD,EACW7B,CAAA,CAAaO,CAAb,CAAqBC,CAAAqB,SAArB,CADf,CAGIE,EAAoB,EAHxB,CAIIC,EAAc,EAJlB,CAKIC,EAAcZ,CAAAa,KALlB,CAMIC,EAAe,CAAA,CANnB,CAOIC,EAAgB,CAGpB,CAAsB,IAAtB,EAAOH,CAAP,CAAA,CAA4B,CAC1BG,CAAA,EACA,KAAIzD,EAAcsD,CAAAI,QAAlB,CAEIC,EAAc,CAAA,CACbH,EAAL,EACExC,CAAA,CAAQZ,CAAR,CAAoB,QAAQ,CAACwD,CAAD,CAAQC,CAAR,CAAa,CAClCF,CAAAA,CAAL,EAAoBnC,CAAA,CAAOoC,CAAP,CAApB,EAAqC5D,CAAAE,KAAA,CAAiB2D,CAAjB,CAArC;AAEM,CAAAR,CAAA,CAAYQ,CAAZ,CAFN,GAKEF,CACA,CAHAN,CAAA,CAAYQ,CAAZ,CAGA,CAHmB,CAAA,CAGnB,CAAA7D,CAAAO,OAAA,EANF,CADuC,CAAzC,CAYEoD,EAAJ,CAGEH,CAHF,CAGiB,CAACN,CAHlB,CAKEE,CAAAb,KAAA,CAAuBvC,CAAvB,CAGFsD,EAAA,CAAcA,CAAAQ,KA1BY,CA6B5B9C,CAAA,CAAQoC,CAAR,CAA2B,QAAQ,CAACpD,CAAD,CAAc,CAC/CA,CAAAc,OAAA,EAD+C,CAAjD,CAIAsC,EAAA7B,OAAA,GAA6BkC,CAA7B,CACKnF,CAAAyF,SAAA,CAAkBpC,CAAlB,CAnEQqC,WAmER,CAlEUC,aAkEV,CADL,CAEK3F,CAAAyF,SAAA,CAAkBpC,CAAlB,CAnEUsC,aAmEV,CApEQD,WAoER,CApD4B,CAuDnCpC,EAAA9B,iBAAA,CAAwB+B,CAAAqC,WAAxB,EAA6CrC,CAAA,CAAO,KAAP,CAA7C,CAA4Da,CAAAM,OAA5D,CAEA,KAAApD,SAAA,CAAgBuE,QAAQ,EAAG,CACpBrB,CAAL,GACEA,CACA,CADc,CAAA,CACd,CAAAlB,CAAAwC,WAAA,CAAkB,QAAQ,EAAG,CACvBtB,CAAJ,EACEC,CADF,EACsBL,CAAAM,OAAA,CAAYD,CAAZ,CAFK,CAA7B,CAFF,CADyB,CAW3B,KAAA9C,SAAA,CAAgBoE,QAAQ,CAACrC,CAAD,CAAUhC,CAAV,CAAuB,CAC7C,IAAIsE,EAAU3B,CAAA4B,SAAA,EACdlC,EAAA,CAASiC,CAAT,CAAA,CAAoB,CAClBZ,QAAS1D,CADS,CAGF,KAAA,EAAA2B,CAAA,CAAS,CAAT,CAAA,CAoCd6C,EAAcnC,CAAA,CApCsBiC,CAoCtB,CACb5B,EAAAa,KAAL,CAIE,CADIkB,CACJ,CADY3C,CAAA,CAAoBC,CAApB,CAxCiBC,CAwCjB,CACZ,GACEwC,CAAAV,KACA,CADmBW,CAAAX,KACnB,CAAAW,CAAAX,KAAA,CAAaU,CAFf,GAIEA,CAAAV,KACA,CADmBpB,CAAAa,KACnB,CAAAb,CAAAa,KAAA,CAAYiB,CALd,CAJF,CACE9B,CAAAa,KADF,CACciB,CArCdxC,EAAAI,gBAAA,CAA0BkC,CAC1B3B,EAAA,EAEAD,EAAA9C,SAAA,EAT6C,CAY/C;IAAAiB,WAAA,CAAkB6D,QAAQ,CAAC1C,CAAD,CAAU,CAClC,IAAI6B,EAAM7B,CAAAI,gBACV,QAAOJ,CAAAI,gBA2CP,KAAIoC,EAAcnC,CAAA,CA1CsBwB,CA0CtB,CAGlB,EADIY,CACJ,CADY3C,CAAA,CA5CMH,CAAAI,CAAS,CAATA,CA4CN,CA5CmBC,CA4CnB,CACZ,EACEyC,CAAAX,KADF,CACeU,CAAAV,KADf,CAGEpB,CAAAa,KAHF,CAGciB,CAAAV,KA/Cd,QAAOzB,CAAA,CAASwB,CAAT,CACPnB,EAAA9C,SAAA,EALkC,CA1F0C,CAApE,CAHP,CAJgD,CAAhC,CA1E5B,CAAAwB,UAAA,CA4Qc,mBA5Qd,CA6QK,CAAC,kBAAD,CAAqB,WAArB,CAAkC,UAAlC,CAA8C,QAAQ,CAACuD,CAAD,CAAmBC,CAAnB,CAA8BC,CAA9B,CAAwC,CAE9F,MAAO,CACLxG,SAAU,IADL,CAELI,QAAS,cAFJ,CAGLC,KAAMA,QAAQ,CAACkD,CAAD,CAAShD,CAAT,CAAkBC,CAAlB,CAAyB,CACrC,IAAIiG,EAAMjG,CAAAkG,kBAAND,EAAiCjG,CAAAiG,IACrCH,EAAA,CAAiBG,CAAjB,CAAAE,KAAA,CAA2B,QAAQ,CAACC,CAAD,CAAO,CACxCJ,CAAA,CAASI,CAAT,CAAA,CAAerD,CAAf,CAAuB,QAAQ,CAACsD,CAAD,CAAW,CACxCtG,CAAAuG,MAAA,CAAcD,CAAd,CAGIE,EAAAA,CAASlE,CAAA,CAAO0D,CAAA,CAAU,CAAV,CAAAS,cAAA,CAA2B,sBAA3B,CAAoDP,CAApD,CAA0D,GAA1D,CAAP,CACblG,EAAAuG,MAAA,CAAcC,CAAd,CAGAxG,EAAA0G,OAAA,EARwC,CAA1C,CADwC,CAA1C,CAFqC,CAHlC,CAFuF,CAA9F,CA7QL,CAAAlE,UAAA,CAoUa,WApUb;AAoU0BhD,CAAA,CAA0B,IAA1B,CApU1B,CAAAgD,UAAA,CAoWa,cApWb,CAoW6BhD,CAAA,CAA0B,GAA1B,CApW7B,CAxQsC,CAArC,CAAD,CA4rBGH,MA5rBH,CA4rBWA,MAAAC,QA5rBX;", +"sources":["angular-messages.js"], +"names":["window","angular","undefined","ngMessageDirectiveFactory","restrict","$animate","transclude","terminal","require","link","scope","element","attrs","ngMessagesCtrl","$transclude","commentNode","records","staticExp","ngMessage","when","dynamicExp","ngMessageExp","whenExp","assignRecords","items","isArray","split","reRender","$eval","$watchCollection","currentElement","messageCtrl","register","test","name","collection","indexOf","hasOwnProperty","attach","elm","enter","$$attachId","getAttachId","on","deregister","detach","leave","forEach","isString","jqLite","module","directive","isAttrTruthy","attr","length","truthy","val","controller","$element","$scope","$attrs","findPreviousMessage","parent","comment","prevNode","parentLookup","prevKey","$$ngMessageNode","messages","childNodes","push","previousSibling","parentNode","ctrl","latestKey","nextAttachId","this.getAttachId","renderLater","cachedCollection","render","this.render","multiple","ngMessagesMultiple","unmatchedMessages","matchedKeys","messageItem","head","messageFound","totalMessages","message","messageUsed","value","key","next","setClass","ACTIVE_CLASS","INACTIVE_CLASS","ngMessages","this.reRender","$evalAsync","this.register","nextKey","toString","messageNode","match","this.deregister","$templateRequest","$document","$compile","src","ngMessagesInclude","then","html","contents","after","anchor","createComment","remove"] +} diff --git a/1.4.10/angular-mocks.js b/1.4.10/angular-mocks.js new file mode 100644 index 0000000000..ab2ba68318 --- /dev/null +++ b/1.4.10/angular-mocks.js @@ -0,0 +1,2550 @@ +/** + * @license AngularJS v1.4.10 + * (c) 2010-2015 Google, Inc. http://angularjs.org + * License: MIT + */ +(function(window, angular, undefined) { + +'use strict'; + +/** + * @ngdoc object + * @name angular.mock + * @description + * + * Namespace from 'angular-mocks.js' which contains testing related code. + */ +angular.mock = {}; + +/** + * ! This is a private undocumented service ! + * + * @name $browser + * + * @description + * This service is a mock implementation of {@link ng.$browser}. It provides fake + * implementation for commonly used browser apis that are hard to test, e.g. setTimeout, xhr, + * cookies, etc... + * + * The api of this service is the same as that of the real {@link ng.$browser $browser}, except + * that there are several helper methods available which can be used in tests. + */ +angular.mock.$BrowserProvider = function() { + this.$get = function() { + return new angular.mock.$Browser(); + }; +}; + +angular.mock.$Browser = function() { + var self = this; + + this.isMock = true; + self.$$url = "http://server/"; + self.$$lastUrl = self.$$url; // used by url polling fn + self.pollFns = []; + + // TODO(vojta): remove this temporary api + self.$$completeOutstandingRequest = angular.noop; + self.$$incOutstandingRequestCount = angular.noop; + + + // register url polling fn + + self.onUrlChange = function(listener) { + self.pollFns.push( + function() { + if (self.$$lastUrl !== self.$$url || self.$$state !== self.$$lastState) { + self.$$lastUrl = self.$$url; + self.$$lastState = self.$$state; + listener(self.$$url, self.$$state); + } + } + ); + + return listener; + }; + + self.$$applicationDestroyed = angular.noop; + self.$$checkUrlChange = angular.noop; + + self.deferredFns = []; + self.deferredNextId = 0; + + self.defer = function(fn, delay) { + delay = delay || 0; + self.deferredFns.push({time:(self.defer.now + delay), fn:fn, id: self.deferredNextId}); + self.deferredFns.sort(function(a, b) { return a.time - b.time;}); + return self.deferredNextId++; + }; + + + /** + * @name $browser#defer.now + * + * @description + * Current milliseconds mock time. + */ + self.defer.now = 0; + + + self.defer.cancel = function(deferId) { + var fnIndex; + + angular.forEach(self.deferredFns, function(fn, index) { + if (fn.id === deferId) fnIndex = index; + }); + + if (angular.isDefined(fnIndex)) { + self.deferredFns.splice(fnIndex, 1); + return true; + } + + return false; + }; + + + /** + * @name $browser#defer.flush + * + * @description + * Flushes all pending requests and executes the defer callbacks. + * + * @param {number=} number of milliseconds to flush. See {@link #defer.now} + */ + self.defer.flush = function(delay) { + if (angular.isDefined(delay)) { + self.defer.now += delay; + } else { + if (self.deferredFns.length) { + self.defer.now = self.deferredFns[self.deferredFns.length - 1].time; + } else { + throw new Error('No deferred tasks to be flushed'); + } + } + + while (self.deferredFns.length && self.deferredFns[0].time <= self.defer.now) { + self.deferredFns.shift().fn(); + } + }; + + self.$$baseHref = '/'; + self.baseHref = function() { + return this.$$baseHref; + }; +}; +angular.mock.$Browser.prototype = { + +/** + * @name $browser#poll + * + * @description + * run all fns in pollFns + */ + poll: function poll() { + angular.forEach(this.pollFns, function(pollFn) { + pollFn(); + }); + }, + + url: function(url, replace, state) { + if (angular.isUndefined(state)) { + state = null; + } + if (url) { + this.$$url = url; + // Native pushState serializes & copies the object; simulate it. + this.$$state = angular.copy(state); + return this; + } + + return this.$$url; + }, + + state: function() { + return this.$$state; + }, + + notifyWhenNoOutstandingRequests: function(fn) { + fn(); + } +}; + + +/** + * @ngdoc provider + * @name $exceptionHandlerProvider + * + * @description + * Configures the mock implementation of {@link ng.$exceptionHandler} to rethrow or to log errors + * passed to the `$exceptionHandler`. + */ + +/** + * @ngdoc service + * @name $exceptionHandler + * + * @description + * Mock implementation of {@link ng.$exceptionHandler} that rethrows or logs errors passed + * to it. See {@link ngMock.$exceptionHandlerProvider $exceptionHandlerProvider} for configuration + * information. + * + * + * ```js + * describe('$exceptionHandlerProvider', function() { + * + * it('should capture log messages and exceptions', function() { + * + * module(function($exceptionHandlerProvider) { + * $exceptionHandlerProvider.mode('log'); + * }); + * + * inject(function($log, $exceptionHandler, $timeout) { + * $timeout(function() { $log.log(1); }); + * $timeout(function() { $log.log(2); throw 'banana peel'; }); + * $timeout(function() { $log.log(3); }); + * expect($exceptionHandler.errors).toEqual([]); + * expect($log.assertEmpty()); + * $timeout.flush(); + * expect($exceptionHandler.errors).toEqual(['banana peel']); + * expect($log.log.logs).toEqual([[1], [2], [3]]); + * }); + * }); + * }); + * ``` + */ + +angular.mock.$ExceptionHandlerProvider = function() { + var handler; + + /** + * @ngdoc method + * @name $exceptionHandlerProvider#mode + * + * @description + * Sets the logging mode. + * + * @param {string} mode Mode of operation, defaults to `rethrow`. + * + * - `log`: Sometimes it is desirable to test that an error is thrown, for this case the `log` + * mode stores an array of errors in `$exceptionHandler.errors`, to allow later + * assertion of them. See {@link ngMock.$log#assertEmpty assertEmpty()} and + * {@link ngMock.$log#reset reset()} + * - `rethrow`: If any errors are passed to the handler in tests, it typically means that there + * is a bug in the application or test, so this mock will make these tests fail. + * For any implementations that expect exceptions to be thrown, the `rethrow` mode + * will also maintain a log of thrown errors. + */ + this.mode = function(mode) { + + switch (mode) { + case 'log': + case 'rethrow': + var errors = []; + handler = function(e) { + if (arguments.length == 1) { + errors.push(e); + } else { + errors.push([].slice.call(arguments, 0)); + } + if (mode === "rethrow") { + throw e; + } + }; + handler.errors = errors; + break; + default: + throw new Error("Unknown mode '" + mode + "', only 'log'/'rethrow' modes are allowed!"); + } + }; + + this.$get = function() { + return handler; + }; + + this.mode('rethrow'); +}; + + +/** + * @ngdoc service + * @name $log + * + * @description + * Mock implementation of {@link ng.$log} that gathers all logged messages in arrays + * (one array per logging level). These arrays are exposed as `logs` property of each of the + * level-specific log function, e.g. for level `error` the array is exposed as `$log.error.logs`. + * + */ +angular.mock.$LogProvider = function() { + var debug = true; + + function concat(array1, array2, index) { + return array1.concat(Array.prototype.slice.call(array2, index)); + } + + this.debugEnabled = function(flag) { + if (angular.isDefined(flag)) { + debug = flag; + return this; + } else { + return debug; + } + }; + + this.$get = function() { + var $log = { + log: function() { $log.log.logs.push(concat([], arguments, 0)); }, + warn: function() { $log.warn.logs.push(concat([], arguments, 0)); }, + info: function() { $log.info.logs.push(concat([], arguments, 0)); }, + error: function() { $log.error.logs.push(concat([], arguments, 0)); }, + debug: function() { + if (debug) { + $log.debug.logs.push(concat([], arguments, 0)); + } + } + }; + + /** + * @ngdoc method + * @name $log#reset + * + * @description + * Reset all of the logging arrays to empty. + */ + $log.reset = function() { + /** + * @ngdoc property + * @name $log#log.logs + * + * @description + * Array of messages logged using {@link ng.$log#log `log()`}. + * + * @example + * ```js + * $log.log('Some Log'); + * var first = $log.log.logs.unshift(); + * ``` + */ + $log.log.logs = []; + /** + * @ngdoc property + * @name $log#info.logs + * + * @description + * Array of messages logged using {@link ng.$log#info `info()`}. + * + * @example + * ```js + * $log.info('Some Info'); + * var first = $log.info.logs.unshift(); + * ``` + */ + $log.info.logs = []; + /** + * @ngdoc property + * @name $log#warn.logs + * + * @description + * Array of messages logged using {@link ng.$log#warn `warn()`}. + * + * @example + * ```js + * $log.warn('Some Warning'); + * var first = $log.warn.logs.unshift(); + * ``` + */ + $log.warn.logs = []; + /** + * @ngdoc property + * @name $log#error.logs + * + * @description + * Array of messages logged using {@link ng.$log#error `error()`}. + * + * @example + * ```js + * $log.error('Some Error'); + * var first = $log.error.logs.unshift(); + * ``` + */ + $log.error.logs = []; + /** + * @ngdoc property + * @name $log#debug.logs + * + * @description + * Array of messages logged using {@link ng.$log#debug `debug()`}. + * + * @example + * ```js + * $log.debug('Some Error'); + * var first = $log.debug.logs.unshift(); + * ``` + */ + $log.debug.logs = []; + }; + + /** + * @ngdoc method + * @name $log#assertEmpty + * + * @description + * Assert that all of the logging methods have no logged messages. If any messages are present, + * an exception is thrown. + */ + $log.assertEmpty = function() { + var errors = []; + angular.forEach(['error', 'warn', 'info', 'log', 'debug'], function(logLevel) { + angular.forEach($log[logLevel].logs, function(log) { + angular.forEach(log, function(logItem) { + errors.push('MOCK $log (' + logLevel + '): ' + String(logItem) + '\n' + + (logItem.stack || '')); + }); + }); + }); + if (errors.length) { + errors.unshift("Expected $log to be empty! Either a message was logged unexpectedly, or " + + "an expected log message was not checked and removed:"); + errors.push(''); + throw new Error(errors.join('\n---------\n')); + } + }; + + $log.reset(); + return $log; + }; +}; + + +/** + * @ngdoc service + * @name $interval + * + * @description + * Mock implementation of the $interval service. + * + * Use {@link ngMock.$interval#flush `$interval.flush(millis)`} to + * move forward by `millis` milliseconds and trigger any functions scheduled to run in that + * time. + * + * @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. + */ +angular.mock.$IntervalProvider = function() { + this.$get = ['$browser', '$rootScope', '$q', '$$q', + function($browser, $rootScope, $q, $$q) { + var repeatFns = [], + nextRepeatId = 0, + now = 0; + + var $interval = function(fn, delay, count, invokeApply) { + var hasParams = arguments.length > 4, + args = hasParams ? Array.prototype.slice.call(arguments, 4) : [], + iteration = 0, + skipApply = (angular.isDefined(invokeApply) && !invokeApply), + deferred = (skipApply ? $$q : $q).defer(), + promise = deferred.promise; + + count = (angular.isDefined(count)) ? count : 0; + promise.then(null, null, (!hasParams) ? fn : function() { + fn.apply(null, args); + }); + + promise.$$intervalId = nextRepeatId; + + function tick() { + deferred.notify(iteration++); + + if (count > 0 && iteration >= count) { + var fnIndex; + deferred.resolve(iteration); + + angular.forEach(repeatFns, function(fn, index) { + if (fn.id === promise.$$intervalId) fnIndex = index; + }); + + if (angular.isDefined(fnIndex)) { + repeatFns.splice(fnIndex, 1); + } + } + + if (skipApply) { + $browser.defer.flush(); + } else { + $rootScope.$apply(); + } + } + + repeatFns.push({ + nextTime:(now + delay), + delay: delay, + fn: tick, + id: nextRepeatId, + deferred: deferred + }); + repeatFns.sort(function(a, b) { return a.nextTime - b.nextTime;}); + + nextRepeatId++; + return promise; + }; + /** + * @ngdoc method + * @name $interval#cancel + * + * @description + * Cancels a task associated with the `promise`. + * + * @param {promise} promise A promise from calling the `$interval` function. + * @returns {boolean} Returns `true` if the task was successfully cancelled. + */ + $interval.cancel = function(promise) { + if (!promise) return false; + var fnIndex; + + angular.forEach(repeatFns, function(fn, index) { + if (fn.id === promise.$$intervalId) fnIndex = index; + }); + + if (angular.isDefined(fnIndex)) { + repeatFns[fnIndex].deferred.reject('canceled'); + repeatFns.splice(fnIndex, 1); + return true; + } + + return false; + }; + + /** + * @ngdoc method + * @name $interval#flush + * @description + * + * Runs interval tasks scheduled to be run in the next `millis` milliseconds. + * + * @param {number=} millis maximum timeout amount to flush up until. + * + * @return {number} The amount of time moved forward. + */ + $interval.flush = function(millis) { + now += millis; + while (repeatFns.length && repeatFns[0].nextTime <= now) { + var task = repeatFns[0]; + task.fn(); + task.nextTime += task.delay; + repeatFns.sort(function(a, b) { return a.nextTime - b.nextTime;}); + } + return millis; + }; + + return $interval; + }]; +}; + + +/* jshint -W101 */ +/* The R_ISO8061_STR regex is never going to fit into the 100 char limit! + * This directive should go inside the anonymous function but a bug in JSHint means that it would + * not be enacted early enough to prevent the warning. + */ +var R_ISO8061_STR = /^(\d{4})-?(\d\d)-?(\d\d)(?:T(\d\d)(?:\:?(\d\d)(?:\:?(\d\d)(?:\.(\d{3}))?)?)?(Z|([+-])(\d\d):?(\d\d)))?$/; + +function jsonStringToDate(string) { + var match; + if (match = string.match(R_ISO8061_STR)) { + var date = new Date(0), + tzHour = 0, + tzMin = 0; + if (match[9]) { + tzHour = toInt(match[9] + match[10]); + tzMin = toInt(match[9] + match[11]); + } + date.setUTCFullYear(toInt(match[1]), toInt(match[2]) - 1, toInt(match[3])); + date.setUTCHours(toInt(match[4] || 0) - tzHour, + toInt(match[5] || 0) - tzMin, + toInt(match[6] || 0), + toInt(match[7] || 0)); + return date; + } + return string; +} + +function toInt(str) { + return parseInt(str, 10); +} + +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; +} + + +/** + * @ngdoc type + * @name angular.mock.TzDate + * @description + * + * *NOTE*: this is not an injectable instance, just a globally available mock class of `Date`. + * + * Mock of the Date type which has its timezone specified via constructor arg. + * + * The main purpose is to create Date-like instances with timezone fixed to the specified timezone + * offset, so that we can test code that depends on local timezone settings without dependency on + * the time zone settings of the machine where the code is running. + * + * @param {number} offset Offset of the *desired* timezone in hours (fractions will be honored) + * @param {(number|string)} timestamp Timestamp representing the desired time in *UTC* + * + * @example + * !!!! WARNING !!!!! + * This is not a complete Date object so only methods that were implemented can be called safely. + * To make matters worse, TzDate instances inherit stuff from Date via a prototype. + * + * We do our best to intercept calls to "unimplemented" methods, but since the list of methods is + * incomplete we might be missing some non-standard methods. This can result in errors like: + * "Date.prototype.foo called on incompatible Object". + * + * ```js + * var newYearInBratislava = new TzDate(-1, '2009-12-31T23:00:00Z'); + * newYearInBratislava.getTimezoneOffset() => -60; + * newYearInBratislava.getFullYear() => 2010; + * newYearInBratislava.getMonth() => 0; + * newYearInBratislava.getDate() => 1; + * newYearInBratislava.getHours() => 0; + * newYearInBratislava.getMinutes() => 0; + * newYearInBratislava.getSeconds() => 0; + * ``` + * + */ +angular.mock.TzDate = function(offset, timestamp) { + var self = new Date(0); + if (angular.isString(timestamp)) { + var tsStr = timestamp; + + self.origDate = jsonStringToDate(timestamp); + + timestamp = self.origDate.getTime(); + if (isNaN(timestamp)) { + throw { + name: "Illegal Argument", + message: "Arg '" + tsStr + "' passed into TzDate constructor is not a valid date string" + }; + } + } else { + self.origDate = new Date(timestamp); + } + + var localOffset = new Date(timestamp).getTimezoneOffset(); + self.offsetDiff = localOffset * 60 * 1000 - offset * 1000 * 60 * 60; + self.date = new Date(timestamp + self.offsetDiff); + + self.getTime = function() { + return self.date.getTime() - self.offsetDiff; + }; + + self.toLocaleDateString = function() { + return self.date.toLocaleDateString(); + }; + + self.getFullYear = function() { + return self.date.getFullYear(); + }; + + self.getMonth = function() { + return self.date.getMonth(); + }; + + self.getDate = function() { + return self.date.getDate(); + }; + + self.getHours = function() { + return self.date.getHours(); + }; + + self.getMinutes = function() { + return self.date.getMinutes(); + }; + + self.getSeconds = function() { + return self.date.getSeconds(); + }; + + self.getMilliseconds = function() { + return self.date.getMilliseconds(); + }; + + self.getTimezoneOffset = function() { + return offset * 60; + }; + + self.getUTCFullYear = function() { + return self.origDate.getUTCFullYear(); + }; + + self.getUTCMonth = function() { + return self.origDate.getUTCMonth(); + }; + + self.getUTCDate = function() { + return self.origDate.getUTCDate(); + }; + + self.getUTCHours = function() { + return self.origDate.getUTCHours(); + }; + + self.getUTCMinutes = function() { + return self.origDate.getUTCMinutes(); + }; + + self.getUTCSeconds = function() { + return self.origDate.getUTCSeconds(); + }; + + self.getUTCMilliseconds = function() { + return self.origDate.getUTCMilliseconds(); + }; + + self.getDay = function() { + return self.date.getDay(); + }; + + // provide this method only on browsers that already have it + if (self.toISOString) { + self.toISOString = function() { + return padNumber(self.origDate.getUTCFullYear(), 4) + '-' + + padNumber(self.origDate.getUTCMonth() + 1, 2) + '-' + + padNumber(self.origDate.getUTCDate(), 2) + 'T' + + padNumber(self.origDate.getUTCHours(), 2) + ':' + + padNumber(self.origDate.getUTCMinutes(), 2) + ':' + + padNumber(self.origDate.getUTCSeconds(), 2) + '.' + + padNumber(self.origDate.getUTCMilliseconds(), 3) + 'Z'; + }; + } + + //hide all methods not implemented in this mock that the Date prototype exposes + var unimplementedMethods = ['getUTCDay', + 'getYear', 'setDate', 'setFullYear', 'setHours', 'setMilliseconds', + 'setMinutes', 'setMonth', 'setSeconds', 'setTime', 'setUTCDate', 'setUTCFullYear', + 'setUTCHours', 'setUTCMilliseconds', 'setUTCMinutes', 'setUTCMonth', 'setUTCSeconds', + 'setYear', 'toDateString', 'toGMTString', 'toJSON', 'toLocaleFormat', 'toLocaleString', + 'toLocaleTimeString', 'toSource', 'toString', 'toTimeString', 'toUTCString', 'valueOf']; + + angular.forEach(unimplementedMethods, function(methodName) { + self[methodName] = function() { + throw new Error("Method '" + methodName + "' is not implemented in the TzDate mock"); + }; + }); + + return self; +}; + +//make "tzDateInstance instanceof Date" return true +angular.mock.TzDate.prototype = Date.prototype; +/* jshint +W101 */ + + +/** + * @ngdoc service + * @name $animate + * + * @description + * Mock implementation of the {@link ng.$animate `$animate`} service. Exposes two additional methods + * for testing animations. + */ +angular.mock.animate = angular.module('ngAnimateMock', ['ng']) + + .config(['$provide', function($provide) { + + $provide.factory('$$forceReflow', function() { + function reflowFn() { + reflowFn.totalReflows++; + } + reflowFn.totalReflows = 0; + return reflowFn; + }); + + $provide.factory('$$animateAsyncRun', function() { + var queue = []; + var queueFn = function() { + return function(fn) { + queue.push(fn); + }; + }; + queueFn.flush = function() { + if (queue.length === 0) return false; + + for (var i = 0; i < queue.length; i++) { + queue[i](); + } + queue = []; + + return true; + }; + return queueFn; + }); + + $provide.decorator('$$animateJs', ['$delegate', function($delegate) { + var runners = []; + + var animateJsConstructor = function() { + var animator = $delegate.apply($delegate, arguments); + // If no javascript animation is found, animator is undefined + if (animator) { + runners.push(animator); + } + return animator; + }; + + animateJsConstructor.$closeAndFlush = function() { + runners.forEach(function(runner) { + runner.end(); + }); + runners = []; + }; + + return animateJsConstructor; + }]); + + $provide.decorator('$animateCss', ['$delegate', function($delegate) { + var runners = []; + + var animateCssConstructor = function(element, options) { + var animator = $delegate(element, options); + runners.push(animator); + return animator; + }; + + animateCssConstructor.$closeAndFlush = function() { + runners.forEach(function(runner) { + runner.end(); + }); + runners = []; + }; + + return animateCssConstructor; + }]); + + $provide.decorator('$animate', ['$delegate', '$timeout', '$browser', '$$rAF', '$animateCss', '$$animateJs', + '$$forceReflow', '$$animateAsyncRun', '$rootScope', + function($delegate, $timeout, $browser, $$rAF, $animateCss, $$animateJs, + $$forceReflow, $$animateAsyncRun, $rootScope) { + var animate = { + queue: [], + cancel: $delegate.cancel, + on: $delegate.on, + off: $delegate.off, + pin: $delegate.pin, + get reflows() { + return $$forceReflow.totalReflows; + }, + enabled: $delegate.enabled, + /** + * @ngdoc method + * @name $animate#closeAndFlush + * @description + * + * This method will close all pending animations (both {@link ngAnimate#javascript-based-animations Javascript} + * and {@link ngAnimate.$animateCss CSS}) and it will also flush any remaining animation frames and/or callbacks. + */ + closeAndFlush: function() { + // we allow the flush command to swallow the errors + // because depending on whether CSS or JS animations are + // used, there may not be a RAF flush. The primary flush + // at the end of this function must throw an exception + // because it will track if there were pending animations + this.flush(true); + $animateCss.$closeAndFlush(); + $$animateJs.$closeAndFlush(); + this.flush(); + }, + /** + * @ngdoc method + * @name $animate#flush + * @description + * + * This method is used to flush the pending callbacks and animation frames to either start + * an animation or conclude an animation. Note that this will not actually close an + * actively running animation (see {@link ngMock.$animate#closeAndFlush `closeAndFlush()`} for that). + */ + flush: function(hideErrors) { + $rootScope.$digest(); + + var doNextRun, somethingFlushed = false; + do { + doNextRun = false; + + if ($$rAF.queue.length) { + $$rAF.flush(); + doNextRun = somethingFlushed = true; + } + + if ($$animateAsyncRun.flush()) { + doNextRun = somethingFlushed = true; + } + } while (doNextRun); + + if (!somethingFlushed && !hideErrors) { + throw new Error('No pending animations ready to be closed or flushed'); + } + + $rootScope.$digest(); + } + }; + + angular.forEach( + ['animate','enter','leave','move','addClass','removeClass','setClass'], function(method) { + animate[method] = function() { + animate.queue.push({ + event: method, + element: arguments[0], + options: arguments[arguments.length - 1], + args: arguments + }); + return $delegate[method].apply($delegate, arguments); + }; + }); + + return animate; + }]); + + }]); + + +/** + * @ngdoc function + * @name angular.mock.dump + * @description + * + * *NOTE*: this is not an injectable instance, just a globally available function. + * + * Method for serializing common angular objects (scope, elements, etc..) into strings, useful for + * debugging. + * + * This method is also available on window, where it can be used to display objects on debug + * console. + * + * @param {*} object - any object to turn into string. + * @return {string} a serialized string of the argument + */ +angular.mock.dump = function(object) { + return serialize(object); + + function serialize(object) { + var out; + + if (angular.isElement(object)) { + object = angular.element(object); + out = angular.element('
'); + angular.forEach(object, function(element) { + out.append(angular.element(element).clone()); + }); + out = out.html(); + } else if (angular.isArray(object)) { + out = []; + angular.forEach(object, function(o) { + out.push(serialize(o)); + }); + out = '[ ' + out.join(', ') + ' ]'; + } else if (angular.isObject(object)) { + if (angular.isFunction(object.$eval) && angular.isFunction(object.$apply)) { + out = serializeScope(object); + } else if (object instanceof Error) { + out = object.stack || ('' + object.name + ': ' + object.message); + } else { + // TODO(i): this prevents methods being logged, + // we should have a better way to serialize objects + out = angular.toJson(object, true); + } + } else { + out = String(object); + } + + return out; + } + + function serializeScope(scope, offset) { + offset = offset || ' '; + var log = [offset + 'Scope(' + scope.$id + '): {']; + for (var key in scope) { + if (Object.prototype.hasOwnProperty.call(scope, key) && !key.match(/^(\$|this)/)) { + log.push(' ' + key + ': ' + angular.toJson(scope[key])); + } + } + var child = scope.$$childHead; + while (child) { + log.push(serializeScope(child, offset + ' ')); + child = child.$$nextSibling; + } + log.push('}'); + return log.join('\n' + offset); + } +}; + +/** + * @ngdoc service + * @name $httpBackend + * @description + * Fake HTTP backend implementation suitable for unit testing applications that use the + * {@link ng.$http $http service}. + * + * *Note*: For fake HTTP backend implementation suitable for end-to-end testing or backend-less + * development please see {@link ngMockE2E.$httpBackend e2e $httpBackend mock}. + * + * During unit testing, we want our unit tests to run quickly and have no external dependencies so + * we don’t want to send [XHR](https://developer.mozilla.org/en/xmlhttprequest) or + * [JSONP](http://en.wikipedia.org/wiki/JSONP) requests to a real server. All we really need is + * to verify whether a certain request has been sent or not, or alternatively just let the + * application make requests, respond with pre-trained responses and assert that the end result is + * what we expect it to be. + * + * This mock implementation can be used to respond with static or dynamic responses via the + * `expect` and `when` apis and their shortcuts (`expectGET`, `whenPOST`, etc). + * + * When an Angular application needs some data from a server, it calls the $http service, which + * sends the request to a real server using $httpBackend service. With dependency injection, it is + * easy to inject $httpBackend mock (which has the same API as $httpBackend) and use it to verify + * the requests and respond with some testing data without sending a request to a real server. + * + * There are two ways to specify what test data should be returned as http responses by the mock + * backend when the code under test makes http requests: + * + * - `$httpBackend.expect` - specifies a request expectation + * - `$httpBackend.when` - specifies a backend definition + * + * + * # Request Expectations vs Backend Definitions + * + * Request expectations provide a way to make assertions about requests made by the application and + * to define responses for those requests. The test will fail if the expected requests are not made + * or they are made in the wrong order. + * + * Backend definitions allow you to define a fake backend for your application which doesn't assert + * if a particular request was made or not, it just returns a trained response if a request is made. + * The test will pass whether or not the request gets made during testing. + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + * + *
Request expectationsBackend definitions
Syntax.expect(...).respond(...).when(...).respond(...)
Typical usagestrict unit testsloose (black-box) unit testing
Fulfills multiple requestsNOYES
Order of requests mattersYESNO
Request requiredYESNO
Response requiredoptional (see below)YES
+ * + * In cases where both backend definitions and request expectations are specified during unit + * testing, the request expectations are evaluated first. + * + * If a request expectation has no response specified, the algorithm will search your backend + * definitions for an appropriate response. + * + * If a request didn't match any expectation or if the expectation doesn't have the response + * defined, the backend definitions are evaluated in sequential order to see if any of them match + * the request. The response from the first matched definition is returned. + * + * + * # Flushing HTTP requests + * + * The $httpBackend used in production always responds to requests asynchronously. If we preserved + * this behavior in unit testing, we'd have to create async unit tests, which are hard to write, + * to follow and to maintain. But neither can the testing mock respond synchronously; that would + * change the execution of the code under test. For this reason, the mock $httpBackend has a + * `flush()` method, which allows the test to explicitly flush pending requests. This preserves + * the async api of the backend, while allowing the test to execute synchronously. + * + * + * # Unit testing with mock $httpBackend + * The following code shows how to setup and use the mock backend when unit testing a controller. + * First we create the controller under test: + * + ```js + // The module code + angular + .module('MyApp', []) + .controller('MyController', MyController); + + // The controller code + function MyController($scope, $http) { + var authToken; + + $http.get('/auth.py').success(function(data, status, headers) { + authToken = headers('A-Token'); + $scope.user = data; + }); + + $scope.saveMessage = function(message) { + var headers = { 'Authorization': authToken }; + $scope.status = 'Saving...'; + + $http.post('/add-msg.py', message, { headers: headers } ).success(function(response) { + $scope.status = ''; + }).error(function() { + $scope.status = 'Failed...'; + }); + }; + } + ``` + * + * Now we setup the mock backend and create the test specs: + * + ```js + // testing controller + describe('MyController', function() { + var $httpBackend, $rootScope, createController, authRequestHandler; + + // Set up the module + beforeEach(module('MyApp')); + + beforeEach(inject(function($injector) { + // Set up the mock http service responses + $httpBackend = $injector.get('$httpBackend'); + // backend definition common for all tests + authRequestHandler = $httpBackend.when('GET', '/auth.py') + .respond({userId: 'userX'}, {'A-Token': 'xxx'}); + + // Get hold of a scope (i.e. the root scope) + $rootScope = $injector.get('$rootScope'); + // The $controller service is used to create instances of controllers + var $controller = $injector.get('$controller'); + + createController = function() { + return $controller('MyController', {'$scope' : $rootScope }); + }; + })); + + + afterEach(function() { + $httpBackend.verifyNoOutstandingExpectation(); + $httpBackend.verifyNoOutstandingRequest(); + }); + + + it('should fetch authentication token', function() { + $httpBackend.expectGET('/auth.py'); + var controller = createController(); + $httpBackend.flush(); + }); + + + it('should fail authentication', function() { + + // Notice how you can change the response even after it was set + authRequestHandler.respond(401, ''); + + $httpBackend.expectGET('/auth.py'); + var controller = createController(); + $httpBackend.flush(); + expect($rootScope.status).toBe('Failed...'); + }); + + + it('should send msg to server', function() { + var controller = createController(); + $httpBackend.flush(); + + // now you don’t care about the authentication, but + // the controller will still send the request and + // $httpBackend will respond without you having to + // specify the expectation and response for this request + + $httpBackend.expectPOST('/add-msg.py', 'message content').respond(201, ''); + $rootScope.saveMessage('message content'); + expect($rootScope.status).toBe('Saving...'); + $httpBackend.flush(); + expect($rootScope.status).toBe(''); + }); + + + it('should send auth header', function() { + var controller = createController(); + $httpBackend.flush(); + + $httpBackend.expectPOST('/add-msg.py', undefined, function(headers) { + // check if the header was sent, if it wasn't the expectation won't + // match the request and the test will fail + return headers['Authorization'] == 'xxx'; + }).respond(201, ''); + + $rootScope.saveMessage('whatever'); + $httpBackend.flush(); + }); + }); + ``` + */ +angular.mock.$HttpBackendProvider = function() { + this.$get = ['$rootScope', '$timeout', createHttpBackendMock]; +}; + +/** + * General factory function for $httpBackend mock. + * Returns instance for unit testing (when no arguments specified): + * - passing through is disabled + * - auto flushing is disabled + * + * Returns instance for e2e testing (when `$delegate` and `$browser` specified): + * - passing through (delegating request to real backend) is enabled + * - auto flushing is enabled + * + * @param {Object=} $delegate Real $httpBackend instance (allow passing through if specified) + * @param {Object=} $browser Auto-flushing enabled if specified + * @return {Object} Instance of $httpBackend mock + */ +function createHttpBackendMock($rootScope, $timeout, $delegate, $browser) { + var definitions = [], + expectations = [], + responses = [], + responsesPush = angular.bind(responses, responses.push), + copy = angular.copy; + + function createResponse(status, data, headers, statusText) { + if (angular.isFunction(status)) return status; + + return function() { + return angular.isNumber(status) + ? [status, data, headers, statusText] + : [200, status, data, headers]; + }; + } + + // TODO(vojta): change params to: method, url, data, headers, callback + function $httpBackend(method, url, data, callback, headers, timeout, withCredentials, responseType) { + + var xhr = new MockXhr(), + expectation = expectations[0], + wasExpected = false; + + function prettyPrint(data) { + return (angular.isString(data) || angular.isFunction(data) || data instanceof RegExp) + ? data + : angular.toJson(data); + } + + function wrapResponse(wrapped) { + if (!$browser && timeout) { + timeout.then ? timeout.then(handleTimeout) : $timeout(handleTimeout, timeout); + } + + return handleResponse; + + function handleResponse() { + var response = wrapped.response(method, url, data, headers); + xhr.$$respHeaders = response[2]; + callback(copy(response[0]), copy(response[1]), xhr.getAllResponseHeaders(), + copy(response[3] || '')); + } + + function handleTimeout() { + for (var i = 0, ii = responses.length; i < ii; i++) { + if (responses[i] === handleResponse) { + responses.splice(i, 1); + callback(-1, undefined, ''); + break; + } + } + } + } + + if (expectation && expectation.match(method, url)) { + if (!expectation.matchData(data)) { + throw new Error('Expected ' + expectation + ' with different data\n' + + 'EXPECTED: ' + prettyPrint(expectation.data) + '\nGOT: ' + data); + } + + if (!expectation.matchHeaders(headers)) { + throw new Error('Expected ' + expectation + ' with different headers\n' + + 'EXPECTED: ' + prettyPrint(expectation.headers) + '\nGOT: ' + + prettyPrint(headers)); + } + + expectations.shift(); + + if (expectation.response) { + responses.push(wrapResponse(expectation)); + return; + } + wasExpected = true; + } + + var i = -1, definition; + while ((definition = definitions[++i])) { + if (definition.match(method, url, data, headers || {})) { + if (definition.response) { + // if $browser specified, we do auto flush all requests + ($browser ? $browser.defer : responsesPush)(wrapResponse(definition)); + } else if (definition.passThrough) { + $delegate(method, url, data, callback, headers, timeout, withCredentials, responseType); + } else throw new Error('No response defined !'); + return; + } + } + throw wasExpected ? + new Error('No response defined !') : + new Error('Unexpected request: ' + method + ' ' + url + '\n' + + (expectation ? 'Expected ' + expectation : 'No more request expected')); + } + + /** + * @ngdoc method + * @name $httpBackend#when + * @description + * Creates a new backend definition. + * + * @param {string} method HTTP method. + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(string|RegExp|function(string))=} data HTTP request body or function that receives + * data string and returns true if the data is as expected. + * @param {(Object|function(Object))=} headers HTTP headers or function that receives http header + * object and returns true if the headers match the current definition. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + * + * - respond – + * `{function([status,] data[, headers, statusText]) + * | function(function(method, url, data, headers)}` + * – The respond method takes a set of static data to be returned or a function that can + * return an array containing response status (number), response data (string), response + * headers (Object), and the text for the status (string). The respond method returns the + * `requestHandler` object for possible overrides. + */ + $httpBackend.when = function(method, url, data, headers) { + var definition = new MockHttpExpectation(method, url, data, headers), + chain = { + respond: function(status, data, headers, statusText) { + definition.passThrough = undefined; + definition.response = createResponse(status, data, headers, statusText); + return chain; + } + }; + + if ($browser) { + chain.passThrough = function() { + definition.response = undefined; + definition.passThrough = true; + return chain; + }; + } + + definitions.push(definition); + return chain; + }; + + /** + * @ngdoc method + * @name $httpBackend#whenGET + * @description + * Creates a new backend definition for GET requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(Object|function(Object))=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + + /** + * @ngdoc method + * @name $httpBackend#whenHEAD + * @description + * Creates a new backend definition for HEAD requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(Object|function(Object))=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + + /** + * @ngdoc method + * @name $httpBackend#whenDELETE + * @description + * Creates a new backend definition for DELETE requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(Object|function(Object))=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + + /** + * @ngdoc method + * @name $httpBackend#whenPOST + * @description + * Creates a new backend definition for POST requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(string|RegExp|function(string))=} data HTTP request body or function that receives + * data string and returns true if the data is as expected. + * @param {(Object|function(Object))=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + + /** + * @ngdoc method + * @name $httpBackend#whenPUT + * @description + * Creates a new backend definition for PUT requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(string|RegExp|function(string))=} data HTTP request body or function that receives + * data string and returns true if the data is as expected. + * @param {(Object|function(Object))=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + + /** + * @ngdoc method + * @name $httpBackend#whenJSONP + * @description + * Creates a new backend definition for JSONP requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + createShortMethods('when'); + + + /** + * @ngdoc method + * @name $httpBackend#expect + * @description + * Creates a new request expectation. + * + * @param {string} method HTTP method. + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(string|RegExp|function(string)|Object)=} data HTTP request body or function that + * receives data string and returns true if the data is as expected, or Object if request body + * is in JSON format. + * @param {(Object|function(Object))=} headers HTTP headers or function that receives http header + * object and returns true if the headers match the current expectation. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + * + * - respond – + * `{function([status,] data[, headers, statusText]) + * | function(function(method, url, data, headers)}` + * – The respond method takes a set of static data to be returned or a function that can + * return an array containing response status (number), response data (string), response + * headers (Object), and the text for the status (string). The respond method returns the + * `requestHandler` object for possible overrides. + */ + $httpBackend.expect = function(method, url, data, headers) { + var expectation = new MockHttpExpectation(method, url, data, headers), + chain = { + respond: function(status, data, headers, statusText) { + expectation.response = createResponse(status, data, headers, statusText); + return chain; + } + }; + + expectations.push(expectation); + return chain; + }; + + + /** + * @ngdoc method + * @name $httpBackend#expectGET + * @description + * Creates a new request expectation for GET requests. For more info see `expect()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {Object=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. See #expect for more info. + */ + + /** + * @ngdoc method + * @name $httpBackend#expectHEAD + * @description + * Creates a new request expectation for HEAD requests. For more info see `expect()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {Object=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + + /** + * @ngdoc method + * @name $httpBackend#expectDELETE + * @description + * Creates a new request expectation for DELETE requests. For more info see `expect()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {Object=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + + /** + * @ngdoc method + * @name $httpBackend#expectPOST + * @description + * Creates a new request expectation for POST requests. For more info see `expect()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(string|RegExp|function(string)|Object)=} data HTTP request body or function that + * receives data string and returns true if the data is as expected, or Object if request body + * is in JSON format. + * @param {Object=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + + /** + * @ngdoc method + * @name $httpBackend#expectPUT + * @description + * Creates a new request expectation for PUT requests. For more info see `expect()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(string|RegExp|function(string)|Object)=} data HTTP request body or function that + * receives data string and returns true if the data is as expected, or Object if request body + * is in JSON format. + * @param {Object=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + + /** + * @ngdoc method + * @name $httpBackend#expectPATCH + * @description + * Creates a new request expectation for PATCH requests. For more info see `expect()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(string|RegExp|function(string)|Object)=} data HTTP request body or function that + * receives data string and returns true if the data is as expected, or Object if request body + * is in JSON format. + * @param {Object=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + + /** + * @ngdoc method + * @name $httpBackend#expectJSONP + * @description + * Creates a new request expectation for JSONP requests. For more info see `expect()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives an url + * and returns true if the url matches the current definition. + * @returns {requestHandler} Returns an object with `respond` method that controls how a matched + * request is handled. You can save this object for later use and invoke `respond` again in + * order to change how a matched request is handled. + */ + createShortMethods('expect'); + + + /** + * @ngdoc method + * @name $httpBackend#flush + * @description + * Flushes all pending requests using the trained responses. + * + * @param {number=} count Number of responses to flush (in the order they arrived). If undefined, + * all pending requests will be flushed. If there are no pending requests when the flush method + * is called an exception is thrown (as this typically a sign of programming error). + */ + $httpBackend.flush = function(count, digest) { + if (digest !== false) $rootScope.$digest(); + if (!responses.length) throw new Error('No pending request to flush !'); + + if (angular.isDefined(count) && count !== null) { + while (count--) { + if (!responses.length) throw new Error('No more pending request to flush !'); + responses.shift()(); + } + } else { + while (responses.length) { + responses.shift()(); + } + } + $httpBackend.verifyNoOutstandingExpectation(digest); + }; + + + /** + * @ngdoc method + * @name $httpBackend#verifyNoOutstandingExpectation + * @description + * Verifies that all of the requests defined via the `expect` api were made. If any of the + * requests were not made, verifyNoOutstandingExpectation throws an exception. + * + * Typically, you would call this method following each test case that asserts requests using an + * "afterEach" clause. + * + * ```js + * afterEach($httpBackend.verifyNoOutstandingExpectation); + * ``` + */ + $httpBackend.verifyNoOutstandingExpectation = function(digest) { + if (digest !== false) $rootScope.$digest(); + if (expectations.length) { + throw new Error('Unsatisfied requests: ' + expectations.join(', ')); + } + }; + + + /** + * @ngdoc method + * @name $httpBackend#verifyNoOutstandingRequest + * @description + * Verifies that there are no outstanding requests that need to be flushed. + * + * Typically, you would call this method following each test case that asserts requests using an + * "afterEach" clause. + * + * ```js + * afterEach($httpBackend.verifyNoOutstandingRequest); + * ``` + */ + $httpBackend.verifyNoOutstandingRequest = function() { + if (responses.length) { + throw new Error('Unflushed requests: ' + responses.length); + } + }; + + + /** + * @ngdoc method + * @name $httpBackend#resetExpectations + * @description + * Resets all request expectations, but preserves all backend definitions. Typically, you would + * call resetExpectations during a multiple-phase test when you want to reuse the same instance of + * $httpBackend mock. + */ + $httpBackend.resetExpectations = function() { + expectations.length = 0; + responses.length = 0; + }; + + return $httpBackend; + + + function createShortMethods(prefix) { + angular.forEach(['GET', 'DELETE', 'JSONP', 'HEAD'], function(method) { + $httpBackend[prefix + method] = function(url, headers) { + return $httpBackend[prefix](method, url, undefined, headers); + }; + }); + + angular.forEach(['PUT', 'POST', 'PATCH'], function(method) { + $httpBackend[prefix + method] = function(url, data, headers) { + return $httpBackend[prefix](method, url, data, headers); + }; + }); + } +} + +function MockHttpExpectation(method, url, data, headers) { + + this.data = data; + this.headers = headers; + + this.match = function(m, u, d, h) { + if (method != m) return false; + if (!this.matchUrl(u)) return false; + if (angular.isDefined(d) && !this.matchData(d)) return false; + if (angular.isDefined(h) && !this.matchHeaders(h)) return false; + return true; + }; + + this.matchUrl = function(u) { + if (!url) return true; + if (angular.isFunction(url.test)) return url.test(u); + if (angular.isFunction(url)) return url(u); + return url == u; + }; + + this.matchHeaders = function(h) { + if (angular.isUndefined(headers)) return true; + if (angular.isFunction(headers)) return headers(h); + return angular.equals(headers, h); + }; + + this.matchData = function(d) { + if (angular.isUndefined(data)) return true; + if (data && angular.isFunction(data.test)) return data.test(d); + if (data && angular.isFunction(data)) return data(d); + if (data && !angular.isString(data)) { + return angular.equals(angular.fromJson(angular.toJson(data)), angular.fromJson(d)); + } + return data == d; + }; + + this.toString = function() { + return method + ' ' + url; + }; +} + +function createMockXhr() { + return new MockXhr(); +} + +function MockXhr() { + + // hack for testing $http, $httpBackend + MockXhr.$$lastInstance = this; + + this.open = function(method, url, async) { + this.$$method = method; + this.$$url = url; + this.$$async = async; + this.$$reqHeaders = {}; + this.$$respHeaders = {}; + }; + + this.send = function(data) { + this.$$data = data; + }; + + this.setRequestHeader = function(key, value) { + this.$$reqHeaders[key] = value; + }; + + this.getResponseHeader = function(name) { + // the lookup must be case insensitive, + // that's why we try two quick lookups first and full scan last + var header = this.$$respHeaders[name]; + if (header) return header; + + name = angular.lowercase(name); + header = this.$$respHeaders[name]; + if (header) return header; + + header = undefined; + angular.forEach(this.$$respHeaders, function(headerVal, headerName) { + if (!header && angular.lowercase(headerName) == name) header = headerVal; + }); + return header; + }; + + this.getAllResponseHeaders = function() { + var lines = []; + + angular.forEach(this.$$respHeaders, function(value, key) { + lines.push(key + ': ' + value); + }); + return lines.join('\n'); + }; + + this.abort = angular.noop; +} + + +/** + * @ngdoc service + * @name $timeout + * @description + * + * This service is just a simple decorator for {@link ng.$timeout $timeout} service + * that adds a "flush" and "verifyNoPendingTasks" methods. + */ + +angular.mock.$TimeoutDecorator = ['$delegate', '$browser', function($delegate, $browser) { + + /** + * @ngdoc method + * @name $timeout#flush + * @description + * + * Flushes the queue of pending tasks. + * + * @param {number=} delay maximum timeout amount to flush up until + */ + $delegate.flush = function(delay) { + $browser.defer.flush(delay); + }; + + /** + * @ngdoc method + * @name $timeout#verifyNoPendingTasks + * @description + * + * Verifies that there are no pending tasks that need to be flushed. + */ + $delegate.verifyNoPendingTasks = function() { + if ($browser.deferredFns.length) { + throw new Error('Deferred tasks to flush (' + $browser.deferredFns.length + '): ' + + formatPendingTasksAsString($browser.deferredFns)); + } + }; + + function formatPendingTasksAsString(tasks) { + var result = []; + angular.forEach(tasks, function(task) { + result.push('{id: ' + task.id + ', ' + 'time: ' + task.time + '}'); + }); + + return result.join(', '); + } + + return $delegate; +}]; + +angular.mock.$RAFDecorator = ['$delegate', function($delegate) { + var rafFn = function(fn) { + var index = rafFn.queue.length; + rafFn.queue.push(fn); + return function() { + rafFn.queue.splice(index, 1); + }; + }; + + rafFn.queue = []; + rafFn.supported = $delegate.supported; + + rafFn.flush = function() { + if (rafFn.queue.length === 0) { + throw new Error('No rAF callbacks present'); + } + + var length = rafFn.queue.length; + for (var i = 0; i < length; i++) { + rafFn.queue[i](); + } + + rafFn.queue = rafFn.queue.slice(i); + }; + + return rafFn; +}]; + +/** + * + */ +angular.mock.$RootElementProvider = function() { + this.$get = function() { + return angular.element('
'); + }; +}; + +/** + * @ngdoc service + * @name $controller + * @description + * A decorator for {@link ng.$controller} with additional `bindings` parameter, useful when testing + * controllers of directives that use {@link $compile#-bindtocontroller- `bindToController`}. + * + * + * ## Example + * + * ```js + * + * // Directive definition ... + * + * myMod.directive('myDirective', { + * controller: 'MyDirectiveController', + * bindToController: { + * name: '@' + * } + * }); + * + * + * // Controller definition ... + * + * myMod.controller('MyDirectiveController', ['$log', function($log) { + * $log.info(this.name); + * }]); + * + * + * // In a test ... + * + * describe('myDirectiveController', function() { + * it('should write the bound name to the log', inject(function($controller, $log) { + * var ctrl = $controller('MyDirectiveController', { /* no locals */ }, { name: 'Clark Kent' }); + * expect(ctrl.name).toEqual('Clark Kent'); + * expect($log.info.logs).toEqual(['Clark Kent']); + * })); + * }); + * + * ``` + * + * @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. + * @param {Object=} bindings Properties to add to the controller before invoking the constructor. This is used + * to simulate the `bindToController` feature and simplify certain kinds of tests. + * @return {Object} Instance of given controller. + */ +angular.mock.$ControllerDecorator = ['$delegate', function($delegate) { + return function(expression, locals, later, ident) { + if (later && typeof later === 'object') { + var create = $delegate(expression, locals, true, ident); + angular.extend(create.instance, later); + return create(); + } + return $delegate(expression, locals, later, ident); + }; +}]; + + +/** + * @ngdoc module + * @name ngMock + * @packageName angular-mocks + * @description + * + * # ngMock + * + * The `ngMock` module provides support to inject and mock Angular services into unit tests. + * In addition, ngMock also extends various core ng services such that they can be + * inspected and controlled in a synchronous manner within test code. + * + * + *
+ * + */ +angular.module('ngMock', ['ng']).provider({ + $browser: angular.mock.$BrowserProvider, + $exceptionHandler: angular.mock.$ExceptionHandlerProvider, + $log: angular.mock.$LogProvider, + $interval: angular.mock.$IntervalProvider, + $httpBackend: angular.mock.$HttpBackendProvider, + $rootElement: angular.mock.$RootElementProvider +}).config(['$provide', function($provide) { + $provide.decorator('$timeout', angular.mock.$TimeoutDecorator); + $provide.decorator('$$rAF', angular.mock.$RAFDecorator); + $provide.decorator('$rootScope', angular.mock.$RootScopeDecorator); + $provide.decorator('$controller', angular.mock.$ControllerDecorator); +}]); + +/** + * @ngdoc module + * @name ngMockE2E + * @module ngMockE2E + * @packageName angular-mocks + * @description + * + * The `ngMockE2E` is an angular module which contains mocks suitable for end-to-end testing. + * Currently there is only one mock present in this module - + * the {@link ngMockE2E.$httpBackend e2e $httpBackend} mock. + */ +angular.module('ngMockE2E', ['ng']).config(['$provide', function($provide) { + $provide.decorator('$httpBackend', angular.mock.e2e.$httpBackendDecorator); +}]); + +/** + * @ngdoc service + * @name $httpBackend + * @module ngMockE2E + * @description + * Fake HTTP backend implementation suitable for end-to-end testing or backend-less development of + * applications that use the {@link ng.$http $http service}. + * + * *Note*: For fake http backend implementation suitable for unit testing please see + * {@link ngMock.$httpBackend unit-testing $httpBackend mock}. + * + * This implementation can be used to respond with static or dynamic responses via the `when` api + * and its shortcuts (`whenGET`, `whenPOST`, etc) and optionally pass through requests to the + * real $httpBackend for specific requests (e.g. to interact with certain remote apis or to fetch + * templates from a webserver). + * + * As opposed to unit-testing, in an end-to-end testing scenario or in scenario when an application + * is being developed with the real backend api replaced with a mock, it is often desirable for + * certain category of requests to bypass the mock and issue a real http request (e.g. to fetch + * templates or static files from the webserver). To configure the backend with this behavior + * use the `passThrough` request handler of `when` instead of `respond`. + * + * Additionally, we don't want to manually have to flush mocked out requests like we do during unit + * testing. For this reason the e2e $httpBackend flushes mocked out requests + * automatically, closely simulating the behavior of the XMLHttpRequest object. + * + * To setup the application to run with this http backend, you have to create a module that depends + * on the `ngMockE2E` and your application modules and defines the fake backend: + * + * ```js + * myAppDev = angular.module('myAppDev', ['myApp', 'ngMockE2E']); + * myAppDev.run(function($httpBackend) { + * phones = [{name: 'phone1'}, {name: 'phone2'}]; + * + * // returns the current list of phones + * $httpBackend.whenGET('/phones').respond(phones); + * + * // adds a new phone to the phones array + * $httpBackend.whenPOST('/phones').respond(function(method, url, data) { + * var phone = angular.fromJson(data); + * phones.push(phone); + * return [200, phone, {}]; + * }); + * $httpBackend.whenGET(/^\/templates\//).passThrough(); + * //... + * }); + * ``` + * + * Afterwards, bootstrap your app with this new module. + */ + +/** + * @ngdoc method + * @name $httpBackend#when + * @module ngMockE2E + * @description + * Creates a new backend definition. + * + * @param {string} method HTTP method. + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(string|RegExp)=} data HTTP request body. + * @param {(Object|function(Object))=} headers HTTP headers or function that receives http header + * object and returns true if the headers match the current definition. + * @returns {requestHandler} Returns an object with `respond` and `passThrough` methods that + * control how a matched request is handled. You can save this object for later use and invoke + * `respond` or `passThrough` again in order to change how a matched request is handled. + * + * - respond – + * `{function([status,] data[, headers, statusText]) + * | function(function(method, url, data, headers)}` + * – The respond method takes a set of static data to be returned or a function that can return + * an array containing response status (number), response data (string), response headers + * (Object), and the text for the status (string). + * - passThrough – `{function()}` – Any request matching a backend definition with + * `passThrough` handler will be passed through to the real backend (an XHR request will be made + * to the server.) + * - Both methods return the `requestHandler` object for possible overrides. + */ + +/** + * @ngdoc method + * @name $httpBackend#whenGET + * @module ngMockE2E + * @description + * Creates a new backend definition for GET requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(Object|function(Object))=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` and `passThrough` methods that + * control how a matched request is handled. You can save this object for later use and invoke + * `respond` or `passThrough` again in order to change how a matched request is handled. + */ + +/** + * @ngdoc method + * @name $httpBackend#whenHEAD + * @module ngMockE2E + * @description + * Creates a new backend definition for HEAD requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(Object|function(Object))=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` and `passThrough` methods that + * control how a matched request is handled. You can save this object for later use and invoke + * `respond` or `passThrough` again in order to change how a matched request is handled. + */ + +/** + * @ngdoc method + * @name $httpBackend#whenDELETE + * @module ngMockE2E + * @description + * Creates a new backend definition for DELETE requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(Object|function(Object))=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` and `passThrough` methods that + * control how a matched request is handled. You can save this object for later use and invoke + * `respond` or `passThrough` again in order to change how a matched request is handled. + */ + +/** + * @ngdoc method + * @name $httpBackend#whenPOST + * @module ngMockE2E + * @description + * Creates a new backend definition for POST requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(string|RegExp)=} data HTTP request body. + * @param {(Object|function(Object))=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` and `passThrough` methods that + * control how a matched request is handled. You can save this object for later use and invoke + * `respond` or `passThrough` again in order to change how a matched request is handled. + */ + +/** + * @ngdoc method + * @name $httpBackend#whenPUT + * @module ngMockE2E + * @description + * Creates a new backend definition for PUT requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(string|RegExp)=} data HTTP request body. + * @param {(Object|function(Object))=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` and `passThrough` methods that + * control how a matched request is handled. You can save this object for later use and invoke + * `respond` or `passThrough` again in order to change how a matched request is handled. + */ + +/** + * @ngdoc method + * @name $httpBackend#whenPATCH + * @module ngMockE2E + * @description + * Creates a new backend definition for PATCH requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @param {(string|RegExp)=} data HTTP request body. + * @param {(Object|function(Object))=} headers HTTP headers. + * @returns {requestHandler} Returns an object with `respond` and `passThrough` methods that + * control how a matched request is handled. You can save this object for later use and invoke + * `respond` or `passThrough` again in order to change how a matched request is handled. + */ + +/** + * @ngdoc method + * @name $httpBackend#whenJSONP + * @module ngMockE2E + * @description + * Creates a new backend definition for JSONP requests. For more info see `when()`. + * + * @param {string|RegExp|function(string)} url HTTP url or function that receives a url + * and returns true if the url matches the current definition. + * @returns {requestHandler} Returns an object with `respond` and `passThrough` methods that + * control how a matched request is handled. You can save this object for later use and invoke + * `respond` or `passThrough` again in order to change how a matched request is handled. + */ +angular.mock.e2e = {}; +angular.mock.e2e.$httpBackendDecorator = + ['$rootScope', '$timeout', '$delegate', '$browser', createHttpBackendMock]; + + +/** + * @ngdoc type + * @name $rootScope.Scope + * @module ngMock + * @description + * {@link ng.$rootScope.Scope Scope} type decorated with helper methods useful for testing. These + * methods are automatically available on any {@link ng.$rootScope.Scope Scope} instance when + * `ngMock` module is loaded. + * + * In addition to all the regular `Scope` methods, the following helper methods are available: + */ +angular.mock.$RootScopeDecorator = ['$delegate', function($delegate) { + + var $rootScopePrototype = Object.getPrototypeOf($delegate); + + $rootScopePrototype.$countChildScopes = countChildScopes; + $rootScopePrototype.$countWatchers = countWatchers; + + return $delegate; + + // ------------------------------------------------------------------------------------------ // + + /** + * @ngdoc method + * @name $rootScope.Scope#$countChildScopes + * @module ngMock + * @description + * Counts all the direct and indirect child scopes of the current scope. + * + * The current scope is excluded from the count. The count includes all isolate child scopes. + * + * @returns {number} Total number of child scopes. + */ + function countChildScopes() { + // jshint validthis: true + var count = 0; // exclude the current scope + var pendingChildHeads = [this.$$childHead]; + var currentScope; + + while (pendingChildHeads.length) { + currentScope = pendingChildHeads.shift(); + + while (currentScope) { + count += 1; + pendingChildHeads.push(currentScope.$$childHead); + currentScope = currentScope.$$nextSibling; + } + } + + return count; + } + + + /** + * @ngdoc method + * @name $rootScope.Scope#$countWatchers + * @module ngMock + * @description + * Counts all the watchers of direct and indirect child scopes of the current scope. + * + * The watchers of the current scope are included in the count and so are all the watchers of + * isolate child scopes. + * + * @returns {number} Total number of watchers. + */ + function countWatchers() { + // jshint validthis: true + var count = this.$$watchers ? this.$$watchers.length : 0; // include the current scope + var pendingChildHeads = [this.$$childHead]; + var currentScope; + + while (pendingChildHeads.length) { + currentScope = pendingChildHeads.shift(); + + while (currentScope) { + count += currentScope.$$watchers ? currentScope.$$watchers.length : 0; + pendingChildHeads.push(currentScope.$$childHead); + currentScope = currentScope.$$nextSibling; + } + } + + return count; + } +}]; + + +if (window.jasmine || window.mocha) { + + var currentSpec = null, + annotatedFunctions = [], + isSpecRunning = function() { + return !!currentSpec; + }; + + angular.mock.$$annotate = angular.injector.$$annotate; + angular.injector.$$annotate = function(fn) { + if (typeof fn === 'function' && !fn.$inject) { + annotatedFunctions.push(fn); + } + return angular.mock.$$annotate.apply(this, arguments); + }; + + + (window.beforeEach || window.setup)(function() { + annotatedFunctions = []; + currentSpec = this; + }); + + (window.afterEach || window.teardown)(function() { + var injector = currentSpec.$injector; + + annotatedFunctions.forEach(function(fn) { + delete fn.$inject; + }); + + angular.forEach(currentSpec.$modules, function(module) { + if (module && module.$$hashKey) { + module.$$hashKey = undefined; + } + }); + + currentSpec.$injector = null; + currentSpec.$modules = null; + currentSpec = null; + + if (injector) { + injector.get('$rootElement').off(); + } + + // clean up jquery's fragment cache + angular.forEach(angular.element.fragments, function(val, key) { + delete angular.element.fragments[key]; + }); + + MockXhr.$$lastInstance = null; + + angular.forEach(angular.callbacks, function(val, key) { + delete angular.callbacks[key]; + }); + angular.callbacks.counter = 0; + }); + + /** + * @ngdoc function + * @name angular.mock.module + * @description + * + * *NOTE*: This function is also published on window for easy access.
+ * *NOTE*: This function is declared ONLY WHEN running tests with jasmine or mocha + * + * This function registers a module configuration code. It collects the configuration information + * which will be used when the injector is created by {@link angular.mock.inject inject}. + * + * See {@link angular.mock.inject inject} for usage example + * + * @param {...(string|Function|Object)} fns any number of modules which are represented as string + * aliases or as anonymous module initialization functions. The modules are used to + * configure the injector. The 'ng' and 'ngMock' modules are automatically loaded. If an + * object literal is passed each key-value pair will be registered on the module via + * {@link auto.$provide $provide}.value, the key being the string name (or token) to associate + * with the value on the injector. + */ + window.module = angular.mock.module = function() { + var moduleFns = Array.prototype.slice.call(arguments, 0); + return isSpecRunning() ? workFn() : workFn; + ///////////////////// + function workFn() { + if (currentSpec.$injector) { + throw new Error('Injector already created, can not register a module!'); + } else { + var modules = currentSpec.$modules || (currentSpec.$modules = []); + angular.forEach(moduleFns, function(module) { + if (angular.isObject(module) && !angular.isArray(module)) { + modules.push(function($provide) { + angular.forEach(module, function(value, key) { + $provide.value(key, value); + }); + }); + } else { + modules.push(module); + } + }); + } + } + }; + + /** + * @ngdoc function + * @name angular.mock.inject + * @description + * + * *NOTE*: This function is also published on window for easy access.
+ * *NOTE*: This function is declared ONLY WHEN running tests with jasmine or mocha + * + * The inject function wraps a function into an injectable function. The inject() creates new + * instance of {@link auto.$injector $injector} per test, which is then used for + * resolving references. + * + * + * ## Resolving References (Underscore Wrapping) + * Often, we would like to inject a reference once, in a `beforeEach()` block and reuse this + * in multiple `it()` clauses. To be able to do this we must assign the reference to a variable + * that is declared in the scope of the `describe()` block. Since we would, most likely, want + * the variable to have the same name of the reference we have a problem, since the parameter + * to the `inject()` function would hide the outer variable. + * + * To help with this, the injected parameters can, optionally, be enclosed with underscores. + * These are ignored by the injector when the reference name is resolved. + * + * For example, the parameter `_myService_` would be resolved as the reference `myService`. + * Since it is available in the function body as _myService_, we can then assign it to a variable + * defined in an outer scope. + * + * ``` + * // Defined out reference variable outside + * var myService; + * + * // Wrap the parameter in underscores + * beforeEach( inject( function(_myService_){ + * myService = _myService_; + * })); + * + * // Use myService in a series of tests. + * it('makes use of myService', function() { + * myService.doStuff(); + * }); + * + * ``` + * + * See also {@link angular.mock.module angular.mock.module} + * + * ## Example + * Example of what a typical jasmine tests looks like with the inject method. + * ```js + * + * angular.module('myApplicationModule', []) + * .value('mode', 'app') + * .value('version', 'v1.0.1'); + * + * + * describe('MyApp', function() { + * + * // You need to load modules that you want to test, + * // it loads only the "ng" module by default. + * beforeEach(module('myApplicationModule')); + * + * + * // inject() is used to inject arguments of all given functions + * it('should provide a version', inject(function(mode, version) { + * expect(version).toEqual('v1.0.1'); + * expect(mode).toEqual('app'); + * })); + * + * + * // The inject and module method can also be used inside of the it or beforeEach + * it('should override a version and test the new version is injected', function() { + * // module() takes functions or strings (module aliases) + * module(function($provide) { + * $provide.value('version', 'overridden'); // override version here + * }); + * + * inject(function(version) { + * expect(version).toEqual('overridden'); + * }); + * }); + * }); + * + * ``` + * + * @param {...Function} fns any number of functions which will be injected using the injector. + */ + + + + var ErrorAddingDeclarationLocationStack = function(e, errorForStack) { + this.message = e.message; + this.name = e.name; + if (e.line) this.line = e.line; + if (e.sourceId) this.sourceId = e.sourceId; + if (e.stack && errorForStack) + this.stack = e.stack + '\n' + errorForStack.stack; + if (e.stackArray) this.stackArray = e.stackArray; + }; + ErrorAddingDeclarationLocationStack.prototype.toString = Error.prototype.toString; + + window.inject = angular.mock.inject = function() { + var blockFns = Array.prototype.slice.call(arguments, 0); + var errorForStack = new Error('Declaration Location'); + return isSpecRunning() ? workFn.call(currentSpec) : workFn; + ///////////////////// + function workFn() { + var modules = currentSpec.$modules || []; + var strictDi = !!currentSpec.$injectorStrict; + modules.unshift('ngMock'); + modules.unshift('ng'); + var injector = currentSpec.$injector; + if (!injector) { + if (strictDi) { + // If strictDi is enabled, annotate the providerInjector blocks + angular.forEach(modules, function(moduleFn) { + if (typeof moduleFn === "function") { + angular.injector.$$annotate(moduleFn); + } + }); + } + injector = currentSpec.$injector = angular.injector(modules, strictDi); + currentSpec.$injectorStrict = strictDi; + } + for (var i = 0, ii = blockFns.length; i < ii; i++) { + if (currentSpec.$injectorStrict) { + // If the injector is strict / strictDi, and the spec wants to inject using automatic + // annotation, then annotate the function here. + injector.annotate(blockFns[i]); + } + try { + /* jshint -W040 *//* Jasmine explicitly provides a `this` object when calling functions */ + injector.invoke(blockFns[i] || angular.noop, this); + /* jshint +W040 */ + } catch (e) { + if (e.stack && errorForStack) { + throw new ErrorAddingDeclarationLocationStack(e, errorForStack); + } + throw e; + } finally { + errorForStack = null; + } + } + } + }; + + + angular.mock.inject.strictDi = function(value) { + value = arguments.length ? !!value : true; + return isSpecRunning() ? workFn() : workFn; + + function workFn() { + if (value !== currentSpec.$injectorStrict) { + if (currentSpec.$injector) { + throw new Error('Injector already created, can not modify strict annotations'); + } else { + currentSpec.$injectorStrict = value; + } + } + } + }; +} + + +})(window, window.angular); diff --git a/1.4.10/angular-resource.js b/1.4.10/angular-resource.js new file mode 100644 index 0000000000..4cde1b74a9 --- /dev/null +++ b/1.4.10/angular-resource.js @@ -0,0 +1,694 @@ +/** + * @license AngularJS v1.4.10 + * (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. + * + * + *
+ * + * 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 a 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.=} 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.}` – + * 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.}` – + * 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}` – timeout in milliseconds.
+ * **Note:** In contrast to {@link ng.$http#usage $http.config}, {@link ng.$q promises} are + * **not** supported in $resource, because the same value would be used for multiple requests. + * If you need support for cancellable $resource actions, you should upgrade to version 1.5 or + * higher. + * - **`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 rejected 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', '$log', '$q', function($http, $log, $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': + if (value && !angular.isNumber(value)) { + $log.debug('ngResource:\n' + + ' Only numeric values are allowed as `timeout`.\n' + + ' Promises are not supported in $resource, because the same value would ' + + 'be used for multiple requests.\n' + + ' If you need support for cancellable $resource actions, you should ' + + 'upgrade to version 1.5 or higher.'); + } + 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); diff --git a/1.4.10/angular-resource.min.js b/1.4.10/angular-resource.min.js new file mode 100644 index 0000000000..8de60b9864 --- /dev/null +++ b/1.4.10/angular-resource.min.js @@ -0,0 +1,14 @@ +/* + AngularJS v1.4.10 + (c) 2010-2015 Google, Inc. http://angularjs.org + License: MIT +*/ +(function(J,f,C){'use strict';function D(t,e){e=e||{};f.forEach(e,function(f,k){delete e[k]});for(var k in t)!t.hasOwnProperty(k)||"$"===k.charAt(0)&&"$"===k.charAt(1)||(e[k]=t[k]);return e}var y=f.$$minErr("$resource"),B=/^(\.[a-zA-Z_$@][0-9a-zA-Z_$@]*)+$/;f.module("ngResource",["ng"]).provider("$resource",function(){var t=/^https?:\/\/[^\/]*/,e=this;this.defaults={stripTrailingSlashes:!0,actions:{get:{method:"GET"},save:{method:"POST"},query:{method:"GET",isArray:!0},remove:{method:"DELETE"},"delete":{method:"DELETE"}}}; +this.$get=["$http","$log","$q",function(k,F,G){function w(f,g){this.template=f;this.defaults=r({},e.defaults,g);this.urlParams={}}function z(l,g,s,h){function c(a,q){var c={};q=r({},g,q);u(q,function(b,q){x(b)&&(b=b());var m;if(b&&b.charAt&&"@"==b.charAt(0)){m=a;var d=b.substr(1);if(null==d||""===d||"hasOwnProperty"===d||!B.test("."+d))throw y("badmember",d);for(var d=d.split("."),n=0,g=d.length;n + */ + /* global -ngRouteModule */ +var ngRouteModule = angular.module('ngRoute', ['ng']). + provider('$route', $RouteProvider), + $routeMinErr = angular.$$minErr('ngRoute'); + +/** + * @ngdoc provider + * @name $routeProvider + * + * @description + * + * Used for configuring routes. + * + * ## Example + * See {@link ngRoute.$route#example $route} for an example of configuring and using `ngRoute`. + * + * ## Dependencies + * Requires the {@link ngRoute `ngRoute`} module to be installed. + */ +function $RouteProvider() { + function inherit(parent, extra) { + return angular.extend(Object.create(parent), extra); + } + + var routes = {}; + + /** + * @ngdoc method + * @name $routeProvider#when + * + * @param {string} path Route path (matched against `$location.path`). If `$location.path` + * contains redundant trailing slash or is missing one, the route will still match and the + * `$location.path` will be updated to add or drop the trailing slash to exactly match the + * route definition. + * + * * `path` can contain named groups starting with a colon: e.g. `:name`. All characters up + * to the next slash are matched and stored in `$routeParams` under the given `name` + * when the route matches. + * * `path` can contain named groups starting with a colon and ending with a star: + * e.g.`:name*`. All characters are eagerly stored in `$routeParams` under the given `name` + * when the route matches. + * * `path` can contain optional named groups with a question mark: e.g.`:name?`. + * + * For example, routes like `/color/:color/largecode/:largecode*\/edit` will match + * `/color/brown/largecode/code/with/slashes/edit` and extract: + * + * * `color: brown` + * * `largecode: code/with/slashes`. + * + * + * @param {Object} route Mapping information to be assigned to `$route.current` on route + * match. + * + * Object properties: + * + * - `controller` – `{(string|function()=}` – Controller fn that should be associated with + * newly created scope or the name of a {@link angular.Module#controller registered + * controller} if passed as a string. + * - `controllerAs` – `{string=}` – An identifier name for a reference to the controller. + * If present, the controller will be published to scope under the `controllerAs` name. + * - `template` – `{string=|function()=}` – html template as a string or a function that + * returns an html template as a string which should be used by {@link + * ngRoute.directive:ngView ngView} or {@link ng.directive:ngInclude ngInclude} directives. + * This property takes precedence over `templateUrl`. + * + * If `template` is a function, it will be called with the following parameters: + * + * - `{Array.}` - route parameters extracted from the current + * `$location.path()` by applying the current route + * + * - `templateUrl` – `{string=|function()=}` – path or function that returns a path to an html + * template that should be used by {@link ngRoute.directive:ngView ngView}. + * + * If `templateUrl` is a function, it will be called with the following parameters: + * + * - `{Array.}` - route parameters extracted from the current + * `$location.path()` by applying the current route + * + * - `resolve` - `{Object.=}` - An optional map of dependencies which should + * be injected into the controller. If any of these dependencies are promises, the router + * will wait for them all to be resolved or one to be rejected before the controller is + * instantiated. + * If all the promises are resolved successfully, the values of the resolved promises are + * injected and {@link ngRoute.$route#$routeChangeSuccess $routeChangeSuccess} event is + * fired. If any of the promises are rejected the + * {@link ngRoute.$route#$routeChangeError $routeChangeError} event is fired. The map object + * is: + * + * - `key` – `{string}`: a name of a dependency to be injected into the controller. + * - `factory` - `{string|function}`: If `string` then it is an alias for a service. + * Otherwise if function, then it is {@link auto.$injector#invoke injected} + * and the return value is treated as the dependency. If the result is a promise, it is + * resolved before its value is injected into the controller. Be aware that + * `ngRoute.$routeParams` will still refer to the previous route within these resolve + * functions. Use `$route.current.params` to access the new route parameters, instead. + * + * - `redirectTo` – {(string|function())=} – value to update + * {@link ng.$location $location} path with and trigger route redirection. + * + * If `redirectTo` is a function, it will be called with the following parameters: + * + * - `{Object.}` - route parameters extracted from the current + * `$location.path()` by applying the current route templateUrl. + * - `{string}` - current `$location.path()` + * - `{Object}` - current `$location.search()` + * + * The custom `redirectTo` function is expected to return a string which will be used + * to update `$location.path()` and `$location.search()`. + * + * - `[reloadOnSearch=true]` - {boolean=} - reload route when only `$location.search()` + * or `$location.hash()` changes. + * + * If the option is set to `false` and url in the browser changes, then + * `$routeUpdate` event is broadcasted on the root scope. + * + * - `[caseInsensitiveMatch=false]` - {boolean=} - match routes without being case sensitive + * + * If the option is set to `true`, then the particular route can be matched without being + * case sensitive + * + * @returns {Object} self + * + * @description + * Adds a new route definition to the `$route` service. + */ + this.when = function(path, route) { + //copy original route object to preserve params inherited from proto chain + var routeCopy = angular.copy(route); + if (angular.isUndefined(routeCopy.reloadOnSearch)) { + routeCopy.reloadOnSearch = true; + } + if (angular.isUndefined(routeCopy.caseInsensitiveMatch)) { + routeCopy.caseInsensitiveMatch = this.caseInsensitiveMatch; + } + routes[path] = angular.extend( + routeCopy, + path && pathRegExp(path, routeCopy) + ); + + // create redirection for trailing slashes + if (path) { + var redirectPath = (path[path.length - 1] == '/') + ? path.substr(0, path.length - 1) + : path + '/'; + + routes[redirectPath] = angular.extend( + {redirectTo: path}, + pathRegExp(redirectPath, routeCopy) + ); + } + + return this; + }; + + /** + * @ngdoc property + * @name $routeProvider#caseInsensitiveMatch + * @description + * + * A boolean property indicating if routes defined + * using this provider should be matched using a case insensitive + * algorithm. Defaults to `false`. + */ + this.caseInsensitiveMatch = false; + + /** + * @param path {string} path + * @param opts {Object} options + * @return {?Object} + * + * @description + * Normalizes the given path, returning a regular expression + * and the original path. + * + * Inspired by pathRexp in visionmedia/express/lib/utils.js. + */ + function pathRegExp(path, opts) { + var insensitive = opts.caseInsensitiveMatch, + ret = { + originalPath: path, + regexp: path + }, + keys = ret.keys = []; + + path = path + .replace(/([().])/g, '\\$1') + .replace(/(\/)?:(\w+)(\*\?|[\?\*])?/g, function(_, slash, key, option) { + var optional = (option === '?' || option === '*?') ? '?' : null; + var star = (option === '*' || option === '*?') ? '*' : null; + keys.push({ name: key, optional: !!optional }); + slash = slash || ''; + return '' + + (optional ? '' : slash) + + '(?:' + + (optional ? slash : '') + + (star && '(.+?)' || '([^/]+)') + + (optional || '') + + ')' + + (optional || ''); + }) + .replace(/([\/$\*])/g, '\\$1'); + + ret.regexp = new RegExp('^' + path + '$', insensitive ? 'i' : ''); + return ret; + } + + /** + * @ngdoc method + * @name $routeProvider#otherwise + * + * @description + * Sets route definition that will be used on route change when no other route definition + * is matched. + * + * @param {Object|string} params Mapping information to be assigned to `$route.current`. + * If called with a string, the value maps to `redirectTo`. + * @returns {Object} self + */ + this.otherwise = function(params) { + if (typeof params === 'string') { + params = {redirectTo: params}; + } + this.when(null, params); + return this; + }; + + + this.$get = ['$rootScope', + '$location', + '$routeParams', + '$q', + '$injector', + '$templateRequest', + '$sce', + function($rootScope, $location, $routeParams, $q, $injector, $templateRequest, $sce) { + + /** + * @ngdoc service + * @name $route + * @requires $location + * @requires $routeParams + * + * @property {Object} current Reference to the current route definition. + * The route definition contains: + * + * - `controller`: The controller constructor as define in route definition. + * - `locals`: A map of locals which is used by {@link ng.$controller $controller} service for + * controller instantiation. The `locals` contain + * the resolved values of the `resolve` map. Additionally the `locals` also contain: + * + * - `$scope` - The current route scope. + * - `$template` - The current route template HTML. + * + * @property {Object} routes Object with all route configuration Objects as its properties. + * + * @description + * `$route` is used for deep-linking URLs to controllers and views (HTML partials). + * It watches `$location.url()` and tries to map the path to an existing route definition. + * + * Requires the {@link ngRoute `ngRoute`} module to be installed. + * + * You can define routes through {@link ngRoute.$routeProvider $routeProvider}'s API. + * + * The `$route` service is typically used in conjunction with the + * {@link ngRoute.directive:ngView `ngView`} directive and the + * {@link ngRoute.$routeParams `$routeParams`} service. + * + * @example + * This example shows how changing the URL hash causes the `$route` to match a route against the + * URL, and the `ngView` pulls in the partial. + * + * + * + *
+ * Choose: + * Moby | + * Moby: Ch1 | + * Gatsby | + * Gatsby: Ch4 | + * Scarlet Letter
+ * + *
+ * + *
+ * + *
$location.path() = {{$location.path()}}
+ *
$route.current.templateUrl = {{$route.current.templateUrl}}
+ *
$route.current.params = {{$route.current.params}}
+ *
$route.current.scope.name = {{$route.current.scope.name}}
+ *
$routeParams = {{$routeParams}}
+ *
+ *
+ * + * + * controller: {{name}}
+ * Book Id: {{params.bookId}}
+ *
+ * + * + * controller: {{name}}
+ * Book Id: {{params.bookId}}
+ * Chapter Id: {{params.chapterId}} + *
+ * + * + * angular.module('ngRouteExample', ['ngRoute']) + * + * .controller('MainController', function($scope, $route, $routeParams, $location) { + * $scope.$route = $route; + * $scope.$location = $location; + * $scope.$routeParams = $routeParams; + * }) + * + * .controller('BookController', function($scope, $routeParams) { + * $scope.name = "BookController"; + * $scope.params = $routeParams; + * }) + * + * .controller('ChapterController', function($scope, $routeParams) { + * $scope.name = "ChapterController"; + * $scope.params = $routeParams; + * }) + * + * .config(function($routeProvider, $locationProvider) { + * $routeProvider + * .when('/Book/:bookId', { + * templateUrl: 'book.html', + * controller: 'BookController', + * resolve: { + * // I will cause a 1 second delay + * delay: function($q, $timeout) { + * var delay = $q.defer(); + * $timeout(delay.resolve, 1000); + * return delay.promise; + * } + * } + * }) + * .when('/Book/:bookId/ch/:chapterId', { + * templateUrl: 'chapter.html', + * controller: 'ChapterController' + * }); + * + * // configure html5 to get links working on jsfiddle + * $locationProvider.html5Mode(true); + * }); + * + * + * + * + * it('should load and compile correct template', function() { + * element(by.linkText('Moby: Ch1')).click(); + * var content = element(by.css('[ng-view]')).getText(); + * expect(content).toMatch(/controller\: ChapterController/); + * expect(content).toMatch(/Book Id\: Moby/); + * expect(content).toMatch(/Chapter Id\: 1/); + * + * element(by.partialLinkText('Scarlet')).click(); + * + * content = element(by.css('[ng-view]')).getText(); + * expect(content).toMatch(/controller\: BookController/); + * expect(content).toMatch(/Book Id\: Scarlet/); + * }); + * + *
+ */ + + /** + * @ngdoc event + * @name $route#$routeChangeStart + * @eventType broadcast on root scope + * @description + * Broadcasted before a route change. At this point the route services starts + * resolving all of the dependencies needed for the route change to occur. + * Typically this involves fetching the view template as well as any dependencies + * defined in `resolve` route property. Once all of the dependencies are resolved + * `$routeChangeSuccess` is fired. + * + * The route change (and the `$location` change that triggered it) can be prevented + * by calling `preventDefault` method of the event. See {@link ng.$rootScope.Scope#$on} + * for more details about event object. + * + * @param {Object} angularEvent Synthetic event object. + * @param {Route} next Future route information. + * @param {Route} current Current route information. + */ + + /** + * @ngdoc event + * @name $route#$routeChangeSuccess + * @eventType broadcast on root scope + * @description + * Broadcasted after a route change has happened successfully. + * The `resolve` dependencies are now available in the `current.locals` property. + * + * {@link ngRoute.directive:ngView ngView} listens for the directive + * to instantiate the controller and render the view. + * + * @param {Object} angularEvent Synthetic event object. + * @param {Route} current Current route information. + * @param {Route|Undefined} previous Previous route information, or undefined if current is + * first route entered. + */ + + /** + * @ngdoc event + * @name $route#$routeChangeError + * @eventType broadcast on root scope + * @description + * Broadcasted if any of the resolve promises are rejected. + * + * @param {Object} angularEvent Synthetic event object + * @param {Route} current Current route information. + * @param {Route} previous Previous route information. + * @param {Route} rejection Rejection of the promise. Usually the error of the failed promise. + */ + + /** + * @ngdoc event + * @name $route#$routeUpdate + * @eventType broadcast on root scope + * @description + * The `reloadOnSearch` property has been set to false, and we are reusing the same + * instance of the Controller. + * + * @param {Object} angularEvent Synthetic event object + * @param {Route} current Current/previous route information. + */ + + var forceReload = false, + preparedRoute, + preparedRouteIsUpdateOnly, + $route = { + routes: routes, + + /** + * @ngdoc method + * @name $route#reload + * + * @description + * Causes `$route` service to reload the current route even if + * {@link ng.$location $location} hasn't changed. + * + * As a result of that, {@link ngRoute.directive:ngView ngView} + * creates new scope and reinstantiates the controller. + */ + reload: function() { + forceReload = true; + + var fakeLocationEvent = { + defaultPrevented: false, + preventDefault: function fakePreventDefault() { + this.defaultPrevented = true; + forceReload = false; + } + }; + + $rootScope.$evalAsync(function() { + prepareRoute(fakeLocationEvent); + if (!fakeLocationEvent.defaultPrevented) commitRoute(); + }); + }, + + /** + * @ngdoc method + * @name $route#updateParams + * + * @description + * Causes `$route` service to update the current URL, replacing + * current route parameters with those specified in `newParams`. + * Provided property names that match the route's path segment + * definitions will be interpolated into the location's path, while + * remaining properties will be treated as query params. + * + * @param {!Object} newParams mapping of URL parameter names to values + */ + updateParams: function(newParams) { + if (this.current && this.current.$$route) { + newParams = angular.extend({}, this.current.params, newParams); + $location.path(interpolate(this.current.$$route.originalPath, newParams)); + // interpolate modifies newParams, only query params are left + $location.search(newParams); + } else { + throw $routeMinErr('norout', 'Tried updating route when with no current route'); + } + } + }; + + $rootScope.$on('$locationChangeStart', prepareRoute); + $rootScope.$on('$locationChangeSuccess', commitRoute); + + return $route; + + ///////////////////////////////////////////////////// + + /** + * @param on {string} current url + * @param route {Object} route regexp to match the url against + * @return {?Object} + * + * @description + * Check if the route matches the current url. + * + * Inspired by match in + * visionmedia/express/lib/router/router.js. + */ + function switchRouteMatcher(on, route) { + var keys = route.keys, + params = {}; + + if (!route.regexp) return null; + + var m = route.regexp.exec(on); + if (!m) return null; + + for (var i = 1, len = m.length; i < len; ++i) { + var key = keys[i - 1]; + + var val = m[i]; + + if (key && val) { + params[key.name] = val; + } + } + return params; + } + + function prepareRoute($locationEvent) { + var lastRoute = $route.current; + + preparedRoute = parseRoute(); + preparedRouteIsUpdateOnly = preparedRoute && lastRoute && preparedRoute.$$route === lastRoute.$$route + && angular.equals(preparedRoute.pathParams, lastRoute.pathParams) + && !preparedRoute.reloadOnSearch && !forceReload; + + if (!preparedRouteIsUpdateOnly && (lastRoute || preparedRoute)) { + if ($rootScope.$broadcast('$routeChangeStart', preparedRoute, lastRoute).defaultPrevented) { + if ($locationEvent) { + $locationEvent.preventDefault(); + } + } + } + } + + function commitRoute() { + var lastRoute = $route.current; + var nextRoute = preparedRoute; + + if (preparedRouteIsUpdateOnly) { + lastRoute.params = nextRoute.params; + angular.copy(lastRoute.params, $routeParams); + $rootScope.$broadcast('$routeUpdate', lastRoute); + } else if (nextRoute || lastRoute) { + forceReload = false; + $route.current = nextRoute; + if (nextRoute) { + if (nextRoute.redirectTo) { + if (angular.isString(nextRoute.redirectTo)) { + $location.path(interpolate(nextRoute.redirectTo, nextRoute.params)).search(nextRoute.params) + .replace(); + } else { + $location.url(nextRoute.redirectTo(nextRoute.pathParams, $location.path(), $location.search())) + .replace(); + } + } + } + + $q.when(nextRoute). + then(function() { + if (nextRoute) { + var locals = angular.extend({}, nextRoute.resolve), + template, templateUrl; + + angular.forEach(locals, function(value, key) { + locals[key] = angular.isString(value) ? + $injector.get(value) : $injector.invoke(value, null, null, key); + }); + + if (angular.isDefined(template = nextRoute.template)) { + if (angular.isFunction(template)) { + template = template(nextRoute.params); + } + } else if (angular.isDefined(templateUrl = nextRoute.templateUrl)) { + if (angular.isFunction(templateUrl)) { + templateUrl = templateUrl(nextRoute.params); + } + if (angular.isDefined(templateUrl)) { + nextRoute.loadedTemplateUrl = $sce.valueOf(templateUrl); + template = $templateRequest(templateUrl); + } + } + if (angular.isDefined(template)) { + locals['$template'] = template; + } + return $q.all(locals); + } + }). + then(function(locals) { + // after route change + if (nextRoute == $route.current) { + if (nextRoute) { + nextRoute.locals = locals; + angular.copy(nextRoute.params, $routeParams); + } + $rootScope.$broadcast('$routeChangeSuccess', nextRoute, lastRoute); + } + }, function(error) { + if (nextRoute == $route.current) { + $rootScope.$broadcast('$routeChangeError', nextRoute, lastRoute, error); + } + }); + } + } + + + /** + * @returns {Object} the current active route, by matching it against the URL + */ + function parseRoute() { + // Match a route + var params, match; + angular.forEach(routes, function(route, path) { + if (!match && (params = switchRouteMatcher($location.path(), route))) { + match = inherit(route, { + params: angular.extend({}, $location.search(), params), + pathParams: params}); + match.$$route = route; + } + }); + // No route matched; fallback to "otherwise" route + return match || routes[null] && inherit(routes[null], {params: {}, pathParams:{}}); + } + + /** + * @returns {string} interpolation of the redirect path with the parameters + */ + function interpolate(string, params) { + var result = []; + angular.forEach((string || '').split(':'), function(segment, i) { + if (i === 0) { + result.push(segment); + } else { + var segmentMatch = segment.match(/(\w+)(?:[?*])?(.*)/); + var key = segmentMatch[1]; + result.push(params[key]); + result.push(segmentMatch[2] || ''); + delete params[key]; + } + }); + return result.join(''); + } + }]; +} + +ngRouteModule.provider('$routeParams', $RouteParamsProvider); + + +/** + * @ngdoc service + * @name $routeParams + * @requires $route + * + * @description + * The `$routeParams` service allows you to retrieve the current set of route parameters. + * + * Requires the {@link ngRoute `ngRoute`} module to be installed. + * + * The route parameters are a combination of {@link ng.$location `$location`}'s + * {@link ng.$location#search `search()`} and {@link ng.$location#path `path()`}. + * The `path` parameters are extracted when the {@link ngRoute.$route `$route`} path is matched. + * + * In case of parameter name collision, `path` params take precedence over `search` params. + * + * The service guarantees that the identity of the `$routeParams` object will remain unchanged + * (but its properties will likely change) even when a route change occurs. + * + * Note that the `$routeParams` are only updated *after* a route change completes successfully. + * This means that you cannot rely on `$routeParams` being correct in route resolve functions. + * Instead you can use `$route.current.params` to access the new route's parameters. + * + * @example + * ```js + * // Given: + * // URL: http://server.com/index.html#/Chapter/1/Section/2?search=moby + * // Route: /Chapter/:chapterId/Section/:sectionId + * // + * // Then + * $routeParams ==> {chapterId:'1', sectionId:'2', search:'moby'} + * ``` + */ +function $RouteParamsProvider() { + this.$get = function() { return {}; }; +} + +ngRouteModule.directive('ngView', ngViewFactory); +ngRouteModule.directive('ngView', ngViewFillContentFactory); + + +/** + * @ngdoc directive + * @name ngView + * @restrict ECA + * + * @description + * # Overview + * `ngView` is a directive that complements the {@link ngRoute.$route $route} service by + * including the rendered template of the current route into the main layout (`index.html`) file. + * Every time the current route changes, the included view changes with it according to the + * configuration of the `$route` service. + * + * Requires the {@link ngRoute `ngRoute`} module to be installed. + * + * @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=} onload Expression to evaluate whenever the view updates. + * + * @param {string=} autoscroll Whether `ngView` should call {@link ng.$anchorScroll + * $anchorScroll} to scroll the viewport after the view is updated. + * + * - If the attribute is not set, disable scrolling. + * - If the attribute is set without value, enable scrolling. + * - Otherwise enable scrolling only if the `autoscroll` attribute value evaluated + * as an expression yields a truthy value. + * @example + + +
+ Choose: + Moby | + Moby: Ch1 | + Gatsby | + Gatsby: Ch4 | + Scarlet Letter
+ +
+
+
+
+ +
$location.path() = {{main.$location.path()}}
+
$route.current.templateUrl = {{main.$route.current.templateUrl}}
+
$route.current.params = {{main.$route.current.params}}
+
$routeParams = {{main.$routeParams}}
+
+
+ + +
+ controller: {{book.name}}
+ Book Id: {{book.params.bookId}}
+
+
+ + +
+ controller: {{chapter.name}}
+ Book Id: {{chapter.params.bookId}}
+ Chapter Id: {{chapter.params.chapterId}} +
+
+ + + .view-animate-container { + position:relative; + height:100px!important; + background:white; + border:1px solid black; + height:40px; + overflow:hidden; + } + + .view-animate { + padding:10px; + } + + .view-animate.ng-enter, .view-animate.ng-leave { + transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 1.5s; + + display:block; + width:100%; + border-left:1px solid black; + + position:absolute; + top:0; + left:0; + right:0; + bottom:0; + padding:10px; + } + + .view-animate.ng-enter { + left:100%; + } + .view-animate.ng-enter.ng-enter-active { + left:0; + } + .view-animate.ng-leave.ng-leave-active { + left:-100%; + } + + + + angular.module('ngViewExample', ['ngRoute', 'ngAnimate']) + .config(['$routeProvider', '$locationProvider', + function($routeProvider, $locationProvider) { + $routeProvider + .when('/Book/:bookId', { + templateUrl: 'book.html', + controller: 'BookCtrl', + controllerAs: 'book' + }) + .when('/Book/:bookId/ch/:chapterId', { + templateUrl: 'chapter.html', + controller: 'ChapterCtrl', + controllerAs: 'chapter' + }); + + $locationProvider.html5Mode(true); + }]) + .controller('MainCtrl', ['$route', '$routeParams', '$location', + function($route, $routeParams, $location) { + this.$route = $route; + this.$location = $location; + this.$routeParams = $routeParams; + }]) + .controller('BookCtrl', ['$routeParams', function($routeParams) { + this.name = "BookCtrl"; + this.params = $routeParams; + }]) + .controller('ChapterCtrl', ['$routeParams', function($routeParams) { + this.name = "ChapterCtrl"; + this.params = $routeParams; + }]); + + + + + it('should load and compile correct template', function() { + element(by.linkText('Moby: Ch1')).click(); + var content = element(by.css('[ng-view]')).getText(); + expect(content).toMatch(/controller\: ChapterCtrl/); + expect(content).toMatch(/Book Id\: Moby/); + expect(content).toMatch(/Chapter Id\: 1/); + + element(by.partialLinkText('Scarlet')).click(); + + content = element(by.css('[ng-view]')).getText(); + expect(content).toMatch(/controller\: BookCtrl/); + expect(content).toMatch(/Book Id\: Scarlet/); + }); + +
+ */ + + +/** + * @ngdoc event + * @name ngView#$viewContentLoaded + * @eventType emit on the current ngView scope + * @description + * Emitted every time the ngView content is reloaded. + */ +ngViewFactory.$inject = ['$route', '$anchorScroll', '$animate']; +function ngViewFactory($route, $anchorScroll, $animate) { + return { + restrict: 'ECA', + terminal: true, + priority: 400, + transclude: 'element', + link: function(scope, $element, attr, ctrl, $transclude) { + var currentScope, + currentElement, + previousLeaveAnimation, + autoScrollExp = attr.autoscroll, + onloadExp = attr.onload || ''; + + scope.$on('$routeChangeSuccess', update); + update(); + + function cleanupLastView() { + if (previousLeaveAnimation) { + $animate.cancel(previousLeaveAnimation); + previousLeaveAnimation = null; + } + + if (currentScope) { + currentScope.$destroy(); + currentScope = null; + } + if (currentElement) { + previousLeaveAnimation = $animate.leave(currentElement); + previousLeaveAnimation.then(function() { + previousLeaveAnimation = null; + }); + currentElement = null; + } + } + + function update() { + var locals = $route.current && $route.current.locals, + template = locals && locals.$template; + + if (angular.isDefined(template)) { + var newScope = scope.$new(); + var current = $route.current; + + // Note: This will also link all children of ng-view that were contained in the original + // html. If that content contains controllers, ... they could pollute/change the scope. + // However, using ng-view 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) { + $animate.enter(clone, null, currentElement || $element).then(function onNgViewEnter() { + if (angular.isDefined(autoScrollExp) + && (!autoScrollExp || scope.$eval(autoScrollExp))) { + $anchorScroll(); + } + }); + cleanupLastView(); + }); + + currentElement = clone; + currentScope = current.scope = newScope; + currentScope.$emit('$viewContentLoaded'); + currentScope.$eval(onloadExp); + } else { + cleanupLastView(); + } + } + } + }; +} + +// This directive is called during the $transclude call of the first `ngView` 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 ngView +// is called. +ngViewFillContentFactory.$inject = ['$compile', '$controller', '$route']; +function ngViewFillContentFactory($compile, $controller, $route) { + return { + restrict: 'ECA', + priority: -400, + link: function(scope, $element) { + var current = $route.current, + locals = current.locals; + + $element.html(locals.$template); + + var link = $compile($element.contents()); + + if (current.controller) { + locals.$scope = scope; + var controller = $controller(current.controller, locals); + if (current.controllerAs) { + scope[current.controllerAs] = controller; + } + $element.data('$ngControllerController', controller); + $element.children().data('$ngControllerController', controller); + } + + link(scope); + } + }; +} + + +})(window, window.angular); diff --git a/1.4.10/angular-route.min.js b/1.4.10/angular-route.min.js new file mode 100644 index 0000000000..2ee50813f4 --- /dev/null +++ b/1.4.10/angular-route.min.js @@ -0,0 +1,15 @@ +/* + AngularJS v1.4.10 + (c) 2010-2015 Google, Inc. http://angularjs.org + License: MIT +*/ +(function(q,d,C){'use strict';function w(s,k,h){return{restrict:"ECA",terminal:!0,priority:400,transclude:"element",link:function(a,f,b,c,y){function z(){l&&(h.cancel(l),l=null);m&&(m.$destroy(),m=null);n&&(l=h.leave(n),l.then(function(){l=null}),n=null)}function x(){var b=s.current&&s.current.locals;if(d.isDefined(b&&b.$template)){var b=a.$new(),c=s.current;n=y(b,function(b){h.enter(b,null,n||f).then(function(){!d.isDefined(u)||u&&!a.$eval(u)||k()});z()});m=c.scope=b;m.$emit("$viewContentLoaded"); +m.$eval(v)}else z()}var m,n,l,u=b.autoscroll,v=b.onload||"";a.$on("$routeChangeSuccess",x);x()}}}function A(d,k,h){return{restrict:"ECA",priority:-400,link:function(a,f){var b=h.current,c=b.locals;f.html(c.$template);var y=d(f.contents());b.controller&&(c.$scope=a,c=k(b.controller,c),b.controllerAs&&(a[b.controllerAs]=c),f.data("$ngControllerController",c),f.children().data("$ngControllerController",c));y(a)}}}q=d.module("ngRoute",["ng"]).provider("$route",function(){function s(a,f){return d.extend(Object.create(a), +f)}function k(a,d){var b=d.caseInsensitiveMatch,c={originalPath:a,regexp:a},h=c.keys=[];a=a.replace(/([().])/g,"\\$1").replace(/(\/)?:(\w+)(\*\?|[\?\*])?/g,function(a,d,b,c){a="?"===c||"*?"===c?"?":null;c="*"===c||"*?"===c?"*":null;h.push({name:b,optional:!!a});d=d||"";return""+(a?"":d)+"(?:"+(a?d:"")+(c&&"(.+?)"||"([^/]+)")+(a||"")+")"+(a||"")}).replace(/([\/$\*])/g,"\\$1");c.regexp=new RegExp("^"+a+"$",b?"i":"");return c}var h={};this.when=function(a,f){var b=d.copy(f);d.isUndefined(b.reloadOnSearch)&& +(b.reloadOnSearch=!0);d.isUndefined(b.caseInsensitiveMatch)&&(b.caseInsensitiveMatch=this.caseInsensitiveMatch);h[a]=d.extend(b,a&&k(a,b));if(a){var c="/"==a[a.length-1]?a.substr(0,a.length-1):a+"/";h[c]=d.extend({redirectTo:a},k(c,b))}return this};this.caseInsensitiveMatch=!1;this.otherwise=function(a){"string"===typeof a&&(a={redirectTo:a});this.when(null,a);return this};this.$get=["$rootScope","$location","$routeParams","$q","$injector","$templateRequest","$sce",function(a,f,b,c,k,q,x){function m(b){var e= +t.current;(w=(p=l())&&e&&p.$$route===e.$$route&&d.equals(p.pathParams,e.pathParams)&&!p.reloadOnSearch&&!v)||!e&&!p||a.$broadcast("$routeChangeStart",p,e).defaultPrevented&&b&&b.preventDefault()}function n(){var g=t.current,e=p;if(w)g.params=e.params,d.copy(g.params,b),a.$broadcast("$routeUpdate",g);else if(e||g)v=!1,(t.current=e)&&e.redirectTo&&(d.isString(e.redirectTo)?f.path(u(e.redirectTo,e.params)).search(e.params).replace():f.url(e.redirectTo(e.pathParams,f.path(),f.search())).replace()),c.when(e).then(function(){if(e){var a= +d.extend({},e.resolve),b,g;d.forEach(a,function(b,e){a[e]=d.isString(b)?k.get(b):k.invoke(b,null,null,e)});d.isDefined(b=e.template)?d.isFunction(b)&&(b=b(e.params)):d.isDefined(g=e.templateUrl)&&(d.isFunction(g)&&(g=g(e.params)),d.isDefined(g)&&(e.loadedTemplateUrl=x.valueOf(g),b=q(g)));d.isDefined(b)&&(a.$template=b);return c.all(a)}}).then(function(c){e==t.current&&(e&&(e.locals=c,d.copy(e.params,b)),a.$broadcast("$routeChangeSuccess",e,g))},function(b){e==t.current&&a.$broadcast("$routeChangeError", +e,g,b)})}function l(){var a,b;d.forEach(h,function(c,h){var r;if(r=!b){var k=f.path();r=c.keys;var m={};if(c.regexp)if(k=c.regexp.exec(k)){for(var l=1,n=k.length;l + * + * 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 + + + +
+ Snippet: + + + + + + + + + + + + + + + + + + + + + + + + + +
DirectiveHowSourceRendered
ng-bind-htmlAutomatically uses $sanitize
<div ng-bind-html="snippet">
</div>
ng-bind-htmlBypass $sanitize by explicitly trusting the dangerous value +
<div ng-bind-html="deliberatelyTrustDangerousSnippet()">
+</div>
+
ng-bindAutomatically escapes
<div ng-bind="snippet">
</div>
+
+
+ + it('should sanitize the html snippet by default', function() { + expect(element(by.css('#bind-html-with-sanitize div')).getInnerHtml()). + toBe('

an html\nclick here\nsnippet

'); + }); + + it('should inline raw snippet if bound to a trusted value', function() { + expect(element(by.css('#bind-html-with-trust div')).getInnerHtml()). + toBe("

an html\n" + + "click here\n" + + "snippet

"); + }); + + it('should escape snippet without any filter', function() { + expect(element(by.css('#bind-default div')).getInnerHtml()). + toBe("<p style=\"color:blue\">an html\n" + + "<em onmouseover=\"this.textContent='PWN3D!'\">click here</em>\n" + + "snippet</p>"); + }); + + it('should update', function() { + element(by.model('snippet')).clear(); + element(by.model('snippet')).sendKeys('new text'); + expect(element(by.css('#bind-html-with-sanitize div')).getInnerHtml()). + toBe('new text'); + expect(element(by.css('#bind-html-with-trust div')).getInnerHtml()).toBe( + 'new text'); + expect(element(by.css('#bind-default div')).getInnerHtml()).toBe( + "new <b onclick=\"alert(1)\">text</b>"); + }); +
+
+ */ +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 = /^/g, + DOCTYPE_REGEXP = /]*?)>/i, + CDATA_REGEXP = //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("", 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, '>'); +} + +/** + * 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(''); + } + 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 + + * + * @example + + + +
+ Snippet: + + + + + + + + + + + + + + + + + + + + + +
FilterSourceRendered
linky filter +
<div ng-bind-html="snippet | linky">
</div>
+
+
+
linky target +
<div ng-bind-html="snippetWithTarget | linky:'_blank'">
</div>
+
+
+
no filter
<div ng-bind="snippet">
</div>
+ + + 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'); + }); + + + */ +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(''); + addText(text); + html.push(''); + } + }; +}]); + + +})(window, window.angular); diff --git a/1.4.10/angular-sanitize.min.js b/1.4.10/angular-sanitize.min.js new file mode 100644 index 0000000000..4558b97d0c --- /dev/null +++ b/1.4.10/angular-sanitize.min.js @@ -0,0 +1,16 @@ +/* + AngularJS v1.4.10 + (c) 2010-2015 Google, Inc. http://angularjs.org + License: MIT +*/ +(function(n,h,p){'use strict';function E(a){var f=[];r(f,h.noop).chars(a);return f.join("")}function g(a,f){var d={},c=a.split(","),b;for(b=0;b=c;d--)f.end&&f.end(e[d]);e.length=c}}"string"!==typeof a&&(a=null===a||"undefined"===typeof a?"":""+a);var b,k,e=[],m=a,l;for(e.last=function(){return e[e.length-1]};a;){l="";k=!0;if(e.last()&&w[e.last()])a=a.replace(new RegExp("([\\W\\w]*)<\\s*\\/\\s*"+e.last()+"[^>]*>","i"),function(a,b){b=b.replace(H,"$1").replace(I,"$1");f.chars&&f.chars(q(b));return""}),c("",e.last());else{if(0===a.indexOf("\x3c!--"))b=a.indexOf("--",4),0<=b&&a.lastIndexOf("--\x3e", +b)===b&&(f.comment&&f.comment(a.substring(4,b)),a=a.substring(b+3),k=!1);else if(x.test(a)){if(b=a.match(x))a=a.replace(b[0],""),k=!1}else if(J.test(a)){if(b=a.match(y))a=a.substring(b[0].length),b[0].replace(y,c),k=!1}else K.test(a)&&((b=a.match(z))?(b[4]&&(a=a.substring(b[0].length),b[0].replace(z,d)),k=!1):(l+="<",a=a.substring(1)));k&&(b=a.indexOf("<"),l+=0>b?a:a.substring(0,b),a=0>b?"":a.substring(b),f.chars&&f.chars(q(l)))}if(a==m)throw L("badparse",a);m=a}c()}function q(a){if(!a)return"";A.innerHTML= +a.replace(//g,">")}function r(a,f){var d=!1,c=h.bind(a,a.push);return{start:function(a,k,e){a=h.lowercase(a);!d&&w[a]&&(d=a);d||!0!==C[a]||(c("<"),c(a),h.forEach(k,function(d,e){var k=h.lowercase(e),g="img"===a&&"src"===k|| +"background"===k;!0!==O[k]||!0===D[k]&&!f(d,g)||(c(" "),c(e),c('="'),c(B(d)),c('"'))}),c(e?"/>":">"))},end:function(a){a=h.lowercase(a);d||!0!==C[a]||(c(""));a==d&&(d=!1)},chars:function(a){d||c(B(a))}}}var L=h.$$minErr("$sanitize"),z=/^<((?:[a-zA-Z])[\w:-]*)((?:\s+[\w:-]+(?:\s*=\s*(?:(?:"[^"]*")|(?:'[^']*')|[^>\s]+))?)*)\s*(\/?)\s*(>?)/,y=/^<\/\s*([\w:-]+)[^>]*>/,G=/([\w:-]+)(?:\s*=\s*(?:(?:"((?:[^"])*)")|(?:'((?:[^'])*)')|([^>\s]+)))?/g,K=/^]*?)>/i, +I=/"\u201d\u2019]/i,d=/^mailto:/i;return function(c,b){function k(a){a&&g.push(E(a))}function e(a, +c){g.push("');k(c);g.push("")}if(!c)return c;for(var m,l=c,g=[],n,p;m=l.match(f);)n=m[0],m[2]||m[4]||(n=(m[3]?"http://":"mailto:")+n),p=m.index,k(l.substr(0,p)),e(n,m[0].replace(d,"")),l=l.substring(p+m[0].length);k(l);return a(g.join(""))}}])})(window,window.angular); +//# sourceMappingURL=angular-sanitize.min.js.map diff --git a/1.4.10/angular-sanitize.min.js.map b/1.4.10/angular-sanitize.min.js.map new file mode 100644 index 0000000000..2f360f4df7 --- /dev/null +++ b/1.4.10/angular-sanitize.min.js.map @@ -0,0 +1,8 @@ +{ +"version":3, +"file":"angular-sanitize.min.js", +"lineCount":15, +"mappings":"A;;;;;aAKC,SAAQ,CAACA,CAAD,CAASC,CAAT,CAAkBC,CAAlB,CAA6B,CA6JtCC,QAASA,EAAY,CAACC,CAAD,CAAQ,CAC3B,IAAIC,EAAM,EACGC,EAAAC,CAAmBF,CAAnBE,CAAwBN,CAAAO,KAAxBD,CACbH,MAAA,CAAaA,CAAb,CACA,OAAOC,EAAAI,KAAA,CAAS,EAAT,CAJoB,CAmG7BC,QAASA,EAAO,CAACC,CAAD,CAAMC,CAAN,CAAqB,CAAA,IAC/BC,EAAM,EADyB,CACrBC,EAAQH,CAAAI,MAAA,CAAU,GAAV,CADa,CACGC,CACtC,KAAKA,CAAL,CAAS,CAAT,CAAYA,CAAZ,CAAgBF,CAAAG,OAAhB,CAA8BD,CAAA,EAA9B,CACEH,CAAA,CAAID,CAAA,CAAgBX,CAAAiB,UAAA,CAAkBJ,CAAA,CAAME,CAAN,CAAlB,CAAhB,CAA8CF,CAAA,CAAME,CAAN,CAAlD,CAAA,CAA8D,CAAA,CAEhE,OAAOH,EAL4B,CAqBrCM,QAASA,EAAU,CAACC,CAAD,CAAOC,CAAP,CAAgB,CAiGjCC,QAASA,EAAa,CAACC,CAAD,CAAMC,CAAN,CAAeC,CAAf,CAAqBC,CAArB,CAA4B,CAChDF,CAAA,CAAUvB,CAAAiB,UAAA,CAAkBM,CAAlB,CACV,IAAIG,CAAA,CAAcH,CAAd,CAAJ,CACE,IAAA,CAAOI,CAAAC,KAAA,EAAP,EAAuBC,CAAA,CAAeF,CAAAC,KAAA,EAAf,CAAvB,CAAA,CACEE,CAAA,CAAY,EAAZ,CAAgBH,CAAAC,KAAA,EAAhB,CAIAG,EAAA,CAAuBR,CAAvB,CAAJ,EAAuCI,CAAAC,KAAA,EAAvC,EAAuDL,CAAvD,EACEO,CAAA,CAAY,EAAZ,CAAgBP,CAAhB,CAKF,EAFAE,CAEA,CAFQO,CAAA,CAAaT,CAAb,CAER,EAFiC,CAAEE,CAAAA,CAEnC,GACEE,CAAAM,KAAA,CAAWV,CAAX,CAGF,KAAIW,EAAQ,EAEZV,EAAAW,QAAA,CAAaC,CAAb,CACE,QAAQ,CAACC,CAAD,CAAQC,CAAR,CAAcC,CAAd,CAAiCC,CAAjC,CAAoDC,CAApD,CAAmE,CAMzEP,CAAA,CAAMI,CAAN,CAAA,CAAcI,CAAA,CALFH,CAKE,EAJTC,CAIS,EAHTC,CAGS,EAFT,EAES,CAN2D,CAD7E,CASIrB,EAAAuB,MAAJ,EAAmBvB,CAAAuB,MAAA,CAAcpB,CAAd,CAAuBW,CAAvB,CAA8BT,CAA9B,CA7B6B,CAgClDK,QAASA,EAAW,CAACR,CAAD,CAAMC,CAAN,CAAe,CAAA,IAC7BqB,EAAM,CADuB,CACpB7B,CAEb,IADAQ,CACA,CADUvB,CAAAiB,UAAA,CAAkBM,CAAlB,CACV,CAEE,IAAKqB,CAAL,CAAWjB,CAAAX,OAAX;AAA0B,CAA1B,CAAoC,CAApC,EAA6B4B,CAA7B,EACMjB,CAAA,CAAMiB,CAAN,CADN,EACoBrB,CADpB,CAAuCqB,CAAA,EAAvC,EAKF,GAAW,CAAX,EAAIA,CAAJ,CAAc,CAEZ,IAAK7B,CAAL,CAASY,CAAAX,OAAT,CAAwB,CAAxB,CAA2BD,CAA3B,EAAgC6B,CAAhC,CAAqC7B,CAAA,EAArC,CACMK,CAAAyB,IAAJ,EAAiBzB,CAAAyB,IAAA,CAAYlB,CAAA,CAAMZ,CAAN,CAAZ,CAGnBY,EAAAX,OAAA,CAAe4B,CANH,CAVmB,CAhIf,QAApB,GAAI,MAAOzB,EAAX,GAEIA,CAFJ,CACe,IAAb,GAAIA,CAAJ,EAAqC,WAArC,GAAqB,MAAOA,EAA5B,CACS,EADT,CAGS,EAHT,CAGcA,CAJhB,CADiC,KAQ7B2B,CAR6B,CAQtB3C,CARsB,CAQRwB,EAAQ,EARA,CAQIC,EAAOT,CARX,CAQiB4B,CAGlD,KAFApB,CAAAC,KAEA,CAFaoB,QAAQ,EAAG,CAAE,MAAOrB,EAAA,CAAMA,CAAAX,OAAN,CAAqB,CAArB,CAAT,CAExB,CAAOG,CAAP,CAAA,CAAa,CACX4B,CAAA,CAAO,EACP5C,EAAA,CAAQ,CAAA,CAGR,IAAKwB,CAAAC,KAAA,EAAL,EAAsBqB,CAAA,CAAgBtB,CAAAC,KAAA,EAAhB,CAAtB,CA2DET,CASA,CATOA,CAAAgB,QAAA,CAAa,IAAIe,MAAJ,CAAW,yBAAX,CAAuCvB,CAAAC,KAAA,EAAvC,CAAsD,QAAtD,CAAgE,GAAhE,CAAb,CACL,QAAQ,CAACuB,CAAD,CAAMJ,CAAN,CAAY,CAClBA,CAAA,CAAOA,CAAAZ,QAAA,CAAaiB,CAAb,CAA6B,IAA7B,CAAAjB,QAAA,CAA2CkB,CAA3C,CAAyD,IAAzD,CAEHjC,EAAAjB,MAAJ,EAAmBiB,CAAAjB,MAAA,CAAcuC,CAAA,CAAeK,CAAf,CAAd,CAEnB,OAAO,EALW,CADf,CASP,CAAAjB,CAAA,CAAY,EAAZ,CAAgBH,CAAAC,KAAA,EAAhB,CApEF,KAAqD,CAGnD,GAA6B,CAA7B,GAAIT,CAAAmC,QAAA,CAAa,SAAb,CAAJ,CAEER,CAEA,CAFQ3B,CAAAmC,QAAA,CAAa,IAAb,CAAmB,CAAnB,CAER,CAAa,CAAb,EAAIR,CAAJ,EAAkB3B,CAAAoC,YAAA,CAAiB,QAAjB;AAAwBT,CAAxB,CAAlB,GAAqDA,CAArD,GACM1B,CAAAoC,QAEJ,EAFqBpC,CAAAoC,QAAA,CAAgBrC,CAAAsC,UAAA,CAAe,CAAf,CAAkBX,CAAlB,CAAhB,CAErB,CADA3B,CACA,CADOA,CAAAsC,UAAA,CAAeX,CAAf,CAAuB,CAAvB,CACP,CAAA3C,CAAA,CAAQ,CAAA,CAHV,CAJF,KAUO,IAAIuD,CAAAC,KAAA,CAAoBxC,CAApB,CAAJ,CAGL,IAFAkB,CAEA,CAFQlB,CAAAkB,MAAA,CAAWqB,CAAX,CAER,CACEvC,CACA,CADOA,CAAAgB,QAAA,CAAaE,CAAA,CAAM,CAAN,CAAb,CAAuB,EAAvB,CACP,CAAAlC,CAAA,CAAQ,CAAA,CAFV,CAHK,IAQA,IAAIyD,CAAAD,KAAA,CAA4BxC,CAA5B,CAAJ,CAGL,IAFAkB,CAEA,CAFQlB,CAAAkB,MAAA,CAAWwB,CAAX,CAER,CACE1C,CAEA,CAFOA,CAAAsC,UAAA,CAAepB,CAAA,CAAM,CAAN,CAAArB,OAAf,CAEP,CADAqB,CAAA,CAAM,CAAN,CAAAF,QAAA,CAAiB0B,CAAjB,CAAiC/B,CAAjC,CACA,CAAA3B,CAAA,CAAQ,CAAA,CAHV,CAHK,IAUI2D,EAAAH,KAAA,CAAsBxC,CAAtB,CAAJ,GAGL,CAFAkB,CAEA,CAFQlB,CAAAkB,MAAA,CAAW0B,CAAX,CAER,GAEM1B,CAAA,CAAM,CAAN,CAIJ,GAHElB,CACA,CADOA,CAAAsC,UAAA,CAAepB,CAAA,CAAM,CAAN,CAAArB,OAAf,CACP,CAAAqB,CAAA,CAAM,CAAN,CAAAF,QAAA,CAAiB4B,CAAjB,CAAmC1C,CAAnC,CAEF,EAAAlB,CAAA,CAAQ,CAAA,CANV,GASE4C,CACA,EADQ,GACR,CAAA5B,CAAA,CAAOA,CAAAsC,UAAA,CAAe,CAAf,CAVT,CAHK,CAiBHtD,EAAJ,GACE2C,CAKA,CALQ3B,CAAAmC,QAAA,CAAa,GAAb,CAKR,CAHAP,CAGA,EAHgB,CAAR,CAAAD,CAAA,CAAY3B,CAAZ,CAAmBA,CAAAsC,UAAA,CAAe,CAAf,CAAkBX,CAAlB,CAG3B,CAFA3B,CAEA,CAFe,CAAR,CAAA2B,CAAA,CAAY,EAAZ,CAAiB3B,CAAAsC,UAAA,CAAeX,CAAf,CAExB,CAAI1B,CAAAjB,MAAJ,EAAmBiB,CAAAjB,MAAA,CAAcuC,CAAA,CAAeK,CAAf,CAAd,CANrB,CAhDmD,CAuErD,GAAI5B,CAAJ,EAAYS,CAAZ,CACE,KAAMoC,EAAA,CAAgB,UAAhB,CAC4C7C,CAD5C,CAAN,CAGFS,CAAA,CAAOT,CAhFI,CAoFbW,CAAA,EA/FiC,CA4JnCY,QAASA,EAAc,CAACuB,CAAD,CAAQ,CAC7B,GAAKA,CAAAA,CAAL,CAAc,MAAO,EAErBC,EAAAC,UAAA;AAAsBF,CAAA9B,QAAA,CAAc,IAAd,CAAmB,MAAnB,CAGtB,OAAO+B,EAAAE,YANsB,CAgB/BC,QAASA,EAAc,CAACJ,CAAD,CAAQ,CAC7B,MAAOA,EAAA9B,QAAA,CACG,IADH,CACS,OADT,CAAAA,QAAA,CAEGmC,CAFH,CAE0B,QAAQ,CAACL,CAAD,CAAQ,CAC7C,IAAIM,EAAKN,CAAAO,WAAA,CAAiB,CAAjB,CACLC,EAAAA,CAAMR,CAAAO,WAAA,CAAiB,CAAjB,CACV,OAAO,IAAP,EAAgC,IAAhC,EAAiBD,CAAjB,CAAsB,KAAtB,GAA0CE,CAA1C,CAAgD,KAAhD,EAA0D,KAA1D,EAAqE,GAHxB,CAF1C,CAAAtC,QAAA,CAOGuC,CAPH,CAO4B,QAAQ,CAACT,CAAD,CAAQ,CAC/C,MAAO,IAAP,CAAcA,CAAAO,WAAA,CAAiB,CAAjB,CAAd,CAAoC,GADW,CAP5C,CAAArC,QAAA,CAUG,IAVH,CAUS,MAVT,CAAAA,QAAA,CAWG,IAXH,CAWS,MAXT,CADsB,CAyB/B9B,QAASA,EAAkB,CAACD,CAAD,CAAMuE,CAAN,CAAoB,CAC7C,IAAIC,EAAS,CAAA,CAAb,CACIC,EAAM7E,CAAA8E,KAAA,CAAa1E,CAAb,CAAkBA,CAAA6B,KAAlB,CACV,OAAO,CACLU,MAAOA,QAAQ,CAACrB,CAAD,CAAMY,CAAN,CAAaT,CAAb,CAAoB,CACjCH,CAAA,CAAMtB,CAAAiB,UAAA,CAAkBK,CAAlB,CACDsD,EAAAA,CAAL,EAAe3B,CAAA,CAAgB3B,CAAhB,CAAf,GACEsD,CADF,CACWtD,CADX,CAGKsD,EAAL,EAAsC,CAAA,CAAtC,GAAeG,CAAA,CAAczD,CAAd,CAAf,GACEuD,CAAA,CAAI,GAAJ,CAcA,CAbAA,CAAA,CAAIvD,CAAJ,CAaA,CAZAtB,CAAAgF,QAAA,CAAgB9C,CAAhB,CAAuB,QAAQ,CAAC+B,CAAD,CAAQgB,CAAR,CAAa,CAC1C,IAAIC,EAAKlF,CAAAiB,UAAA,CAAkBgE,CAAlB,CAAT,CACIE,EAAmB,KAAnBA,GAAW7D,CAAX6D,EAAqC,KAArCA,GAA4BD,CAA5BC;AAAyD,YAAzDA,GAAgDD,CAC3B,EAAA,CAAzB,GAAIE,CAAA,CAAWF,CAAX,CAAJ,EACsB,CAAA,CADtB,GACGG,CAAA,CAASH,CAAT,CADH,EAC8B,CAAAP,CAAA,CAAaV,CAAb,CAAoBkB,CAApB,CAD9B,GAEEN,CAAA,CAAI,GAAJ,CAIA,CAHAA,CAAA,CAAII,CAAJ,CAGA,CAFAJ,CAAA,CAAI,IAAJ,CAEA,CADAA,CAAA,CAAIR,CAAA,CAAeJ,CAAf,CAAJ,CACA,CAAAY,CAAA,CAAI,GAAJ,CANF,CAH0C,CAA5C,CAYA,CAAAA,CAAA,CAAIpD,CAAA,CAAQ,IAAR,CAAe,GAAnB,CAfF,CALiC,CAD9B,CAwBLoB,IAAKA,QAAQ,CAACvB,CAAD,CAAM,CACfA,CAAA,CAAMtB,CAAAiB,UAAA,CAAkBK,CAAlB,CACDsD,EAAL,EAAsC,CAAA,CAAtC,GAAeG,CAAA,CAAczD,CAAd,CAAf,GACEuD,CAAA,CAAI,IAAJ,CAEA,CADAA,CAAA,CAAIvD,CAAJ,CACA,CAAAuD,CAAA,CAAI,GAAJ,CAHF,CAKIvD,EAAJ,EAAWsD,CAAX,GACEA,CADF,CACW,CAAA,CADX,CAPe,CAxBd,CAmCLzE,MAAOA,QAAQ,CAACA,CAAD,CAAQ,CACdyE,CAAL,EACEC,CAAA,CAAIR,CAAA,CAAelE,CAAf,CAAJ,CAFiB,CAnClB,CAHsC,CA7c/C,IAAI6D,EAAkBhE,CAAAsF,SAAA,CAAiB,WAAjB,CAAtB,CAyJIvB,EACG,wGA1JP,CA2JEF,EAAiB,wBA3JnB,CA4JEzB,EAAc,yEA5JhB,CA6JE0B,EAAmB,IA7JrB,CA8JEF,EAAyB,MA9J3B,CA+JER,EAAiB,qBA/JnB,CAgKEM,EAAiB,qBAhKnB;AAiKEL,EAAe,yBAjKjB,CAkKEiB,EAAwB,iCAlK1B,CAoKEI,EAA0B,gBApK5B,CA6KI1C,EAAevB,CAAA,CAAQ,wBAAR,CAIf8E,EAAAA,CAA8B9E,CAAA,CAAQ,gDAAR,CAC9B+E,EAAAA,CAA+B/E,CAAA,CAAQ,OAAR,CADnC,KAEIsB,EAAyB/B,CAAAyF,OAAA,CAAe,EAAf,CACeD,CADf,CAEeD,CAFf,CAF7B,CAOI7D,EAAgB1B,CAAAyF,OAAA,CAAe,EAAf,CAAmBF,CAAnB,CAAgD9E,CAAA,CAAQ,4KAAR,CAAhD,CAPpB,CAYIoB,EAAiB7B,CAAAyF,OAAA,CAAe,EAAf,CAAmBD,CAAnB,CAAiD/E,CAAA,CAAQ,2JAAR,CAAjD,CAQjBiF;CAAAA,CAAcjF,CAAA,CAAQ,4NAAR,CAKlB,KAAIwC,EAAkBxC,CAAA,CAAQ,cAAR,CAAtB,CAEIsE,EAAgB/E,CAAAyF,OAAA,CAAe,EAAf,CACezD,CADf,CAEeN,CAFf,CAGeG,CAHf,CAIeE,CAJf,CAKe2D,CALf,CAFpB,CAUIL,EAAW5E,CAAA,CAAQ,qDAAR,CAEXkF,EAAAA,CAAYlF,CAAA,CAAQ,kTAAR,CAQZmF;CAAAA,CAAWnF,CAAA,CAAQ,guCAAR;AAcoE,CAAA,CAdpE,CAgBf,KAAI2E,EAAapF,CAAAyF,OAAA,CAAe,EAAf,CACeJ,CADf,CAEeO,CAFf,CAGeD,CAHf,CAAjB,CAgLIzB,EAAU2B,QAAAC,cAAA,CAAuB,KAAvB,CA+Fd9F,EAAA+F,OAAA,CAAe,YAAf,CAA6B,EAA7B,CAAAC,SAAA,CAA0C,WAA1C,CAzXAC,QAA0B,EAAG,CAC3B,IAAAC,KAAA,CAAY,CAAC,eAAD,CAAkB,QAAQ,CAACC,CAAD,CAAgB,CACpD,MAAO,SAAQ,CAAChF,CAAD,CAAO,CACpB,IAAIf,EAAM,EACVc,EAAA,CAAWC,CAAX,CAAiBd,CAAA,CAAmBD,CAAnB,CAAwB,QAAQ,CAACgG,CAAD,CAAMjB,CAAN,CAAe,CAC9D,MAAO,CAAC,SAAAxB,KAAA,CAAewC,CAAA,CAAcC,CAAd,CAAmBjB,CAAnB,CAAf,CADsD,CAA/C,CAAjB,CAGA,OAAO/E,EAAAI,KAAA,CAAS,EAAT,CALa,CAD8B,CAA1C,CADe,CAyX7B,CAwGAR,EAAA+F,OAAA,CAAe,YAAf,CAAAM,OAAA,CAAoC,OAApC,CAA6C,CAAC,WAAD,CAAc,QAAQ,CAACC,CAAD,CAAY,CAAA,IACzEC,EACE,yFAFuE,CAGzEC,EAAgB,WAEpB,OAAO,SAAQ,CAACzD,CAAD,CAAO0D,CAAP,CAAe,CAsB5BC,QAASA,EAAO,CAAC3D,CAAD,CAAO,CAChBA,CAAL,EAGA5B,CAAAc,KAAA,CAAU/B,CAAA,CAAa6C,CAAb,CAAV,CAJqB,CAOvB4D,QAASA,EAAO,CAACC,CAAD;AAAM7D,CAAN,CAAY,CAC1B5B,CAAAc,KAAA,CAAU,KAAV,CACIjC,EAAA6G,UAAA,CAAkBJ,CAAlB,CAAJ,EACEtF,CAAAc,KAAA,CAAU,UAAV,CACUwE,CADV,CAEU,IAFV,CAIFtF,EAAAc,KAAA,CAAU,QAAV,CACU2E,CAAAzE,QAAA,CAAY,IAAZ,CAAkB,QAAlB,CADV,CAEU,IAFV,CAGAuE,EAAA,CAAQ3D,CAAR,CACA5B,EAAAc,KAAA,CAAU,MAAV,CAX0B,CA5B5B,GAAKc,CAAAA,CAAL,CAAW,MAAOA,EAMlB,KALA,IAAIV,CAAJ,CACIyE,EAAM/D,CADV,CAEI5B,EAAO,EAFX,CAGIyF,CAHJ,CAII7F,CACJ,CAAQsB,CAAR,CAAgByE,CAAAzE,MAAA,CAAUkE,CAAV,CAAhB,CAAA,CAEEK,CAQA,CARMvE,CAAA,CAAM,CAAN,CAQN,CANKA,CAAA,CAAM,CAAN,CAML,EANkBA,CAAA,CAAM,CAAN,CAMlB,GALEuE,CAKF,EALSvE,CAAA,CAAM,CAAN,CAAA,CAAW,SAAX,CAAuB,SAKhC,EAL6CuE,CAK7C,EAHA7F,CAGA,CAHIsB,CAAAS,MAGJ,CAFA4D,CAAA,CAAQI,CAAAC,OAAA,CAAW,CAAX,CAAchG,CAAd,CAAR,CAEA,CADA4F,CAAA,CAAQC,CAAR,CAAavE,CAAA,CAAM,CAAN,CAAAF,QAAA,CAAiBqE,CAAjB,CAAgC,EAAhC,CAAb,CACA,CAAAM,CAAA,CAAMA,CAAArD,UAAA,CAAc1C,CAAd,CAAkBsB,CAAA,CAAM,CAAN,CAAArB,OAAlB,CAER0F,EAAA,CAAQI,CAAR,CACA,OAAOR,EAAA,CAAUnF,CAAAX,KAAA,CAAU,EAAV,CAAV,CApBqB,CAL+C,CAAlC,CAA7C,CAlnBsC,CAArC,CAAD,CAqqBGT,MArqBH,CAqqBWA,MAAAC,QArqBX;", +"sources":["angular-sanitize.js"], +"names":["window","angular","undefined","sanitizeText","chars","buf","htmlSanitizeWriter","writer","noop","join","makeMap","str","lowercaseKeys","obj","items","split","i","length","lowercase","htmlParser","html","handler","parseStartTag","tag","tagName","rest","unary","blockElements","stack","last","inlineElements","parseEndTag","optionalEndTagElements","voidElements","push","attrs","replace","ATTR_REGEXP","match","name","doubleQuotedValue","singleQuotedValue","unquotedValue","decodeEntities","start","pos","end","index","text","stack.last","specialElements","RegExp","all","COMMENT_REGEXP","CDATA_REGEXP","indexOf","lastIndexOf","comment","substring","DOCTYPE_REGEXP","test","BEGING_END_TAGE_REGEXP","END_TAG_REGEXP","BEGIN_TAG_REGEXP","START_TAG_REGEXP","$sanitizeMinErr","value","hiddenPre","innerHTML","textContent","encodeEntities","SURROGATE_PAIR_REGEXP","hi","charCodeAt","low","NON_ALPHANUMERIC_REGEXP","uriValidator","ignore","out","bind","validElements","forEach","key","lkey","isImage","validAttrs","uriAttrs","$$minErr","optionalEndTagBlockElements","optionalEndTagInlineElements","extend","svgElements","htmlAttrs","svgAttrs","document","createElement","module","provider","$SanitizeProvider","$get","$$sanitizeUri","uri","filter","$sanitize","LINKY_URL_REGEXP","MAILTO_REGEXP","target","addText","addLink","url","isDefined","raw","substr"] +} diff --git a/1.4.10/angular-scenario.js b/1.4.10/angular-scenario.js new file mode 100644 index 0000000000..42e5394063 --- /dev/null +++ b/1.4.10/angular-scenario.js @@ -0,0 +1,41240 @@ +/*! + * jQuery JavaScript Library v2.1.1 + * http://jquery.com/ + * + * Includes Sizzle.js + * http://sizzlejs.com/ + * + * Copyright 2005, 2014 jQuery Foundation, Inc. and other contributors + * Released under the MIT license + * http://jquery.org/license + * + * Date: 2014-05-01T17:11Z + */ + +(function( global, factory ) {'use strict'; + + if ( typeof module === "object" && typeof module.exports === "object" ) { + // For CommonJS and CommonJS-like environments where a proper window is present, + // execute the factory and get jQuery + // For environments that do not inherently posses a window with a document + // (such as Node.js), expose a jQuery-making factory as module.exports + // This accentuates the need for the creation of a real window + // e.g. var jQuery = require("jquery")(window); + // See ticket #14549 for more info + module.exports = global.document ? + factory( global, true ) : + function( w ) { + if ( !w.document ) { + throw new Error( "jQuery requires a window with a document" ); + } + return factory( w ); + }; + } else { + factory( global ); + } + +// Pass this if window is not defined yet +}(typeof window !== "undefined" ? window : this, function( window, noGlobal ) { + +// Can't do this because several apps including ASP.NET trace +// the stack via arguments.caller.callee and Firefox dies if +// you try to trace through "use strict" call chains. (#13335) +// Support: Firefox 18+ +// + +var arr = []; + +var slice = arr.slice; + +var concat = arr.concat; + +var push = arr.push; + +var indexOf = arr.indexOf; + +var class2type = {}; + +var toString = class2type.toString; + +var hasOwn = class2type.hasOwnProperty; + +var support = {}; + + + +var + // Use the correct document accordingly with window argument (sandbox) + document = window.document, + + version = "2.1.1", + + // Define a local copy of jQuery + jQuery = function( selector, context ) { + // The jQuery object is actually just the init constructor 'enhanced' + // Need init if jQuery is called (just allow error to be thrown if not included) + return new jQuery.fn.init( selector, context ); + }, + + // Support: Android<4.1 + // Make sure we trim BOM and NBSP + rtrim = /^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, + + // Matches dashed string for camelizing + rmsPrefix = /^-ms-/, + rdashAlpha = /-([\da-z])/gi, + + // Used by jQuery.camelCase as callback to replace() + fcamelCase = function( all, letter ) { + return letter.toUpperCase(); + }; + +jQuery.fn = jQuery.prototype = { + // The current version of jQuery being used + jquery: version, + + constructor: jQuery, + + // Start with an empty selector + selector: "", + + // The default length of a jQuery object is 0 + length: 0, + + toArray: function() { + return slice.call( this ); + }, + + // Get the Nth element in the matched element set OR + // Get the whole matched element set as a clean array + get: function( num ) { + return num != null ? + + // Return just the one element from the set + ( num < 0 ? this[ num + this.length ] : this[ num ] ) : + + // Return all the elements in a clean array + slice.call( this ); + }, + + // Take an array of elements and push it onto the stack + // (returning the new matched element set) + pushStack: function( elems ) { + + // Build a new jQuery matched element set + var ret = jQuery.merge( this.constructor(), elems ); + + // Add the old object onto the stack (as a reference) + ret.prevObject = this; + ret.context = this.context; + + // Return the newly-formed element set + return ret; + }, + + // Execute a callback for every element in the matched set. + // (You can seed the arguments with an array of args, but this is + // only used internally.) + each: function( callback, args ) { + return jQuery.each( this, callback, args ); + }, + + map: function( callback ) { + return this.pushStack( jQuery.map(this, function( elem, i ) { + return callback.call( elem, i, elem ); + })); + }, + + slice: function() { + return this.pushStack( slice.apply( this, arguments ) ); + }, + + first: function() { + return this.eq( 0 ); + }, + + last: function() { + return this.eq( -1 ); + }, + + eq: function( i ) { + var len = this.length, + j = +i + ( i < 0 ? len : 0 ); + return this.pushStack( j >= 0 && j < len ? [ this[j] ] : [] ); + }, + + end: function() { + return this.prevObject || this.constructor(null); + }, + + // For internal use only. + // Behaves like an Array's method, not like a jQuery method. + push: push, + sort: arr.sort, + splice: arr.splice +}; + +jQuery.extend = jQuery.fn.extend = function() { + var options, name, src, copy, copyIsArray, clone, + target = arguments[0] || {}, + i = 1, + length = arguments.length, + deep = false; + + // Handle a deep copy situation + if ( typeof target === "boolean" ) { + deep = target; + + // skip the boolean and the target + target = arguments[ i ] || {}; + i++; + } + + // Handle case when target is a string or something (possible in deep copy) + if ( typeof target !== "object" && !jQuery.isFunction(target) ) { + target = {}; + } + + // extend jQuery itself if only one argument is passed + if ( i === length ) { + target = this; + i--; + } + + for ( ; i < length; i++ ) { + // Only deal with non-null/undefined values + if ( (options = arguments[ i ]) != null ) { + // Extend the base object + for ( name in options ) { + src = target[ name ]; + copy = options[ name ]; + + // Prevent never-ending loop + if ( target === copy ) { + continue; + } + + // Recurse if we're merging plain objects or arrays + if ( deep && copy && ( jQuery.isPlainObject(copy) || (copyIsArray = jQuery.isArray(copy)) ) ) { + if ( copyIsArray ) { + copyIsArray = false; + clone = src && jQuery.isArray(src) ? src : []; + + } else { + clone = src && jQuery.isPlainObject(src) ? src : {}; + } + + // Never move original objects, clone them + target[ name ] = jQuery.extend( deep, clone, copy ); + + // Don't bring in undefined values + } else if ( copy !== undefined ) { + target[ name ] = copy; + } + } + } + } + + // Return the modified object + return target; +}; + +jQuery.extend({ + // Unique for each copy of jQuery on the page + expando: "jQuery" + ( version + Math.random() ).replace( /\D/g, "" ), + + // Assume jQuery is ready without the ready module + isReady: true, + + error: function( msg ) { + throw new Error( msg ); + }, + + noop: function() {}, + + // See test/unit/core.js for details concerning isFunction. + // Since version 1.3, DOM methods and functions like alert + // aren't supported. They return false on IE (#2968). + isFunction: function( obj ) { + return jQuery.type(obj) === "function"; + }, + + isArray: Array.isArray, + + isWindow: function( obj ) { + return obj != null && obj === obj.window; + }, + + isNumeric: function( obj ) { + // parseFloat NaNs numeric-cast false positives (null|true|false|"") + // ...but misinterprets leading-number strings, particularly hex literals ("0x...") + // subtraction forces infinities to NaN + return !jQuery.isArray( obj ) && obj - parseFloat( obj ) >= 0; + }, + + isPlainObject: function( obj ) { + // Not plain objects: + // - Any object or value whose internal [[Class]] property is not "[object Object]" + // - DOM nodes + // - window + if ( jQuery.type( obj ) !== "object" || obj.nodeType || jQuery.isWindow( obj ) ) { + return false; + } + + if ( obj.constructor && + !hasOwn.call( obj.constructor.prototype, "isPrototypeOf" ) ) { + return false; + } + + // If the function hasn't returned already, we're confident that + // |obj| is a plain object, created by {} or constructed with new Object + return true; + }, + + isEmptyObject: function( obj ) { + var name; + for ( name in obj ) { + return false; + } + return true; + }, + + type: function( obj ) { + if ( obj == null ) { + return obj + ""; + } + // Support: Android < 4.0, iOS < 6 (functionish RegExp) + return typeof obj === "object" || typeof obj === "function" ? + class2type[ toString.call(obj) ] || "object" : + typeof obj; + }, + + // Evaluates a script in a global context + globalEval: function( code ) { + var script, + indirect = eval; + + code = jQuery.trim( code ); + + if ( code ) { + // If the code includes a valid, prologue position + // strict mode pragma, execute code by injecting a + // script tag into the document. + if ( code.indexOf("use strict") === 1 ) { + script = document.createElement("script"); + script.text = code; + document.head.appendChild( script ).parentNode.removeChild( script ); + } else { + // Otherwise, avoid the DOM node creation, insertion + // and removal by using an indirect global eval + indirect( code ); + } + } + }, + + // Convert dashed to camelCase; used by the css and data modules + // Microsoft forgot to hump their vendor prefix (#9572) + camelCase: function( string ) { + return string.replace( rmsPrefix, "ms-" ).replace( rdashAlpha, fcamelCase ); + }, + + nodeName: function( elem, name ) { + return elem.nodeName && elem.nodeName.toLowerCase() === name.toLowerCase(); + }, + + // args is for internal usage only + each: function( obj, callback, args ) { + var value, + i = 0, + length = obj.length, + isArray = isArraylike( obj ); + + if ( args ) { + if ( isArray ) { + for ( ; i < length; i++ ) { + value = callback.apply( obj[ i ], args ); + + if ( value === false ) { + break; + } + } + } else { + for ( i in obj ) { + value = callback.apply( obj[ i ], args ); + + if ( value === false ) { + break; + } + } + } + + // A special, fast, case for the most common use of each + } else { + if ( isArray ) { + for ( ; i < length; i++ ) { + value = callback.call( obj[ i ], i, obj[ i ] ); + + if ( value === false ) { + break; + } + } + } else { + for ( i in obj ) { + value = callback.call( obj[ i ], i, obj[ i ] ); + + if ( value === false ) { + break; + } + } + } + } + + return obj; + }, + + // Support: Android<4.1 + trim: function( text ) { + return text == null ? + "" : + ( text + "" ).replace( rtrim, "" ); + }, + + // results is for internal usage only + makeArray: function( arr, results ) { + var ret = results || []; + + if ( arr != null ) { + if ( isArraylike( Object(arr) ) ) { + jQuery.merge( ret, + typeof arr === "string" ? + [ arr ] : arr + ); + } else { + push.call( ret, arr ); + } + } + + return ret; + }, + + inArray: function( elem, arr, i ) { + return arr == null ? -1 : indexOf.call( arr, elem, i ); + }, + + merge: function( first, second ) { + var len = +second.length, + j = 0, + i = first.length; + + for ( ; j < len; j++ ) { + first[ i++ ] = second[ j ]; + } + + first.length = i; + + return first; + }, + + grep: function( elems, callback, invert ) { + var callbackInverse, + matches = [], + i = 0, + length = elems.length, + callbackExpect = !invert; + + // Go through the array, only saving the items + // that pass the validator function + for ( ; i < length; i++ ) { + callbackInverse = !callback( elems[ i ], i ); + if ( callbackInverse !== callbackExpect ) { + matches.push( elems[ i ] ); + } + } + + return matches; + }, + + // arg is for internal usage only + map: function( elems, callback, arg ) { + var value, + i = 0, + length = elems.length, + isArray = isArraylike( elems ), + ret = []; + + // Go through the array, translating each of the items to their new values + if ( isArray ) { + for ( ; i < length; i++ ) { + value = callback( elems[ i ], i, arg ); + + if ( value != null ) { + ret.push( value ); + } + } + + // Go through every key on the object, + } else { + for ( i in elems ) { + value = callback( elems[ i ], i, arg ); + + if ( value != null ) { + ret.push( value ); + } + } + } + + // Flatten any nested arrays + return concat.apply( [], ret ); + }, + + // A global GUID counter for objects + guid: 1, + + // Bind a function to a context, optionally partially applying any + // arguments. + proxy: function( fn, context ) { + var tmp, args, proxy; + + if ( typeof context === "string" ) { + tmp = fn[ context ]; + context = fn; + fn = tmp; + } + + // Quick check to determine if target is callable, in the spec + // this throws a TypeError, but we will just return undefined. + if ( !jQuery.isFunction( fn ) ) { + return undefined; + } + + // Simulated bind + args = slice.call( arguments, 2 ); + proxy = function() { + return fn.apply( context || this, args.concat( slice.call( arguments ) ) ); + }; + + // Set the guid of unique handler to the same of original handler, so it can be removed + proxy.guid = fn.guid = fn.guid || jQuery.guid++; + + return proxy; + }, + + now: Date.now, + + // jQuery.support is not used in Core but other projects attach their + // properties to it so it needs to exist. + support: support +}); + +// Populate the class2type map +jQuery.each("Boolean Number String Function Array Date RegExp Object Error".split(" "), function(i, name) { + class2type[ "[object " + name + "]" ] = name.toLowerCase(); +}); + +function isArraylike( obj ) { + var length = obj.length, + type = jQuery.type( obj ); + + if ( type === "function" || jQuery.isWindow( obj ) ) { + return false; + } + + if ( obj.nodeType === 1 && length ) { + return true; + } + + return type === "array" || length === 0 || + typeof length === "number" && length > 0 && ( length - 1 ) in obj; +} +var Sizzle = +/*! + * Sizzle CSS Selector Engine v1.10.19 + * http://sizzlejs.com/ + * + * Copyright 2013 jQuery Foundation, Inc. and other contributors + * Released under the MIT license + * http://jquery.org/license + * + * Date: 2014-04-18 + */ +(function( window ) { + +var i, + support, + Expr, + getText, + isXML, + tokenize, + compile, + select, + outermostContext, + sortInput, + hasDuplicate, + + // Local document vars + setDocument, + document, + docElem, + documentIsHTML, + rbuggyQSA, + rbuggyMatches, + matches, + contains, + + // Instance-specific data + expando = "sizzle" + -(new Date()), + preferredDoc = window.document, + dirruns = 0, + done = 0, + classCache = createCache(), + tokenCache = createCache(), + compilerCache = createCache(), + sortOrder = function( a, b ) { + if ( a === b ) { + hasDuplicate = true; + } + return 0; + }, + + // General-purpose constants + strundefined = typeof undefined, + MAX_NEGATIVE = 1 << 31, + + // Instance methods + hasOwn = ({}).hasOwnProperty, + arr = [], + pop = arr.pop, + push_native = arr.push, + push = arr.push, + slice = arr.slice, + // Use a stripped-down indexOf if we can't use a native one + indexOf = arr.indexOf || function( elem ) { + var i = 0, + len = this.length; + for ( ; i < len; i++ ) { + if ( this[i] === elem ) { + return i; + } + } + return -1; + }, + + booleans = "checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped", + + // Regular expressions + + // Whitespace characters http://www.w3.org/TR/css3-selectors/#whitespace + whitespace = "[\\x20\\t\\r\\n\\f]", + // http://www.w3.org/TR/css3-syntax/#characters + characterEncoding = "(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+", + + // Loosely modeled on CSS identifier characters + // An unquoted value should be a CSS identifier http://www.w3.org/TR/css3-selectors/#attribute-selectors + // Proper syntax: http://www.w3.org/TR/CSS21/syndata.html#value-def-identifier + identifier = characterEncoding.replace( "w", "w#" ), + + // Attribute selectors: http://www.w3.org/TR/selectors/#attribute-selectors + attributes = "\\[" + whitespace + "*(" + characterEncoding + ")(?:" + whitespace + + // Operator (capture 2) + "*([*^$|!~]?=)" + whitespace + + // "Attribute values must be CSS identifiers [capture 5] or strings [capture 3 or capture 4]" + "*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|(" + identifier + "))|)" + whitespace + + "*\\]", + + pseudos = ":(" + characterEncoding + ")(?:\\((" + + // To reduce the number of selectors needing tokenize in the preFilter, prefer arguments: + // 1. quoted (capture 3; capture 4 or capture 5) + "('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|" + + // 2. simple (capture 6) + "((?:\\\\.|[^\\\\()[\\]]|" + attributes + ")*)|" + + // 3. anything else (capture 2) + ".*" + + ")\\)|)", + + // Leading and non-escaped trailing whitespace, capturing some non-whitespace characters preceding the latter + rtrim = new RegExp( "^" + whitespace + "+|((?:^|[^\\\\])(?:\\\\.)*)" + whitespace + "+$", "g" ), + + rcomma = new RegExp( "^" + whitespace + "*," + whitespace + "*" ), + rcombinators = new RegExp( "^" + whitespace + "*([>+~]|" + whitespace + ")" + whitespace + "*" ), + + rattributeQuotes = new RegExp( "=" + whitespace + "*([^\\]'\"]*?)" + whitespace + "*\\]", "g" ), + + rpseudo = new RegExp( pseudos ), + ridentifier = new RegExp( "^" + identifier + "$" ), + + matchExpr = { + "ID": new RegExp( "^#(" + characterEncoding + ")" ), + "CLASS": new RegExp( "^\\.(" + characterEncoding + ")" ), + "TAG": new RegExp( "^(" + characterEncoding.replace( "w", "w*" ) + ")" ), + "ATTR": new RegExp( "^" + attributes ), + "PSEUDO": new RegExp( "^" + pseudos ), + "CHILD": new RegExp( "^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\(" + whitespace + + "*(even|odd|(([+-]|)(\\d*)n|)" + whitespace + "*(?:([+-]|)" + whitespace + + "*(\\d+)|))" + whitespace + "*\\)|)", "i" ), + "bool": new RegExp( "^(?:" + booleans + ")$", "i" ), + // For use in libraries implementing .is() + // We use this for POS matching in `select` + "needsContext": new RegExp( "^" + whitespace + "*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\(" + + whitespace + "*((?:-\\d)?\\d*)" + whitespace + "*\\)|)(?=[^-]|$)", "i" ) + }, + + rinputs = /^(?:input|select|textarea|button)$/i, + rheader = /^h\d$/i, + + rnative = /^[^{]+\{\s*\[native \w/, + + // Easily-parseable/retrievable ID or TAG or CLASS selectors + rquickExpr = /^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/, + + rsibling = /[+~]/, + rescape = /'|\\/g, + + // CSS escapes http://www.w3.org/TR/CSS21/syndata.html#escaped-characters + runescape = new RegExp( "\\\\([\\da-f]{1,6}" + whitespace + "?|(" + whitespace + ")|.)", "ig" ), + funescape = function( _, escaped, escapedWhitespace ) { + var high = "0x" + escaped - 0x10000; + // NaN means non-codepoint + // Support: Firefox<24 + // Workaround erroneous numeric interpretation of +"0x" + return high !== high || escapedWhitespace ? + escaped : + high < 0 ? + // BMP codepoint + String.fromCharCode( high + 0x10000 ) : + // Supplemental Plane codepoint (surrogate pair) + String.fromCharCode( high >> 10 | 0xD800, high & 0x3FF | 0xDC00 ); + }; + +// Optimize for push.apply( _, NodeList ) +try { + push.apply( + (arr = slice.call( preferredDoc.childNodes )), + preferredDoc.childNodes + ); + // Support: Android<4.0 + // Detect silently failing push.apply + arr[ preferredDoc.childNodes.length ].nodeType; +} catch ( e ) { + push = { apply: arr.length ? + + // Leverage slice if possible + function( target, els ) { + push_native.apply( target, slice.call(els) ); + } : + + // Support: IE<9 + // Otherwise append directly + function( target, els ) { + var j = target.length, + i = 0; + // Can't trust NodeList.length + while ( (target[j++] = els[i++]) ) {} + target.length = j - 1; + } + }; +} + +function Sizzle( selector, context, results, seed ) { + var match, elem, m, nodeType, + // QSA vars + i, groups, old, nid, newContext, newSelector; + + if ( ( context ? context.ownerDocument || context : preferredDoc ) !== document ) { + setDocument( context ); + } + + context = context || document; + results = results || []; + + if ( !selector || typeof selector !== "string" ) { + return results; + } + + if ( (nodeType = context.nodeType) !== 1 && nodeType !== 9 ) { + return []; + } + + if ( documentIsHTML && !seed ) { + + // Shortcuts + if ( (match = rquickExpr.exec( selector )) ) { + // Speed-up: Sizzle("#ID") + if ( (m = match[1]) ) { + if ( nodeType === 9 ) { + elem = context.getElementById( m ); + // Check parentNode to catch when Blackberry 4.6 returns + // nodes that are no longer in the document (jQuery #6963) + if ( elem && elem.parentNode ) { + // Handle the case where IE, Opera, and Webkit return items + // by name instead of ID + if ( elem.id === m ) { + results.push( elem ); + return results; + } + } else { + return results; + } + } else { + // Context is not a document + if ( context.ownerDocument && (elem = context.ownerDocument.getElementById( m )) && + contains( context, elem ) && elem.id === m ) { + results.push( elem ); + return results; + } + } + + // Speed-up: Sizzle("TAG") + } else if ( match[2] ) { + push.apply( results, context.getElementsByTagName( selector ) ); + return results; + + // Speed-up: Sizzle(".CLASS") + } else if ( (m = match[3]) && support.getElementsByClassName && context.getElementsByClassName ) { + push.apply( results, context.getElementsByClassName( m ) ); + return results; + } + } + + // QSA path + if ( support.qsa && (!rbuggyQSA || !rbuggyQSA.test( selector )) ) { + nid = old = expando; + newContext = context; + newSelector = nodeType === 9 && selector; + + // qSA works strangely on Element-rooted queries + // We can work around this by specifying an extra ID on the root + // and working up from there (Thanks to Andrew Dupont for the technique) + // IE 8 doesn't work on object elements + if ( nodeType === 1 && context.nodeName.toLowerCase() !== "object" ) { + groups = tokenize( selector ); + + if ( (old = context.getAttribute("id")) ) { + nid = old.replace( rescape, "\\$&" ); + } else { + context.setAttribute( "id", nid ); + } + nid = "[id='" + nid + "'] "; + + i = groups.length; + while ( i-- ) { + groups[i] = nid + toSelector( groups[i] ); + } + newContext = rsibling.test( selector ) && testContext( context.parentNode ) || context; + newSelector = groups.join(","); + } + + if ( newSelector ) { + try { + push.apply( results, + newContext.querySelectorAll( newSelector ) + ); + return results; + } catch(qsaError) { + } finally { + if ( !old ) { + context.removeAttribute("id"); + } + } + } + } + } + + // All others + return select( selector.replace( rtrim, "$1" ), context, results, seed ); +} + +/** + * Create key-value caches of limited size + * @returns {Function(string, Object)} Returns the Object data after storing it on itself with + * property name the (space-suffixed) string and (if the cache is larger than Expr.cacheLength) + * deleting the oldest entry + */ +function createCache() { + var keys = []; + + function cache( key, value ) { + // Use (key + " ") to avoid collision with native prototype properties (see Issue #157) + if ( keys.push( key + " " ) > Expr.cacheLength ) { + // Only keep the most recent entries + delete cache[ keys.shift() ]; + } + return (cache[ key + " " ] = value); + } + return cache; +} + +/** + * Mark a function for special use by Sizzle + * @param {Function} fn The function to mark + */ +function markFunction( fn ) { + fn[ expando ] = true; + return fn; +} + +/** + * Support testing using an element + * @param {Function} fn Passed the created div and expects a boolean result + */ +function assert( fn ) { + var div = document.createElement("div"); + + try { + return !!fn( div ); + } catch (e) { + return false; + } finally { + // Remove from its parent by default + if ( div.parentNode ) { + div.parentNode.removeChild( div ); + } + // release memory in IE + div = null; + } +} + +/** + * Adds the same handler for all of the specified attrs + * @param {String} attrs Pipe-separated list of attributes + * @param {Function} handler The method that will be applied + */ +function addHandle( attrs, handler ) { + var arr = attrs.split("|"), + i = attrs.length; + + while ( i-- ) { + Expr.attrHandle[ arr[i] ] = handler; + } +} + +/** + * Checks document order of two siblings + * @param {Element} a + * @param {Element} b + * @returns {Number} Returns less than 0 if a precedes b, greater than 0 if a follows b + */ +function siblingCheck( a, b ) { + var cur = b && a, + diff = cur && a.nodeType === 1 && b.nodeType === 1 && + ( ~b.sourceIndex || MAX_NEGATIVE ) - + ( ~a.sourceIndex || MAX_NEGATIVE ); + + // Use IE sourceIndex if available on both nodes + if ( diff ) { + return diff; + } + + // Check if b follows a + if ( cur ) { + while ( (cur = cur.nextSibling) ) { + if ( cur === b ) { + return -1; + } + } + } + + return a ? 1 : -1; +} + +/** + * Returns a function to use in pseudos for input types + * @param {String} type + */ +function createInputPseudo( type ) { + return function( elem ) { + var name = elem.nodeName.toLowerCase(); + return name === "input" && elem.type === type; + }; +} + +/** + * Returns a function to use in pseudos for buttons + * @param {String} type + */ +function createButtonPseudo( type ) { + return function( elem ) { + var name = elem.nodeName.toLowerCase(); + return (name === "input" || name === "button") && elem.type === type; + }; +} + +/** + * Returns a function to use in pseudos for positionals + * @param {Function} fn + */ +function createPositionalPseudo( fn ) { + return markFunction(function( argument ) { + argument = +argument; + return markFunction(function( seed, matches ) { + var j, + matchIndexes = fn( [], seed.length, argument ), + i = matchIndexes.length; + + // Match elements found at the specified indexes + while ( i-- ) { + if ( seed[ (j = matchIndexes[i]) ] ) { + seed[j] = !(matches[j] = seed[j]); + } + } + }); + }); +} + +/** + * Checks a node for validity as a Sizzle context + * @param {Element|Object=} context + * @returns {Element|Object|Boolean} The input node if acceptable, otherwise a falsy value + */ +function testContext( context ) { + return context && typeof context.getElementsByTagName !== strundefined && context; +} + +// Expose support vars for convenience +support = Sizzle.support = {}; + +/** + * Detects XML nodes + * @param {Element|Object} elem An element or a document + * @returns {Boolean} True iff elem is a non-HTML XML node + */ +isXML = Sizzle.isXML = function( elem ) { + // documentElement is verified for cases where it doesn't yet exist + // (such as loading iframes in IE - #4833) + var documentElement = elem && (elem.ownerDocument || elem).documentElement; + return documentElement ? documentElement.nodeName !== "HTML" : false; +}; + +/** + * Sets document-related variables once based on the current document + * @param {Element|Object} [doc] An element or document object to use to set the document + * @returns {Object} Returns the current document + */ +setDocument = Sizzle.setDocument = function( node ) { + var hasCompare, + doc = node ? node.ownerDocument || node : preferredDoc, + parent = doc.defaultView; + + // If no document and documentElement is available, return + if ( doc === document || doc.nodeType !== 9 || !doc.documentElement ) { + return document; + } + + // Set our document + document = doc; + docElem = doc.documentElement; + + // Support tests + documentIsHTML = !isXML( doc ); + + // Support: IE>8 + // If iframe document is assigned to "document" variable and if iframe has been reloaded, + // IE will throw "permission denied" error when accessing "document" variable, see jQuery #13936 + // IE6-8 do not support the defaultView property so parent will be undefined + if ( parent && parent !== parent.top ) { + // IE11 does not have attachEvent, so all must suffer + if ( parent.addEventListener ) { + parent.addEventListener( "unload", function() { + setDocument(); + }, false ); + } else if ( parent.attachEvent ) { + parent.attachEvent( "onunload", function() { + setDocument(); + }); + } + } + + /* Attributes + ---------------------------------------------------------------------- */ + + // Support: IE<8 + // Verify that getAttribute really returns attributes and not properties (excepting IE8 booleans) + support.attributes = assert(function( div ) { + div.className = "i"; + return !div.getAttribute("className"); + }); + + /* getElement(s)By* + ---------------------------------------------------------------------- */ + + // Check if getElementsByTagName("*") returns only elements + support.getElementsByTagName = assert(function( div ) { + div.appendChild( doc.createComment("") ); + return !div.getElementsByTagName("*").length; + }); + + // Check if getElementsByClassName can be trusted + support.getElementsByClassName = rnative.test( doc.getElementsByClassName ) && assert(function( div ) { + div.innerHTML = "
"; + + // Support: Safari<4 + // Catch class over-caching + div.firstChild.className = "i"; + // Support: Opera<10 + // Catch gEBCN failure to find non-leading classes + return div.getElementsByClassName("i").length === 2; + }); + + // Support: IE<10 + // Check if getElementById returns elements by name + // The broken getElementById methods don't pick up programatically-set names, + // so use a roundabout getElementsByName test + support.getById = assert(function( div ) { + docElem.appendChild( div ).id = expando; + return !doc.getElementsByName || !doc.getElementsByName( expando ).length; + }); + + // ID find and filter + if ( support.getById ) { + Expr.find["ID"] = function( id, context ) { + if ( typeof context.getElementById !== strundefined && documentIsHTML ) { + var m = context.getElementById( id ); + // Check parentNode to catch when Blackberry 4.6 returns + // nodes that are no longer in the document #6963 + return m && m.parentNode ? [ m ] : []; + } + }; + Expr.filter["ID"] = function( id ) { + var attrId = id.replace( runescape, funescape ); + return function( elem ) { + return elem.getAttribute("id") === attrId; + }; + }; + } else { + // Support: IE6/7 + // getElementById is not reliable as a find shortcut + delete Expr.find["ID"]; + + Expr.filter["ID"] = function( id ) { + var attrId = id.replace( runescape, funescape ); + return function( elem ) { + var node = typeof elem.getAttributeNode !== strundefined && elem.getAttributeNode("id"); + return node && node.value === attrId; + }; + }; + } + + // Tag + Expr.find["TAG"] = support.getElementsByTagName ? + function( tag, context ) { + if ( typeof context.getElementsByTagName !== strundefined ) { + return context.getElementsByTagName( tag ); + } + } : + function( tag, context ) { + var elem, + tmp = [], + i = 0, + results = context.getElementsByTagName( tag ); + + // Filter out possible comments + if ( tag === "*" ) { + while ( (elem = results[i++]) ) { + if ( elem.nodeType === 1 ) { + tmp.push( elem ); + } + } + + return tmp; + } + return results; + }; + + // Class + Expr.find["CLASS"] = support.getElementsByClassName && function( className, context ) { + if ( typeof context.getElementsByClassName !== strundefined && documentIsHTML ) { + return context.getElementsByClassName( className ); + } + }; + + /* QSA/matchesSelector + ---------------------------------------------------------------------- */ + + // QSA and matchesSelector support + + // matchesSelector(:active) reports false when true (IE9/Opera 11.5) + rbuggyMatches = []; + + // qSa(:focus) reports false when true (Chrome 21) + // We allow this because of a bug in IE8/9 that throws an error + // whenever `document.activeElement` is accessed on an iframe + // So, we allow :focus to pass through QSA all the time to avoid the IE error + // See http://bugs.jquery.com/ticket/13378 + rbuggyQSA = []; + + if ( (support.qsa = rnative.test( doc.querySelectorAll )) ) { + // Build QSA regex + // Regex strategy adopted from Diego Perini + assert(function( div ) { + // Select is set to empty string on purpose + // This is to test IE's treatment of not explicitly + // setting a boolean content attribute, + // since its presence should be enough + // http://bugs.jquery.com/ticket/12359 + div.innerHTML = ""; + + // Support: IE8, Opera 11-12.16 + // Nothing should be selected when empty strings follow ^= or $= or *= + // The test attribute must be unknown in Opera but "safe" for WinRT + // http://msdn.microsoft.com/en-us/library/ie/hh465388.aspx#attribute_section + if ( div.querySelectorAll("[msallowclip^='']").length ) { + rbuggyQSA.push( "[*^$]=" + whitespace + "*(?:''|\"\")" ); + } + + // Support: IE8 + // Boolean attributes and "value" are not treated correctly + if ( !div.querySelectorAll("[selected]").length ) { + rbuggyQSA.push( "\\[" + whitespace + "*(?:value|" + booleans + ")" ); + } + + // Webkit/Opera - :checked should return selected option elements + // http://www.w3.org/TR/2011/REC-css3-selectors-20110929/#checked + // IE8 throws error here and will not see later tests + if ( !div.querySelectorAll(":checked").length ) { + rbuggyQSA.push(":checked"); + } + }); + + assert(function( div ) { + // Support: Windows 8 Native Apps + // The type and name attributes are restricted during .innerHTML assignment + var input = doc.createElement("input"); + input.setAttribute( "type", "hidden" ); + div.appendChild( input ).setAttribute( "name", "D" ); + + // Support: IE8 + // Enforce case-sensitivity of name attribute + if ( div.querySelectorAll("[name=d]").length ) { + rbuggyQSA.push( "name" + whitespace + "*[*^$|!~]?=" ); + } + + // FF 3.5 - :enabled/:disabled and hidden elements (hidden elements are still enabled) + // IE8 throws error here and will not see later tests + if ( !div.querySelectorAll(":enabled").length ) { + rbuggyQSA.push( ":enabled", ":disabled" ); + } + + // Opera 10-11 does not throw on post-comma invalid pseudos + div.querySelectorAll("*,:x"); + rbuggyQSA.push(",.*:"); + }); + } + + if ( (support.matchesSelector = rnative.test( (matches = docElem.matches || + docElem.webkitMatchesSelector || + docElem.mozMatchesSelector || + docElem.oMatchesSelector || + docElem.msMatchesSelector) )) ) { + + assert(function( div ) { + // Check to see if it's possible to do matchesSelector + // on a disconnected node (IE 9) + support.disconnectedMatch = matches.call( div, "div" ); + + // This should fail with an exception + // Gecko does not error, returns false instead + matches.call( div, "[s!='']:x" ); + rbuggyMatches.push( "!=", pseudos ); + }); + } + + rbuggyQSA = rbuggyQSA.length && new RegExp( rbuggyQSA.join("|") ); + rbuggyMatches = rbuggyMatches.length && new RegExp( rbuggyMatches.join("|") ); + + /* Contains + ---------------------------------------------------------------------- */ + hasCompare = rnative.test( docElem.compareDocumentPosition ); + + // Element contains another + // Purposefully does not implement inclusive descendent + // As in, an element does not contain itself + contains = hasCompare || rnative.test( docElem.contains ) ? + function( a, b ) { + var adown = a.nodeType === 9 ? a.documentElement : a, + bup = b && b.parentNode; + return a === bup || !!( bup && bup.nodeType === 1 && ( + adown.contains ? + adown.contains( bup ) : + a.compareDocumentPosition && a.compareDocumentPosition( bup ) & 16 + )); + } : + function( a, b ) { + if ( b ) { + while ( (b = b.parentNode) ) { + if ( b === a ) { + return true; + } + } + } + return false; + }; + + /* Sorting + ---------------------------------------------------------------------- */ + + // Document order sorting + sortOrder = hasCompare ? + function( a, b ) { + + // Flag for duplicate removal + if ( a === b ) { + hasDuplicate = true; + return 0; + } + + // Sort on method existence if only one input has compareDocumentPosition + var compare = !a.compareDocumentPosition - !b.compareDocumentPosition; + if ( compare ) { + return compare; + } + + // Calculate position if both inputs belong to the same document + compare = ( a.ownerDocument || a ) === ( b.ownerDocument || b ) ? + a.compareDocumentPosition( b ) : + + // Otherwise we know they are disconnected + 1; + + // Disconnected nodes + if ( compare & 1 || + (!support.sortDetached && b.compareDocumentPosition( a ) === compare) ) { + + // Choose the first element that is related to our preferred document + if ( a === doc || a.ownerDocument === preferredDoc && contains(preferredDoc, a) ) { + return -1; + } + if ( b === doc || b.ownerDocument === preferredDoc && contains(preferredDoc, b) ) { + return 1; + } + + // Maintain original order + return sortInput ? + ( indexOf.call( sortInput, a ) - indexOf.call( sortInput, b ) ) : + 0; + } + + return compare & 4 ? -1 : 1; + } : + function( a, b ) { + // Exit early if the nodes are identical + if ( a === b ) { + hasDuplicate = true; + return 0; + } + + var cur, + i = 0, + aup = a.parentNode, + bup = b.parentNode, + ap = [ a ], + bp = [ b ]; + + // Parentless nodes are either documents or disconnected + if ( !aup || !bup ) { + return a === doc ? -1 : + b === doc ? 1 : + aup ? -1 : + bup ? 1 : + sortInput ? + ( indexOf.call( sortInput, a ) - indexOf.call( sortInput, b ) ) : + 0; + + // If the nodes are siblings, we can do a quick check + } else if ( aup === bup ) { + return siblingCheck( a, b ); + } + + // Otherwise we need full lists of their ancestors for comparison + cur = a; + while ( (cur = cur.parentNode) ) { + ap.unshift( cur ); + } + cur = b; + while ( (cur = cur.parentNode) ) { + bp.unshift( cur ); + } + + // Walk down the tree looking for a discrepancy + while ( ap[i] === bp[i] ) { + i++; + } + + return i ? + // Do a sibling check if the nodes have a common ancestor + siblingCheck( ap[i], bp[i] ) : + + // Otherwise nodes in our document sort first + ap[i] === preferredDoc ? -1 : + bp[i] === preferredDoc ? 1 : + 0; + }; + + return doc; +}; + +Sizzle.matches = function( expr, elements ) { + return Sizzle( expr, null, null, elements ); +}; + +Sizzle.matchesSelector = function( elem, expr ) { + // Set document vars if needed + if ( ( elem.ownerDocument || elem ) !== document ) { + setDocument( elem ); + } + + // Make sure that attribute selectors are quoted + expr = expr.replace( rattributeQuotes, "='$1']" ); + + if ( support.matchesSelector && documentIsHTML && + ( !rbuggyMatches || !rbuggyMatches.test( expr ) ) && + ( !rbuggyQSA || !rbuggyQSA.test( expr ) ) ) { + + try { + var ret = matches.call( elem, expr ); + + // IE 9's matchesSelector returns false on disconnected nodes + if ( ret || support.disconnectedMatch || + // As well, disconnected nodes are said to be in a document + // fragment in IE 9 + elem.document && elem.document.nodeType !== 11 ) { + return ret; + } + } catch(e) {} + } + + return Sizzle( expr, document, null, [ elem ] ).length > 0; +}; + +Sizzle.contains = function( context, elem ) { + // Set document vars if needed + if ( ( context.ownerDocument || context ) !== document ) { + setDocument( context ); + } + return contains( context, elem ); +}; + +Sizzle.attr = function( elem, name ) { + // Set document vars if needed + if ( ( elem.ownerDocument || elem ) !== document ) { + setDocument( elem ); + } + + var fn = Expr.attrHandle[ name.toLowerCase() ], + // Don't get fooled by Object.prototype properties (jQuery #13807) + val = fn && hasOwn.call( Expr.attrHandle, name.toLowerCase() ) ? + fn( elem, name, !documentIsHTML ) : + undefined; + + return val !== undefined ? + val : + support.attributes || !documentIsHTML ? + elem.getAttribute( name ) : + (val = elem.getAttributeNode(name)) && val.specified ? + val.value : + null; +}; + +Sizzle.error = function( msg ) { + throw new Error( "Syntax error, unrecognized expression: " + msg ); +}; + +/** + * Document sorting and removing duplicates + * @param {ArrayLike} results + */ +Sizzle.uniqueSort = function( results ) { + var elem, + duplicates = [], + j = 0, + i = 0; + + // Unless we *know* we can detect duplicates, assume their presence + hasDuplicate = !support.detectDuplicates; + sortInput = !support.sortStable && results.slice( 0 ); + results.sort( sortOrder ); + + if ( hasDuplicate ) { + while ( (elem = results[i++]) ) { + if ( elem === results[ i ] ) { + j = duplicates.push( i ); + } + } + while ( j-- ) { + results.splice( duplicates[ j ], 1 ); + } + } + + // Clear input after sorting to release objects + // See https://github.com/jquery/sizzle/pull/225 + sortInput = null; + + return results; +}; + +/** + * Utility function for retrieving the text value of an array of DOM nodes + * @param {Array|Element} elem + */ +getText = Sizzle.getText = function( elem ) { + var node, + ret = "", + i = 0, + nodeType = elem.nodeType; + + if ( !nodeType ) { + // If no nodeType, this is expected to be an array + while ( (node = elem[i++]) ) { + // Do not traverse comment nodes + ret += getText( node ); + } + } else if ( nodeType === 1 || nodeType === 9 || nodeType === 11 ) { + // Use textContent for elements + // innerText usage removed for consistency of new lines (jQuery #11153) + if ( typeof elem.textContent === "string" ) { + return elem.textContent; + } else { + // Traverse its children + for ( elem = elem.firstChild; elem; elem = elem.nextSibling ) { + ret += getText( elem ); + } + } + } else if ( nodeType === 3 || nodeType === 4 ) { + return elem.nodeValue; + } + // Do not include comment or processing instruction nodes + + return ret; +}; + +Expr = Sizzle.selectors = { + + // Can be adjusted by the user + cacheLength: 50, + + createPseudo: markFunction, + + match: matchExpr, + + attrHandle: {}, + + find: {}, + + relative: { + ">": { dir: "parentNode", first: true }, + " ": { dir: "parentNode" }, + "+": { dir: "previousSibling", first: true }, + "~": { dir: "previousSibling" } + }, + + preFilter: { + "ATTR": function( match ) { + match[1] = match[1].replace( runescape, funescape ); + + // Move the given value to match[3] whether quoted or unquoted + match[3] = ( match[3] || match[4] || match[5] || "" ).replace( runescape, funescape ); + + if ( match[2] === "~=" ) { + match[3] = " " + match[3] + " "; + } + + return match.slice( 0, 4 ); + }, + + "CHILD": function( match ) { + /* matches from matchExpr["CHILD"] + 1 type (only|nth|...) + 2 what (child|of-type) + 3 argument (even|odd|\d*|\d*n([+-]\d+)?|...) + 4 xn-component of xn+y argument ([+-]?\d*n|) + 5 sign of xn-component + 6 x of xn-component + 7 sign of y-component + 8 y of y-component + */ + match[1] = match[1].toLowerCase(); + + if ( match[1].slice( 0, 3 ) === "nth" ) { + // nth-* requires argument + if ( !match[3] ) { + Sizzle.error( match[0] ); + } + + // numeric x and y parameters for Expr.filter.CHILD + // remember that false/true cast respectively to 0/1 + match[4] = +( match[4] ? match[5] + (match[6] || 1) : 2 * ( match[3] === "even" || match[3] === "odd" ) ); + match[5] = +( ( match[7] + match[8] ) || match[3] === "odd" ); + + // other types prohibit arguments + } else if ( match[3] ) { + Sizzle.error( match[0] ); + } + + return match; + }, + + "PSEUDO": function( match ) { + var excess, + unquoted = !match[6] && match[2]; + + if ( matchExpr["CHILD"].test( match[0] ) ) { + return null; + } + + // Accept quoted arguments as-is + if ( match[3] ) { + match[2] = match[4] || match[5] || ""; + + // Strip excess characters from unquoted arguments + } else if ( unquoted && rpseudo.test( unquoted ) && + // Get excess from tokenize (recursively) + (excess = tokenize( unquoted, true )) && + // advance to the next closing parenthesis + (excess = unquoted.indexOf( ")", unquoted.length - excess ) - unquoted.length) ) { + + // excess is a negative index + match[0] = match[0].slice( 0, excess ); + match[2] = unquoted.slice( 0, excess ); + } + + // Return only captures needed by the pseudo filter method (type and argument) + return match.slice( 0, 3 ); + } + }, + + filter: { + + "TAG": function( nodeNameSelector ) { + var nodeName = nodeNameSelector.replace( runescape, funescape ).toLowerCase(); + return nodeNameSelector === "*" ? + function() { return true; } : + function( elem ) { + return elem.nodeName && elem.nodeName.toLowerCase() === nodeName; + }; + }, + + "CLASS": function( className ) { + var pattern = classCache[ className + " " ]; + + return pattern || + (pattern = new RegExp( "(^|" + whitespace + ")" + className + "(" + whitespace + "|$)" )) && + classCache( className, function( elem ) { + return pattern.test( typeof elem.className === "string" && elem.className || typeof elem.getAttribute !== strundefined && elem.getAttribute("class") || "" ); + }); + }, + + "ATTR": function( name, operator, check ) { + return function( elem ) { + var result = Sizzle.attr( elem, name ); + + if ( result == null ) { + return operator === "!="; + } + if ( !operator ) { + return true; + } + + result += ""; + + return operator === "=" ? result === check : + operator === "!=" ? result !== check : + operator === "^=" ? check && result.indexOf( check ) === 0 : + operator === "*=" ? check && result.indexOf( check ) > -1 : + operator === "$=" ? check && result.slice( -check.length ) === check : + operator === "~=" ? ( " " + result + " " ).indexOf( check ) > -1 : + operator === "|=" ? result === check || result.slice( 0, check.length + 1 ) === check + "-" : + false; + }; + }, + + "CHILD": function( type, what, argument, first, last ) { + var simple = type.slice( 0, 3 ) !== "nth", + forward = type.slice( -4 ) !== "last", + ofType = what === "of-type"; + + return first === 1 && last === 0 ? + + // Shortcut for :nth-*(n) + function( elem ) { + return !!elem.parentNode; + } : + + function( elem, context, xml ) { + var cache, outerCache, node, diff, nodeIndex, start, + dir = simple !== forward ? "nextSibling" : "previousSibling", + parent = elem.parentNode, + name = ofType && elem.nodeName.toLowerCase(), + useCache = !xml && !ofType; + + if ( parent ) { + + // :(first|last|only)-(child|of-type) + if ( simple ) { + while ( dir ) { + node = elem; + while ( (node = node[ dir ]) ) { + if ( ofType ? node.nodeName.toLowerCase() === name : node.nodeType === 1 ) { + return false; + } + } + // Reverse direction for :only-* (if we haven't yet done so) + start = dir = type === "only" && !start && "nextSibling"; + } + return true; + } + + start = [ forward ? parent.firstChild : parent.lastChild ]; + + // non-xml :nth-child(...) stores cache data on `parent` + if ( forward && useCache ) { + // Seek `elem` from a previously-cached index + outerCache = parent[ expando ] || (parent[ expando ] = {}); + cache = outerCache[ type ] || []; + nodeIndex = cache[0] === dirruns && cache[1]; + diff = cache[0] === dirruns && cache[2]; + node = nodeIndex && parent.childNodes[ nodeIndex ]; + + while ( (node = ++nodeIndex && node && node[ dir ] || + + // Fallback to seeking `elem` from the start + (diff = nodeIndex = 0) || start.pop()) ) { + + // When found, cache indexes on `parent` and break + if ( node.nodeType === 1 && ++diff && node === elem ) { + outerCache[ type ] = [ dirruns, nodeIndex, diff ]; + break; + } + } + + // Use previously-cached element index if available + } else if ( useCache && (cache = (elem[ expando ] || (elem[ expando ] = {}))[ type ]) && cache[0] === dirruns ) { + diff = cache[1]; + + // xml :nth-child(...) or :nth-last-child(...) or :nth(-last)?-of-type(...) + } else { + // Use the same loop as above to seek `elem` from the start + while ( (node = ++nodeIndex && node && node[ dir ] || + (diff = nodeIndex = 0) || start.pop()) ) { + + if ( ( ofType ? node.nodeName.toLowerCase() === name : node.nodeType === 1 ) && ++diff ) { + // Cache the index of each encountered element + if ( useCache ) { + (node[ expando ] || (node[ expando ] = {}))[ type ] = [ dirruns, diff ]; + } + + if ( node === elem ) { + break; + } + } + } + } + + // Incorporate the offset, then check against cycle size + diff -= last; + return diff === first || ( diff % first === 0 && diff / first >= 0 ); + } + }; + }, + + "PSEUDO": function( pseudo, argument ) { + // pseudo-class names are case-insensitive + // http://www.w3.org/TR/selectors/#pseudo-classes + // Prioritize by case sensitivity in case custom pseudos are added with uppercase letters + // Remember that setFilters inherits from pseudos + var args, + fn = Expr.pseudos[ pseudo ] || Expr.setFilters[ pseudo.toLowerCase() ] || + Sizzle.error( "unsupported pseudo: " + pseudo ); + + // The user may use createPseudo to indicate that + // arguments are needed to create the filter function + // just as Sizzle does + if ( fn[ expando ] ) { + return fn( argument ); + } + + // But maintain support for old signatures + if ( fn.length > 1 ) { + args = [ pseudo, pseudo, "", argument ]; + return Expr.setFilters.hasOwnProperty( pseudo.toLowerCase() ) ? + markFunction(function( seed, matches ) { + var idx, + matched = fn( seed, argument ), + i = matched.length; + while ( i-- ) { + idx = indexOf.call( seed, matched[i] ); + seed[ idx ] = !( matches[ idx ] = matched[i] ); + } + }) : + function( elem ) { + return fn( elem, 0, args ); + }; + } + + return fn; + } + }, + + pseudos: { + // Potentially complex pseudos + "not": markFunction(function( selector ) { + // Trim the selector passed to compile + // to avoid treating leading and trailing + // spaces as combinators + var input = [], + results = [], + matcher = compile( selector.replace( rtrim, "$1" ) ); + + return matcher[ expando ] ? + markFunction(function( seed, matches, context, xml ) { + var elem, + unmatched = matcher( seed, null, xml, [] ), + i = seed.length; + + // Match elements unmatched by `matcher` + while ( i-- ) { + if ( (elem = unmatched[i]) ) { + seed[i] = !(matches[i] = elem); + } + } + }) : + function( elem, context, xml ) { + input[0] = elem; + matcher( input, null, xml, results ); + return !results.pop(); + }; + }), + + "has": markFunction(function( selector ) { + return function( elem ) { + return Sizzle( selector, elem ).length > 0; + }; + }), + + "contains": markFunction(function( text ) { + return function( elem ) { + return ( elem.textContent || elem.innerText || getText( elem ) ).indexOf( text ) > -1; + }; + }), + + // "Whether an element is represented by a :lang() selector + // is based solely on the element's language value + // being equal to the identifier C, + // or beginning with the identifier C immediately followed by "-". + // The matching of C against the element's language value is performed case-insensitively. + // The identifier C does not have to be a valid language name." + // http://www.w3.org/TR/selectors/#lang-pseudo + "lang": markFunction( function( lang ) { + // lang value must be a valid identifier + if ( !ridentifier.test(lang || "") ) { + Sizzle.error( "unsupported lang: " + lang ); + } + lang = lang.replace( runescape, funescape ).toLowerCase(); + return function( elem ) { + var elemLang; + do { + if ( (elemLang = documentIsHTML ? + elem.lang : + elem.getAttribute("xml:lang") || elem.getAttribute("lang")) ) { + + elemLang = elemLang.toLowerCase(); + return elemLang === lang || elemLang.indexOf( lang + "-" ) === 0; + } + } while ( (elem = elem.parentNode) && elem.nodeType === 1 ); + return false; + }; + }), + + // Miscellaneous + "target": function( elem ) { + var hash = window.location && window.location.hash; + return hash && hash.slice( 1 ) === elem.id; + }, + + "root": function( elem ) { + return elem === docElem; + }, + + "focus": function( elem ) { + return elem === document.activeElement && (!document.hasFocus || document.hasFocus()) && !!(elem.type || elem.href || ~elem.tabIndex); + }, + + // Boolean properties + "enabled": function( elem ) { + return elem.disabled === false; + }, + + "disabled": function( elem ) { + return elem.disabled === true; + }, + + "checked": function( elem ) { + // In CSS3, :checked should return both checked and selected elements + // http://www.w3.org/TR/2011/REC-css3-selectors-20110929/#checked + var nodeName = elem.nodeName.toLowerCase(); + return (nodeName === "input" && !!elem.checked) || (nodeName === "option" && !!elem.selected); + }, + + "selected": function( elem ) { + // Accessing this property makes selected-by-default + // options in Safari work properly + if ( elem.parentNode ) { + elem.parentNode.selectedIndex; + } + + return elem.selected === true; + }, + + // Contents + "empty": function( elem ) { + // http://www.w3.org/TR/selectors/#empty-pseudo + // :empty is negated by element (1) or content nodes (text: 3; cdata: 4; entity ref: 5), + // but not by others (comment: 8; processing instruction: 7; etc.) + // nodeType < 6 works because attributes (2) do not appear as children + for ( elem = elem.firstChild; elem; elem = elem.nextSibling ) { + if ( elem.nodeType < 6 ) { + return false; + } + } + return true; + }, + + "parent": function( elem ) { + return !Expr.pseudos["empty"]( elem ); + }, + + // Element/input types + "header": function( elem ) { + return rheader.test( elem.nodeName ); + }, + + "input": function( elem ) { + return rinputs.test( elem.nodeName ); + }, + + "button": function( elem ) { + var name = elem.nodeName.toLowerCase(); + return name === "input" && elem.type === "button" || name === "button"; + }, + + "text": function( elem ) { + var attr; + return elem.nodeName.toLowerCase() === "input" && + elem.type === "text" && + + // Support: IE<8 + // New HTML5 attribute values (e.g., "search") appear with elem.type === "text" + ( (attr = elem.getAttribute("type")) == null || attr.toLowerCase() === "text" ); + }, + + // Position-in-collection + "first": createPositionalPseudo(function() { + return [ 0 ]; + }), + + "last": createPositionalPseudo(function( matchIndexes, length ) { + return [ length - 1 ]; + }), + + "eq": createPositionalPseudo(function( matchIndexes, length, argument ) { + return [ argument < 0 ? argument + length : argument ]; + }), + + "even": createPositionalPseudo(function( matchIndexes, length ) { + var i = 0; + for ( ; i < length; i += 2 ) { + matchIndexes.push( i ); + } + return matchIndexes; + }), + + "odd": createPositionalPseudo(function( matchIndexes, length ) { + var i = 1; + for ( ; i < length; i += 2 ) { + matchIndexes.push( i ); + } + return matchIndexes; + }), + + "lt": createPositionalPseudo(function( matchIndexes, length, argument ) { + var i = argument < 0 ? argument + length : argument; + for ( ; --i >= 0; ) { + matchIndexes.push( i ); + } + return matchIndexes; + }), + + "gt": createPositionalPseudo(function( matchIndexes, length, argument ) { + var i = argument < 0 ? argument + length : argument; + for ( ; ++i < length; ) { + matchIndexes.push( i ); + } + return matchIndexes; + }) + } +}; + +Expr.pseudos["nth"] = Expr.pseudos["eq"]; + +// Add button/input type pseudos +for ( i in { radio: true, checkbox: true, file: true, password: true, image: true } ) { + Expr.pseudos[ i ] = createInputPseudo( i ); +} +for ( i in { submit: true, reset: true } ) { + Expr.pseudos[ i ] = createButtonPseudo( i ); +} + +// Easy API for creating new setFilters +function setFilters() {} +setFilters.prototype = Expr.filters = Expr.pseudos; +Expr.setFilters = new setFilters(); + +tokenize = Sizzle.tokenize = function( selector, parseOnly ) { + var matched, match, tokens, type, + soFar, groups, preFilters, + cached = tokenCache[ selector + " " ]; + + if ( cached ) { + return parseOnly ? 0 : cached.slice( 0 ); + } + + soFar = selector; + groups = []; + preFilters = Expr.preFilter; + + while ( soFar ) { + + // Comma and first run + if ( !matched || (match = rcomma.exec( soFar )) ) { + if ( match ) { + // Don't consume trailing commas as valid + soFar = soFar.slice( match[0].length ) || soFar; + } + groups.push( (tokens = []) ); + } + + matched = false; + + // Combinators + if ( (match = rcombinators.exec( soFar )) ) { + matched = match.shift(); + tokens.push({ + value: matched, + // Cast descendant combinators to space + type: match[0].replace( rtrim, " " ) + }); + soFar = soFar.slice( matched.length ); + } + + // Filters + for ( type in Expr.filter ) { + if ( (match = matchExpr[ type ].exec( soFar )) && (!preFilters[ type ] || + (match = preFilters[ type ]( match ))) ) { + matched = match.shift(); + tokens.push({ + value: matched, + type: type, + matches: match + }); + soFar = soFar.slice( matched.length ); + } + } + + if ( !matched ) { + break; + } + } + + // Return the length of the invalid excess + // if we're just parsing + // Otherwise, throw an error or return tokens + return parseOnly ? + soFar.length : + soFar ? + Sizzle.error( selector ) : + // Cache the tokens + tokenCache( selector, groups ).slice( 0 ); +}; + +function toSelector( tokens ) { + var i = 0, + len = tokens.length, + selector = ""; + for ( ; i < len; i++ ) { + selector += tokens[i].value; + } + return selector; +} + +function addCombinator( matcher, combinator, base ) { + var dir = combinator.dir, + checkNonElements = base && dir === "parentNode", + doneName = done++; + + return combinator.first ? + // Check against closest ancestor/preceding element + function( elem, context, xml ) { + while ( (elem = elem[ dir ]) ) { + if ( elem.nodeType === 1 || checkNonElements ) { + return matcher( elem, context, xml ); + } + } + } : + + // Check against all ancestor/preceding elements + function( elem, context, xml ) { + var oldCache, outerCache, + newCache = [ dirruns, doneName ]; + + // We can't set arbitrary data on XML nodes, so they don't benefit from dir caching + if ( xml ) { + while ( (elem = elem[ dir ]) ) { + if ( elem.nodeType === 1 || checkNonElements ) { + if ( matcher( elem, context, xml ) ) { + return true; + } + } + } + } else { + while ( (elem = elem[ dir ]) ) { + if ( elem.nodeType === 1 || checkNonElements ) { + outerCache = elem[ expando ] || (elem[ expando ] = {}); + if ( (oldCache = outerCache[ dir ]) && + oldCache[ 0 ] === dirruns && oldCache[ 1 ] === doneName ) { + + // Assign to newCache so results back-propagate to previous elements + return (newCache[ 2 ] = oldCache[ 2 ]); + } else { + // Reuse newcache so results back-propagate to previous elements + outerCache[ dir ] = newCache; + + // A match means we're done; a fail means we have to keep checking + if ( (newCache[ 2 ] = matcher( elem, context, xml )) ) { + return true; + } + } + } + } + } + }; +} + +function elementMatcher( matchers ) { + return matchers.length > 1 ? + function( elem, context, xml ) { + var i = matchers.length; + while ( i-- ) { + if ( !matchers[i]( elem, context, xml ) ) { + return false; + } + } + return true; + } : + matchers[0]; +} + +function multipleContexts( selector, contexts, results ) { + var i = 0, + len = contexts.length; + for ( ; i < len; i++ ) { + Sizzle( selector, contexts[i], results ); + } + return results; +} + +function condense( unmatched, map, filter, context, xml ) { + var elem, + newUnmatched = [], + i = 0, + len = unmatched.length, + mapped = map != null; + + for ( ; i < len; i++ ) { + if ( (elem = unmatched[i]) ) { + if ( !filter || filter( elem, context, xml ) ) { + newUnmatched.push( elem ); + if ( mapped ) { + map.push( i ); + } + } + } + } + + return newUnmatched; +} + +function setMatcher( preFilter, selector, matcher, postFilter, postFinder, postSelector ) { + if ( postFilter && !postFilter[ expando ] ) { + postFilter = setMatcher( postFilter ); + } + if ( postFinder && !postFinder[ expando ] ) { + postFinder = setMatcher( postFinder, postSelector ); + } + return markFunction(function( seed, results, context, xml ) { + var temp, i, elem, + preMap = [], + postMap = [], + preexisting = results.length, + + // Get initial elements from seed or context + elems = seed || multipleContexts( selector || "*", context.nodeType ? [ context ] : context, [] ), + + // Prefilter to get matcher input, preserving a map for seed-results synchronization + matcherIn = preFilter && ( seed || !selector ) ? + condense( elems, preMap, preFilter, context, xml ) : + elems, + + matcherOut = matcher ? + // If we have a postFinder, or filtered seed, or non-seed postFilter or preexisting results, + postFinder || ( seed ? preFilter : preexisting || postFilter ) ? + + // ...intermediate processing is necessary + [] : + + // ...otherwise use results directly + results : + matcherIn; + + // Find primary matches + if ( matcher ) { + matcher( matcherIn, matcherOut, context, xml ); + } + + // Apply postFilter + if ( postFilter ) { + temp = condense( matcherOut, postMap ); + postFilter( temp, [], context, xml ); + + // Un-match failing elements by moving them back to matcherIn + i = temp.length; + while ( i-- ) { + if ( (elem = temp[i]) ) { + matcherOut[ postMap[i] ] = !(matcherIn[ postMap[i] ] = elem); + } + } + } + + if ( seed ) { + if ( postFinder || preFilter ) { + if ( postFinder ) { + // Get the final matcherOut by condensing this intermediate into postFinder contexts + temp = []; + i = matcherOut.length; + while ( i-- ) { + if ( (elem = matcherOut[i]) ) { + // Restore matcherIn since elem is not yet a final match + temp.push( (matcherIn[i] = elem) ); + } + } + postFinder( null, (matcherOut = []), temp, xml ); + } + + // Move matched elements from seed to results to keep them synchronized + i = matcherOut.length; + while ( i-- ) { + if ( (elem = matcherOut[i]) && + (temp = postFinder ? indexOf.call( seed, elem ) : preMap[i]) > -1 ) { + + seed[temp] = !(results[temp] = elem); + } + } + } + + // Add elements to results, through postFinder if defined + } else { + matcherOut = condense( + matcherOut === results ? + matcherOut.splice( preexisting, matcherOut.length ) : + matcherOut + ); + if ( postFinder ) { + postFinder( null, results, matcherOut, xml ); + } else { + push.apply( results, matcherOut ); + } + } + }); +} + +function matcherFromTokens( tokens ) { + var checkContext, matcher, j, + len = tokens.length, + leadingRelative = Expr.relative[ tokens[0].type ], + implicitRelative = leadingRelative || Expr.relative[" "], + i = leadingRelative ? 1 : 0, + + // The foundational matcher ensures that elements are reachable from top-level context(s) + matchContext = addCombinator( function( elem ) { + return elem === checkContext; + }, implicitRelative, true ), + matchAnyContext = addCombinator( function( elem ) { + return indexOf.call( checkContext, elem ) > -1; + }, implicitRelative, true ), + matchers = [ function( elem, context, xml ) { + return ( !leadingRelative && ( xml || context !== outermostContext ) ) || ( + (checkContext = context).nodeType ? + matchContext( elem, context, xml ) : + matchAnyContext( elem, context, xml ) ); + } ]; + + for ( ; i < len; i++ ) { + if ( (matcher = Expr.relative[ tokens[i].type ]) ) { + matchers = [ addCombinator(elementMatcher( matchers ), matcher) ]; + } else { + matcher = Expr.filter[ tokens[i].type ].apply( null, tokens[i].matches ); + + // Return special upon seeing a positional matcher + if ( matcher[ expando ] ) { + // Find the next relative operator (if any) for proper handling + j = ++i; + for ( ; j < len; j++ ) { + if ( Expr.relative[ tokens[j].type ] ) { + break; + } + } + return setMatcher( + i > 1 && elementMatcher( matchers ), + i > 1 && toSelector( + // If the preceding token was a descendant combinator, insert an implicit any-element `*` + tokens.slice( 0, i - 1 ).concat({ value: tokens[ i - 2 ].type === " " ? "*" : "" }) + ).replace( rtrim, "$1" ), + matcher, + i < j && matcherFromTokens( tokens.slice( i, j ) ), + j < len && matcherFromTokens( (tokens = tokens.slice( j )) ), + j < len && toSelector( tokens ) + ); + } + matchers.push( matcher ); + } + } + + return elementMatcher( matchers ); +} + +function matcherFromGroupMatchers( elementMatchers, setMatchers ) { + var bySet = setMatchers.length > 0, + byElement = elementMatchers.length > 0, + superMatcher = function( seed, context, xml, results, outermost ) { + var elem, j, matcher, + matchedCount = 0, + i = "0", + unmatched = seed && [], + setMatched = [], + contextBackup = outermostContext, + // We must always have either seed elements or outermost context + elems = seed || byElement && Expr.find["TAG"]( "*", outermost ), + // Use integer dirruns iff this is the outermost matcher + dirrunsUnique = (dirruns += contextBackup == null ? 1 : Math.random() || 0.1), + len = elems.length; + + if ( outermost ) { + outermostContext = context !== document && context; + } + + // Add elements passing elementMatchers directly to results + // Keep `i` a string if there are no elements so `matchedCount` will be "00" below + // Support: IE<9, Safari + // Tolerate NodeList properties (IE: "length"; Safari: ) matching elements by id + for ( ; i !== len && (elem = elems[i]) != null; i++ ) { + if ( byElement && elem ) { + j = 0; + while ( (matcher = elementMatchers[j++]) ) { + if ( matcher( elem, context, xml ) ) { + results.push( elem ); + break; + } + } + if ( outermost ) { + dirruns = dirrunsUnique; + } + } + + // Track unmatched elements for set filters + if ( bySet ) { + // They will have gone through all possible matchers + if ( (elem = !matcher && elem) ) { + matchedCount--; + } + + // Lengthen the array for every element, matched or not + if ( seed ) { + unmatched.push( elem ); + } + } + } + + // Apply set filters to unmatched elements + matchedCount += i; + if ( bySet && i !== matchedCount ) { + j = 0; + while ( (matcher = setMatchers[j++]) ) { + matcher( unmatched, setMatched, context, xml ); + } + + if ( seed ) { + // Reintegrate element matches to eliminate the need for sorting + if ( matchedCount > 0 ) { + while ( i-- ) { + if ( !(unmatched[i] || setMatched[i]) ) { + setMatched[i] = pop.call( results ); + } + } + } + + // Discard index placeholder values to get only actual matches + setMatched = condense( setMatched ); + } + + // Add matches to results + push.apply( results, setMatched ); + + // Seedless set matches succeeding multiple successful matchers stipulate sorting + if ( outermost && !seed && setMatched.length > 0 && + ( matchedCount + setMatchers.length ) > 1 ) { + + Sizzle.uniqueSort( results ); + } + } + + // Override manipulation of globals by nested matchers + if ( outermost ) { + dirruns = dirrunsUnique; + outermostContext = contextBackup; + } + + return unmatched; + }; + + return bySet ? + markFunction( superMatcher ) : + superMatcher; +} + +compile = Sizzle.compile = function( selector, match /* Internal Use Only */ ) { + var i, + setMatchers = [], + elementMatchers = [], + cached = compilerCache[ selector + " " ]; + + if ( !cached ) { + // Generate a function of recursive functions that can be used to check each element + if ( !match ) { + match = tokenize( selector ); + } + i = match.length; + while ( i-- ) { + cached = matcherFromTokens( match[i] ); + if ( cached[ expando ] ) { + setMatchers.push( cached ); + } else { + elementMatchers.push( cached ); + } + } + + // Cache the compiled function + cached = compilerCache( selector, matcherFromGroupMatchers( elementMatchers, setMatchers ) ); + + // Save selector and tokenization + cached.selector = selector; + } + return cached; +}; + +/** + * A low-level selection function that works with Sizzle's compiled + * selector functions + * @param {String|Function} selector A selector or a pre-compiled + * selector function built with Sizzle.compile + * @param {Element} context + * @param {Array} [results] + * @param {Array} [seed] A set of elements to match against + */ +select = Sizzle.select = function( selector, context, results, seed ) { + var i, tokens, token, type, find, + compiled = typeof selector === "function" && selector, + match = !seed && tokenize( (selector = compiled.selector || selector) ); + + results = results || []; + + // Try to minimize operations if there is no seed and only one group + if ( match.length === 1 ) { + + // Take a shortcut and set the context if the root selector is an ID + tokens = match[0] = match[0].slice( 0 ); + if ( tokens.length > 2 && (token = tokens[0]).type === "ID" && + support.getById && context.nodeType === 9 && documentIsHTML && + Expr.relative[ tokens[1].type ] ) { + + context = ( Expr.find["ID"]( token.matches[0].replace(runescape, funescape), context ) || [] )[0]; + if ( !context ) { + return results; + + // Precompiled matchers will still verify ancestry, so step up a level + } else if ( compiled ) { + context = context.parentNode; + } + + selector = selector.slice( tokens.shift().value.length ); + } + + // Fetch a seed set for right-to-left matching + i = matchExpr["needsContext"].test( selector ) ? 0 : tokens.length; + while ( i-- ) { + token = tokens[i]; + + // Abort if we hit a combinator + if ( Expr.relative[ (type = token.type) ] ) { + break; + } + if ( (find = Expr.find[ type ]) ) { + // Search, expanding context for leading sibling combinators + if ( (seed = find( + token.matches[0].replace( runescape, funescape ), + rsibling.test( tokens[0].type ) && testContext( context.parentNode ) || context + )) ) { + + // If seed is empty or no tokens remain, we can return early + tokens.splice( i, 1 ); + selector = seed.length && toSelector( tokens ); + if ( !selector ) { + push.apply( results, seed ); + return results; + } + + break; + } + } + } + } + + // Compile and execute a filtering function if one is not provided + // Provide `match` to avoid retokenization if we modified the selector above + ( compiled || compile( selector, match ) )( + seed, + context, + !documentIsHTML, + results, + rsibling.test( selector ) && testContext( context.parentNode ) || context + ); + return results; +}; + +// One-time assignments + +// Sort stability +support.sortStable = expando.split("").sort( sortOrder ).join("") === expando; + +// Support: Chrome<14 +// Always assume duplicates if they aren't passed to the comparison function +support.detectDuplicates = !!hasDuplicate; + +// Initialize against the default document +setDocument(); + +// Support: Webkit<537.32 - Safari 6.0.3/Chrome 25 (fixed in Chrome 27) +// Detached nodes confoundingly follow *each other* +support.sortDetached = assert(function( div1 ) { + // Should return 1, but returns 4 (following) + return div1.compareDocumentPosition( document.createElement("div") ) & 1; +}); + +// Support: IE<8 +// Prevent attribute/property "interpolation" +// http://msdn.microsoft.com/en-us/library/ms536429%28VS.85%29.aspx +if ( !assert(function( div ) { + div.innerHTML = ""; + return div.firstChild.getAttribute("href") === "#" ; +}) ) { + addHandle( "type|href|height|width", function( elem, name, isXML ) { + if ( !isXML ) { + return elem.getAttribute( name, name.toLowerCase() === "type" ? 1 : 2 ); + } + }); +} + +// Support: IE<9 +// Use defaultValue in place of getAttribute("value") +if ( !support.attributes || !assert(function( div ) { + div.innerHTML = ""; + div.firstChild.setAttribute( "value", "" ); + return div.firstChild.getAttribute( "value" ) === ""; +}) ) { + addHandle( "value", function( elem, name, isXML ) { + if ( !isXML && elem.nodeName.toLowerCase() === "input" ) { + return elem.defaultValue; + } + }); +} + +// Support: IE<9 +// Use getAttributeNode to fetch booleans when getAttribute lies +if ( !assert(function( div ) { + return div.getAttribute("disabled") == null; +}) ) { + addHandle( booleans, function( elem, name, isXML ) { + var val; + if ( !isXML ) { + return elem[ name ] === true ? name.toLowerCase() : + (val = elem.getAttributeNode( name )) && val.specified ? + val.value : + null; + } + }); +} + +return Sizzle; + +})( window ); + + + +jQuery.find = Sizzle; +jQuery.expr = Sizzle.selectors; +jQuery.expr[":"] = jQuery.expr.pseudos; +jQuery.unique = Sizzle.uniqueSort; +jQuery.text = Sizzle.getText; +jQuery.isXMLDoc = Sizzle.isXML; +jQuery.contains = Sizzle.contains; + + + +var rneedsContext = jQuery.expr.match.needsContext; + +var rsingleTag = (/^<(\w+)\s*\/?>(?:<\/\1>|)$/); + + + +var risSimple = /^.[^:#\[\.,]*$/; + +// Implement the identical functionality for filter and not +function winnow( elements, qualifier, not ) { + if ( jQuery.isFunction( qualifier ) ) { + return jQuery.grep( elements, function( elem, i ) { + /* jshint -W018 */ + return !!qualifier.call( elem, i, elem ) !== not; + }); + + } + + if ( qualifier.nodeType ) { + return jQuery.grep( elements, function( elem ) { + return ( elem === qualifier ) !== not; + }); + + } + + if ( typeof qualifier === "string" ) { + if ( risSimple.test( qualifier ) ) { + return jQuery.filter( qualifier, elements, not ); + } + + qualifier = jQuery.filter( qualifier, elements ); + } + + return jQuery.grep( elements, function( elem ) { + return ( indexOf.call( qualifier, elem ) >= 0 ) !== not; + }); +} + +jQuery.filter = function( expr, elems, not ) { + var elem = elems[ 0 ]; + + if ( not ) { + expr = ":not(" + expr + ")"; + } + + return elems.length === 1 && elem.nodeType === 1 ? + jQuery.find.matchesSelector( elem, expr ) ? [ elem ] : [] : + jQuery.find.matches( expr, jQuery.grep( elems, function( elem ) { + return elem.nodeType === 1; + })); +}; + +jQuery.fn.extend({ + find: function( selector ) { + var i, + len = this.length, + ret = [], + self = this; + + if ( typeof selector !== "string" ) { + return this.pushStack( jQuery( selector ).filter(function() { + for ( i = 0; i < len; i++ ) { + if ( jQuery.contains( self[ i ], this ) ) { + return true; + } + } + }) ); + } + + for ( i = 0; i < len; i++ ) { + jQuery.find( selector, self[ i ], ret ); + } + + // Needed because $( selector, context ) becomes $( context ).find( selector ) + ret = this.pushStack( len > 1 ? jQuery.unique( ret ) : ret ); + ret.selector = this.selector ? this.selector + " " + selector : selector; + return ret; + }, + filter: function( selector ) { + return this.pushStack( winnow(this, selector || [], false) ); + }, + not: function( selector ) { + return this.pushStack( winnow(this, selector || [], true) ); + }, + is: function( selector ) { + return !!winnow( + this, + + // If this is a positional/relative selector, check membership in the returned set + // so $("p:first").is("p:last") won't return true for a doc with two "p". + typeof selector === "string" && rneedsContext.test( selector ) ? + jQuery( selector ) : + selector || [], + false + ).length; + } +}); + + +// Initialize a jQuery object + + +// A central reference to the root jQuery(document) +var rootjQuery, + + // A simple way to check for HTML strings + // Prioritize #id over to avoid XSS via location.hash (#9521) + // Strict HTML recognition (#11290: must start with <) + rquickExpr = /^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]*))$/, + + init = jQuery.fn.init = function( selector, context ) { + var match, elem; + + // HANDLE: $(""), $(null), $(undefined), $(false) + if ( !selector ) { + return this; + } + + // Handle HTML strings + if ( typeof selector === "string" ) { + if ( selector[0] === "<" && selector[ selector.length - 1 ] === ">" && selector.length >= 3 ) { + // Assume that strings that start and end with <> are HTML and skip the regex check + match = [ null, selector, null ]; + + } else { + match = rquickExpr.exec( selector ); + } + + // Match html or make sure no context is specified for #id + if ( match && (match[1] || !context) ) { + + // HANDLE: $(html) -> $(array) + if ( match[1] ) { + context = context instanceof jQuery ? context[0] : context; + + // scripts is true for back-compat + // Intentionally let the error be thrown if parseHTML is not present + jQuery.merge( this, jQuery.parseHTML( + match[1], + context && context.nodeType ? context.ownerDocument || context : document, + true + ) ); + + // HANDLE: $(html, props) + if ( rsingleTag.test( match[1] ) && jQuery.isPlainObject( context ) ) { + for ( match in context ) { + // Properties of context are called as methods if possible + if ( jQuery.isFunction( this[ match ] ) ) { + this[ match ]( context[ match ] ); + + // ...and otherwise set as attributes + } else { + this.attr( match, context[ match ] ); + } + } + } + + return this; + + // HANDLE: $(#id) + } else { + elem = document.getElementById( match[2] ); + + // Check parentNode to catch when Blackberry 4.6 returns + // nodes that are no longer in the document #6963 + if ( elem && elem.parentNode ) { + // Inject the element directly into the jQuery object + this.length = 1; + this[0] = elem; + } + + this.context = document; + this.selector = selector; + return this; + } + + // HANDLE: $(expr, $(...)) + } else if ( !context || context.jquery ) { + return ( context || rootjQuery ).find( selector ); + + // HANDLE: $(expr, context) + // (which is just equivalent to: $(context).find(expr) + } else { + return this.constructor( context ).find( selector ); + } + + // HANDLE: $(DOMElement) + } else if ( selector.nodeType ) { + this.context = this[0] = selector; + this.length = 1; + return this; + + // HANDLE: $(function) + // Shortcut for document ready + } else if ( jQuery.isFunction( selector ) ) { + return typeof rootjQuery.ready !== "undefined" ? + rootjQuery.ready( selector ) : + // Execute immediately if ready is not present + selector( jQuery ); + } + + if ( selector.selector !== undefined ) { + this.selector = selector.selector; + this.context = selector.context; + } + + return jQuery.makeArray( selector, this ); + }; + +// Give the init function the jQuery prototype for later instantiation +init.prototype = jQuery.fn; + +// Initialize central reference +rootjQuery = jQuery( document ); + + +var rparentsprev = /^(?:parents|prev(?:Until|All))/, + // methods guaranteed to produce a unique set when starting from a unique set + guaranteedUnique = { + children: true, + contents: true, + next: true, + prev: true + }; + +jQuery.extend({ + dir: function( elem, dir, until ) { + var matched = [], + truncate = until !== undefined; + + while ( (elem = elem[ dir ]) && elem.nodeType !== 9 ) { + if ( elem.nodeType === 1 ) { + if ( truncate && jQuery( elem ).is( until ) ) { + break; + } + matched.push( elem ); + } + } + return matched; + }, + + sibling: function( n, elem ) { + var matched = []; + + for ( ; n; n = n.nextSibling ) { + if ( n.nodeType === 1 && n !== elem ) { + matched.push( n ); + } + } + + return matched; + } +}); + +jQuery.fn.extend({ + has: function( target ) { + var targets = jQuery( target, this ), + l = targets.length; + + return this.filter(function() { + var i = 0; + for ( ; i < l; i++ ) { + if ( jQuery.contains( this, targets[i] ) ) { + return true; + } + } + }); + }, + + closest: function( selectors, context ) { + var cur, + i = 0, + l = this.length, + matched = [], + pos = rneedsContext.test( selectors ) || typeof selectors !== "string" ? + jQuery( selectors, context || this.context ) : + 0; + + for ( ; i < l; i++ ) { + for ( cur = this[i]; cur && cur !== context; cur = cur.parentNode ) { + // Always skip document fragments + if ( cur.nodeType < 11 && (pos ? + pos.index(cur) > -1 : + + // Don't pass non-elements to Sizzle + cur.nodeType === 1 && + jQuery.find.matchesSelector(cur, selectors)) ) { + + matched.push( cur ); + break; + } + } + } + + return this.pushStack( matched.length > 1 ? jQuery.unique( matched ) : matched ); + }, + + // Determine the position of an element within + // the matched set of elements + index: function( elem ) { + + // No argument, return index in parent + if ( !elem ) { + return ( this[ 0 ] && this[ 0 ].parentNode ) ? this.first().prevAll().length : -1; + } + + // index in selector + if ( typeof elem === "string" ) { + return indexOf.call( jQuery( elem ), this[ 0 ] ); + } + + // Locate the position of the desired element + return indexOf.call( this, + + // If it receives a jQuery object, the first element is used + elem.jquery ? elem[ 0 ] : elem + ); + }, + + add: function( selector, context ) { + return this.pushStack( + jQuery.unique( + jQuery.merge( this.get(), jQuery( selector, context ) ) + ) + ); + }, + + addBack: function( selector ) { + return this.add( selector == null ? + this.prevObject : this.prevObject.filter(selector) + ); + } +}); + +function sibling( cur, dir ) { + while ( (cur = cur[dir]) && cur.nodeType !== 1 ) {} + return cur; +} + +jQuery.each({ + parent: function( elem ) { + var parent = elem.parentNode; + return parent && parent.nodeType !== 11 ? parent : null; + }, + parents: function( elem ) { + return jQuery.dir( elem, "parentNode" ); + }, + parentsUntil: function( elem, i, until ) { + return jQuery.dir( elem, "parentNode", until ); + }, + next: function( elem ) { + return sibling( elem, "nextSibling" ); + }, + prev: function( elem ) { + return sibling( elem, "previousSibling" ); + }, + nextAll: function( elem ) { + return jQuery.dir( elem, "nextSibling" ); + }, + prevAll: function( elem ) { + return jQuery.dir( elem, "previousSibling" ); + }, + nextUntil: function( elem, i, until ) { + return jQuery.dir( elem, "nextSibling", until ); + }, + prevUntil: function( elem, i, until ) { + return jQuery.dir( elem, "previousSibling", until ); + }, + siblings: function( elem ) { + return jQuery.sibling( ( elem.parentNode || {} ).firstChild, elem ); + }, + children: function( elem ) { + return jQuery.sibling( elem.firstChild ); + }, + contents: function( elem ) { + return elem.contentDocument || jQuery.merge( [], elem.childNodes ); + } +}, function( name, fn ) { + jQuery.fn[ name ] = function( until, selector ) { + var matched = jQuery.map( this, fn, until ); + + if ( name.slice( -5 ) !== "Until" ) { + selector = until; + } + + if ( selector && typeof selector === "string" ) { + matched = jQuery.filter( selector, matched ); + } + + if ( this.length > 1 ) { + // Remove duplicates + if ( !guaranteedUnique[ name ] ) { + jQuery.unique( matched ); + } + + // Reverse order for parents* and prev-derivatives + if ( rparentsprev.test( name ) ) { + matched.reverse(); + } + } + + return this.pushStack( matched ); + }; +}); +var rnotwhite = (/\S+/g); + + + +// String to Object options format cache +var optionsCache = {}; + +// Convert String-formatted options into Object-formatted ones and store in cache +function createOptions( options ) { + var object = optionsCache[ options ] = {}; + jQuery.each( options.match( rnotwhite ) || [], function( _, flag ) { + object[ flag ] = true; + }); + return object; +} + +/* + * Create a callback list using the following parameters: + * + * options: an optional list of space-separated options that will change how + * the callback list behaves or a more traditional option object + * + * By default a callback list will act like an event callback list and can be + * "fired" multiple times. + * + * Possible options: + * + * once: will ensure the callback list can only be fired once (like a Deferred) + * + * memory: will keep track of previous values and will call any callback added + * after the list has been fired right away with the latest "memorized" + * values (like a Deferred) + * + * unique: will ensure a callback can only be added once (no duplicate in the list) + * + * stopOnFalse: interrupt callings when a callback returns false + * + */ +jQuery.Callbacks = function( options ) { + + // Convert options from String-formatted to Object-formatted if needed + // (we check in cache first) + options = typeof options === "string" ? + ( optionsCache[ options ] || createOptions( options ) ) : + jQuery.extend( {}, options ); + + var // Last fire value (for non-forgettable lists) + memory, + // Flag to know if list was already fired + fired, + // Flag to know if list is currently firing + firing, + // First callback to fire (used internally by add and fireWith) + firingStart, + // End of the loop when firing + firingLength, + // Index of currently firing callback (modified by remove if needed) + firingIndex, + // Actual callback list + list = [], + // Stack of fire calls for repeatable lists + stack = !options.once && [], + // Fire callbacks + fire = function( data ) { + memory = options.memory && data; + fired = true; + firingIndex = firingStart || 0; + firingStart = 0; + firingLength = list.length; + firing = true; + for ( ; list && firingIndex < firingLength; firingIndex++ ) { + if ( list[ firingIndex ].apply( data[ 0 ], data[ 1 ] ) === false && options.stopOnFalse ) { + memory = false; // To prevent further calls using add + break; + } + } + firing = false; + if ( list ) { + if ( stack ) { + if ( stack.length ) { + fire( stack.shift() ); + } + } else if ( memory ) { + list = []; + } else { + self.disable(); + } + } + }, + // Actual Callbacks object + self = { + // Add a callback or a collection of callbacks to the list + add: function() { + if ( list ) { + // First, we save the current length + var start = list.length; + (function add( args ) { + jQuery.each( args, function( _, arg ) { + var type = jQuery.type( arg ); + if ( type === "function" ) { + if ( !options.unique || !self.has( arg ) ) { + list.push( arg ); + } + } else if ( arg && arg.length && type !== "string" ) { + // Inspect recursively + add( arg ); + } + }); + })( arguments ); + // Do we need to add the callbacks to the + // current firing batch? + if ( firing ) { + firingLength = list.length; + // With memory, if we're not firing then + // we should call right away + } else if ( memory ) { + firingStart = start; + fire( memory ); + } + } + return this; + }, + // Remove a callback from the list + remove: function() { + if ( list ) { + jQuery.each( arguments, function( _, arg ) { + var index; + while ( ( index = jQuery.inArray( arg, list, index ) ) > -1 ) { + list.splice( index, 1 ); + // Handle firing indexes + if ( firing ) { + if ( index <= firingLength ) { + firingLength--; + } + if ( index <= firingIndex ) { + firingIndex--; + } + } + } + }); + } + return this; + }, + // Check if a given callback is in the list. + // If no argument is given, return whether or not list has callbacks attached. + has: function( fn ) { + return fn ? jQuery.inArray( fn, list ) > -1 : !!( list && list.length ); + }, + // Remove all callbacks from the list + empty: function() { + list = []; + firingLength = 0; + return this; + }, + // Have the list do nothing anymore + disable: function() { + list = stack = memory = undefined; + return this; + }, + // Is it disabled? + disabled: function() { + return !list; + }, + // Lock the list in its current state + lock: function() { + stack = undefined; + if ( !memory ) { + self.disable(); + } + return this; + }, + // Is it locked? + locked: function() { + return !stack; + }, + // Call all callbacks with the given context and arguments + fireWith: function( context, args ) { + if ( list && ( !fired || stack ) ) { + args = args || []; + args = [ context, args.slice ? args.slice() : args ]; + if ( firing ) { + stack.push( args ); + } else { + fire( args ); + } + } + return this; + }, + // Call all the callbacks with the given arguments + fire: function() { + self.fireWith( this, arguments ); + return this; + }, + // To know if the callbacks have already been called at least once + fired: function() { + return !!fired; + } + }; + + return self; +}; + + +jQuery.extend({ + + Deferred: function( func ) { + var tuples = [ + // action, add listener, listener list, final state + [ "resolve", "done", jQuery.Callbacks("once memory"), "resolved" ], + [ "reject", "fail", jQuery.Callbacks("once memory"), "rejected" ], + [ "notify", "progress", jQuery.Callbacks("memory") ] + ], + state = "pending", + promise = { + state: function() { + return state; + }, + always: function() { + deferred.done( arguments ).fail( arguments ); + return this; + }, + then: function( /* fnDone, fnFail, fnProgress */ ) { + var fns = arguments; + return jQuery.Deferred(function( newDefer ) { + jQuery.each( tuples, function( i, tuple ) { + var fn = jQuery.isFunction( fns[ i ] ) && fns[ i ]; + // deferred[ done | fail | progress ] for forwarding actions to newDefer + deferred[ tuple[1] ](function() { + var returned = fn && fn.apply( this, arguments ); + if ( returned && jQuery.isFunction( returned.promise ) ) { + returned.promise() + .done( newDefer.resolve ) + .fail( newDefer.reject ) + .progress( newDefer.notify ); + } else { + newDefer[ tuple[ 0 ] + "With" ]( this === promise ? newDefer.promise() : this, fn ? [ returned ] : arguments ); + } + }); + }); + fns = null; + }).promise(); + }, + // Get a promise for this deferred + // If obj is provided, the promise aspect is added to the object + promise: function( obj ) { + return obj != null ? jQuery.extend( obj, promise ) : promise; + } + }, + deferred = {}; + + // Keep pipe for back-compat + promise.pipe = promise.then; + + // Add list-specific methods + jQuery.each( tuples, function( i, tuple ) { + var list = tuple[ 2 ], + stateString = tuple[ 3 ]; + + // promise[ done | fail | progress ] = list.add + promise[ tuple[1] ] = list.add; + + // Handle state + if ( stateString ) { + list.add(function() { + // state = [ resolved | rejected ] + state = stateString; + + // [ reject_list | resolve_list ].disable; progress_list.lock + }, tuples[ i ^ 1 ][ 2 ].disable, tuples[ 2 ][ 2 ].lock ); + } + + // deferred[ resolve | reject | notify ] + deferred[ tuple[0] ] = function() { + deferred[ tuple[0] + "With" ]( this === deferred ? promise : this, arguments ); + return this; + }; + deferred[ tuple[0] + "With" ] = list.fireWith; + }); + + // Make the deferred a promise + promise.promise( deferred ); + + // Call given func if any + if ( func ) { + func.call( deferred, deferred ); + } + + // All done! + return deferred; + }, + + // Deferred helper + when: function( subordinate /* , ..., subordinateN */ ) { + var i = 0, + resolveValues = slice.call( arguments ), + length = resolveValues.length, + + // the count of uncompleted subordinates + remaining = length !== 1 || ( subordinate && jQuery.isFunction( subordinate.promise ) ) ? length : 0, + + // the master Deferred. If resolveValues consist of only a single Deferred, just use that. + deferred = remaining === 1 ? subordinate : jQuery.Deferred(), + + // Update function for both resolve and progress values + updateFunc = function( i, contexts, values ) { + return function( value ) { + contexts[ i ] = this; + values[ i ] = arguments.length > 1 ? slice.call( arguments ) : value; + if ( values === progressValues ) { + deferred.notifyWith( contexts, values ); + } else if ( !( --remaining ) ) { + deferred.resolveWith( contexts, values ); + } + }; + }, + + progressValues, progressContexts, resolveContexts; + + // add listeners to Deferred subordinates; treat others as resolved + if ( length > 1 ) { + progressValues = new Array( length ); + progressContexts = new Array( length ); + resolveContexts = new Array( length ); + for ( ; i < length; i++ ) { + if ( resolveValues[ i ] && jQuery.isFunction( resolveValues[ i ].promise ) ) { + resolveValues[ i ].promise() + .done( updateFunc( i, resolveContexts, resolveValues ) ) + .fail( deferred.reject ) + .progress( updateFunc( i, progressContexts, progressValues ) ); + } else { + --remaining; + } + } + } + + // if we're not waiting on anything, resolve the master + if ( !remaining ) { + deferred.resolveWith( resolveContexts, resolveValues ); + } + + return deferred.promise(); + } +}); + + +// The deferred used on DOM ready +var readyList; + +jQuery.fn.ready = function( fn ) { + // Add the callback + jQuery.ready.promise().done( fn ); + + return this; +}; + +jQuery.extend({ + // Is the DOM ready to be used? Set to true once it occurs. + isReady: false, + + // A counter to track how many items to wait for before + // the ready event fires. See #6781 + readyWait: 1, + + // Hold (or release) the ready event + holdReady: function( hold ) { + if ( hold ) { + jQuery.readyWait++; + } else { + jQuery.ready( true ); + } + }, + + // Handle when the DOM is ready + ready: function( wait ) { + + // Abort if there are pending holds or we're already ready + if ( wait === true ? --jQuery.readyWait : jQuery.isReady ) { + return; + } + + // Remember that the DOM is ready + jQuery.isReady = true; + + // If a normal DOM Ready event fired, decrement, and wait if need be + if ( wait !== true && --jQuery.readyWait > 0 ) { + return; + } + + // If there are functions bound, to execute + readyList.resolveWith( document, [ jQuery ] ); + + // Trigger any bound ready events + if ( jQuery.fn.triggerHandler ) { + jQuery( document ).triggerHandler( "ready" ); + jQuery( document ).off( "ready" ); + } + } +}); + +/** + * The ready event handler and self cleanup method + */ +function completed() { + document.removeEventListener( "DOMContentLoaded", completed, false ); + window.removeEventListener( "load", completed, false ); + jQuery.ready(); +} + +jQuery.ready.promise = function( obj ) { + if ( !readyList ) { + + readyList = jQuery.Deferred(); + + // Catch cases where $(document).ready() is called after the browser event has already occurred. + // we once tried to use readyState "interactive" here, but it caused issues like the one + // discovered by ChrisS here: http://bugs.jquery.com/ticket/12282#comment:15 + if ( document.readyState === "complete" ) { + // Handle it asynchronously to allow scripts the opportunity to delay ready + setTimeout( jQuery.ready ); + + } else { + + // Use the handy event callback + document.addEventListener( "DOMContentLoaded", completed, false ); + + // A fallback to window.onload, that will always work + window.addEventListener( "load", completed, false ); + } + } + return readyList.promise( obj ); +}; + +// Kick off the DOM ready check even if the user does not +jQuery.ready.promise(); + + + + +// Multifunctional method to get and set values of a collection +// The value/s can optionally be executed if it's a function +var access = jQuery.access = function( elems, fn, key, value, chainable, emptyGet, raw ) { + var i = 0, + len = elems.length, + bulk = key == null; + + // Sets many values + if ( jQuery.type( key ) === "object" ) { + chainable = true; + for ( i in key ) { + jQuery.access( elems, fn, i, key[i], true, emptyGet, raw ); + } + + // Sets one value + } else if ( value !== undefined ) { + chainable = true; + + if ( !jQuery.isFunction( value ) ) { + raw = true; + } + + if ( bulk ) { + // Bulk operations run against the entire set + if ( raw ) { + fn.call( elems, value ); + fn = null; + + // ...except when executing function values + } else { + bulk = fn; + fn = function( elem, key, value ) { + return bulk.call( jQuery( elem ), value ); + }; + } + } + + if ( fn ) { + for ( ; i < len; i++ ) { + fn( elems[i], key, raw ? value : value.call( elems[i], i, fn( elems[i], key ) ) ); + } + } + } + + return chainable ? + elems : + + // Gets + bulk ? + fn.call( elems ) : + len ? fn( elems[0], key ) : emptyGet; +}; + + +/** + * Determines whether an object can have data + */ +jQuery.acceptData = function( owner ) { + // Accepts only: + // - Node + // - Node.ELEMENT_NODE + // - Node.DOCUMENT_NODE + // - Object + // - Any + /* jshint -W018 */ + return owner.nodeType === 1 || owner.nodeType === 9 || !( +owner.nodeType ); +}; + + +function Data() { + // Support: Android < 4, + // Old WebKit does not have Object.preventExtensions/freeze method, + // return new empty object instead with no [[set]] accessor + Object.defineProperty( this.cache = {}, 0, { + get: function() { + return {}; + } + }); + + this.expando = jQuery.expando + Math.random(); +} + +Data.uid = 1; +Data.accepts = jQuery.acceptData; + +Data.prototype = { + key: function( owner ) { + // We can accept data for non-element nodes in modern browsers, + // but we should not, see #8335. + // Always return the key for a frozen object. + if ( !Data.accepts( owner ) ) { + return 0; + } + + var descriptor = {}, + // Check if the owner object already has a cache key + unlock = owner[ this.expando ]; + + // If not, create one + if ( !unlock ) { + unlock = Data.uid++; + + // Secure it in a non-enumerable, non-writable property + try { + descriptor[ this.expando ] = { value: unlock }; + Object.defineProperties( owner, descriptor ); + + // Support: Android < 4 + // Fallback to a less secure definition + } catch ( e ) { + descriptor[ this.expando ] = unlock; + jQuery.extend( owner, descriptor ); + } + } + + // Ensure the cache object + if ( !this.cache[ unlock ] ) { + this.cache[ unlock ] = {}; + } + + return unlock; + }, + set: function( owner, data, value ) { + var prop, + // There may be an unlock assigned to this node, + // if there is no entry for this "owner", create one inline + // and set the unlock as though an owner entry had always existed + unlock = this.key( owner ), + cache = this.cache[ unlock ]; + + // Handle: [ owner, key, value ] args + if ( typeof data === "string" ) { + cache[ data ] = value; + + // Handle: [ owner, { properties } ] args + } else { + // Fresh assignments by object are shallow copied + if ( jQuery.isEmptyObject( cache ) ) { + jQuery.extend( this.cache[ unlock ], data ); + // Otherwise, copy the properties one-by-one to the cache object + } else { + for ( prop in data ) { + cache[ prop ] = data[ prop ]; + } + } + } + return cache; + }, + get: function( owner, key ) { + // Either a valid cache is found, or will be created. + // New caches will be created and the unlock returned, + // allowing direct access to the newly created + // empty data object. A valid owner object must be provided. + var cache = this.cache[ this.key( owner ) ]; + + return key === undefined ? + cache : cache[ key ]; + }, + access: function( owner, key, value ) { + var stored; + // In cases where either: + // + // 1. No key was specified + // 2. A string key was specified, but no value provided + // + // Take the "read" path and allow the get method to determine + // which value to return, respectively either: + // + // 1. The entire cache object + // 2. The data stored at the key + // + if ( key === undefined || + ((key && typeof key === "string") && value === undefined) ) { + + stored = this.get( owner, key ); + + return stored !== undefined ? + stored : this.get( owner, jQuery.camelCase(key) ); + } + + // [*]When the key is not a string, or both a key and value + // are specified, set or extend (existing objects) with either: + // + // 1. An object of properties + // 2. A key and value + // + this.set( owner, key, value ); + + // Since the "set" path can have two possible entry points + // return the expected data based on which path was taken[*] + return value !== undefined ? value : key; + }, + remove: function( owner, key ) { + var i, name, camel, + unlock = this.key( owner ), + cache = this.cache[ unlock ]; + + if ( key === undefined ) { + this.cache[ unlock ] = {}; + + } else { + // Support array or space separated string of keys + if ( jQuery.isArray( key ) ) { + // If "name" is an array of keys... + // When data is initially created, via ("key", "val") signature, + // keys will be converted to camelCase. + // Since there is no way to tell _how_ a key was added, remove + // both plain key and camelCase key. #12786 + // This will only penalize the array argument path. + name = key.concat( key.map( jQuery.camelCase ) ); + } else { + camel = jQuery.camelCase( key ); + // Try the string as a key before any manipulation + if ( key in cache ) { + name = [ key, camel ]; + } else { + // If a key with the spaces exists, use it. + // Otherwise, create an array by matching non-whitespace + name = camel; + name = name in cache ? + [ name ] : ( name.match( rnotwhite ) || [] ); + } + } + + i = name.length; + while ( i-- ) { + delete cache[ name[ i ] ]; + } + } + }, + hasData: function( owner ) { + return !jQuery.isEmptyObject( + this.cache[ owner[ this.expando ] ] || {} + ); + }, + discard: function( owner ) { + if ( owner[ this.expando ] ) { + delete this.cache[ owner[ this.expando ] ]; + } + } +}; +var data_priv = new Data(); + +var data_user = new Data(); + + + +/* + Implementation Summary + + 1. Enforce API surface and semantic compatibility with 1.9.x branch + 2. Improve the module's maintainability by reducing the storage + paths to a single mechanism. + 3. Use the same single mechanism to support "private" and "user" data. + 4. _Never_ expose "private" data to user code (TODO: Drop _data, _removeData) + 5. Avoid exposing implementation details on user objects (eg. expando properties) + 6. Provide a clear path for implementation upgrade to WeakMap in 2014 +*/ +var rbrace = /^(?:\{[\w\W]*\}|\[[\w\W]*\])$/, + rmultiDash = /([A-Z])/g; + +function dataAttr( elem, key, data ) { + var name; + + // If nothing was found internally, try to fetch any + // data from the HTML5 data-* attribute + if ( data === undefined && elem.nodeType === 1 ) { + name = "data-" + key.replace( rmultiDash, "-$1" ).toLowerCase(); + data = elem.getAttribute( name ); + + if ( typeof data === "string" ) { + try { + data = data === "true" ? true : + data === "false" ? false : + data === "null" ? null : + // Only convert to a number if it doesn't change the string + +data + "" === data ? +data : + rbrace.test( data ) ? jQuery.parseJSON( data ) : + data; + } catch( e ) {} + + // Make sure we set the data so it isn't changed later + data_user.set( elem, key, data ); + } else { + data = undefined; + } + } + return data; +} + +jQuery.extend({ + hasData: function( elem ) { + return data_user.hasData( elem ) || data_priv.hasData( elem ); + }, + + data: function( elem, name, data ) { + return data_user.access( elem, name, data ); + }, + + removeData: function( elem, name ) { + data_user.remove( elem, name ); + }, + + // TODO: Now that all calls to _data and _removeData have been replaced + // with direct calls to data_priv methods, these can be deprecated. + _data: function( elem, name, data ) { + return data_priv.access( elem, name, data ); + }, + + _removeData: function( elem, name ) { + data_priv.remove( elem, name ); + } +}); + +jQuery.fn.extend({ + data: function( key, value ) { + var i, name, data, + elem = this[ 0 ], + attrs = elem && elem.attributes; + + // Gets all values + if ( key === undefined ) { + if ( this.length ) { + data = data_user.get( elem ); + + if ( elem.nodeType === 1 && !data_priv.get( elem, "hasDataAttrs" ) ) { + i = attrs.length; + while ( i-- ) { + + // Support: IE11+ + // The attrs elements can be null (#14894) + if ( attrs[ i ] ) { + name = attrs[ i ].name; + if ( name.indexOf( "data-" ) === 0 ) { + name = jQuery.camelCase( name.slice(5) ); + dataAttr( elem, name, data[ name ] ); + } + } + } + data_priv.set( elem, "hasDataAttrs", true ); + } + } + + return data; + } + + // Sets multiple values + if ( typeof key === "object" ) { + return this.each(function() { + data_user.set( this, key ); + }); + } + + return access( this, function( value ) { + var data, + camelKey = jQuery.camelCase( key ); + + // The calling jQuery object (element matches) is not empty + // (and therefore has an element appears at this[ 0 ]) and the + // `value` parameter was not undefined. An empty jQuery object + // will result in `undefined` for elem = this[ 0 ] which will + // throw an exception if an attempt to read a data cache is made. + if ( elem && value === undefined ) { + // Attempt to get data from the cache + // with the key as-is + data = data_user.get( elem, key ); + if ( data !== undefined ) { + return data; + } + + // Attempt to get data from the cache + // with the key camelized + data = data_user.get( elem, camelKey ); + if ( data !== undefined ) { + return data; + } + + // Attempt to "discover" the data in + // HTML5 custom data-* attrs + data = dataAttr( elem, camelKey, undefined ); + if ( data !== undefined ) { + return data; + } + + // We tried really hard, but the data doesn't exist. + return; + } + + // Set the data... + this.each(function() { + // First, attempt to store a copy or reference of any + // data that might've been store with a camelCased key. + var data = data_user.get( this, camelKey ); + + // For HTML5 data-* attribute interop, we have to + // store property names with dashes in a camelCase form. + // This might not apply to all properties...* + data_user.set( this, camelKey, value ); + + // *... In the case of properties that might _actually_ + // have dashes, we need to also store a copy of that + // unchanged property. + if ( key.indexOf("-") !== -1 && data !== undefined ) { + data_user.set( this, key, value ); + } + }); + }, null, value, arguments.length > 1, null, true ); + }, + + removeData: function( key ) { + return this.each(function() { + data_user.remove( this, key ); + }); + } +}); + + +jQuery.extend({ + queue: function( elem, type, data ) { + var queue; + + if ( elem ) { + type = ( type || "fx" ) + "queue"; + queue = data_priv.get( elem, type ); + + // Speed up dequeue by getting out quickly if this is just a lookup + if ( data ) { + if ( !queue || jQuery.isArray( data ) ) { + queue = data_priv.access( elem, type, jQuery.makeArray(data) ); + } else { + queue.push( data ); + } + } + return queue || []; + } + }, + + dequeue: function( elem, type ) { + type = type || "fx"; + + var queue = jQuery.queue( elem, type ), + startLength = queue.length, + fn = queue.shift(), + hooks = jQuery._queueHooks( elem, type ), + next = function() { + jQuery.dequeue( elem, type ); + }; + + // If the fx queue is dequeued, always remove the progress sentinel + if ( fn === "inprogress" ) { + fn = queue.shift(); + startLength--; + } + + if ( fn ) { + + // Add a progress sentinel to prevent the fx queue from being + // automatically dequeued + if ( type === "fx" ) { + queue.unshift( "inprogress" ); + } + + // clear up the last queue stop function + delete hooks.stop; + fn.call( elem, next, hooks ); + } + + if ( !startLength && hooks ) { + hooks.empty.fire(); + } + }, + + // not intended for public consumption - generates a queueHooks object, or returns the current one + _queueHooks: function( elem, type ) { + var key = type + "queueHooks"; + return data_priv.get( elem, key ) || data_priv.access( elem, key, { + empty: jQuery.Callbacks("once memory").add(function() { + data_priv.remove( elem, [ type + "queue", key ] ); + }) + }); + } +}); + +jQuery.fn.extend({ + queue: function( type, data ) { + var setter = 2; + + if ( typeof type !== "string" ) { + data = type; + type = "fx"; + setter--; + } + + if ( arguments.length < setter ) { + return jQuery.queue( this[0], type ); + } + + return data === undefined ? + this : + this.each(function() { + var queue = jQuery.queue( this, type, data ); + + // ensure a hooks for this queue + jQuery._queueHooks( this, type ); + + if ( type === "fx" && queue[0] !== "inprogress" ) { + jQuery.dequeue( this, type ); + } + }); + }, + dequeue: function( type ) { + return this.each(function() { + jQuery.dequeue( this, type ); + }); + }, + clearQueue: function( type ) { + return this.queue( type || "fx", [] ); + }, + // Get a promise resolved when queues of a certain type + // are emptied (fx is the type by default) + promise: function( type, obj ) { + var tmp, + count = 1, + defer = jQuery.Deferred(), + elements = this, + i = this.length, + resolve = function() { + if ( !( --count ) ) { + defer.resolveWith( elements, [ elements ] ); + } + }; + + if ( typeof type !== "string" ) { + obj = type; + type = undefined; + } + type = type || "fx"; + + while ( i-- ) { + tmp = data_priv.get( elements[ i ], type + "queueHooks" ); + if ( tmp && tmp.empty ) { + count++; + tmp.empty.add( resolve ); + } + } + resolve(); + return defer.promise( obj ); + } +}); +var pnum = (/[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/).source; + +var cssExpand = [ "Top", "Right", "Bottom", "Left" ]; + +var isHidden = function( elem, el ) { + // isHidden might be called from jQuery#filter function; + // in that case, element will be second argument + elem = el || elem; + return jQuery.css( elem, "display" ) === "none" || !jQuery.contains( elem.ownerDocument, elem ); + }; + +var rcheckableType = (/^(?:checkbox|radio)$/i); + + + +(function() { + var fragment = document.createDocumentFragment(), + div = fragment.appendChild( document.createElement( "div" ) ), + input = document.createElement( "input" ); + + // #11217 - WebKit loses check when the name is after the checked attribute + // Support: Windows Web Apps (WWA) + // `name` and `type` need .setAttribute for WWA + input.setAttribute( "type", "radio" ); + input.setAttribute( "checked", "checked" ); + input.setAttribute( "name", "t" ); + + div.appendChild( input ); + + // Support: Safari 5.1, iOS 5.1, Android 4.x, Android 2.3 + // old WebKit doesn't clone checked state correctly in fragments + support.checkClone = div.cloneNode( true ).cloneNode( true ).lastChild.checked; + + // Make sure textarea (and checkbox) defaultValue is properly cloned + // Support: IE9-IE11+ + div.innerHTML = ""; + support.noCloneChecked = !!div.cloneNode( true ).lastChild.defaultValue; +})(); +var strundefined = typeof undefined; + + + +support.focusinBubbles = "onfocusin" in window; + + +var + rkeyEvent = /^key/, + rmouseEvent = /^(?:mouse|pointer|contextmenu)|click/, + rfocusMorph = /^(?:focusinfocus|focusoutblur)$/, + rtypenamespace = /^([^.]*)(?:\.(.+)|)$/; + +function returnTrue() { + return true; +} + +function returnFalse() { + return false; +} + +function safeActiveElement() { + try { + return document.activeElement; + } catch ( err ) { } +} + +/* + * Helper functions for managing events -- not part of the public interface. + * Props to Dean Edwards' addEvent library for many of the ideas. + */ +jQuery.event = { + + global: {}, + + add: function( elem, types, handler, data, selector ) { + + var handleObjIn, eventHandle, tmp, + events, t, handleObj, + special, handlers, type, namespaces, origType, + elemData = data_priv.get( elem ); + + // Don't attach events to noData or text/comment nodes (but allow plain objects) + if ( !elemData ) { + return; + } + + // Caller can pass in an object of custom data in lieu of the handler + if ( handler.handler ) { + handleObjIn = handler; + handler = handleObjIn.handler; + selector = handleObjIn.selector; + } + + // Make sure that the handler has a unique ID, used to find/remove it later + if ( !handler.guid ) { + handler.guid = jQuery.guid++; + } + + // Init the element's event structure and main handler, if this is the first + if ( !(events = elemData.events) ) { + events = elemData.events = {}; + } + if ( !(eventHandle = elemData.handle) ) { + eventHandle = elemData.handle = function( e ) { + // Discard the second event of a jQuery.event.trigger() and + // when an event is called after a page has unloaded + return typeof jQuery !== strundefined && jQuery.event.triggered !== e.type ? + jQuery.event.dispatch.apply( elem, arguments ) : undefined; + }; + } + + // Handle multiple events separated by a space + types = ( types || "" ).match( rnotwhite ) || [ "" ]; + t = types.length; + while ( t-- ) { + tmp = rtypenamespace.exec( types[t] ) || []; + type = origType = tmp[1]; + namespaces = ( tmp[2] || "" ).split( "." ).sort(); + + // There *must* be a type, no attaching namespace-only handlers + if ( !type ) { + continue; + } + + // If event changes its type, use the special event handlers for the changed type + special = jQuery.event.special[ type ] || {}; + + // If selector defined, determine special event api type, otherwise given type + type = ( selector ? special.delegateType : special.bindType ) || type; + + // Update special based on newly reset type + special = jQuery.event.special[ type ] || {}; + + // handleObj is passed to all event handlers + handleObj = jQuery.extend({ + type: type, + origType: origType, + data: data, + handler: handler, + guid: handler.guid, + selector: selector, + needsContext: selector && jQuery.expr.match.needsContext.test( selector ), + namespace: namespaces.join(".") + }, handleObjIn ); + + // Init the event handler queue if we're the first + if ( !(handlers = events[ type ]) ) { + handlers = events[ type ] = []; + handlers.delegateCount = 0; + + // Only use addEventListener if the special events handler returns false + if ( !special.setup || special.setup.call( elem, data, namespaces, eventHandle ) === false ) { + if ( elem.addEventListener ) { + elem.addEventListener( type, eventHandle, false ); + } + } + } + + if ( special.add ) { + special.add.call( elem, handleObj ); + + if ( !handleObj.handler.guid ) { + handleObj.handler.guid = handler.guid; + } + } + + // Add to the element's handler list, delegates in front + if ( selector ) { + handlers.splice( handlers.delegateCount++, 0, handleObj ); + } else { + handlers.push( handleObj ); + } + + // Keep track of which events have ever been used, for event optimization + jQuery.event.global[ type ] = true; + } + + }, + + // Detach an event or set of events from an element + remove: function( elem, types, handler, selector, mappedTypes ) { + + var j, origCount, tmp, + events, t, handleObj, + special, handlers, type, namespaces, origType, + elemData = data_priv.hasData( elem ) && data_priv.get( elem ); + + if ( !elemData || !(events = elemData.events) ) { + return; + } + + // Once for each type.namespace in types; type may be omitted + types = ( types || "" ).match( rnotwhite ) || [ "" ]; + t = types.length; + while ( t-- ) { + tmp = rtypenamespace.exec( types[t] ) || []; + type = origType = tmp[1]; + namespaces = ( tmp[2] || "" ).split( "." ).sort(); + + // Unbind all events (on this namespace, if provided) for the element + if ( !type ) { + for ( type in events ) { + jQuery.event.remove( elem, type + types[ t ], handler, selector, true ); + } + continue; + } + + special = jQuery.event.special[ type ] || {}; + type = ( selector ? special.delegateType : special.bindType ) || type; + handlers = events[ type ] || []; + tmp = tmp[2] && new RegExp( "(^|\\.)" + namespaces.join("\\.(?:.*\\.|)") + "(\\.|$)" ); + + // Remove matching events + origCount = j = handlers.length; + while ( j-- ) { + handleObj = handlers[ j ]; + + if ( ( mappedTypes || origType === handleObj.origType ) && + ( !handler || handler.guid === handleObj.guid ) && + ( !tmp || tmp.test( handleObj.namespace ) ) && + ( !selector || selector === handleObj.selector || selector === "**" && handleObj.selector ) ) { + handlers.splice( j, 1 ); + + if ( handleObj.selector ) { + handlers.delegateCount--; + } + if ( special.remove ) { + special.remove.call( elem, handleObj ); + } + } + } + + // Remove generic event handler if we removed something and no more handlers exist + // (avoids potential for endless recursion during removal of special event handlers) + if ( origCount && !handlers.length ) { + if ( !special.teardown || special.teardown.call( elem, namespaces, elemData.handle ) === false ) { + jQuery.removeEvent( elem, type, elemData.handle ); + } + + delete events[ type ]; + } + } + + // Remove the expando if it's no longer used + if ( jQuery.isEmptyObject( events ) ) { + delete elemData.handle; + data_priv.remove( elem, "events" ); + } + }, + + trigger: function( event, data, elem, onlyHandlers ) { + + var i, cur, tmp, bubbleType, ontype, handle, special, + eventPath = [ elem || document ], + type = hasOwn.call( event, "type" ) ? event.type : event, + namespaces = hasOwn.call( event, "namespace" ) ? event.namespace.split(".") : []; + + cur = tmp = elem = elem || document; + + // Don't do events on text and comment nodes + if ( elem.nodeType === 3 || elem.nodeType === 8 ) { + return; + } + + // focus/blur morphs to focusin/out; ensure we're not firing them right now + if ( rfocusMorph.test( type + jQuery.event.triggered ) ) { + return; + } + + if ( type.indexOf(".") >= 0 ) { + // Namespaced trigger; create a regexp to match event type in handle() + namespaces = type.split("."); + type = namespaces.shift(); + namespaces.sort(); + } + ontype = type.indexOf(":") < 0 && "on" + type; + + // Caller can pass in a jQuery.Event object, Object, or just an event type string + event = event[ jQuery.expando ] ? + event : + new jQuery.Event( type, typeof event === "object" && event ); + + // Trigger bitmask: & 1 for native handlers; & 2 for jQuery (always true) + event.isTrigger = onlyHandlers ? 2 : 3; + event.namespace = namespaces.join("."); + event.namespace_re = event.namespace ? + new RegExp( "(^|\\.)" + namespaces.join("\\.(?:.*\\.|)") + "(\\.|$)" ) : + null; + + // Clean up the event in case it is being reused + event.result = undefined; + if ( !event.target ) { + event.target = elem; + } + + // Clone any incoming data and prepend the event, creating the handler arg list + data = data == null ? + [ event ] : + jQuery.makeArray( data, [ event ] ); + + // Allow special events to draw outside the lines + special = jQuery.event.special[ type ] || {}; + if ( !onlyHandlers && special.trigger && special.trigger.apply( elem, data ) === false ) { + return; + } + + // Determine event propagation path in advance, per W3C events spec (#9951) + // Bubble up to document, then to window; watch for a global ownerDocument var (#9724) + if ( !onlyHandlers && !special.noBubble && !jQuery.isWindow( elem ) ) { + + bubbleType = special.delegateType || type; + if ( !rfocusMorph.test( bubbleType + type ) ) { + cur = cur.parentNode; + } + for ( ; cur; cur = cur.parentNode ) { + eventPath.push( cur ); + tmp = cur; + } + + // Only add window if we got to document (e.g., not plain obj or detached DOM) + if ( tmp === (elem.ownerDocument || document) ) { + eventPath.push( tmp.defaultView || tmp.parentWindow || window ); + } + } + + // Fire handlers on the event path + i = 0; + while ( (cur = eventPath[i++]) && !event.isPropagationStopped() ) { + + event.type = i > 1 ? + bubbleType : + special.bindType || type; + + // jQuery handler + handle = ( data_priv.get( cur, "events" ) || {} )[ event.type ] && data_priv.get( cur, "handle" ); + if ( handle ) { + handle.apply( cur, data ); + } + + // Native handler + handle = ontype && cur[ ontype ]; + if ( handle && handle.apply && jQuery.acceptData( cur ) ) { + event.result = handle.apply( cur, data ); + if ( event.result === false ) { + event.preventDefault(); + } + } + } + event.type = type; + + // If nobody prevented the default action, do it now + if ( !onlyHandlers && !event.isDefaultPrevented() ) { + + if ( (!special._default || special._default.apply( eventPath.pop(), data ) === false) && + jQuery.acceptData( elem ) ) { + + // Call a native DOM method on the target with the same name name as the event. + // Don't do default actions on window, that's where global variables be (#6170) + if ( ontype && jQuery.isFunction( elem[ type ] ) && !jQuery.isWindow( elem ) ) { + + // Don't re-trigger an onFOO event when we call its FOO() method + tmp = elem[ ontype ]; + + if ( tmp ) { + elem[ ontype ] = null; + } + + // Prevent re-triggering of the same event, since we already bubbled it above + jQuery.event.triggered = type; + elem[ type ](); + jQuery.event.triggered = undefined; + + if ( tmp ) { + elem[ ontype ] = tmp; + } + } + } + } + + return event.result; + }, + + dispatch: function( event ) { + + // Make a writable jQuery.Event from the native event object + event = jQuery.event.fix( event ); + + var i, j, ret, matched, handleObj, + handlerQueue = [], + args = slice.call( arguments ), + handlers = ( data_priv.get( this, "events" ) || {} )[ event.type ] || [], + special = jQuery.event.special[ event.type ] || {}; + + // Use the fix-ed jQuery.Event rather than the (read-only) native event + args[0] = event; + event.delegateTarget = this; + + // Call the preDispatch hook for the mapped type, and let it bail if desired + if ( special.preDispatch && special.preDispatch.call( this, event ) === false ) { + return; + } + + // Determine handlers + handlerQueue = jQuery.event.handlers.call( this, event, handlers ); + + // Run delegates first; they may want to stop propagation beneath us + i = 0; + while ( (matched = handlerQueue[ i++ ]) && !event.isPropagationStopped() ) { + event.currentTarget = matched.elem; + + j = 0; + while ( (handleObj = matched.handlers[ j++ ]) && !event.isImmediatePropagationStopped() ) { + + // Triggered event must either 1) have no namespace, or + // 2) have namespace(s) a subset or equal to those in the bound event (both can have no namespace). + if ( !event.namespace_re || event.namespace_re.test( handleObj.namespace ) ) { + + event.handleObj = handleObj; + event.data = handleObj.data; + + ret = ( (jQuery.event.special[ handleObj.origType ] || {}).handle || handleObj.handler ) + .apply( matched.elem, args ); + + if ( ret !== undefined ) { + if ( (event.result = ret) === false ) { + event.preventDefault(); + event.stopPropagation(); + } + } + } + } + } + + // Call the postDispatch hook for the mapped type + if ( special.postDispatch ) { + special.postDispatch.call( this, event ); + } + + return event.result; + }, + + handlers: function( event, handlers ) { + var i, matches, sel, handleObj, + handlerQueue = [], + delegateCount = handlers.delegateCount, + cur = event.target; + + // Find delegate handlers + // Black-hole SVG instance trees (#13180) + // Avoid non-left-click bubbling in Firefox (#3861) + if ( delegateCount && cur.nodeType && (!event.button || event.type !== "click") ) { + + for ( ; cur !== this; cur = cur.parentNode || this ) { + + // Don't process clicks on disabled elements (#6911, #8165, #11382, #11764) + if ( cur.disabled !== true || event.type !== "click" ) { + matches = []; + for ( i = 0; i < delegateCount; i++ ) { + handleObj = handlers[ i ]; + + // Don't conflict with Object.prototype properties (#13203) + sel = handleObj.selector + " "; + + if ( matches[ sel ] === undefined ) { + matches[ sel ] = handleObj.needsContext ? + jQuery( sel, this ).index( cur ) >= 0 : + jQuery.find( sel, this, null, [ cur ] ).length; + } + if ( matches[ sel ] ) { + matches.push( handleObj ); + } + } + if ( matches.length ) { + handlerQueue.push({ elem: cur, handlers: matches }); + } + } + } + } + + // Add the remaining (directly-bound) handlers + if ( delegateCount < handlers.length ) { + handlerQueue.push({ elem: this, handlers: handlers.slice( delegateCount ) }); + } + + return handlerQueue; + }, + + // Includes some event props shared by KeyEvent and MouseEvent + props: "altKey bubbles cancelable ctrlKey currentTarget eventPhase metaKey relatedTarget shiftKey target timeStamp view which".split(" "), + + fixHooks: {}, + + keyHooks: { + props: "char charCode key keyCode".split(" "), + filter: function( event, original ) { + + // Add which for key events + if ( event.which == null ) { + event.which = original.charCode != null ? original.charCode : original.keyCode; + } + + return event; + } + }, + + mouseHooks: { + props: "button buttons clientX clientY offsetX offsetY pageX pageY screenX screenY toElement".split(" "), + filter: function( event, original ) { + var eventDoc, doc, body, + button = original.button; + + // Calculate pageX/Y if missing and clientX/Y available + if ( event.pageX == null && original.clientX != null ) { + eventDoc = event.target.ownerDocument || document; + doc = eventDoc.documentElement; + body = eventDoc.body; + + event.pageX = original.clientX + ( doc && doc.scrollLeft || body && body.scrollLeft || 0 ) - ( doc && doc.clientLeft || body && body.clientLeft || 0 ); + event.pageY = original.clientY + ( doc && doc.scrollTop || body && body.scrollTop || 0 ) - ( doc && doc.clientTop || body && body.clientTop || 0 ); + } + + // Add which for click: 1 === left; 2 === middle; 3 === right + // Note: button is not normalized, so don't use it + if ( !event.which && button !== undefined ) { + event.which = ( button & 1 ? 1 : ( button & 2 ? 3 : ( button & 4 ? 2 : 0 ) ) ); + } + + return event; + } + }, + + fix: function( event ) { + if ( event[ jQuery.expando ] ) { + return event; + } + + // Create a writable copy of the event object and normalize some properties + var i, prop, copy, + type = event.type, + originalEvent = event, + fixHook = this.fixHooks[ type ]; + + if ( !fixHook ) { + this.fixHooks[ type ] = fixHook = + rmouseEvent.test( type ) ? this.mouseHooks : + rkeyEvent.test( type ) ? this.keyHooks : + {}; + } + copy = fixHook.props ? this.props.concat( fixHook.props ) : this.props; + + event = new jQuery.Event( originalEvent ); + + i = copy.length; + while ( i-- ) { + prop = copy[ i ]; + event[ prop ] = originalEvent[ prop ]; + } + + // Support: Cordova 2.5 (WebKit) (#13255) + // All events should have a target; Cordova deviceready doesn't + if ( !event.target ) { + event.target = document; + } + + // Support: Safari 6.0+, Chrome < 28 + // Target should not be a text node (#504, #13143) + if ( event.target.nodeType === 3 ) { + event.target = event.target.parentNode; + } + + return fixHook.filter ? fixHook.filter( event, originalEvent ) : event; + }, + + special: { + load: { + // Prevent triggered image.load events from bubbling to window.load + noBubble: true + }, + focus: { + // Fire native event if possible so blur/focus sequence is correct + trigger: function() { + if ( this !== safeActiveElement() && this.focus ) { + this.focus(); + return false; + } + }, + delegateType: "focusin" + }, + blur: { + trigger: function() { + if ( this === safeActiveElement() && this.blur ) { + this.blur(); + return false; + } + }, + delegateType: "focusout" + }, + click: { + // For checkbox, fire native event so checked state will be right + trigger: function() { + if ( this.type === "checkbox" && this.click && jQuery.nodeName( this, "input" ) ) { + this.click(); + return false; + } + }, + + // For cross-browser consistency, don't fire native .click() on links + _default: function( event ) { + return jQuery.nodeName( event.target, "a" ); + } + }, + + beforeunload: { + postDispatch: function( event ) { + + // Support: Firefox 20+ + // Firefox doesn't alert if the returnValue field is not set. + if ( event.result !== undefined && event.originalEvent ) { + event.originalEvent.returnValue = event.result; + } + } + } + }, + + simulate: function( type, elem, event, bubble ) { + // Piggyback on a donor event to simulate a different one. + // Fake originalEvent to avoid donor's stopPropagation, but if the + // simulated event prevents default then we do the same on the donor. + var e = jQuery.extend( + new jQuery.Event(), + event, + { + type: type, + isSimulated: true, + originalEvent: {} + } + ); + if ( bubble ) { + jQuery.event.trigger( e, null, elem ); + } else { + jQuery.event.dispatch.call( elem, e ); + } + if ( e.isDefaultPrevented() ) { + event.preventDefault(); + } + } +}; + +jQuery.removeEvent = function( elem, type, handle ) { + if ( elem.removeEventListener ) { + elem.removeEventListener( type, handle, false ); + } +}; + +jQuery.Event = function( src, props ) { + // Allow instantiation without the 'new' keyword + if ( !(this instanceof jQuery.Event) ) { + return new jQuery.Event( src, props ); + } + + // Event object + if ( src && src.type ) { + this.originalEvent = src; + this.type = src.type; + + // Events bubbling up the document may have been marked as prevented + // by a handler lower down the tree; reflect the correct value. + this.isDefaultPrevented = src.defaultPrevented || + src.defaultPrevented === undefined && + // Support: Android < 4.0 + src.returnValue === false ? + returnTrue : + returnFalse; + + // Event type + } else { + this.type = src; + } + + // Put explicitly provided properties onto the event object + if ( props ) { + jQuery.extend( this, props ); + } + + // Create a timestamp if incoming event doesn't have one + this.timeStamp = src && src.timeStamp || jQuery.now(); + + // Mark it as fixed + this[ jQuery.expando ] = true; +}; + +// jQuery.Event is based on DOM3 Events as specified by the ECMAScript Language Binding +// http://www.w3.org/TR/2003/WD-DOM-Level-3-Events-20030331/ecma-script-binding.html +jQuery.Event.prototype = { + isDefaultPrevented: returnFalse, + isPropagationStopped: returnFalse, + isImmediatePropagationStopped: returnFalse, + + preventDefault: function() { + var e = this.originalEvent; + + this.isDefaultPrevented = returnTrue; + + if ( e && e.preventDefault ) { + e.preventDefault(); + } + }, + stopPropagation: function() { + var e = this.originalEvent; + + this.isPropagationStopped = returnTrue; + + if ( e && e.stopPropagation ) { + e.stopPropagation(); + } + }, + stopImmediatePropagation: function() { + var e = this.originalEvent; + + this.isImmediatePropagationStopped = returnTrue; + + if ( e && e.stopImmediatePropagation ) { + e.stopImmediatePropagation(); + } + + this.stopPropagation(); + } +}; + +// Create mouseenter/leave events using mouseover/out and event-time checks +// Support: Chrome 15+ +jQuery.each({ + mouseenter: "mouseover", + mouseleave: "mouseout", + pointerenter: "pointerover", + pointerleave: "pointerout" +}, function( orig, fix ) { + jQuery.event.special[ orig ] = { + delegateType: fix, + bindType: fix, + + handle: function( event ) { + var ret, + target = this, + related = event.relatedTarget, + handleObj = event.handleObj; + + // 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 && !jQuery.contains( target, related )) ) { + event.type = handleObj.origType; + ret = handleObj.handler.apply( this, arguments ); + event.type = fix; + } + return ret; + } + }; +}); + +// Create "bubbling" focus and blur events +// Support: Firefox, Chrome, Safari +if ( !support.focusinBubbles ) { + jQuery.each({ focus: "focusin", blur: "focusout" }, function( orig, fix ) { + + // Attach a single capturing handler on the document while someone wants focusin/focusout + var handler = function( event ) { + jQuery.event.simulate( fix, event.target, jQuery.event.fix( event ), true ); + }; + + jQuery.event.special[ fix ] = { + setup: function() { + var doc = this.ownerDocument || this, + attaches = data_priv.access( doc, fix ); + + if ( !attaches ) { + doc.addEventListener( orig, handler, true ); + } + data_priv.access( doc, fix, ( attaches || 0 ) + 1 ); + }, + teardown: function() { + var doc = this.ownerDocument || this, + attaches = data_priv.access( doc, fix ) - 1; + + if ( !attaches ) { + doc.removeEventListener( orig, handler, true ); + data_priv.remove( doc, fix ); + + } else { + data_priv.access( doc, fix, attaches ); + } + } + }; + }); +} + +jQuery.fn.extend({ + + on: function( types, selector, data, fn, /*INTERNAL*/ one ) { + var origFn, type; + + // Types can be a map of types/handlers + if ( typeof types === "object" ) { + // ( types-Object, selector, data ) + if ( typeof selector !== "string" ) { + // ( types-Object, data ) + data = data || selector; + selector = undefined; + } + for ( type in types ) { + this.on( type, selector, data, types[ type ], one ); + } + return this; + } + + if ( data == null && fn == null ) { + // ( types, fn ) + fn = selector; + data = selector = undefined; + } else if ( fn == null ) { + if ( typeof selector === "string" ) { + // ( types, selector, fn ) + fn = data; + data = undefined; + } else { + // ( types, data, fn ) + fn = data; + data = selector; + selector = undefined; + } + } + if ( fn === false ) { + fn = returnFalse; + } else if ( !fn ) { + return this; + } + + if ( one === 1 ) { + origFn = fn; + fn = function( event ) { + // Can use an empty set, since event contains the info + jQuery().off( event ); + return origFn.apply( this, arguments ); + }; + // Use same guid so caller can remove using origFn + fn.guid = origFn.guid || ( origFn.guid = jQuery.guid++ ); + } + return this.each( function() { + jQuery.event.add( this, types, fn, data, selector ); + }); + }, + one: function( types, selector, data, fn ) { + return this.on( types, selector, data, fn, 1 ); + }, + off: function( types, selector, fn ) { + var handleObj, type; + if ( types && types.preventDefault && types.handleObj ) { + // ( event ) dispatched jQuery.Event + handleObj = types.handleObj; + jQuery( types.delegateTarget ).off( + handleObj.namespace ? handleObj.origType + "." + handleObj.namespace : handleObj.origType, + handleObj.selector, + handleObj.handler + ); + return this; + } + if ( typeof types === "object" ) { + // ( types-object [, selector] ) + for ( type in types ) { + this.off( type, selector, types[ type ] ); + } + return this; + } + if ( selector === false || typeof selector === "function" ) { + // ( types [, fn] ) + fn = selector; + selector = undefined; + } + if ( fn === false ) { + fn = returnFalse; + } + return this.each(function() { + jQuery.event.remove( this, types, fn, selector ); + }); + }, + + trigger: function( type, data ) { + return this.each(function() { + jQuery.event.trigger( type, data, this ); + }); + }, + triggerHandler: function( type, data ) { + var elem = this[0]; + if ( elem ) { + return jQuery.event.trigger( type, data, elem, true ); + } + } +}); + + +var + rxhtmlTag = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/gi, + rtagName = /<([\w:]+)/, + rhtml = /<|&#?\w+;/, + rnoInnerhtml = /<(?:script|style|link)/i, + // checked="checked" or checked + rchecked = /checked\s*(?:[^=]|=\s*.checked.)/i, + rscriptType = /^$|\/(?:java|ecma)script/i, + rscriptTypeMasked = /^true\/(.*)/, + rcleanScript = /^\s*\s*$/g, + + // We have to close these tags to support XHTML (#13200) + wrapMap = { + + // Support: IE 9 + option: [ 1, "" ], + + thead: [ 1, "", "
" ], + col: [ 2, "", "
" ], + tr: [ 2, "", "
" ], + td: [ 3, "", "
" ], + + _default: [ 0, "", "" ] + }; + +// Support: IE 9 +wrapMap.optgroup = wrapMap.option; + +wrapMap.tbody = wrapMap.tfoot = wrapMap.colgroup = wrapMap.caption = wrapMap.thead; +wrapMap.th = wrapMap.td; + +// Support: 1.x compatibility +// Manipulating tables requires a tbody +function manipulationTarget( elem, content ) { + return jQuery.nodeName( elem, "table" ) && + jQuery.nodeName( content.nodeType !== 11 ? content : content.firstChild, "tr" ) ? + + elem.getElementsByTagName("tbody")[0] || + elem.appendChild( elem.ownerDocument.createElement("tbody") ) : + elem; +} + +// Replace/restore the type attribute of script elements for safe DOM manipulation +function disableScript( elem ) { + elem.type = (elem.getAttribute("type") !== null) + "/" + elem.type; + return elem; +} +function restoreScript( elem ) { + var match = rscriptTypeMasked.exec( elem.type ); + + if ( match ) { + elem.type = match[ 1 ]; + } else { + elem.removeAttribute("type"); + } + + return elem; +} + +// Mark scripts as having already been evaluated +function setGlobalEval( elems, refElements ) { + var i = 0, + l = elems.length; + + for ( ; i < l; i++ ) { + data_priv.set( + elems[ i ], "globalEval", !refElements || data_priv.get( refElements[ i ], "globalEval" ) + ); + } +} + +function cloneCopyEvent( src, dest ) { + var i, l, type, pdataOld, pdataCur, udataOld, udataCur, events; + + if ( dest.nodeType !== 1 ) { + return; + } + + // 1. Copy private data: events, handlers, etc. + if ( data_priv.hasData( src ) ) { + pdataOld = data_priv.access( src ); + pdataCur = data_priv.set( dest, pdataOld ); + events = pdataOld.events; + + if ( events ) { + delete pdataCur.handle; + pdataCur.events = {}; + + for ( type in events ) { + for ( i = 0, l = events[ type ].length; i < l; i++ ) { + jQuery.event.add( dest, type, events[ type ][ i ] ); + } + } + } + } + + // 2. Copy user data + if ( data_user.hasData( src ) ) { + udataOld = data_user.access( src ); + udataCur = jQuery.extend( {}, udataOld ); + + data_user.set( dest, udataCur ); + } +} + +function getAll( context, tag ) { + var ret = context.getElementsByTagName ? context.getElementsByTagName( tag || "*" ) : + context.querySelectorAll ? context.querySelectorAll( tag || "*" ) : + []; + + return tag === undefined || tag && jQuery.nodeName( context, tag ) ? + jQuery.merge( [ context ], ret ) : + ret; +} + +// Support: IE >= 9 +function fixInput( src, dest ) { + var nodeName = dest.nodeName.toLowerCase(); + + // Fails to persist the checked state of a cloned checkbox or radio button. + if ( nodeName === "input" && rcheckableType.test( src.type ) ) { + dest.checked = src.checked; + + // Fails to return the selected option to the default selected state when cloning options + } else if ( nodeName === "input" || nodeName === "textarea" ) { + dest.defaultValue = src.defaultValue; + } +} + +jQuery.extend({ + clone: function( elem, dataAndEvents, deepDataAndEvents ) { + var i, l, srcElements, destElements, + clone = elem.cloneNode( true ), + inPage = jQuery.contains( elem.ownerDocument, elem ); + + // Support: IE >= 9 + // Fix Cloning issues + if ( !support.noCloneChecked && ( elem.nodeType === 1 || elem.nodeType === 11 ) && + !jQuery.isXMLDoc( elem ) ) { + + // We eschew Sizzle here for performance reasons: http://jsperf.com/getall-vs-sizzle/2 + destElements = getAll( clone ); + srcElements = getAll( elem ); + + for ( i = 0, l = srcElements.length; i < l; i++ ) { + fixInput( srcElements[ i ], destElements[ i ] ); + } + } + + // Copy the events from the original to the clone + if ( dataAndEvents ) { + if ( deepDataAndEvents ) { + srcElements = srcElements || getAll( elem ); + destElements = destElements || getAll( clone ); + + for ( i = 0, l = srcElements.length; i < l; i++ ) { + cloneCopyEvent( srcElements[ i ], destElements[ i ] ); + } + } else { + cloneCopyEvent( elem, clone ); + } + } + + // Preserve script evaluation history + destElements = getAll( clone, "script" ); + if ( destElements.length > 0 ) { + setGlobalEval( destElements, !inPage && getAll( elem, "script" ) ); + } + + // Return the cloned set + return clone; + }, + + buildFragment: function( elems, context, scripts, selection ) { + var elem, tmp, tag, wrap, contains, j, + fragment = context.createDocumentFragment(), + nodes = [], + i = 0, + l = elems.length; + + for ( ; i < l; i++ ) { + elem = elems[ i ]; + + if ( elem || elem === 0 ) { + + // Add nodes directly + if ( jQuery.type( elem ) === "object" ) { + // Support: QtWebKit + // jQuery.merge because push.apply(_, arraylike) throws + jQuery.merge( nodes, elem.nodeType ? [ elem ] : elem ); + + // Convert non-html into a text node + } else if ( !rhtml.test( elem ) ) { + nodes.push( context.createTextNode( elem ) ); + + // Convert html into DOM nodes + } else { + tmp = tmp || fragment.appendChild( context.createElement("div") ); + + // Deserialize a standard representation + tag = ( rtagName.exec( elem ) || [ "", "" ] )[ 1 ].toLowerCase(); + wrap = wrapMap[ tag ] || wrapMap._default; + tmp.innerHTML = wrap[ 1 ] + elem.replace( rxhtmlTag, "<$1>" ) + wrap[ 2 ]; + + // Descend through wrappers to the right content + j = wrap[ 0 ]; + while ( j-- ) { + tmp = tmp.lastChild; + } + + // Support: QtWebKit + // jQuery.merge because push.apply(_, arraylike) throws + jQuery.merge( nodes, tmp.childNodes ); + + // Remember the top-level container + tmp = fragment.firstChild; + + // Fixes #12346 + // Support: Webkit, IE + tmp.textContent = ""; + } + } + } + + // Remove wrapper from fragment + fragment.textContent = ""; + + i = 0; + while ( (elem = nodes[ i++ ]) ) { + + // #4087 - If origin and destination elements are the same, and this is + // that element, do not do anything + if ( selection && jQuery.inArray( elem, selection ) !== -1 ) { + continue; + } + + contains = jQuery.contains( elem.ownerDocument, elem ); + + // Append to fragment + tmp = getAll( fragment.appendChild( elem ), "script" ); + + // Preserve script evaluation history + if ( contains ) { + setGlobalEval( tmp ); + } + + // Capture executables + if ( scripts ) { + j = 0; + while ( (elem = tmp[ j++ ]) ) { + if ( rscriptType.test( elem.type || "" ) ) { + scripts.push( elem ); + } + } + } + } + + return fragment; + }, + + cleanData: function( elems ) { + var data, elem, type, key, + special = jQuery.event.special, + i = 0; + + for ( ; (elem = elems[ i ]) !== undefined; i++ ) { + if ( jQuery.acceptData( elem ) ) { + key = elem[ data_priv.expando ]; + + if ( key && (data = data_priv.cache[ key ]) ) { + if ( data.events ) { + for ( type in data.events ) { + if ( special[ type ] ) { + jQuery.event.remove( elem, type ); + + // This is a shortcut to avoid jQuery.event.remove's overhead + } else { + jQuery.removeEvent( elem, type, data.handle ); + } + } + } + if ( data_priv.cache[ key ] ) { + // Discard any remaining `private` data + delete data_priv.cache[ key ]; + } + } + } + // Discard any remaining `user` data + delete data_user.cache[ elem[ data_user.expando ] ]; + } + } +}); + +jQuery.fn.extend({ + text: function( value ) { + return access( this, function( value ) { + return value === undefined ? + jQuery.text( this ) : + this.empty().each(function() { + if ( this.nodeType === 1 || this.nodeType === 11 || this.nodeType === 9 ) { + this.textContent = value; + } + }); + }, null, value, arguments.length ); + }, + + append: function() { + return this.domManip( arguments, function( elem ) { + if ( this.nodeType === 1 || this.nodeType === 11 || this.nodeType === 9 ) { + var target = manipulationTarget( this, elem ); + target.appendChild( elem ); + } + }); + }, + + prepend: function() { + return this.domManip( arguments, function( elem ) { + if ( this.nodeType === 1 || this.nodeType === 11 || this.nodeType === 9 ) { + var target = manipulationTarget( this, elem ); + target.insertBefore( elem, target.firstChild ); + } + }); + }, + + before: function() { + return this.domManip( arguments, function( elem ) { + if ( this.parentNode ) { + this.parentNode.insertBefore( elem, this ); + } + }); + }, + + after: function() { + return this.domManip( arguments, function( elem ) { + if ( this.parentNode ) { + this.parentNode.insertBefore( elem, this.nextSibling ); + } + }); + }, + + remove: function( selector, keepData /* Internal Use Only */ ) { + var elem, + elems = selector ? jQuery.filter( selector, this ) : this, + i = 0; + + for ( ; (elem = elems[i]) != null; i++ ) { + if ( !keepData && elem.nodeType === 1 ) { + jQuery.cleanData( getAll( elem ) ); + } + + if ( elem.parentNode ) { + if ( keepData && jQuery.contains( elem.ownerDocument, elem ) ) { + setGlobalEval( getAll( elem, "script" ) ); + } + elem.parentNode.removeChild( elem ); + } + } + + return this; + }, + + empty: function() { + var elem, + i = 0; + + for ( ; (elem = this[i]) != null; i++ ) { + if ( elem.nodeType === 1 ) { + + // Prevent memory leaks + jQuery.cleanData( getAll( elem, false ) ); + + // Remove any remaining nodes + elem.textContent = ""; + } + } + + return this; + }, + + clone: function( dataAndEvents, deepDataAndEvents ) { + dataAndEvents = dataAndEvents == null ? false : dataAndEvents; + deepDataAndEvents = deepDataAndEvents == null ? dataAndEvents : deepDataAndEvents; + + return this.map(function() { + return jQuery.clone( this, dataAndEvents, deepDataAndEvents ); + }); + }, + + html: function( value ) { + return access( this, function( value ) { + var elem = this[ 0 ] || {}, + i = 0, + l = this.length; + + if ( value === undefined && elem.nodeType === 1 ) { + return elem.innerHTML; + } + + // See if we can take a shortcut and just use innerHTML + if ( typeof value === "string" && !rnoInnerhtml.test( value ) && + !wrapMap[ ( rtagName.exec( value ) || [ "", "" ] )[ 1 ].toLowerCase() ] ) { + + value = value.replace( rxhtmlTag, "<$1>" ); + + try { + for ( ; i < l; i++ ) { + elem = this[ i ] || {}; + + // Remove element nodes and prevent memory leaks + if ( elem.nodeType === 1 ) { + jQuery.cleanData( getAll( elem, false ) ); + elem.innerHTML = value; + } + } + + elem = 0; + + // If using innerHTML throws an exception, use the fallback method + } catch( e ) {} + } + + if ( elem ) { + this.empty().append( value ); + } + }, null, value, arguments.length ); + }, + + replaceWith: function() { + var arg = arguments[ 0 ]; + + // Make the changes, replacing each context element with the new content + this.domManip( arguments, function( elem ) { + arg = this.parentNode; + + jQuery.cleanData( getAll( this ) ); + + if ( arg ) { + arg.replaceChild( elem, this ); + } + }); + + // Force removal if there was no new content (e.g., from empty arguments) + return arg && (arg.length || arg.nodeType) ? this : this.remove(); + }, + + detach: function( selector ) { + return this.remove( selector, true ); + }, + + domManip: function( args, callback ) { + + // Flatten any nested arrays + args = concat.apply( [], args ); + + var fragment, first, scripts, hasScripts, node, doc, + i = 0, + l = this.length, + set = this, + iNoClone = l - 1, + value = args[ 0 ], + isFunction = jQuery.isFunction( value ); + + // We can't cloneNode fragments that contain checked, in WebKit + if ( isFunction || + ( l > 1 && typeof value === "string" && + !support.checkClone && rchecked.test( value ) ) ) { + return this.each(function( index ) { + var self = set.eq( index ); + if ( isFunction ) { + args[ 0 ] = value.call( this, index, self.html() ); + } + self.domManip( args, callback ); + }); + } + + if ( l ) { + fragment = jQuery.buildFragment( args, this[ 0 ].ownerDocument, false, this ); + first = fragment.firstChild; + + if ( fragment.childNodes.length === 1 ) { + fragment = first; + } + + if ( first ) { + scripts = jQuery.map( getAll( fragment, "script" ), disableScript ); + hasScripts = scripts.length; + + // Use the original fragment for the last item instead of the first because it can end up + // being emptied incorrectly in certain situations (#8070). + for ( ; i < l; i++ ) { + node = fragment; + + if ( i !== iNoClone ) { + node = jQuery.clone( node, true, true ); + + // Keep references to cloned scripts for later restoration + if ( hasScripts ) { + // Support: QtWebKit + // jQuery.merge because push.apply(_, arraylike) throws + jQuery.merge( scripts, getAll( node, "script" ) ); + } + } + + callback.call( this[ i ], node, i ); + } + + if ( hasScripts ) { + doc = scripts[ scripts.length - 1 ].ownerDocument; + + // Reenable scripts + jQuery.map( scripts, restoreScript ); + + // Evaluate executable scripts on first document insertion + for ( i = 0; i < hasScripts; i++ ) { + node = scripts[ i ]; + if ( rscriptType.test( node.type || "" ) && + !data_priv.access( node, "globalEval" ) && jQuery.contains( doc, node ) ) { + + if ( node.src ) { + // Optional AJAX dependency, but won't run scripts if not present + if ( jQuery._evalUrl ) { + jQuery._evalUrl( node.src ); + } + } else { + jQuery.globalEval( node.textContent.replace( rcleanScript, "" ) ); + } + } + } + } + } + } + + return this; + } +}); + +jQuery.each({ + appendTo: "append", + prependTo: "prepend", + insertBefore: "before", + insertAfter: "after", + replaceAll: "replaceWith" +}, function( name, original ) { + jQuery.fn[ name ] = function( selector ) { + var elems, + ret = [], + insert = jQuery( selector ), + last = insert.length - 1, + i = 0; + + for ( ; i <= last; i++ ) { + elems = i === last ? this : this.clone( true ); + jQuery( insert[ i ] )[ original ]( elems ); + + // Support: QtWebKit + // .get() because push.apply(_, arraylike) throws + push.apply( ret, elems.get() ); + } + + return this.pushStack( ret ); + }; +}); + + +var iframe, + elemdisplay = {}; + +/** + * Retrieve the actual display of a element + * @param {String} name nodeName of the element + * @param {Object} doc Document object + */ +// Called only from within defaultDisplay +function actualDisplay( name, doc ) { + var style, + elem = jQuery( doc.createElement( name ) ).appendTo( doc.body ), + + // getDefaultComputedStyle might be reliably used only on attached element + display = window.getDefaultComputedStyle && ( style = window.getDefaultComputedStyle( elem[ 0 ] ) ) ? + + // Use of this method is a temporary fix (more like optmization) until something better comes along, + // since it was removed from specification and supported only in FF + style.display : jQuery.css( elem[ 0 ], "display" ); + + // We don't have any data stored on the element, + // so use "detach" method as fast way to get rid of the element + elem.detach(); + + return display; +} + +/** + * Try to determine the default display value of an element + * @param {String} nodeName + */ +function defaultDisplay( nodeName ) { + var doc = document, + display = elemdisplay[ nodeName ]; + + if ( !display ) { + display = actualDisplay( nodeName, doc ); + + // If the simple way fails, read from inside an iframe + if ( display === "none" || !display ) { + + // Use the already-created iframe if possible + iframe = (iframe || jQuery( " + +
+ + Light 300 +
+ AaBbCcDdEeFfGgHhJjKkLlMmNnOoPpQqRrSsTtUuVvWwXxYyZz +
+ + Light 300 Italic +
+ AaBbCcDdEeFfGgHhJjKkLlMmNnOoPpQqRrSsTtUuVvWwXxYyZz +
+ + Normal 400 +
+ AaBbCcDdEeFfGgHhJjKkLlMmNnOoPpQqRrSsTtUuVvWwXxYyZz +
+ + Normal 400 Italic +
+ AaBbCcDdEeFfGgHhJjKkLlMmNnOoPpQqRrSsTtUuVvWwXxYyZz +
+ + Semibold 600 +
+ AaBbCcDdEeFfGgHhJjKkLlMmNnOoPpQqRrSsTtUuVvWwXxYyZz +
+ + Semibold 600 Italic +
+ AaBbCcDdEeFfGgHhJjKkLlMmNnOoPpQqRrSsTtUuVvWwXxYyZz +
+ + Bold 700 +
+ AaBbCcDdEeFfGgHhJjKkLlMmNnOoPpQqRrSsTtUuVvWwXxYyZz +
+ + Bold 700 Italic +
+ AaBbCcDdEeFfGgHhJjKkLlMmNnOoPpQqRrSsTtUuVvWwXxYyZz +
+ + Extrabold 800 +
+ AaBbCcDdEeFfGgHhJjKkLlMmNnOoPpQqRrSsTtUuVvWwXxYyZz +
+ + Extrabold 800 Italic +
+ AaBbCcDdEeFfGgHhJjKkLlMmNnOoPpQqRrSsTtUuVvWwXxYyZz +
+ + \ No newline at end of file diff --git a/1.4.10/docs/components/open-sans-fontface-1.0.4/open-sans.css b/1.4.10/docs/components/open-sans-fontface-1.0.4/open-sans.css new file mode 100644 index 0000000000..76c36b935c --- /dev/null +++ b/1.4.10/docs/components/open-sans-fontface-1.0.4/open-sans.css @@ -0,0 +1,131 @@ +/* Open Sans @font-face kit */ + +/* BEGIN Light */ +@font-face { + font-family: 'Open Sans'; + src: url('fonts/Light/OpenSans-Light.eot'); + src: url('fonts/Light/OpenSans-Light.eot?#iefix') format('embedded-opentype'), + url('fonts/Light/OpenSans-Light.woff') format('woff'), + url('fonts/Light/OpenSans-Light.ttf') format('truetype'), + url('fonts/Light/OpenSans-Light.svg#OpenSansLight') format('svg'); + font-weight: 300; + font-style: normal; +} +/* END Light */ + +/* BEGIN Light Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('fonts/LightItalic/OpenSans-LightItalic.eot'); + src: url('fonts/LightItalic/OpenSans-LightItalic.eot?#iefix') format('embedded-opentype'), + url('fonts/LightItalic/OpenSans-LightItalic.woff') format('woff'), + url('fonts/LightItalic/OpenSans-LightItalic.ttf') format('truetype'), + url('fonts/LightItalic/OpenSans-LightItalic.svg#OpenSansLightItalic') format('svg'); + font-weight: 300; + font-style: italic; +} +/* END Light Italic */ + +/* BEGIN Regular */ +@font-face { + font-family: 'Open Sans'; + src: url('fonts/Regular/OpenSans-Regular.eot'); + src: url('fonts/Regular/OpenSans-Regular.eot?#iefix') format('embedded-opentype'), + url('fonts/Regular/OpenSans-Regular.woff') format('woff'), + url('fonts/Regular/OpenSans-Regular.ttf') format('truetype'), + url('fonts/Regular/OpenSans-Regular.svg#OpenSansRegular') format('svg'); + font-weight: normal; + font-style: normal; +} +/* END Regular */ + +/* BEGIN Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('fonts/Italic/OpenSans-Italic.eot'); + src: url('fonts/Italic/OpenSans-Italic.eot?#iefix') format('embedded-opentype'), + url('fonts/Italic/OpenSans-Italic.woff') format('woff'), + url('fonts/Italic/OpenSans-Italic.ttf') format('truetype'), + url('fonts/Italic/OpenSans-Italic.svg#OpenSansItalic') format('svg'); + font-weight: normal; + font-style: italic; +} +/* END Italic */ + +/* BEGIN Semibold */ +@font-face { + font-family: 'Open Sans'; + src: url('fonts/Semibold/OpenSans-Semibold.eot'); + src: url('fonts/Semibold/OpenSans-Semibold.eot?#iefix') format('embedded-opentype'), + url('fonts/Semibold/OpenSans-Semibold.woff') format('woff'), + url('fonts/Semibold/OpenSans-Semibold.ttf') format('truetype'), + url('fonts/Semibold/OpenSans-Semibold.svg#OpenSansSemibold') format('svg'); + font-weight: 600; + font-style: normal; +} +/* END Semibold */ + +/* BEGIN Semibold Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('fonts/SemiboldItalic/OpenSans-SemiboldItalic.eot'); + src: url('fonts/SemiboldItalic/OpenSans-SemiboldItalic.eot?#iefix') format('embedded-opentype'), + url('fonts/SemiboldItalic/OpenSans-SemiboldItalic.woff') format('woff'), + url('fonts/SemiboldItalic/OpenSans-SemiboldItalic.ttf') format('truetype'), + url('fonts/SemiboldItalic/OpenSans-SemiboldItalic.svg#OpenSansSemiboldItalic') format('svg'); + font-weight: 600; + font-style: italic; +} +/* END Semibold Italic */ + +/* BEGIN Bold */ +@font-face { + font-family: 'Open Sans'; + src: url('fonts/Bold/OpenSans-Bold.eot'); + src: url('fonts/Bold/OpenSans-Bold.eot?#iefix') format('embedded-opentype'), + url('fonts/Bold/OpenSans-Bold.woff') format('woff'), + url('fonts/Bold/OpenSans-Bold.ttf') format('truetype'), + url('fonts/Bold/OpenSans-Bold.svg#OpenSansBold') format('svg'); + font-weight: bold; + font-style: normal; +} +/* END Bold */ + +/* BEGIN Bold Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('fonts/BoldItalic/OpenSans-BoldItalic.eot'); + src: url('fonts/BoldItalic/OpenSans-BoldItalic.eot?#iefix') format('embedded-opentype'), + url('fonts/BoldItalic/OpenSans-BoldItalic.woff') format('woff'), + url('fonts/BoldItalic/OpenSans-BoldItalic.ttf') format('truetype'), + url('fonts/BoldItalic/OpenSans-BoldItalic.svg#OpenSansBoldItalic') format('svg'); + font-weight: bold; + font-style: italic; +} +/* END Bold Italic */ + +/* BEGIN Extrabold */ +@font-face { + font-family: 'Open Sans'; + src: url('fonts/ExtraBold/OpenSans-ExtraBold.eot'); + src: url('fonts/ExtraBold/OpenSans-ExtraBold.eot?#iefix') format('embedded-opentype'), + url('fonts/ExtraBold/OpenSans-ExtraBold.woff') format('woff'), + url('fonts/ExtraBold/OpenSans-ExtraBold.ttf') format('truetype'), + url('fonts/ExtraBold/OpenSans-ExtraBold.svg#OpenSansExtrabold') format('svg'); + font-weight: 800; + font-style: normal; +} +/* END Extrabold */ + +/* BEGIN Extrabold Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('fonts/ExtraBoldItalic/OpenSans-ExtraBoldItalic.eot'); + src: url('fonts/ExtraBoldItalic/OpenSans-ExtraBoldItalic.eot?#iefix') format('embedded-opentype'), + url('fonts/ExtraBoldItalic/OpenSans-ExtraBoldItalic.woff') format('woff'), + url('fonts/ExtraBoldItalic/OpenSans-ExtraBoldItalic.ttf') format('truetype'), + url('fonts/ExtraBoldItalic/OpenSans-ExtraBoldItalic.svg#OpenSansExtraboldItalic') format('svg'); + font-weight: 800; + font-style: italic; +} +/* END Extrabold Italic */ diff --git a/1.4.10/docs/components/open-sans-fontface-1.0.4/open-sans.less b/1.4.10/docs/components/open-sans-fontface-1.0.4/open-sans.less new file mode 100644 index 0000000000..236a402e96 --- /dev/null +++ b/1.4.10/docs/components/open-sans-fontface-1.0.4/open-sans.less @@ -0,0 +1,133 @@ +/* Open Sans @font-face kit */ + +@OpenSansPath: "./fonts"; + +/* BEGIN Light */ +@font-face { + font-family: 'Open Sans'; + src: url('@{OpenSansPath}/Light/OpenSans-Light.eot'); + src: url('@{OpenSansPath}/Light/OpenSans-Light.eot?#iefix') format('embedded-opentype'), + url('@{OpenSansPath}/Light/OpenSans-Light.woff') format('woff'), + url('@{OpenSansPath}/Light/OpenSans-Light.ttf') format('truetype'), + url('@{OpenSansPath}/Light/OpenSans-Light.svg#OpenSansLight') format('svg'); + font-weight: 300; + font-style: normal; +} +/* END Light */ + +/* BEGIN Light Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('@{OpenSansPath}/LightItalic/OpenSans-LightItalic.eot'); + src: url('@{OpenSansPath}/LightItalic/OpenSans-LightItalic.eot?#iefix') format('embedded-opentype'), + url('@{OpenSansPath}/LightItalic/OpenSans-LightItalic.woff') format('woff'), + url('@{OpenSansPath}/LightItalic/OpenSans-LightItalic.ttf') format('truetype'), + url('@{OpenSansPath}/LightItalic/OpenSans-LightItalic.svg#OpenSansLightItalic') format('svg'); + font-weight: 300; + font-style: italic; +} +/* END Light Italic */ + +/* BEGIN Regular */ +@font-face { + font-family: 'Open Sans'; + src: url('@{OpenSansPath}/Regular/OpenSans-Regular.eot'); + src: url('@{OpenSansPath}/Regular/OpenSans-Regular.eot?#iefix') format('embedded-opentype'), + url('@{OpenSansPath}/Regular/OpenSans-Regular.woff') format('woff'), + url('@{OpenSansPath}/Regular/OpenSans-Regular.ttf') format('truetype'), + url('@{OpenSansPath}/Regular/OpenSans-Regular.svg#OpenSansRegular') format('svg'); + font-weight: normal; + font-style: normal; +} +/* END Regular */ + +/* BEGIN Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('@{OpenSansPath}/Italic/OpenSans-Italic.eot'); + src: url('@{OpenSansPath}/Italic/OpenSans-Italic.eot?#iefix') format('embedded-opentype'), + url('@{OpenSansPath}/Italic/OpenSans-Italic.woff') format('woff'), + url('@{OpenSansPath}/Italic/OpenSans-Italic.ttf') format('truetype'), + url('@{OpenSansPath}/Italic/OpenSans-Italic.svg#OpenSansItalic') format('svg'); + font-weight: normal; + font-style: italic; +} +/* END Italic */ + +/* BEGIN Semibold */ +@font-face { + font-family: 'Open Sans'; + src: url('@{OpenSansPath}/Semibold/OpenSans-Semibold.eot'); + src: url('@{OpenSansPath}/Semibold/OpenSans-Semibold.eot?#iefix') format('embedded-opentype'), + url('@{OpenSansPath}/Semibold/OpenSans-Semibold.woff') format('woff'), + url('@{OpenSansPath}/Semibold/OpenSans-Semibold.ttf') format('truetype'), + url('@{OpenSansPath}/Semibold/OpenSans-Semibold.svg#OpenSansSemibold') format('svg'); + font-weight: 600; + font-style: normal; +} +/* END Semibold */ + +/* BEGIN Semibold Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('@{OpenSansPath}/SemiboldItalic/OpenSans-SemiboldItalic.eot'); + src: url('@{OpenSansPath}/SemiboldItalic/OpenSans-SemiboldItalic.eot?#iefix') format('embedded-opentype'), + url('@{OpenSansPath}/SemiboldItalic/OpenSans-SemiboldItalic.woff') format('woff'), + url('@{OpenSansPath}/SemiboldItalic/OpenSans-SemiboldItalic.ttf') format('truetype'), + url('@{OpenSansPath}/SemiboldItalic/OpenSans-SemiboldItalic.svg#OpenSansSemiboldItalic') format('svg'); + font-weight: 600; + font-style: italic; +} +/* END Semibold Italic */ + +/* BEGIN Bold */ +@font-face { + font-family: 'Open Sans'; + src: url('@{OpenSansPath}/Bold/OpenSans-Bold.eot'); + src: url('@{OpenSansPath}/Bold/OpenSans-Bold.eot?#iefix') format('embedded-opentype'), + url('@{OpenSansPath}/Bold/OpenSans-Bold.woff') format('woff'), + url('@{OpenSansPath}/Bold/OpenSans-Bold.ttf') format('truetype'), + url('@{OpenSansPath}/Bold/OpenSans-Bold.svg#OpenSansBold') format('svg'); + font-weight: bold; + font-style: normal; +} +/* END Bold */ + +/* BEGIN Bold Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('@{OpenSansPath}/BoldItalic/OpenSans-BoldItalic.eot'); + src: url('@{OpenSansPath}/BoldItalic/OpenSans-BoldItalic.eot?#iefix') format('embedded-opentype'), + url('@{OpenSansPath}/BoldItalic/OpenSans-BoldItalic.woff') format('woff'), + url('@{OpenSansPath}/BoldItalic/OpenSans-BoldItalic.ttf') format('truetype'), + url('@{OpenSansPath}/BoldItalic/OpenSans-BoldItalic.svg#OpenSansBoldItalic') format('svg'); + font-weight: bold; + font-style: italic; +} +/* END Bold Italic */ + +/* BEGIN Extrabold */ +@font-face { + font-family: 'Open Sans'; + src: url('@{OpenSansPath}/ExtraBold/OpenSans-ExtraBold.eot'); + src: url('@{OpenSansPath}/ExtraBold/OpenSans-ExtraBold.eot?#iefix') format('embedded-opentype'), + url('@{OpenSansPath}/ExtraBold/OpenSans-ExtraBold.woff') format('woff'), + url('@{OpenSansPath}/ExtraBold/OpenSans-ExtraBold.ttf') format('truetype'), + url('@{OpenSansPath}/ExtraBold/OpenSans-ExtraBold.svg#OpenSansExtrabold') format('svg'); + font-weight: 800; + font-style: normal; +} +/* END Extrabold */ + +/* BEGIN Extrabold Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('@{OpenSansPath}/ExtraBoldItalic/OpenSans-ExtraBoldItalic.eot'); + src: url('@{OpenSansPath}/ExtraBoldItalic/OpenSans-ExtraBoldItalic.eot?#iefix') format('embedded-opentype'), + url('@{OpenSansPath}/ExtraBoldItalic/OpenSans-ExtraBoldItalic.woff') format('woff'), + url('@{OpenSansPath}/ExtraBoldItalic/OpenSans-ExtraBoldItalic.ttf') format('truetype'), + url('@{OpenSansPath}/ExtraBoldItalic/OpenSans-ExtraBoldItalic.svg#OpenSansExtraboldItalic') format('svg'); + font-weight: 800; + font-style: italic; +} +/* END Extrabold Italic */ diff --git a/1.4.10/docs/components/open-sans-fontface-1.0.4/open-sans.scss b/1.4.10/docs/components/open-sans-fontface-1.0.4/open-sans.scss new file mode 100644 index 0000000000..a46063fbb3 --- /dev/null +++ b/1.4.10/docs/components/open-sans-fontface-1.0.4/open-sans.scss @@ -0,0 +1,133 @@ +/* Open Sans @font-face kit */ + +$OpenSansPath: "./fonts" !default; + +/* BEGIN Light */ +@font-face { + font-family: 'Open Sans'; + src: url('#{$OpenSansPath}/Light/OpenSans-Light.eot'); + src: url('#{$OpenSansPath}/Light/OpenSans-Light.eot?#iefix') format('embedded-opentype'), + url('#{$OpenSansPath}/Light/OpenSans-Light.woff') format('woff'), + url('#{$OpenSansPath}/Light/OpenSans-Light.ttf') format('truetype'), + url('#{$OpenSansPath}/Light/OpenSans-Light.svg#OpenSansLight') format('svg'); + font-weight: 300; + font-style: normal; +} +/* END Light */ + +/* BEGIN Light Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('#{$OpenSansPath}/LightItalic/OpenSans-LightItalic.eot'); + src: url('#{$OpenSansPath}/LightItalic/OpenSans-LightItalic.eot?#iefix') format('embedded-opentype'), + url('#{$OpenSansPath}/LightItalic/OpenSans-LightItalic.woff') format('woff'), + url('#{$OpenSansPath}/LightItalic/OpenSans-LightItalic.ttf') format('truetype'), + url('#{$OpenSansPath}/LightItalic/OpenSans-LightItalic.svg#OpenSansLightItalic') format('svg'); + font-weight: 300; + font-style: italic; +} +/* END Light Italic */ + +/* BEGIN Regular */ +@font-face { + font-family: 'Open Sans'; + src: url('#{$OpenSansPath}/Regular/OpenSans-Regular.eot'); + src: url('#{$OpenSansPath}/Regular/OpenSans-Regular.eot?#iefix') format('embedded-opentype'), + url('#{$OpenSansPath}/Regular/OpenSans-Regular.woff') format('woff'), + url('#{$OpenSansPath}/Regular/OpenSans-Regular.ttf') format('truetype'), + url('#{$OpenSansPath}/Regular/OpenSans-Regular.svg#OpenSansRegular') format('svg'); + font-weight: normal; + font-style: normal; +} +/* END Regular */ + +/* BEGIN Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('#{$OpenSansPath}/Italic/OpenSans-Italic.eot'); + src: url('#{$OpenSansPath}/Italic/OpenSans-Italic.eot?#iefix') format('embedded-opentype'), + url('#{$OpenSansPath}/Italic/OpenSans-Italic.woff') format('woff'), + url('#{$OpenSansPath}/Italic/OpenSans-Italic.ttf') format('truetype'), + url('#{$OpenSansPath}/Italic/OpenSans-Italic.svg#OpenSansItalic') format('svg'); + font-weight: normal; + font-style: italic; +} +/* END Italic */ + +/* BEGIN Semibold */ +@font-face { + font-family: 'Open Sans'; + src: url('#{$OpenSansPath}/Semibold/OpenSans-Semibold.eot'); + src: url('#{$OpenSansPath}/Semibold/OpenSans-Semibold.eot?#iefix') format('embedded-opentype'), + url('#{$OpenSansPath}/Semibold/OpenSans-Semibold.woff') format('woff'), + url('#{$OpenSansPath}/Semibold/OpenSans-Semibold.ttf') format('truetype'), + url('#{$OpenSansPath}/Semibold/OpenSans-Semibold.svg#OpenSansSemibold') format('svg'); + font-weight: 600; + font-style: normal; +} +/* END Semibold */ + +/* BEGIN Semibold Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('#{$OpenSansPath}/SemiboldItalic/OpenSans-SemiboldItalic.eot'); + src: url('#{$OpenSansPath}/SemiboldItalic/OpenSans-SemiboldItalic.eot?#iefix') format('embedded-opentype'), + url('#{$OpenSansPath}/SemiboldItalic/OpenSans-SemiboldItalic.woff') format('woff'), + url('#{$OpenSansPath}/SemiboldItalic/OpenSans-SemiboldItalic.ttf') format('truetype'), + url('#{$OpenSansPath}/SemiboldItalic/OpenSans-SemiboldItalic.svg#OpenSansSemiboldItalic') format('svg'); + font-weight: 600; + font-style: italic; +} +/* END Semibold Italic */ + +/* BEGIN Bold */ +@font-face { + font-family: 'Open Sans'; + src: url('#{$OpenSansPath}/Bold/OpenSans-Bold.eot'); + src: url('#{$OpenSansPath}/Bold/OpenSans-Bold.eot?#iefix') format('embedded-opentype'), + url('#{$OpenSansPath}/Bold/OpenSans-Bold.woff') format('woff'), + url('#{$OpenSansPath}/Bold/OpenSans-Bold.ttf') format('truetype'), + url('#{$OpenSansPath}/Bold/OpenSans-Bold.svg#OpenSansBold') format('svg'); + font-weight: bold; + font-style: normal; +} +/* END Bold */ + +/* BEGIN Bold Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('#{$OpenSansPath}/BoldItalic/OpenSans-BoldItalic.eot'); + src: url('#{$OpenSansPath}/BoldItalic/OpenSans-BoldItalic.eot?#iefix') format('embedded-opentype'), + url('#{$OpenSansPath}/BoldItalic/OpenSans-BoldItalic.woff') format('woff'), + url('#{$OpenSansPath}/BoldItalic/OpenSans-BoldItalic.ttf') format('truetype'), + url('#{$OpenSansPath}/BoldItalic/OpenSans-BoldItalic.svg#OpenSansBoldItalic') format('svg'); + font-weight: bold; + font-style: italic; +} +/* END Bold Italic */ + +/* BEGIN Extrabold */ +@font-face { + font-family: 'Open Sans'; + src: url('#{$OpenSansPath}/ExtraBold/OpenSans-ExtraBold.eot'); + src: url('#{$OpenSansPath}/ExtraBold/OpenSans-ExtraBold.eot?#iefix') format('embedded-opentype'), + url('#{$OpenSansPath}/ExtraBold/OpenSans-ExtraBold.woff') format('woff'), + url('#{$OpenSansPath}/ExtraBold/OpenSans-ExtraBold.ttf') format('truetype'), + url('#{$OpenSansPath}/ExtraBold/OpenSans-ExtraBold.svg#OpenSansExtrabold') format('svg'); + font-weight: 800; + font-style: normal; +} +/* END Extrabold */ + +/* BEGIN Extrabold Italic */ +@font-face { + font-family: 'Open Sans'; + src: url('#{$OpenSansPath}/ExtraBoldItalic/OpenSans-ExtraBoldItalic.eot'); + src: url('#{$OpenSansPath}/ExtraBoldItalic/OpenSans-ExtraBoldItalic.eot?#iefix') format('embedded-opentype'), + url('#{$OpenSansPath}/ExtraBoldItalic/OpenSans-ExtraBoldItalic.woff') format('woff'), + url('#{$OpenSansPath}/ExtraBoldItalic/OpenSans-ExtraBoldItalic.ttf') format('truetype'), + url('#{$OpenSansPath}/ExtraBoldItalic/OpenSans-ExtraBoldItalic.svg#OpenSansExtraboldItalic') format('svg'); + font-weight: 800; + font-style: italic; +} +/* END Extrabold Italic */ diff --git a/1.4.10/docs/css/animations.css b/1.4.10/docs/css/animations.css new file mode 100644 index 0000000000..e69de29bb2 diff --git a/1.4.10/docs/css/doc_widgets.css b/1.4.10/docs/css/doc_widgets.css new file mode 100644 index 0000000000..c87db30355 --- /dev/null +++ b/1.4.10/docs/css/doc_widgets.css @@ -0,0 +1,150 @@ +ul.doc-example { + list-style-type: none; + position: relative; + font-size: 14px; +} + +ul.doc-example > li { + border: 2px solid gray; + border-radius: 5px; + -moz-border-radius: 5px; + -webkit-border-radius: 5px; + background-color: white; + margin-bottom: 20px; +} + +ul.doc-example > li.doc-example-heading { + border: none; + border-radius: 0; + margin-bottom: -10px; +} + +span.nojsfiddle { + float: right; + font-size: 14px; + margin-right:10px; + margin-top: 10px; +} + +form.jsfiddle { + position: absolute; + right: 0; + z-index: 1; + height: 14px; +} + +form.jsfiddle button { + cursor: pointer; + padding: 4px 10px; + margin: 10px; + background-color: #FFF; + font-weight: bold; + color: #7989D6; + border-color: #7989D6; + -moz-border-radius: 8px; + -webkit-border-radius:8px; + border-radius: 8px; +} + +form.jsfiddle textarea, form.jsfiddle input { + display: none; +} + +li.doc-example-live { + padding: 10px; + font-size: 1.2em; +} + +div.syntaxhighlighter { + padding-bottom: 1px !important; /* fix to remove unnecessary scrollbars */ +} + +/* TABS - tutorial environment navigation */ + +div.tabs-nav { + height: 25px; + position: relative; +} + +div.tabs-nav ul li { + list-style: none; + display: inline-block; + padding: 5px 10px; +} + +div.tabs-nav ul li.current a { + color: white; + text-decoration: none; +} + +div.tabs-nav ul li.current { + background: #7989D6; + -moz-box-shadow: 4px 4px 6px #48577D; + -moz-border-radius-topright: 8px; + -moz-border-radius-topleft: 8px; + box-shadow: 4px 4px 6px #48577D; + border-radius-topright: 8px; + border-radius-topleft: 8px; + -webkit-box-shadow: 4px 4px 6px #48577D; + -webkit-border-top-right-radius: 8px; + -webkit-border-top-left-radius: 8px; + border-top-right-radius: 8px; + border-top-left-radius: 8px; +} + +div.tabs-content { + padding: 4px; + position: relative; + background: #7989D6; + -moz-border-radius: 8px; + border-radius: 8px; + -webkit-border-radius: 8px; +} + +div.tabs-content-inner { + margin: 1px; + padding: 10px; + background: white; + border-radius: 6px; + -moz-border-radius: 6px; + -webkit-border-radius: 6px; +} + + +/* Tutorial Nav Bar */ + +#tutorial-nav { + margin: 0.5em 0 1em 0; + padding: 0; + list-style-type: none; + background: #7989D6; + + -moz-border-radius: 15px; + -webkit-border-radius: 15px; + border-radius: 15px; + + -moz-box-shadow: 4px 4px 6px #48577D; + -webkit-box-shadow: 4px 4px 6px #48577D; + box-shadow: 4px 4px 6px #48577D; +} + + +#tutorial-nav li { + display: inline; +} + + +#tutorial-nav a:link, #tutorial-nav a:visited { + font-size: 1.2em; + color: #FFF; + text-decoration: none; + text-align: center; + display: inline-block; + width: 11em; + padding: 0.2em 0; +} + + +#tutorial-nav a:hover { + color: #000; +} diff --git a/1.4.10/docs/css/docs.css b/1.4.10/docs/css/docs.css new file mode 100644 index 0000000000..89a15c4928 --- /dev/null +++ b/1.4.10/docs/css/docs.css @@ -0,0 +1,713 @@ +html, body { + position:relative; + height:100%; +} + +#wrapper { + min-height:100%; + position:relative; + padding-bottom:120px; +} + +.footer { + border-top:20px solid white; + position:absolute; + bottom:0; + left:0; + right:0; + z-index:100; + padding-top: 2em; + background-color: #333; + color: white; + padding-bottom: 2em; +} + +.header-fixed { + position:fixed; + z-index:1000; + top:0; + left:0; + right:0; +} + +.header-branding { + min-height:41px!important; +} + +.docs-navbar-primary { + border-radius:0!important; + margin-bottom:0!important; +} + +/* Logo */ +/*.dropdown-menu { + display:none; +} +*/ +h1,h2,h3,h4,h5,h6 { + font-family: "Open Sans"; +} + +.subnav-body { + margin:70px 0 20px; +} + +.header .brand { + padding-top: 6px; + padding-bottom: 0px; +} + +.header .brand img { + margin-top:5px; + height: 30px; +} + +.docs-search { + margin:10px 0; + padding:4px 0 4px 20px; + background:white; + border-radius:20px; + vertical-align:middle; +} + +.docs-search > .search-query { + font-size:14px; + border:0; + width:80%; + color:#555; +} + +.docs-search > .search-icon { + font-size:15px; + margin-right:10px; +} + +.docs-search > .search-query:focus { + outline:0; +} + +/* end: Logo */ + + +.spacer { + height: 1em; +} + + +.icon-cog { + line-height: 13px; +} + +.naked-list, +.naked-list ul, +.naked-list li { + list-style:none; + margin:0; + padding:0; +} + +.nav-index-section a { + font-weight:bold; + font-family: "Open Sans"; + color:black!important; + margin-top:10px; + display:block; +} + +.nav-index-group { + margin-bottom:20px!important; +} + +.nav-index-group-heading { + color:#6F0101; + font-weight:bold; + font-size:1.2em; + padding:0; + margin:0; + border-bottom:1px solid #aaa; + margin-bottom:5px; +} + +.nav-index-group .nav-index-listing.current a { + color: #B52E31; +} + +.nav-breadcrumb { + margin:4px 0; + padding:0; +} + +.nav-breadcrumb-entry { + font-family: "Open Sans"; + padding:0; + margin:0; + font-size:18px; + display:inline-block; + vertical-align:middle; +} + +.nav-breadcrumb-entry > .divider { + color:#555; + display:inline-block; + padding-left:8px; +} + +.nav-breadcrumb-entry > span, +.nav-breadcrumb-entry > a { + color:#6F0101; +} + +.step-list > li:nth-child(1) { + padding-left:20px; +} + +.step-list > li:nth-child(2) { + padding-left:40px; +} + +.step-list > li:nth-child(3) { + padding-left:60px; +} + +.api-profile-header-heading { + margin:0; + padding:0; +} + +.api-profile-header-structure, +.api-profile-header-structure a { + font-family: "Open Sans"; + font-weight:bold; + color:#999; +} + +.api-profile-section { + margin-top:30px; + padding-top:30px; + border-top:1px solid #aaa; +} + +pre { + white-space: pre-wrap; + word-break: normal; +} + +.aside-nav a, +.aside-nav a:link, +.aside-nav a:visited, +.aside-nav a:active { + color:#999; +} +.aside-nav a:hover { + color:black; +} + +.api-profile-description > p:first-child { + margin:15px 0; + font-size:18px; +} + +p > code, +code.highlighted { + background:#f4f4f4; + border-radius:5px; + padding:2px 5px; + color:maroon; +} + +ul + p { + margin-top: 10px; +} + +.docs-version-jump { + min-width:100%; + max-width:100%; +} + +.picker { + position: relative; + width: auto; + display: inline-block; + margin: 0 0 2px 1.2%; + overflow: hidden; + border: 1px solid #e5e5e5; + -webkit-border-radius: 4px; + -moz-border-radius: 4px; + -ms-border-radius: 4px; + -o-border-radius: 4px; + border-radius: 4px; + font-family: "Open Sans"; + font-weight: 600; + height: auto; + background-image: -webkit-gradient(linear, 50% 0%, 50% 100%, color-stop(0%, #ffffff), color-stop(100%, #f2f2f2)); + background-image: -webkit-linear-gradient(#ffffff, #f2f2f2); + background-image: -moz-linear-gradient(#ffffff, #f2f2f2); + background-image: -o-linear-gradient(#ffffff, #f2f2f2); + background-image: linear-gradient(#ffffff, #f2f2f2); +} + +.picker select { + position: relative; + display: block; + min-width: 100%; + width: 120%; + height: 34px; + padding: 6px 30px 6px 15px; + color: #555555; + border: none; + background: transparent; + outline: none; + -webkit-appearance: none; + z-index: 99; + cursor: pointer; + font-size: 16px; + -moz-appearance: none; + text-indent: 0.01px; + text-overflow: ''; +} + +.picker:after { + content:""; + position: absolute; + right: 8%; + top: 50%; + z-index: 0; + color: #999; + width: 0; + margin-top:-2px; + height: 0; + border-top: 6px solid; + border-right: 6px solid transparent; + border-left: 6px solid transparent; +} + +iframe.example { + width: 100%; + border: 1px solid black; +} + +.search-results-frame { + clear:both; + display:table; + width:100%; +} + +.search-results.ng-hide { + display:none; +} + +.search-results-container { + padding-bottom:1em; + border-top:1px solid #111; + background:#181818; + box-shadow:inset 0 0 10px #111; +} + +.search-results-container .search-results-group { + vertical-align:top; + padding:10px 10px; + display:inline-block; +} + +.search-results-group-heading { + font-family: "Open Sans"; + padding-left:10px; + color:white; +} + +.search-results-frame > .search-results-group:first-child > .search-results { + border-right:1px solid #050505; +} + +.search-results-group.col-group-api { width:30%; } +.search-results-group.col-group-guide, +.search-results-group.col-group-tutorial { width:20%; } +.search-results-group.col-group-misc, +.search-results-group.col-group-error { width:15%; float: right; } + + +.search-results-group.col-group-api .search-result { + width:48%; + display:inline-block; +} + +.search-close { + position: absolute; + bottom: 0; + left: 50%; + margin-left: -100px; + color: white; + text-align: center; + padding: 5px; + background: #333; + border-top-right-radius: 5px; + border-top-left-radius: 5px; + width: 200px; + box-shadow:0 0 10px #111; +} + +.variables-matrix { + border:1px solid #ddd; + width:100%; + margin:10px 0; +} + +.variables-matrix td, +.variables-matrix th { + padding:10px; +} + +.variables-matrix td { + border-top:1px solid #eee; +} + +.variables-matrix td + td, +.variables-matrix th + th { + border-left:1px solid #eee; +} + +.variables-matrix tr:nth-child(even) td { + background:#f5f5f5; +} + +.variables-matrix th { + background:#f1f1f1; +} + +.sup-header { + padding-top:10px; + padding-bottom:5px; + background:rgba(245,245,245,0.88); + box-shadow:0 0 2px #999; +} + +.main-body-grid { + margin-top:120px; + position:relative; +} + +.main-body-grid > .grid-left, +.main-body-grid > .grid-right { + padding:20px 0; +} + +.main-body-grid > .grid-left { + position:fixed; + top:120px; + bottom:0; + overflow:auto; +} + +.main-header-grid > .grid-left, +.main-body-grid > .grid-left { + width:260px; +} + +.main-header-grid > .grid-right, +.main-body-grid > .grid-right { + margin-left:270px; + position:relative; +} + +.main-header-grid > .grid-left { + float:left; +} + +.main-body-grid .side-navigation { + position:relative; + padding-bottom:120px; +} + +.main-body-grid .side-navigation.ng-hide { + display:block!important; +} + +.variables-matrix td { + vertical-align:top; + padding:5px; +} + +.type-hint { + display:inline-block; + background: gray; +} + +.variables-matrix .type-hint { + text-align:center; + min-width:60px; + margin:1px 5px; +} + +.type-hint + .type-hint { + margin-top:5px; +} + +.type-hint-expression { + background:purple; +} + +.type-hint-date { + background:pink; +} + +.type-hint-string { + background:#3a87ad; +} + +.type-hint-function { + background:green; +} + +.type-hint-object { + background:#999; +} + +.type-hint-array { + background:#F90;; +} + +.type-hint-boolean { + background:rgb(18, 131, 39); +} + +.type-hint-number { + background:rgb(189, 63, 66); +} + +.type-hint-regexp { + background: rgb(90, 84, 189); +} + +.type-hint-domelement { + background: rgb(95, 158, 160); +} + +.runnable-example-frame { + width:100%; + height:300px; + border: 1px solid #ddd; + border-radius:5px; +} + +.runnable-example-tabs { + margin-top:10px; + margin-bottom:20px; +} + +.tutorial-nav { + display:block; +} + +h1 + ul, h1 + ul > li, +h2 + ul, h2 + ul > li, +ul.tutorial-nav, ul.tutorial-nav > li, +.usage > ul, .usage > ul > li, +ul.methods, ul.methods > li, +ul.events, ul.events > li { + list-style:none; + padding:0; +} + +h2 { + border-top:1px solid #eee; + margin-top:30px; + padding-top:30px; +} + +h4 { + margin-top:20px; + padding-top:20px; +} + +.btn { + color:#428bca; + position: relative; + width: auto; + display: inline-block; + margin: 0 0 2px; + overflow: hidden; + border: 1px solid #e5e5e5; + -webkit-border-radius: 4px; + -moz-border-radius: 4px; + -ms-border-radius: 4px; + -o-border-radius: 4px; + border-radius: 4px; + font-family: "Open Sans"; + font-weight: 600; + height: auto; + background-image: -webkit-gradient(linear, 50% 0%, 50% 100%, color-stop(0%, #ffffff), color-stop(100%, #f2f2f2)); + background-image: -webkit-linear-gradient(#ffffff, #f2f2f2); + background-image: -moz-linear-gradient(#ffffff, #f2f2f2); + background-image: -o-linear-gradient(#ffffff, #f2f2f2); + background-image: linear-gradient(#ffffff, #f2f2f2); +} + +.btn + .btn { + margin-left:10px; +} + +.btn:hover, .btn:focus { + color: black!important; + border: 1px solid #ddd!important; + background: white!important; +} + +.view-source, .improve-docs { + position:relative; + z-index:100; +} + +.view-source { + margin-right:10px; +} + +.improve-docs { + float:right; +} + +.return-arguments, +.return-arguments th, +.return-arguments th + th, +.return-arguments td, +.return-arguments td + td { + border-radius:0; + border:0; +} + +.return-arguments td:first-child { + width:100px; +} + +ul.methods > li, +ul.events > li { + margin-bottom:40px; +} + +.definition-table td { + padding: 8px; + border: 1px solid #eee; + vertical-align: top; +} + +.table > tbody > tr.head > td, +.table > tbody > tr.head > th { + border-bottom: 2px solid #ddd; + padding-top: 50px; +} + +@media only screen and (min-width: 769px) and (max-width: 991px) { + .main-body-grid { + margin-top: 160px; + } + .main-body-grid > .grid-left { + top: 160px; + } +} + +@media only screen and (max-width : 768px) { + .picker, .picker select { + width:auto; + display:block; + margin-bottom:10px; + } + .docs-navbar-primary { + text-align:center; + } + .main-body-grid { + margin-top:0; + } + .main-header-grid > .grid-left, + .main-body-grid > .grid-left, + .main-header-grid > .grid-right, + .main-body-grid > .grid-right { + display:block; + float:none; + width:auto!important; + margin-left:0; + } + .main-body-grid > .grid-left, + .header-fixed, .footer { + position:static!important; + } + .main-body-grid > .grid-left { + background:#efefef; + margin-left:-1em; + margin-right:-1em; + padding:1em; + width:auto!important; + overflow:visible; + } + .main-header-grid > .grid-right, + .main-body-grid > .grid-right { + margin-left:0; + } + .main-body-grid .side-navigation { + display:block!important; + padding-bottom:50px; + } + .main-body-grid .side-navigation.ng-hide { + display:none!important; + } + .nav-index-group .nav-index-listing { + display:inline-block; + padding:3px 0; + } + .nav-index-group .nav-index-listing:not(.nav-index-section):after { + padding-right:5px; + margin-left:-3px; + content:", "; + } + .nav-index-group .nav-index-listing:last-child:after { + content:""; + display:inline-block; + } + .nav-index-group .nav-index-section { + display:block; + } + .toc-toggle { + margin-bottom:20px; + } + .toc-close { + position: absolute; + bottom: 5px; + left: 50%; + margin-left: -50%; + text-align: center; + padding: 5px; + background: #eee; + border-radius: 5px; + width: 100%; + border:1px solid #ddd; + box-shadow:0 0 10px #bbb; + } + .navbar-brand { + float:none; + text-align:center; + } + .search-results-container { + padding-bottom:60px; + text-align:left; + } + .search-results-group { + float:none!important; + display:block!important; + width:auto!important; + border:0!important; + padding:0!important; + } + .search-results-group .search-result { + display:inline-block!important; + padding:0 5px; + width:auto!important; + } + .search-results-group .search-result:after { + content:", "; + } + #wrapper { + padding-bottom:0px; + } +} + +iframe[name="example-anchoringExample"] { + height:400px; +} diff --git a/1.4.10/docs/css/prettify-theme.css b/1.4.10/docs/css/prettify-theme.css new file mode 100644 index 0000000000..7308e1ec71 --- /dev/null +++ b/1.4.10/docs/css/prettify-theme.css @@ -0,0 +1,142 @@ +/* GitHub Theme */ +.prettyprint { + background: white; + font-family: Menlo, 'Bitstream Vera Sans Mono', 'DejaVu Sans Mono', Monaco, Consolas, monospace; + font-size: 12px; + line-height: 1.5; +} + +.lang-text * { + color: #333333!important; +} + +.pln { + color: #333333; +} + +@media screen { + .str { + color: #dd1144; + } + + .kwd { + color: #333333; + } + + .com { + color: #999988; + } + + .typ { + color: #445588; + } + + .lit { + color: #445588; + } + + .pun { + color: #333333; + } + + .opn { + color: #333333; + } + + .clo { + color: #333333; + } + + .tag { + color: navy; + } + + .atn { + color: teal; + } + + .atv { + color: #dd1144; + } + + .dec { + color: #333333; + } + + .var { + color: teal; + } + + .fun { + color: #990000; + } +} +@media print, projection { + .str { + color: #006600; + } + + .kwd { + color: #006; + font-weight: bold; + } + + .com { + color: #600; + font-style: italic; + } + + .typ { + color: #404; + font-weight: bold; + } + + .lit { + color: #004444; + } + + .pun, .opn, .clo { + color: #444400; + } + + .tag { + color: #006; + font-weight: bold; + } + + .atn { + color: #440044; + } + + .atv { + color: #006600; + } +} +/* Specify class=linenums on a pre to get line numbering */ +ol.linenums { + margin-top: 0; + margin-bottom: 0; +} + +/* IE indents via margin-left */ +li.L0, +li.L1, +li.L2, +li.L3, +li.L4, +li.L5, +li.L6, +li.L7, +li.L8, +li.L9 { + /* */ +} + +/* Alternate shading for lines */ +li.L1, +li.L3, +li.L5, +li.L7, +li.L9 { + /* */ +} diff --git a/1.4.10/docs/css/prettify.css b/1.4.10/docs/css/prettify.css new file mode 100644 index 0000000000..354bbd544f --- /dev/null +++ b/1.4.10/docs/css/prettify.css @@ -0,0 +1,51 @@ +.pln { color: #000 } /* plain text */ + +@media screen { + .str { color: #080 } /* string content */ + .kwd { color: #008 } /* a keyword */ + .com { color: #800 } /* a comment */ + .typ { color: #606 } /* a type name */ + .lit { color: #066 } /* a literal value */ + /* punctuation, lisp open bracket, lisp close bracket */ + .pun, .opn, .clo { color: #660 } + .tag { color: #008 } /* a markup tag name */ + .atn { color: #606 } /* a markup attribute name */ + .atv { color: #080 } /* a markup attribute value */ + .dec, .var { color: #606 } /* a declaration; a variable name */ + .fun { color: red } /* a function name */ +} + +/* Use higher contrast and text-weight for printable form. */ +@media print, projection { + .str { color: #060 } + .kwd { color: #006; font-weight: bold } + .com { color: #600; font-style: italic } + .typ { color: #404; font-weight: bold } + .lit { color: #044 } + .pun, .opn, .clo { color: #440 } + .tag { color: #006; font-weight: bold } + .atn { color: #404 } + .atv { color: #060 } +} + +pre.prettyprint { + padding: 8px; + background-color: #f7f7f9; + border: 1px solid #e1e1e8; +} +pre.prettyprint.linenums { + -webkit-box-shadow: inset 40px 0 0 #fbfbfc, inset 41px 0 0 #ececf0; + -moz-box-shadow: inset 40px 0 0 #fbfbfc, inset 41px 0 0 #ececf0; + box-shadow: inset 40px 0 0 #fbfbfc, inset 41px 0 0 #ececf0; +} +ol.linenums { + margin: 0 0 0 33px; /* IE indents via margin-left */ +} +ol.linenums li { + padding-left: 12px; + font-size:12px; + color: #bebec5; + line-height: 18px; + text-shadow: 0 1px 0 #fff; + list-style-type:decimal!important; +} diff --git a/1.4.10/docs/examples/example-$filter/index-debug.html b/1.4.10/docs/examples/example-$filter/index-debug.html new file mode 100644 index 0000000000..217b757127 --- /dev/null +++ b/1.4.10/docs/examples/example-$filter/index-debug.html @@ -0,0 +1,20 @@ + + + + + Example - example-$filter-debug + + + + + + + + + +
+

{{ originalText }}

+

{{ filteredText }}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$filter/index-jquery.html b/1.4.10/docs/examples/example-$filter/index-jquery.html new file mode 100644 index 0000000000..b0baadd060 --- /dev/null +++ b/1.4.10/docs/examples/example-$filter/index-jquery.html @@ -0,0 +1,21 @@ + + + + + Example - example-$filter-jquery + + + + + + + + + + +
+

{{ originalText }}

+

{{ filteredText }}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$filter/index-production.html b/1.4.10/docs/examples/example-$filter/index-production.html new file mode 100644 index 0000000000..d1b232107b --- /dev/null +++ b/1.4.10/docs/examples/example-$filter/index-production.html @@ -0,0 +1,20 @@ + + + + + Example - example-$filter-production + + + + + + + + + +
+

{{ originalText }}

+

{{ filteredText }}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$filter/index.html b/1.4.10/docs/examples/example-$filter/index.html new file mode 100644 index 0000000000..5dd116a524 --- /dev/null +++ b/1.4.10/docs/examples/example-$filter/index.html @@ -0,0 +1,20 @@ + + + + + Example - example-$filter + + + + + + + + + +
+

{{ originalText }}

+

{{ filteredText }}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$filter/manifest.json b/1.4.10/docs/examples/example-$filter/manifest.json new file mode 100644 index 0000000000..0d2a67bb53 --- /dev/null +++ b/1.4.10/docs/examples/example-$filter/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-$filter", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$filter/script.js b/1.4.10/docs/examples/example-$filter/script.js new file mode 100644 index 0000000000..f06afee1ff --- /dev/null +++ b/1.4.10/docs/examples/example-$filter/script.js @@ -0,0 +1,8 @@ +(function(angular) { + 'use strict'; +angular.module('filterExample', []) +.controller('MainCtrl', function($scope, $filter) { + $scope.originalText = 'hello'; + $scope.filteredText = $filter('uppercase')($scope.originalText); +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$route-service/book.html b/1.4.10/docs/examples/example-$route-service/book.html new file mode 100644 index 0000000000..0ecdd75ce4 --- /dev/null +++ b/1.4.10/docs/examples/example-$route-service/book.html @@ -0,0 +1,2 @@ +controller: {{name}}
+Book Id: {{params.bookId}}
\ No newline at end of file diff --git a/1.4.10/docs/examples/example-$route-service/chapter.html b/1.4.10/docs/examples/example-$route-service/chapter.html new file mode 100644 index 0000000000..8775d07ddf --- /dev/null +++ b/1.4.10/docs/examples/example-$route-service/chapter.html @@ -0,0 +1,3 @@ +controller: {{name}}
+Book Id: {{params.bookId}}
+Chapter Id: {{params.chapterId}} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$route-service/index-debug.html b/1.4.10/docs/examples/example-$route-service/index-debug.html new file mode 100644 index 0000000000..2e1e4a638f --- /dev/null +++ b/1.4.10/docs/examples/example-$route-service/index-debug.html @@ -0,0 +1,37 @@ + + + + + Example - example-$route-service-debug + + + + + + + + + + +
+ Choose: + Moby | + Moby: Ch1 | + Gatsby | + Gatsby: Ch4 | + Scarlet Letter
+ +
+ +
+ +
$location.path() = {{$location.path()}}
+
$route.current.templateUrl = {{$route.current.templateUrl}}
+
$route.current.params = {{$route.current.params}}
+
$route.current.scope.name = {{$route.current.scope.name}}
+
$routeParams = {{$routeParams}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$route-service/index-jquery.html b/1.4.10/docs/examples/example-$route-service/index-jquery.html new file mode 100644 index 0000000000..770372e923 --- /dev/null +++ b/1.4.10/docs/examples/example-$route-service/index-jquery.html @@ -0,0 +1,38 @@ + + + + + Example - example-$route-service-jquery + + + + + + + + + + + +
+ Choose: + Moby | + Moby: Ch1 | + Gatsby | + Gatsby: Ch4 | + Scarlet Letter
+ +
+ +
+ +
$location.path() = {{$location.path()}}
+
$route.current.templateUrl = {{$route.current.templateUrl}}
+
$route.current.params = {{$route.current.params}}
+
$route.current.scope.name = {{$route.current.scope.name}}
+
$routeParams = {{$routeParams}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$route-service/index-production.html b/1.4.10/docs/examples/example-$route-service/index-production.html new file mode 100644 index 0000000000..dc99e95356 --- /dev/null +++ b/1.4.10/docs/examples/example-$route-service/index-production.html @@ -0,0 +1,37 @@ + + + + + Example - example-$route-service-production + + + + + + + + + + +
+ Choose: + Moby | + Moby: Ch1 | + Gatsby | + Gatsby: Ch4 | + Scarlet Letter
+ +
+ +
+ +
$location.path() = {{$location.path()}}
+
$route.current.templateUrl = {{$route.current.templateUrl}}
+
$route.current.params = {{$route.current.params}}
+
$route.current.scope.name = {{$route.current.scope.name}}
+
$routeParams = {{$routeParams}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$route-service/index.html b/1.4.10/docs/examples/example-$route-service/index.html new file mode 100644 index 0000000000..e2cc058109 --- /dev/null +++ b/1.4.10/docs/examples/example-$route-service/index.html @@ -0,0 +1,37 @@ + + + + + Example - example-$route-service + + + + + + + + + + +
+ Choose: + Moby | + Moby: Ch1 | + Gatsby | + Gatsby: Ch4 | + Scarlet Letter
+ +
+ +
+ +
$location.path() = {{$location.path()}}
+
$route.current.templateUrl = {{$route.current.templateUrl}}
+
$route.current.params = {{$route.current.params}}
+
$route.current.scope.name = {{$route.current.scope.name}}
+
$routeParams = {{$routeParams}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$route-service/manifest.json b/1.4.10/docs/examples/example-$route-service/manifest.json new file mode 100644 index 0000000000..0156fd11f9 --- /dev/null +++ b/1.4.10/docs/examples/example-$route-service/manifest.json @@ -0,0 +1,10 @@ +{ + "name": "example-$route-service", + "files": [ + "index-production.html", + "book.html", + "chapter.html", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$route-service/protractor.js b/1.4.10/docs/examples/example-$route-service/protractor.js new file mode 100644 index 0000000000..d996c20cf3 --- /dev/null +++ b/1.4.10/docs/examples/example-$route-service/protractor.js @@ -0,0 +1,13 @@ +it('should load and compile correct template', function() { + element(by.linkText('Moby: Ch1')).click(); + var content = element(by.css('[ng-view]')).getText(); + expect(content).toMatch(/controller\: ChapterController/); + expect(content).toMatch(/Book Id\: Moby/); + expect(content).toMatch(/Chapter Id\: 1/); + + element(by.partialLinkText('Scarlet')).click(); + + content = element(by.css('[ng-view]')).getText(); + expect(content).toMatch(/controller\: BookController/); + expect(content).toMatch(/Book Id\: Scarlet/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-$route-service/script.js b/1.4.10/docs/examples/example-$route-service/script.js new file mode 100644 index 0000000000..73633960c2 --- /dev/null +++ b/1.4.10/docs/examples/example-$route-service/script.js @@ -0,0 +1,43 @@ +(function(angular) { + 'use strict'; +angular.module('ngRouteExample', ['ngRoute']) + + .controller('MainController', function($scope, $route, $routeParams, $location) { + $scope.$route = $route; + $scope.$location = $location; + $scope.$routeParams = $routeParams; + }) + + .controller('BookController', function($scope, $routeParams) { + $scope.name = "BookController"; + $scope.params = $routeParams; + }) + + .controller('ChapterController', function($scope, $routeParams) { + $scope.name = "ChapterController"; + $scope.params = $routeParams; + }) + +.config(function($routeProvider, $locationProvider) { + $routeProvider + .when('/Book/:bookId', { + templateUrl: 'book.html', + controller: 'BookController', + resolve: { + // I will cause a 1 second delay + delay: function($q, $timeout) { + var delay = $q.defer(); + $timeout(delay.resolve, 1000); + return delay.promise; + } + } + }) + .when('/Book/:bookId/ch/:chapterId', { + templateUrl: 'chapter.html', + controller: 'ChapterController' + }); + + // configure html5 to get links working on jsfiddle + $locationProvider.html5Mode(true); +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-NgModelController/index-debug.html b/1.4.10/docs/examples/example-NgModelController/index-debug.html new file mode 100644 index 0000000000..26137f3e82 --- /dev/null +++ b/1.4.10/docs/examples/example-NgModelController/index-debug.html @@ -0,0 +1,27 @@ + + + + + Example - example-NgModelController-debug + + + + + + + + + + + +
+
Change me!
+ Required! +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-NgModelController/index-jquery.html b/1.4.10/docs/examples/example-NgModelController/index-jquery.html new file mode 100644 index 0000000000..6b89a3fdcc --- /dev/null +++ b/1.4.10/docs/examples/example-NgModelController/index-jquery.html @@ -0,0 +1,28 @@ + + + + + Example - example-NgModelController-jquery + + + + + + + + + + + + +
+
Change me!
+ Required! +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-NgModelController/index-production.html b/1.4.10/docs/examples/example-NgModelController/index-production.html new file mode 100644 index 0000000000..64239e939a --- /dev/null +++ b/1.4.10/docs/examples/example-NgModelController/index-production.html @@ -0,0 +1,27 @@ + + + + + Example - example-NgModelController-production + + + + + + + + + + + +
+
Change me!
+ Required! +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-NgModelController/index.html b/1.4.10/docs/examples/example-NgModelController/index.html new file mode 100644 index 0000000000..908e61639c --- /dev/null +++ b/1.4.10/docs/examples/example-NgModelController/index.html @@ -0,0 +1,27 @@ + + + + + Example - example-NgModelController + + + + + + + + + + + +
+
Change me!
+ Required! +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-NgModelController/manifest.json b/1.4.10/docs/examples/example-NgModelController/manifest.json new file mode 100644 index 0000000000..77bb9769d8 --- /dev/null +++ b/1.4.10/docs/examples/example-NgModelController/manifest.json @@ -0,0 +1,9 @@ +{ + "name": "example-NgModelController", + "files": [ + "index-production.html", + "style.css", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-NgModelController/protractor.js b/1.4.10/docs/examples/example-NgModelController/protractor.js new file mode 100644 index 0000000000..a524f929cc --- /dev/null +++ b/1.4.10/docs/examples/example-NgModelController/protractor.js @@ -0,0 +1,16 @@ +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/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-NgModelController/script.js b/1.4.10/docs/examples/example-NgModelController/script.js new file mode 100644 index 0000000000..415bd0fafe --- /dev/null +++ b/1.4.10/docs/examples/example-NgModelController/script.js @@ -0,0 +1,35 @@ +(function(angular) { + 'use strict'; +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
behind + // If strip-br attribute is provided then we strip this out + if ( attrs.stripBr && html == '
' ) { + html = ''; + } + ngModel.$setViewValue(html); + } + } + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-NgModelController/style.css b/1.4.10/docs/examples/example-NgModelController/style.css new file mode 100644 index 0000000000..a96ba9e6e9 --- /dev/null +++ b/1.4.10/docs/examples/example-NgModelController/style.css @@ -0,0 +1,9 @@ +[contenteditable] { + border: 1px solid black; + background-color: white; + min-height: 20px; +} + +.ng-invalid { + border: 1px solid red; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-anchoringExample/animations.css b/1.4.10/docs/examples/example-anchoringExample/animations.css new file mode 100644 index 0000000000..5f9cf01161 --- /dev/null +++ b/1.4.10/docs/examples/example-anchoringExample/animations.css @@ -0,0 +1,35 @@ +.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; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-anchoringExample/home.html b/1.4.10/docs/examples/example-anchoringExample/home.html new file mode 100644 index 0000000000..4d35076389 --- /dev/null +++ b/1.4.10/docs/examples/example-anchoringExample/home.html @@ -0,0 +1,8 @@ +

Welcome to the home page

+

Please click on an element

+ + {{ record.title }} + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-anchoringExample/index-debug.html b/1.4.10/docs/examples/example-anchoringExample/index-debug.html new file mode 100644 index 0000000000..e423ff06ed --- /dev/null +++ b/1.4.10/docs/examples/example-anchoringExample/index-debug.html @@ -0,0 +1,24 @@ + + + + + Example - example-anchoringExample-debug + + + + + + + + + + + + + Home +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-anchoringExample/index-jquery.html b/1.4.10/docs/examples/example-anchoringExample/index-jquery.html new file mode 100644 index 0000000000..5935c1fe1c --- /dev/null +++ b/1.4.10/docs/examples/example-anchoringExample/index-jquery.html @@ -0,0 +1,25 @@ + + + + + Example - example-anchoringExample-jquery + + + + + + + + + + + + + + Home +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-anchoringExample/index-production.html b/1.4.10/docs/examples/example-anchoringExample/index-production.html new file mode 100644 index 0000000000..647a19b202 --- /dev/null +++ b/1.4.10/docs/examples/example-anchoringExample/index-production.html @@ -0,0 +1,24 @@ + + + + + Example - example-anchoringExample-production + + + + + + + + + + + + + Home +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-anchoringExample/index.html b/1.4.10/docs/examples/example-anchoringExample/index.html new file mode 100644 index 0000000000..c7847dc9e7 --- /dev/null +++ b/1.4.10/docs/examples/example-anchoringExample/index.html @@ -0,0 +1,24 @@ + + + + + Example - example-anchoringExample + + + + + + + + + + + + + Home +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-anchoringExample/manifest.json b/1.4.10/docs/examples/example-anchoringExample/manifest.json new file mode 100644 index 0000000000..1e8ba5e5b5 --- /dev/null +++ b/1.4.10/docs/examples/example-anchoringExample/manifest.json @@ -0,0 +1,10 @@ +{ + "name": "example-anchoringExample", + "files": [ + "index-production.html", + "script.js", + "home.html", + "profile.html", + "animations.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-anchoringExample/profile.html b/1.4.10/docs/examples/example-anchoringExample/profile.html new file mode 100644 index 0000000000..446b1e0465 --- /dev/null +++ b/1.4.10/docs/examples/example-anchoringExample/profile.html @@ -0,0 +1,3 @@ +
+ {{ profile.title }} +
\ No newline at end of file diff --git a/1.4.10/docs/examples/example-anchoringExample/script.js b/1.4.10/docs/examples/example-anchoringExample/script.js new file mode 100644 index 0000000000..5e5d35fac5 --- /dev/null +++ b/1.4.10/docs/examples/example-anchoringExample/script.js @@ -0,0 +1,38 @@ +(function(angular) { + 'use strict'; +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; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-checkbox-input-directive/index-debug.html b/1.4.10/docs/examples/example-checkbox-input-directive/index-debug.html new file mode 100644 index 0000000000..40805be027 --- /dev/null +++ b/1.4.10/docs/examples/example-checkbox-input-directive/index-debug.html @@ -0,0 +1,35 @@ + + + + + Example - example-checkbox-input-directive-debug + + + + + + + + + +
+
+
+ value1 = {{checkboxModel.value1}}
+ value2 = {{checkboxModel.value2}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-checkbox-input-directive/index-jquery.html b/1.4.10/docs/examples/example-checkbox-input-directive/index-jquery.html new file mode 100644 index 0000000000..ef9cd0a634 --- /dev/null +++ b/1.4.10/docs/examples/example-checkbox-input-directive/index-jquery.html @@ -0,0 +1,36 @@ + + + + + Example - example-checkbox-input-directive-jquery + + + + + + + + + + +
+
+
+ value1 = {{checkboxModel.value1}}
+ value2 = {{checkboxModel.value2}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-checkbox-input-directive/index-production.html b/1.4.10/docs/examples/example-checkbox-input-directive/index-production.html new file mode 100644 index 0000000000..6cffb5a40a --- /dev/null +++ b/1.4.10/docs/examples/example-checkbox-input-directive/index-production.html @@ -0,0 +1,35 @@ + + + + + Example - example-checkbox-input-directive-production + + + + + + + + + +
+
+
+ value1 = {{checkboxModel.value1}}
+ value2 = {{checkboxModel.value2}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-checkbox-input-directive/index.html b/1.4.10/docs/examples/example-checkbox-input-directive/index.html new file mode 100644 index 0000000000..df0dda8b0a --- /dev/null +++ b/1.4.10/docs/examples/example-checkbox-input-directive/index.html @@ -0,0 +1,35 @@ + + + + + Example - example-checkbox-input-directive + + + + + + + + + +
+
+
+ value1 = {{checkboxModel.value1}}
+ value2 = {{checkboxModel.value2}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-checkbox-input-directive/manifest.json b/1.4.10/docs/examples/example-checkbox-input-directive/manifest.json new file mode 100644 index 0000000000..7c0bf82470 --- /dev/null +++ b/1.4.10/docs/examples/example-checkbox-input-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-checkbox-input-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-checkbox-input-directive/protractor.js b/1.4.10/docs/examples/example-checkbox-input-directive/protractor.js new file mode 100644 index 0000000000..33614faa90 --- /dev/null +++ b/1.4.10/docs/examples/example-checkbox-input-directive/protractor.js @@ -0,0 +1,13 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-custom-interpolation-markup/index-debug.html b/1.4.10/docs/examples/example-custom-interpolation-markup/index-debug.html new file mode 100644 index 0000000000..6364188fd2 --- /dev/null +++ b/1.4.10/docs/examples/example-custom-interpolation-markup/index-debug.html @@ -0,0 +1,31 @@ + + + + + Example - example-custom-interpolation-markup-debug + + + + + + + + + +
+ //demo.label// +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-custom-interpolation-markup/index-jquery.html b/1.4.10/docs/examples/example-custom-interpolation-markup/index-jquery.html new file mode 100644 index 0000000000..04d23c9a76 --- /dev/null +++ b/1.4.10/docs/examples/example-custom-interpolation-markup/index-jquery.html @@ -0,0 +1,32 @@ + + + + + Example - example-custom-interpolation-markup-jquery + + + + + + + + + + +
+ //demo.label// +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-custom-interpolation-markup/index-production.html b/1.4.10/docs/examples/example-custom-interpolation-markup/index-production.html new file mode 100644 index 0000000000..d96341ee0a --- /dev/null +++ b/1.4.10/docs/examples/example-custom-interpolation-markup/index-production.html @@ -0,0 +1,31 @@ + + + + + Example - example-custom-interpolation-markup-production + + + + + + + + + +
+ //demo.label// +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-custom-interpolation-markup/index.html b/1.4.10/docs/examples/example-custom-interpolation-markup/index.html new file mode 100644 index 0000000000..b90a82fc52 --- /dev/null +++ b/1.4.10/docs/examples/example-custom-interpolation-markup/index.html @@ -0,0 +1,31 @@ + + + + + Example - example-custom-interpolation-markup + + + + + + + + + +
+ //demo.label// +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-custom-interpolation-markup/manifest.json b/1.4.10/docs/examples/example-custom-interpolation-markup/manifest.json new file mode 100644 index 0000000000..9ad01dbea1 --- /dev/null +++ b/1.4.10/docs/examples/example-custom-interpolation-markup/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-custom-interpolation-markup", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-custom-interpolation-markup/protractor.js b/1.4.10/docs/examples/example-custom-interpolation-markup/protractor.js new file mode 100644 index 0000000000..088c6693da --- /dev/null +++ b/1.4.10/docs/examples/example-custom-interpolation-markup/protractor.js @@ -0,0 +1,3 @@ +it('should interpolate binding with custom symbols', function() { + expect(element(by.binding('demo.label')).getText()).toBe('This binding is brought you by // interpolation symbols.'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-date-input-directive/index-debug.html b/1.4.10/docs/examples/example-date-input-directive/index-debug.html new file mode 100644 index 0000000000..647f4539e0 --- /dev/null +++ b/1.4.10/docs/examples/example-date-input-directive/index-debug.html @@ -0,0 +1,39 @@ + + + + + Example - example-date-input-directive-debug + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "yyyy-MM-dd"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-date-input-directive/index-jquery.html b/1.4.10/docs/examples/example-date-input-directive/index-jquery.html new file mode 100644 index 0000000000..a697074189 --- /dev/null +++ b/1.4.10/docs/examples/example-date-input-directive/index-jquery.html @@ -0,0 +1,40 @@ + + + + + Example - example-date-input-directive-jquery + + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "yyyy-MM-dd"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-date-input-directive/index-production.html b/1.4.10/docs/examples/example-date-input-directive/index-production.html new file mode 100644 index 0000000000..7f008f6b5d --- /dev/null +++ b/1.4.10/docs/examples/example-date-input-directive/index-production.html @@ -0,0 +1,39 @@ + + + + + Example - example-date-input-directive-production + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "yyyy-MM-dd"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-date-input-directive/index.html b/1.4.10/docs/examples/example-date-input-directive/index.html new file mode 100644 index 0000000000..0aa90cd967 --- /dev/null +++ b/1.4.10/docs/examples/example-date-input-directive/index.html @@ -0,0 +1,39 @@ + + + + + Example - example-date-input-directive + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "yyyy-MM-dd"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-date-input-directive/manifest.json b/1.4.10/docs/examples/example-date-input-directive/manifest.json new file mode 100644 index 0000000000..ac13615c08 --- /dev/null +++ b/1.4.10/docs/examples/example-date-input-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-date-input-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-date-input-directive/protractor.js b/1.4.10/docs/examples/example-date-input-directive/protractor.js new file mode 100644 index 0000000000..9934c68b10 --- /dev/null +++ b/1.4.10/docs/examples/example-date-input-directive/protractor.js @@ -0,0 +1,31 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-datetimelocal-input-directive/index-debug.html b/1.4.10/docs/examples/example-datetimelocal-input-directive/index-debug.html new file mode 100644 index 0000000000..aa03bdadf8 --- /dev/null +++ b/1.4.10/docs/examples/example-datetimelocal-input-directive/index-debug.html @@ -0,0 +1,39 @@ + + + + + Example - example-datetimelocal-input-directive-debug + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "yyyy-MM-ddTHH:mm:ss"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-datetimelocal-input-directive/index-jquery.html b/1.4.10/docs/examples/example-datetimelocal-input-directive/index-jquery.html new file mode 100644 index 0000000000..2d87cbd083 --- /dev/null +++ b/1.4.10/docs/examples/example-datetimelocal-input-directive/index-jquery.html @@ -0,0 +1,40 @@ + + + + + Example - example-datetimelocal-input-directive-jquery + + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "yyyy-MM-ddTHH:mm:ss"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-datetimelocal-input-directive/index-production.html b/1.4.10/docs/examples/example-datetimelocal-input-directive/index-production.html new file mode 100644 index 0000000000..92f5b8f345 --- /dev/null +++ b/1.4.10/docs/examples/example-datetimelocal-input-directive/index-production.html @@ -0,0 +1,39 @@ + + + + + Example - example-datetimelocal-input-directive-production + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "yyyy-MM-ddTHH:mm:ss"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-datetimelocal-input-directive/index.html b/1.4.10/docs/examples/example-datetimelocal-input-directive/index.html new file mode 100644 index 0000000000..fb7ff39fcf --- /dev/null +++ b/1.4.10/docs/examples/example-datetimelocal-input-directive/index.html @@ -0,0 +1,39 @@ + + + + + Example - example-datetimelocal-input-directive + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "yyyy-MM-ddTHH:mm:ss"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-datetimelocal-input-directive/manifest.json b/1.4.10/docs/examples/example-datetimelocal-input-directive/manifest.json new file mode 100644 index 0000000000..34b85a6353 --- /dev/null +++ b/1.4.10/docs/examples/example-datetimelocal-input-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-datetimelocal-input-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-datetimelocal-input-directive/protractor.js b/1.4.10/docs/examples/example-datetimelocal-input-directive/protractor.js new file mode 100644 index 0000000000..14c86e8514 --- /dev/null +++ b/1.4.10/docs/examples/example-datetimelocal-input-directive/protractor.js @@ -0,0 +1,31 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-email-input-directive/index-debug.html b/1.4.10/docs/examples/example-email-input-directive/index-debug.html new file mode 100644 index 0000000000..f2e80c2ef8 --- /dev/null +++ b/1.4.10/docs/examples/example-email-input-directive/index-debug.html @@ -0,0 +1,40 @@ + + + + + Example - example-email-input-directive-debug + + + + + + + + + +
+ +
+ + Required! + + Not valid email! +
+ text = {{email.text}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+ myForm.$error.email = {{!!myForm.$error.email}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-email-input-directive/index-jquery.html b/1.4.10/docs/examples/example-email-input-directive/index-jquery.html new file mode 100644 index 0000000000..019f4013e7 --- /dev/null +++ b/1.4.10/docs/examples/example-email-input-directive/index-jquery.html @@ -0,0 +1,41 @@ + + + + + Example - example-email-input-directive-jquery + + + + + + + + + + +
+ +
+ + Required! + + Not valid email! +
+ text = {{email.text}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+ myForm.$error.email = {{!!myForm.$error.email}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-email-input-directive/index-production.html b/1.4.10/docs/examples/example-email-input-directive/index-production.html new file mode 100644 index 0000000000..2286ecb232 --- /dev/null +++ b/1.4.10/docs/examples/example-email-input-directive/index-production.html @@ -0,0 +1,40 @@ + + + + + Example - example-email-input-directive-production + + + + + + + + + +
+ +
+ + Required! + + Not valid email! +
+ text = {{email.text}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+ myForm.$error.email = {{!!myForm.$error.email}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-email-input-directive/index.html b/1.4.10/docs/examples/example-email-input-directive/index.html new file mode 100644 index 0000000000..b11ba39b63 --- /dev/null +++ b/1.4.10/docs/examples/example-email-input-directive/index.html @@ -0,0 +1,40 @@ + + + + + Example - example-email-input-directive + + + + + + + + + +
+ +
+ + Required! + + Not valid email! +
+ text = {{email.text}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+ myForm.$error.email = {{!!myForm.$error.email}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-email-input-directive/manifest.json b/1.4.10/docs/examples/example-email-input-directive/manifest.json new file mode 100644 index 0000000000..a7348f2b92 --- /dev/null +++ b/1.4.10/docs/examples/example-email-input-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-email-input-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-email-input-directive/protractor.js b/1.4.10/docs/examples/example-email-input-directive/protractor.js new file mode 100644 index 0000000000..156e285a1b --- /dev/null +++ b/1.4.10/docs/examples/example-email-input-directive/protractor.js @@ -0,0 +1,22 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-error-$rootScope-inprog/app.js b/1.4.10/docs/examples/example-error-$rootScope-inprog/app.js new file mode 100644 index 0000000000..a933541f16 --- /dev/null +++ b/1.4.10/docs/examples/example-error-$rootScope-inprog/app.js @@ -0,0 +1,10 @@ +(function(angular) { + 'use strict'; +angular.module('app', []).directive('setFocusIf', function() { + return function link($scope, $element, $attr) { + $scope.$watch($attr.setFocusIf, function(value) { + if ( value ) { $element[0].focus(); } + }); + }; +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-error-$rootScope-inprog/index-debug.html b/1.4.10/docs/examples/example-error-$rootScope-inprog/index-debug.html new file mode 100644 index 0000000000..5396025f2d --- /dev/null +++ b/1.4.10/docs/examples/example-error-$rootScope-inprog/index-debug.html @@ -0,0 +1,18 @@ + + + + + Example - example-error-$rootScope-inprog-debug + + + + + + + + + + + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-error-$rootScope-inprog/index-jquery.html b/1.4.10/docs/examples/example-error-$rootScope-inprog/index-jquery.html new file mode 100644 index 0000000000..92a28230d7 --- /dev/null +++ b/1.4.10/docs/examples/example-error-$rootScope-inprog/index-jquery.html @@ -0,0 +1,19 @@ + + + + + Example - example-error-$rootScope-inprog-jquery + + + + + + + + + + + + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-error-$rootScope-inprog/index-production.html b/1.4.10/docs/examples/example-error-$rootScope-inprog/index-production.html new file mode 100644 index 0000000000..3b7152fe8e --- /dev/null +++ b/1.4.10/docs/examples/example-error-$rootScope-inprog/index-production.html @@ -0,0 +1,18 @@ + + + + + Example - example-error-$rootScope-inprog-production + + + + + + + + + + + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-error-$rootScope-inprog/index.html b/1.4.10/docs/examples/example-error-$rootScope-inprog/index.html new file mode 100644 index 0000000000..5e7229d8df --- /dev/null +++ b/1.4.10/docs/examples/example-error-$rootScope-inprog/index.html @@ -0,0 +1,18 @@ + + + + + Example - example-error-$rootScope-inprog + + + + + + + + + + + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-error-$rootScope-inprog/manifest.json b/1.4.10/docs/examples/example-error-$rootScope-inprog/manifest.json new file mode 100644 index 0000000000..1c988f9fab --- /dev/null +++ b/1.4.10/docs/examples/example-error-$rootScope-inprog/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-error-$rootScope-inprog", + "files": [ + "index-production.html", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example.csp/index-debug.html b/1.4.10/docs/examples/example-example.csp/index-debug.html new file mode 100644 index 0000000000..0d2af2075b --- /dev/null +++ b/1.4.10/docs/examples/example-example.csp/index-debug.html @@ -0,0 +1,31 @@ + + + + + Example - example-example.csp-debug + + + + + + + + + +
+
+ + + {{ctrl.counter}} + +
+ +
+ + + {{ctrl.evilError}} + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example.csp/index-jquery.html b/1.4.10/docs/examples/example-example.csp/index-jquery.html new file mode 100644 index 0000000000..d4cbed8cb3 --- /dev/null +++ b/1.4.10/docs/examples/example-example.csp/index-jquery.html @@ -0,0 +1,32 @@ + + + + + Example - example-example.csp-jquery + + + + + + + + + + +
+
+ + + {{ctrl.counter}} + +
+ +
+ + + {{ctrl.evilError}} + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example.csp/index-production.html b/1.4.10/docs/examples/example-example.csp/index-production.html new file mode 100644 index 0000000000..ae0307bf5b --- /dev/null +++ b/1.4.10/docs/examples/example-example.csp/index-production.html @@ -0,0 +1,31 @@ + + + + + Example - example-example.csp-production + + + + + + + + + +
+
+ + + {{ctrl.counter}} + +
+ +
+ + + {{ctrl.evilError}} + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example.csp/index.html b/1.4.10/docs/examples/example-example.csp/index.html new file mode 100644 index 0000000000..b9da98b298 --- /dev/null +++ b/1.4.10/docs/examples/example-example.csp/index.html @@ -0,0 +1,31 @@ + + + + + Example - example-example.csp + + + + + + + + + +
+
+ + + {{ctrl.counter}} + +
+ +
+ + + {{ctrl.evilError}} + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example.csp/manifest.json b/1.4.10/docs/examples/example-example.csp/manifest.json new file mode 100644 index 0000000000..7a6feb4359 --- /dev/null +++ b/1.4.10/docs/examples/example-example.csp/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example.csp", + "files": [ + "index-production.html", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example.csp/protractor.js b/1.4.10/docs/examples/example-example.csp/protractor.js new file mode 100644 index 0000000000..1346bb2958 --- /dev/null +++ b/1.4.10/docs/examples/example-example.csp/protractor.js @@ -0,0 +1,77 @@ +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/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example.csp/script.js b/1.4.10/docs/examples/example-example.csp/script.js new file mode 100644 index 0000000000..67db803bed --- /dev/null +++ b/1.4.10/docs/examples/example-example.csp/script.js @@ -0,0 +1,18 @@ +(function(angular) { + 'use strict'; +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; + } + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example/app.js b/1.4.10/docs/examples/example-example/app.js new file mode 100644 index 0000000000..a1dd224781 --- /dev/null +++ b/1.4.10/docs/examples/example-example/app.js @@ -0,0 +1,24 @@ +(function(angular) { + 'use strict'; +angular.module('numfmt-error-module', []) + +.run(function($rootScope) { + $rootScope.typeOf = function(value) { + return typeof value; + }; +}) + +.directive('stringToNumber', function() { + return { + require: 'ngModel', + link: function(scope, element, attrs, ngModel) { + ngModel.$parsers.push(function(value) { + return '' + value; + }); + ngModel.$formatters.push(function(value) { + return parseFloat(value, 10); + }); + } + }; +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example/index-debug.html b/1.4.10/docs/examples/example-example/index-debug.html new file mode 100644 index 0000000000..7750142bd9 --- /dev/null +++ b/1.4.10/docs/examples/example-example/index-debug.html @@ -0,0 +1,23 @@ + + + + + Example - example-example-debug + + + + + + + + + + + + + +
+ {{ x }} : {{ typeOf(x) }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example/index-jquery.html b/1.4.10/docs/examples/example-example/index-jquery.html new file mode 100644 index 0000000000..cc17c45712 --- /dev/null +++ b/1.4.10/docs/examples/example-example/index-jquery.html @@ -0,0 +1,24 @@ + + + + + Example - example-example-jquery + + + + + + + + + + + + + + +
+ {{ x }} : {{ typeOf(x) }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example/index-production.html b/1.4.10/docs/examples/example-example/index-production.html new file mode 100644 index 0000000000..2c9d357ef8 --- /dev/null +++ b/1.4.10/docs/examples/example-example/index-production.html @@ -0,0 +1,23 @@ + + + + + Example - example-example-production + + + + + + + + + + + + + +
+ {{ x }} : {{ typeOf(x) }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example/index.html b/1.4.10/docs/examples/example-example/index.html new file mode 100644 index 0000000000..78807d62f6 --- /dev/null +++ b/1.4.10/docs/examples/example-example/index.html @@ -0,0 +1,23 @@ + + + + + Example - example-example + + + + + + + + + + + + + +
+ {{ x }} : {{ typeOf(x) }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example/manifest.json b/1.4.10/docs/examples/example-example/manifest.json new file mode 100644 index 0000000000..6f0d5079e6 --- /dev/null +++ b/1.4.10/docs/examples/example-example/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example", + "files": [ + "index-production.html", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example1/index-debug.html b/1.4.10/docs/examples/example-example1/index-debug.html new file mode 100644 index 0000000000..e1d1057123 --- /dev/null +++ b/1.4.10/docs/examples/example-example1/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example1-debug + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example1/index-jquery.html b/1.4.10/docs/examples/example-example1/index-jquery.html new file mode 100644 index 0000000000..f29ecd5c0e --- /dev/null +++ b/1.4.10/docs/examples/example-example1/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example1-jquery + + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example1/index-production.html b/1.4.10/docs/examples/example-example1/index-production.html new file mode 100644 index 0000000000..50a4f84a76 --- /dev/null +++ b/1.4.10/docs/examples/example-example1/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example1-production + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example1/index.html b/1.4.10/docs/examples/example-example1/index.html new file mode 100644 index 0000000000..c7c13584d4 --- /dev/null +++ b/1.4.10/docs/examples/example-example1/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example1 + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example1/manifest.json b/1.4.10/docs/examples/example-example1/manifest.json new file mode 100644 index 0000000000..add5f56bdc --- /dev/null +++ b/1.4.10/docs/examples/example-example1/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example1", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example1/script.js b/1.4.10/docs/examples/example-example1/script.js new file mode 100644 index 0000000000..c3beb90bfb --- /dev/null +++ b/1.4.10/docs/examples/example-example1/script.js @@ -0,0 +1,9 @@ +(function(angular) { + 'use strict'; +angular.module('locationExample', []) + .controller('LocationController', ['$scope', '$location', function($scope, $location) { + $scope.locationPath = function (newLocation) { + return $location.path(newLocation); + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example10/index-debug.html b/1.4.10/docs/examples/example-example10/index-debug.html new file mode 100644 index 0000000000..a5506f66ee --- /dev/null +++ b/1.4.10/docs/examples/example-example10/index-debug.html @@ -0,0 +1,24 @@ + + + + + Example - example-example10-debug + + + + + + + + + +
+ Hello
+
+
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example10/index-jquery.html b/1.4.10/docs/examples/example-example10/index-jquery.html new file mode 100644 index 0000000000..fb8c9479dc --- /dev/null +++ b/1.4.10/docs/examples/example-example10/index-jquery.html @@ -0,0 +1,25 @@ + + + + + Example - example-example10-jquery + + + + + + + + + + +
+ Hello
+
+
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example10/index-production.html b/1.4.10/docs/examples/example-example10/index-production.html new file mode 100644 index 0000000000..b5310f3f42 --- /dev/null +++ b/1.4.10/docs/examples/example-example10/index-production.html @@ -0,0 +1,24 @@ + + + + + Example - example-example10-production + + + + + + + + + +
+ Hello
+
+
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example10/index.html b/1.4.10/docs/examples/example-example10/index.html new file mode 100644 index 0000000000..2f05b3d44d --- /dev/null +++ b/1.4.10/docs/examples/example-example10/index.html @@ -0,0 +1,24 @@ + + + + + Example - example-example10 + + + + + + + + + +
+ Hello
+
+
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example10/manifest.json b/1.4.10/docs/examples/example-example10/manifest.json new file mode 100644 index 0000000000..9da50028a1 --- /dev/null +++ b/1.4.10/docs/examples/example-example10/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example10", + "files": [ + "index-production.html", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example10/protractor.js b/1.4.10/docs/examples/example-example10/protractor.js new file mode 100644 index 0000000000..33dd20e160 --- /dev/null +++ b/1.4.10/docs/examples/example-example10/protractor.js @@ -0,0 +1,9 @@ +it('should show off bindings', function() { + var containerElm = element(by.css('div[ng-controller="Controller"]')); + var nameBindings = containerElm.all(by.binding('name')); + + expect(nameBindings.count()).toBe(5); + nameBindings.each(function(elem) { + expect(elem.getText()).toEqual('Max Karl Ernst Ludwig Planck (April 23, 1858 – October 4, 1947)'); + }); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example10/script.js b/1.4.10/docs/examples/example-example10/script.js new file mode 100644 index 0000000000..5fedf009b8 --- /dev/null +++ b/1.4.10/docs/examples/example-example10/script.js @@ -0,0 +1,7 @@ +(function(angular) { + 'use strict'; +angular.module('docsBindExample', []) + .controller('Controller', ['$scope', function($scope) { + $scope.name = 'Max Karl Ernst Ludwig Planck (April 23, 1858 – October 4, 1947)'; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example100/index-debug.html b/1.4.10/docs/examples/example-example100/index-debug.html new file mode 100644 index 0000000000..59bbf013a4 --- /dev/null +++ b/1.4.10/docs/examples/example-example100/index-debug.html @@ -0,0 +1,27 @@ + + + + + Example - example-example100-debug + + + + + + + + + +
+
+ Default formatting: {{val | number}}
+ No fractions: {{val | number:0}}
+ Negative number: {{-val | number:4}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example100/index-jquery.html b/1.4.10/docs/examples/example-example100/index-jquery.html new file mode 100644 index 0000000000..f13b90f967 --- /dev/null +++ b/1.4.10/docs/examples/example-example100/index-jquery.html @@ -0,0 +1,28 @@ + + + + + Example - example-example100-jquery + + + + + + + + + + +
+
+ Default formatting: {{val | number}}
+ No fractions: {{val | number:0}}
+ Negative number: {{-val | number:4}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example100/index-production.html b/1.4.10/docs/examples/example-example100/index-production.html new file mode 100644 index 0000000000..205581ae29 --- /dev/null +++ b/1.4.10/docs/examples/example-example100/index-production.html @@ -0,0 +1,27 @@ + + + + + Example - example-example100-production + + + + + + + + + +
+
+ Default formatting: {{val | number}}
+ No fractions: {{val | number:0}}
+ Negative number: {{-val | number:4}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example100/index.html b/1.4.10/docs/examples/example-example100/index.html new file mode 100644 index 0000000000..d20012b97a --- /dev/null +++ b/1.4.10/docs/examples/example-example100/index.html @@ -0,0 +1,27 @@ + + + + + Example - example-example100 + + + + + + + + + +
+
+ Default formatting: {{val | number}}
+ No fractions: {{val | number:0}}
+ Negative number: {{-val | number:4}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example100/manifest.json b/1.4.10/docs/examples/example-example100/manifest.json new file mode 100644 index 0000000000..5f6b2326cd --- /dev/null +++ b/1.4.10/docs/examples/example-example100/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example100", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example100/protractor.js b/1.4.10/docs/examples/example-example100/protractor.js new file mode 100644 index 0000000000..b6beaa4cd4 --- /dev/null +++ b/1.4.10/docs/examples/example-example100/protractor.js @@ -0,0 +1,13 @@ + 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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example101/index-debug.html b/1.4.10/docs/examples/example-example101/index-debug.html new file mode 100644 index 0000000000..a84dadc5c5 --- /dev/null +++ b/1.4.10/docs/examples/example-example101/index-debug.html @@ -0,0 +1,23 @@ + + + + + Example - example-example101-debug + + + + + + + + + {{1288323623006 | date:'medium'}}: + {{1288323623006 | date:'medium'}}
+{{1288323623006 | date:'yyyy-MM-dd HH:mm:ss Z'}}: + {{1288323623006 | date:'yyyy-MM-dd HH:mm:ss Z'}}
+{{1288323623006 | date:'MM/dd/yyyy @ h:mma'}}: + {{'1288323623006' | date:'MM/dd/yyyy @ h:mma'}}
+{{1288323623006 | date:"MM/dd/yyyy 'at' h:mma"}}: + {{'1288323623006' | date:"MM/dd/yyyy 'at' h:mma"}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example101/index-jquery.html b/1.4.10/docs/examples/example-example101/index-jquery.html new file mode 100644 index 0000000000..649f3953aa --- /dev/null +++ b/1.4.10/docs/examples/example-example101/index-jquery.html @@ -0,0 +1,24 @@ + + + + + Example - example-example101-jquery + + + + + + + + + + {{1288323623006 | date:'medium'}}: + {{1288323623006 | date:'medium'}}
+{{1288323623006 | date:'yyyy-MM-dd HH:mm:ss Z'}}: + {{1288323623006 | date:'yyyy-MM-dd HH:mm:ss Z'}}
+{{1288323623006 | date:'MM/dd/yyyy @ h:mma'}}: + {{'1288323623006' | date:'MM/dd/yyyy @ h:mma'}}
+{{1288323623006 | date:"MM/dd/yyyy 'at' h:mma"}}: + {{'1288323623006' | date:"MM/dd/yyyy 'at' h:mma"}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example101/index-production.html b/1.4.10/docs/examples/example-example101/index-production.html new file mode 100644 index 0000000000..adc97b36ae --- /dev/null +++ b/1.4.10/docs/examples/example-example101/index-production.html @@ -0,0 +1,23 @@ + + + + + Example - example-example101-production + + + + + + + + + {{1288323623006 | date:'medium'}}: + {{1288323623006 | date:'medium'}}
+{{1288323623006 | date:'yyyy-MM-dd HH:mm:ss Z'}}: + {{1288323623006 | date:'yyyy-MM-dd HH:mm:ss Z'}}
+{{1288323623006 | date:'MM/dd/yyyy @ h:mma'}}: + {{'1288323623006' | date:'MM/dd/yyyy @ h:mma'}}
+{{1288323623006 | date:"MM/dd/yyyy 'at' h:mma"}}: + {{'1288323623006' | date:"MM/dd/yyyy 'at' h:mma"}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example101/index.html b/1.4.10/docs/examples/example-example101/index.html new file mode 100644 index 0000000000..9758483f8d --- /dev/null +++ b/1.4.10/docs/examples/example-example101/index.html @@ -0,0 +1,23 @@ + + + + + Example - example-example101 + + + + + + + + + {{1288323623006 | date:'medium'}}: + {{1288323623006 | date:'medium'}}
+{{1288323623006 | date:'yyyy-MM-dd HH:mm:ss Z'}}: + {{1288323623006 | date:'yyyy-MM-dd HH:mm:ss Z'}}
+{{1288323623006 | date:'MM/dd/yyyy @ h:mma'}}: + {{'1288323623006' | date:'MM/dd/yyyy @ h:mma'}}
+{{1288323623006 | date:"MM/dd/yyyy 'at' h:mma"}}: + {{'1288323623006' | date:"MM/dd/yyyy 'at' h:mma"}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example101/manifest.json b/1.4.10/docs/examples/example-example101/manifest.json new file mode 100644 index 0000000000..3b379009d7 --- /dev/null +++ b/1.4.10/docs/examples/example-example101/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example101", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example101/protractor.js b/1.4.10/docs/examples/example-example101/protractor.js new file mode 100644 index 0000000000..75faac8f50 --- /dev/null +++ b/1.4.10/docs/examples/example-example101/protractor.js @@ -0,0 +1,10 @@ +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)/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example102/index-debug.html b/1.4.10/docs/examples/example-example102/index-debug.html new file mode 100644 index 0000000000..0ab19c3ad4 --- /dev/null +++ b/1.4.10/docs/examples/example-example102/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example102-debug + + + + + + + + +
{{ {'name':'value'} | json }}
+
{{ {'name':'value'} | json:4 }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example102/index-jquery.html b/1.4.10/docs/examples/example-example102/index-jquery.html new file mode 100644 index 0000000000..8fc209a6dd --- /dev/null +++ b/1.4.10/docs/examples/example-example102/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example102-jquery + + + + + + + + + +
{{ {'name':'value'} | json }}
+
{{ {'name':'value'} | json:4 }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example102/index-production.html b/1.4.10/docs/examples/example-example102/index-production.html new file mode 100644 index 0000000000..e62919291d --- /dev/null +++ b/1.4.10/docs/examples/example-example102/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example102-production + + + + + + + + +
{{ {'name':'value'} | json }}
+
{{ {'name':'value'} | json:4 }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example102/index.html b/1.4.10/docs/examples/example-example102/index.html new file mode 100644 index 0000000000..4921cdc573 --- /dev/null +++ b/1.4.10/docs/examples/example-example102/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example102 + + + + + + + + +
{{ {'name':'value'} | json }}
+
{{ {'name':'value'} | json:4 }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example102/manifest.json b/1.4.10/docs/examples/example-example102/manifest.json new file mode 100644 index 0000000000..cfc1705a77 --- /dev/null +++ b/1.4.10/docs/examples/example-example102/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example102", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example102/protractor.js b/1.4.10/docs/examples/example-example102/protractor.js new file mode 100644 index 0000000000..809d95f2f9 --- /dev/null +++ b/1.4.10/docs/examples/example-example102/protractor.js @@ -0,0 +1,4 @@ +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}/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example103/index-debug.html b/1.4.10/docs/examples/example-example103/index-debug.html new file mode 100644 index 0000000000..89da87d1e2 --- /dev/null +++ b/1.4.10/docs/examples/example-example103/index-debug.html @@ -0,0 +1,43 @@ + + + + + Example - example-example103-debug + + + + + + + + + +
+ +

Output numbers: {{ numbers | limitTo:numLimit }}

+ +

Output letters: {{ letters | limitTo:letterLimit }}

+ +

Output long number: {{ longNumber | limitTo:longNumberLimit }}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example103/index-jquery.html b/1.4.10/docs/examples/example-example103/index-jquery.html new file mode 100644 index 0000000000..6bc2c94ca0 --- /dev/null +++ b/1.4.10/docs/examples/example-example103/index-jquery.html @@ -0,0 +1,44 @@ + + + + + Example - example-example103-jquery + + + + + + + + + + +
+ +

Output numbers: {{ numbers | limitTo:numLimit }}

+ +

Output letters: {{ letters | limitTo:letterLimit }}

+ +

Output long number: {{ longNumber | limitTo:longNumberLimit }}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example103/index-production.html b/1.4.10/docs/examples/example-example103/index-production.html new file mode 100644 index 0000000000..dfcb5b02c0 --- /dev/null +++ b/1.4.10/docs/examples/example-example103/index-production.html @@ -0,0 +1,43 @@ + + + + + Example - example-example103-production + + + + + + + + + +
+ +

Output numbers: {{ numbers | limitTo:numLimit }}

+ +

Output letters: {{ letters | limitTo:letterLimit }}

+ +

Output long number: {{ longNumber | limitTo:longNumberLimit }}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example103/index.html b/1.4.10/docs/examples/example-example103/index.html new file mode 100644 index 0000000000..e46720742d --- /dev/null +++ b/1.4.10/docs/examples/example-example103/index.html @@ -0,0 +1,43 @@ + + + + + Example - example-example103 + + + + + + + + + +
+ +

Output numbers: {{ numbers | limitTo:numLimit }}

+ +

Output letters: {{ letters | limitTo:letterLimit }}

+ +

Output long number: {{ longNumber | limitTo:longNumberLimit }}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example103/manifest.json b/1.4.10/docs/examples/example-example103/manifest.json new file mode 100644 index 0000000000..581fca581a --- /dev/null +++ b/1.4.10/docs/examples/example-example103/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example103", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example103/protractor.js b/1.4.10/docs/examples/example-example103/protractor.js new file mode 100644 index 0000000000..de30ced4c3 --- /dev/null +++ b/1.4.10/docs/examples/example-example103/protractor.js @@ -0,0 +1,40 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example104/index-debug.html b/1.4.10/docs/examples/example-example104/index-debug.html new file mode 100644 index 0000000000..a8337db02c --- /dev/null +++ b/1.4.10/docs/examples/example-example104/index-debug.html @@ -0,0 +1,30 @@ + + + + + Example - example-example104-debug + + + + + + + + + +
+ + + + + + + + + + + +
NamePhone NumberAge
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example104/index-jquery.html b/1.4.10/docs/examples/example-example104/index-jquery.html new file mode 100644 index 0000000000..a652fd2051 --- /dev/null +++ b/1.4.10/docs/examples/example-example104/index-jquery.html @@ -0,0 +1,31 @@ + + + + + Example - example-example104-jquery + + + + + + + + + + +
+ + + + + + + + + + + +
NamePhone NumberAge
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example104/index-production.html b/1.4.10/docs/examples/example-example104/index-production.html new file mode 100644 index 0000000000..531aca6884 --- /dev/null +++ b/1.4.10/docs/examples/example-example104/index-production.html @@ -0,0 +1,30 @@ + + + + + Example - example-example104-production + + + + + + + + + +
+ + + + + + + + + + + +
NamePhone NumberAge
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example104/index.html b/1.4.10/docs/examples/example-example104/index.html new file mode 100644 index 0000000000..2d8a06a3be --- /dev/null +++ b/1.4.10/docs/examples/example-example104/index.html @@ -0,0 +1,30 @@ + + + + + Example - example-example104 + + + + + + + + + +
+ + + + + + + + + + + +
NamePhone NumberAge
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example104/manifest.json b/1.4.10/docs/examples/example-example104/manifest.json new file mode 100644 index 0000000000..7c0e5a0530 --- /dev/null +++ b/1.4.10/docs/examples/example-example104/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example104", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example104/script.js b/1.4.10/docs/examples/example-example104/script.js new file mode 100644 index 0000000000..9ea6e91c23 --- /dev/null +++ b/1.4.10/docs/examples/example-example104/script.js @@ -0,0 +1,12 @@ +(function(angular) { + 'use strict'; +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}]; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example105/index-debug.html b/1.4.10/docs/examples/example-example105/index-debug.html new file mode 100644 index 0000000000..5fdc34963b --- /dev/null +++ b/1.4.10/docs/examples/example-example105/index-debug.html @@ -0,0 +1,43 @@ + + + + + Example - example-example105-debug + + + + + + + + + + +
+
Sorting predicate = {{predicate}}; reverse = {{reverse}}
+
+ + + + + + + + + + + + +
+ + + + + + + + +
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example105/index-jquery.html b/1.4.10/docs/examples/example-example105/index-jquery.html new file mode 100644 index 0000000000..84869df791 --- /dev/null +++ b/1.4.10/docs/examples/example-example105/index-jquery.html @@ -0,0 +1,44 @@ + + + + + Example - example-example105-jquery + + + + + + + + + + + +
+
Sorting predicate = {{predicate}}; reverse = {{reverse}}
+
+ + + + + + + + + + + + +
+ + + + + + + + +
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example105/index-production.html b/1.4.10/docs/examples/example-example105/index-production.html new file mode 100644 index 0000000000..c39a35eb38 --- /dev/null +++ b/1.4.10/docs/examples/example-example105/index-production.html @@ -0,0 +1,43 @@ + + + + + Example - example-example105-production + + + + + + + + + + +
+
Sorting predicate = {{predicate}}; reverse = {{reverse}}
+
+ + + + + + + + + + + + +
+ + + + + + + + +
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example105/index.html b/1.4.10/docs/examples/example-example105/index.html new file mode 100644 index 0000000000..6ce3491154 --- /dev/null +++ b/1.4.10/docs/examples/example-example105/index.html @@ -0,0 +1,43 @@ + + + + + Example - example-example105 + + + + + + + + + + +
+
Sorting predicate = {{predicate}}; reverse = {{reverse}}
+
+ + + + + + + + + + + + +
+ + + + + + + + +
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example105/manifest.json b/1.4.10/docs/examples/example-example105/manifest.json new file mode 100644 index 0000000000..37651dcdb1 --- /dev/null +++ b/1.4.10/docs/examples/example-example105/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example105", + "files": [ + "index-production.html", + "script.js", + "style.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example105/script.js b/1.4.10/docs/examples/example-example105/script.js new file mode 100644 index 0000000000..655982a166 --- /dev/null +++ b/1.4.10/docs/examples/example-example105/script.js @@ -0,0 +1,18 @@ +(function(angular) { + 'use strict'; +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; + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example105/style.css b/1.4.10/docs/examples/example-example105/style.css new file mode 100644 index 0000000000..95b1075f76 --- /dev/null +++ b/1.4.10/docs/examples/example-example105/style.css @@ -0,0 +1,6 @@ +.sortorder:after { + content: '\25b2'; +} +.sortorder.reverse:after { + content: '\25bc'; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example106/index-debug.html b/1.4.10/docs/examples/example-example106/index-debug.html new file mode 100644 index 0000000000..14564ebf50 --- /dev/null +++ b/1.4.10/docs/examples/example-example106/index-debug.html @@ -0,0 +1,41 @@ + + + + + Example - example-example106-debug + + + + + + + + + + +
+
Sorting predicate = {{predicate}}; reverse = {{reverse}}
+ + + + + + + + + + + +
+ + + + + + + + +
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example106/index-jquery.html b/1.4.10/docs/examples/example-example106/index-jquery.html new file mode 100644 index 0000000000..267d474efb --- /dev/null +++ b/1.4.10/docs/examples/example-example106/index-jquery.html @@ -0,0 +1,42 @@ + + + + + Example - example-example106-jquery + + + + + + + + + + + +
+
Sorting predicate = {{predicate}}; reverse = {{reverse}}
+ + + + + + + + + + + +
+ + + + + + + + +
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example106/index-production.html b/1.4.10/docs/examples/example-example106/index-production.html new file mode 100644 index 0000000000..dd102220f7 --- /dev/null +++ b/1.4.10/docs/examples/example-example106/index-production.html @@ -0,0 +1,41 @@ + + + + + Example - example-example106-production + + + + + + + + + + +
+
Sorting predicate = {{predicate}}; reverse = {{reverse}}
+ + + + + + + + + + + +
+ + + + + + + + +
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example106/index.html b/1.4.10/docs/examples/example-example106/index.html new file mode 100644 index 0000000000..e1187795d5 --- /dev/null +++ b/1.4.10/docs/examples/example-example106/index.html @@ -0,0 +1,41 @@ + + + + + Example - example-example106 + + + + + + + + + + +
+
Sorting predicate = {{predicate}}; reverse = {{reverse}}
+ + + + + + + + + + + +
+ + + + + + + + +
{{friend.name}}{{friend.phone}}{{friend.age}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example106/manifest.json b/1.4.10/docs/examples/example-example106/manifest.json new file mode 100644 index 0000000000..66946313d8 --- /dev/null +++ b/1.4.10/docs/examples/example-example106/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example106", + "files": [ + "index-production.html", + "script.js", + "style.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example106/script.js b/1.4.10/docs/examples/example-example106/script.js new file mode 100644 index 0000000000..2068455441 --- /dev/null +++ b/1.4.10/docs/examples/example-example106/script.js @@ -0,0 +1,20 @@ +(function(angular) { + 'use strict'; +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) { + $scope.predicate = predicate; + $scope.reverse = ($scope.predicate === predicate) ? !$scope.reverse : false; + $scope.friends = orderBy($scope.friends, predicate, $scope.reverse); + }; + $scope.order('age', true); + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example106/style.css b/1.4.10/docs/examples/example-example106/style.css new file mode 100644 index 0000000000..95b1075f76 --- /dev/null +++ b/1.4.10/docs/examples/example-example106/style.css @@ -0,0 +1,6 @@ +.sortorder:after { + content: '\25b2'; +} +.sortorder.reverse:after { + content: '\25bc'; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example107/http-hello.html b/1.4.10/docs/examples/example-example107/http-hello.html new file mode 100644 index 0000000000..7b24164aa1 --- /dev/null +++ b/1.4.10/docs/examples/example-example107/http-hello.html @@ -0,0 +1 @@ +Hello, $http! \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example107/index-debug.html b/1.4.10/docs/examples/example-example107/index-debug.html new file mode 100644 index 0000000000..765813e5fb --- /dev/null +++ b/1.4.10/docs/examples/example-example107/index-debug.html @@ -0,0 +1,36 @@ + + + + + Example - example-example107-debug + + + + + + + + + +
+ + +
+ + + +
http status code: {{status}}
+
http response data: {{data}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example107/index-jquery.html b/1.4.10/docs/examples/example-example107/index-jquery.html new file mode 100644 index 0000000000..6f1fa9f38d --- /dev/null +++ b/1.4.10/docs/examples/example-example107/index-jquery.html @@ -0,0 +1,37 @@ + + + + + Example - example-example107-jquery + + + + + + + + + + +
+ + +
+ + + +
http status code: {{status}}
+
http response data: {{data}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example107/index-production.html b/1.4.10/docs/examples/example-example107/index-production.html new file mode 100644 index 0000000000..2b637ae71b --- /dev/null +++ b/1.4.10/docs/examples/example-example107/index-production.html @@ -0,0 +1,36 @@ + + + + + Example - example-example107-production + + + + + + + + + +
+ + +
+ + + +
http status code: {{status}}
+
http response data: {{data}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example107/index.html b/1.4.10/docs/examples/example-example107/index.html new file mode 100644 index 0000000000..4ba016b257 --- /dev/null +++ b/1.4.10/docs/examples/example-example107/index.html @@ -0,0 +1,36 @@ + + + + + Example - example-example107 + + + + + + + + + +
+ + +
+ + + +
http status code: {{status}}
+
http response data: {{data}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example107/manifest.json b/1.4.10/docs/examples/example-example107/manifest.json new file mode 100644 index 0000000000..904dcb840a --- /dev/null +++ b/1.4.10/docs/examples/example-example107/manifest.json @@ -0,0 +1,9 @@ +{ + "name": "example-example107", + "files": [ + "index-production.html", + "script.js", + "http-hello.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example107/protractor.js b/1.4.10/docs/examples/example-example107/protractor.js new file mode 100644 index 0000000000..a350c345c4 --- /dev/null +++ b/1.4.10/docs/examples/example-example107/protractor.js @@ -0,0 +1,29 @@ + 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'); + }); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example107/script.js b/1.4.10/docs/examples/example-example107/script.js new file mode 100644 index 0000000000..880637655f --- /dev/null +++ b/1.4.10/docs/examples/example-example107/script.js @@ -0,0 +1,28 @@ +(function(angular) { + 'use strict'; +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; + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example108/index-debug.html b/1.4.10/docs/examples/example-example108/index-debug.html new file mode 100644 index 0000000000..94d9e09be4 --- /dev/null +++ b/1.4.10/docs/examples/example-example108/index-debug.html @@ -0,0 +1,25 @@ + + + + + Example - example-example108-debug + + + + + + + + +
+

{{apptitle}}: \{\{ username = "defaced value"; \}\} +

+

{{username}} 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.

+

Instead, the result of the attempted script injection is visible, and can be removed + from the database by an administrator.

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example108/index-jquery.html b/1.4.10/docs/examples/example-example108/index-jquery.html new file mode 100644 index 0000000000..56768ddc30 --- /dev/null +++ b/1.4.10/docs/examples/example-example108/index-jquery.html @@ -0,0 +1,26 @@ + + + + + Example - example-example108-jquery + + + + + + + + + +
+

{{apptitle}}: \{\{ username = "defaced value"; \}\} +

+

{{username}} 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.

+

Instead, the result of the attempted script injection is visible, and can be removed + from the database by an administrator.

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example108/index-production.html b/1.4.10/docs/examples/example-example108/index-production.html new file mode 100644 index 0000000000..17f289f293 --- /dev/null +++ b/1.4.10/docs/examples/example-example108/index-production.html @@ -0,0 +1,25 @@ + + + + + Example - example-example108-production + + + + + + + + +
+

{{apptitle}}: \{\{ username = "defaced value"; \}\} +

+

{{username}} 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.

+

Instead, the result of the attempted script injection is visible, and can be removed + from the database by an administrator.

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example108/index.html b/1.4.10/docs/examples/example-example108/index.html new file mode 100644 index 0000000000..5226b87f95 --- /dev/null +++ b/1.4.10/docs/examples/example-example108/index.html @@ -0,0 +1,25 @@ + + + + + Example - example-example108 + + + + + + + + +
+

{{apptitle}}: \{\{ username = "defaced value"; \}\} +

+

{{username}} 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.

+

Instead, the result of the attempted script injection is visible, and can be removed + from the database by an administrator.

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example108/manifest.json b/1.4.10/docs/examples/example-example108/manifest.json new file mode 100644 index 0000000000..6cb64acab5 --- /dev/null +++ b/1.4.10/docs/examples/example-example108/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example108", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example109/index-debug.html b/1.4.10/docs/examples/example-example109/index-debug.html new file mode 100644 index 0000000000..df37ca2d83 --- /dev/null +++ b/1.4.10/docs/examples/example-example109/index-debug.html @@ -0,0 +1,98 @@ + + + + + Example - example-example109-debug + + + + + + + + + + +
+
+
+ Current time is: +
+ Blood 1 : {{blood_1}} + Blood 2 : {{blood_2}} + + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example109/index-jquery.html b/1.4.10/docs/examples/example-example109/index-jquery.html new file mode 100644 index 0000000000..ef12988238 --- /dev/null +++ b/1.4.10/docs/examples/example-example109/index-jquery.html @@ -0,0 +1,99 @@ + + + + + Example - example-example109-jquery + + + + + + + + + + + +
+
+
+ Current time is: +
+ Blood 1 : {{blood_1}} + Blood 2 : {{blood_2}} + + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example109/index-production.html b/1.4.10/docs/examples/example-example109/index-production.html new file mode 100644 index 0000000000..3191fb8819 --- /dev/null +++ b/1.4.10/docs/examples/example-example109/index-production.html @@ -0,0 +1,98 @@ + + + + + Example - example-example109-production + + + + + + + + + + +
+
+
+ Current time is: +
+ Blood 1 : {{blood_1}} + Blood 2 : {{blood_2}} + + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example109/index.html b/1.4.10/docs/examples/example-example109/index.html new file mode 100644 index 0000000000..80aad4889a --- /dev/null +++ b/1.4.10/docs/examples/example-example109/index.html @@ -0,0 +1,98 @@ + + + + + Example - example-example109 + + + + + + + + + + +
+
+
+ Current time is: +
+ Blood 1 : {{blood_1}} + Blood 2 : {{blood_2}} + + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example109/manifest.json b/1.4.10/docs/examples/example-example109/manifest.json new file mode 100644 index 0000000000..c0d7073208 --- /dev/null +++ b/1.4.10/docs/examples/example-example109/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example109", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example11/index-debug.html b/1.4.10/docs/examples/example-example11/index-debug.html new file mode 100644 index 0000000000..522b9ccc8f --- /dev/null +++ b/1.4.10/docs/examples/example-example11/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example11-debug + + + + + + + + + +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example11/index-jquery.html b/1.4.10/docs/examples/example-example11/index-jquery.html new file mode 100644 index 0000000000..bf22c1fc80 --- /dev/null +++ b/1.4.10/docs/examples/example-example11/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example11-jquery + + + + + + + + + + +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example11/index-production.html b/1.4.10/docs/examples/example-example11/index-production.html new file mode 100644 index 0000000000..4ec4700517 --- /dev/null +++ b/1.4.10/docs/examples/example-example11/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example11-production + + + + + + + + + +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example11/index.html b/1.4.10/docs/examples/example-example11/index.html new file mode 100644 index 0000000000..7e2c7e9263 --- /dev/null +++ b/1.4.10/docs/examples/example-example11/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example11 + + + + + + + + + +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example11/manifest.json b/1.4.10/docs/examples/example-example11/manifest.json new file mode 100644 index 0000000000..e907bf3982 --- /dev/null +++ b/1.4.10/docs/examples/example-example11/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example11", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example11/script.js b/1.4.10/docs/examples/example-example11/script.js new file mode 100644 index 0000000000..0fe2f23757 --- /dev/null +++ b/1.4.10/docs/examples/example-example11/script.js @@ -0,0 +1,15 @@ +(function(angular) { + 'use strict'; +angular.module('docsSimpleDirective', []) + .controller('Controller', ['$scope', function($scope) { + $scope.customer = { + name: 'Naomi', + address: '1600 Amphitheatre' + }; + }]) + .directive('myCustomer', function() { + return { + template: 'Name: {{customer.name}} Address: {{customer.address}}' + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example110/index-debug.html b/1.4.10/docs/examples/example-example110/index-debug.html new file mode 100644 index 0000000000..11ed3d8b9c --- /dev/null +++ b/1.4.10/docs/examples/example-example110/index-debug.html @@ -0,0 +1,26 @@ + + + + + Example - example-example110-debug + + + + + + + + + +
+

Reload this page with open console, enter text and hit the log button...

+ + + + + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example110/index-jquery.html b/1.4.10/docs/examples/example-example110/index-jquery.html new file mode 100644 index 0000000000..b69636caff --- /dev/null +++ b/1.4.10/docs/examples/example-example110/index-jquery.html @@ -0,0 +1,27 @@ + + + + + Example - example-example110-jquery + + + + + + + + + + +
+

Reload this page with open console, enter text and hit the log button...

+ + + + + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example110/index-production.html b/1.4.10/docs/examples/example-example110/index-production.html new file mode 100644 index 0000000000..b83fc7fa73 --- /dev/null +++ b/1.4.10/docs/examples/example-example110/index-production.html @@ -0,0 +1,26 @@ + + + + + Example - example-example110-production + + + + + + + + + +
+

Reload this page with open console, enter text and hit the log button...

+ + + + + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example110/index.html b/1.4.10/docs/examples/example-example110/index.html new file mode 100644 index 0000000000..70f0c9ffd6 --- /dev/null +++ b/1.4.10/docs/examples/example-example110/index.html @@ -0,0 +1,26 @@ + + + + + Example - example-example110 + + + + + + + + + +
+

Reload this page with open console, enter text and hit the log button...

+ + + + + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example110/manifest.json b/1.4.10/docs/examples/example-example110/manifest.json new file mode 100644 index 0000000000..437cd7875b --- /dev/null +++ b/1.4.10/docs/examples/example-example110/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example110", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example110/script.js b/1.4.10/docs/examples/example-example110/script.js new file mode 100644 index 0000000000..2f51cab4b2 --- /dev/null +++ b/1.4.10/docs/examples/example-example110/script.js @@ -0,0 +1,8 @@ +(function(angular) { + 'use strict'; +angular.module('logExample', []) + .controller('LogController', ['$scope', '$log', function($scope, $log) { + $scope.$log = $log; + $scope.message = 'Hello World!'; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example111/index-debug.html b/1.4.10/docs/examples/example-example111/index-debug.html new file mode 100644 index 0000000000..c346d83031 --- /dev/null +++ b/1.4.10/docs/examples/example-example111/index-debug.html @@ -0,0 +1,31 @@ + + + + + Example - example-example111-debug + + + + + + + + + + +
+

+ User comments
+ 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. +
+
+ {{userComment.name}}: + +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example111/index-jquery.html b/1.4.10/docs/examples/example-example111/index-jquery.html new file mode 100644 index 0000000000..f78397fd45 --- /dev/null +++ b/1.4.10/docs/examples/example-example111/index-jquery.html @@ -0,0 +1,32 @@ + + + + + Example - example-example111-jquery + + + + + + + + + + + +
+

+ User comments
+ 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. +
+
+ {{userComment.name}}: + +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example111/index-production.html b/1.4.10/docs/examples/example-example111/index-production.html new file mode 100644 index 0000000000..b6b50c1ff0 --- /dev/null +++ b/1.4.10/docs/examples/example-example111/index-production.html @@ -0,0 +1,31 @@ + + + + + Example - example-example111-production + + + + + + + + + + +
+

+ User comments
+ 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. +
+
+ {{userComment.name}}: + +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example111/index.html b/1.4.10/docs/examples/example-example111/index.html new file mode 100644 index 0000000000..838c550e89 --- /dev/null +++ b/1.4.10/docs/examples/example-example111/index.html @@ -0,0 +1,31 @@ + + + + + Example - example-example111 + + + + + + + + + + +
+

+ User comments
+ 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. +
+
+ {{userComment.name}}: + +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example111/manifest.json b/1.4.10/docs/examples/example-example111/manifest.json new file mode 100644 index 0000000000..896b579790 --- /dev/null +++ b/1.4.10/docs/examples/example-example111/manifest.json @@ -0,0 +1,9 @@ +{ + "name": "example-example111", + "files": [ + "index-production.html", + "script.js", + "test_data.json", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example111/protractor.js b/1.4.10/docs/examples/example-example111/protractor.js new file mode 100644 index 0000000000..2fea13e447 --- /dev/null +++ b/1.4.10/docs/examples/example-example111/protractor.js @@ -0,0 +1,12 @@ +describe('SCE doc demo', function() { + it('should sanitize untrusted values', function() { + expect(element.all(by.css('.htmlComment')).first().getInnerHtml()) + .toBe('Is anyone reading this?'); + }); + + it('should NOT sanitize explicitly trusted values', function() { + expect(element(by.id('explicitlyTrustedHtml')).getInnerHtml()).toBe( + 'Hover over this text.'); + }); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example111/script.js b/1.4.10/docs/examples/example-example111/script.js new file mode 100644 index 0000000000..057e1add5d --- /dev/null +++ b/1.4.10/docs/examples/example-example111/script.js @@ -0,0 +1,14 @@ +(function(angular) { + 'use strict'; +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( + 'Hover over this text.'); + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example111/test_data.json b/1.4.10/docs/examples/example-example111/test_data.json new file mode 100644 index 0000000000..e086b707e3 --- /dev/null +++ b/1.4.10/docs/examples/example-example111/test_data.json @@ -0,0 +1,9 @@ +[ + { "name": "Alice", + "htmlComment": + "Is anyone reading this?" + }, + { "name": "Bob", + "htmlComment": "Yes! Am I the only other one?" + } +] \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example112/index-debug.html b/1.4.10/docs/examples/example-example112/index-debug.html new file mode 100644 index 0000000000..bcb5335ea2 --- /dev/null +++ b/1.4.10/docs/examples/example-example112/index-debug.html @@ -0,0 +1,28 @@ + + + + + Example - example-example112-debug + + + + + + + + + +
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example112/index-jquery.html b/1.4.10/docs/examples/example-example112/index-jquery.html new file mode 100644 index 0000000000..48096c390e --- /dev/null +++ b/1.4.10/docs/examples/example-example112/index-jquery.html @@ -0,0 +1,29 @@ + + + + + Example - example-example112-jquery + + + + + + + + + + +
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example112/index-production.html b/1.4.10/docs/examples/example-example112/index-production.html new file mode 100644 index 0000000000..95af35448e --- /dev/null +++ b/1.4.10/docs/examples/example-example112/index-production.html @@ -0,0 +1,28 @@ + + + + + Example - example-example112-production + + + + + + + + + +
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example112/index.html b/1.4.10/docs/examples/example-example112/index.html new file mode 100644 index 0000000000..873b604b14 --- /dev/null +++ b/1.4.10/docs/examples/example-example112/index.html @@ -0,0 +1,28 @@ + + + + + Example - example-example112 + + + + + + + + + +
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example112/manifest.json b/1.4.10/docs/examples/example-example112/manifest.json new file mode 100644 index 0000000000..513e84dc2d --- /dev/null +++ b/1.4.10/docs/examples/example-example112/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example112", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example112/protractor.js b/1.4.10/docs/examples/example-example112/protractor.js new file mode 100644 index 0000000000..ab3238e810 --- /dev/null +++ b/1.4.10/docs/examples/example-example112/protractor.js @@ -0,0 +1,5 @@ +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(); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example113/index-debug.html b/1.4.10/docs/examples/example-example113/index-debug.html new file mode 100644 index 0000000000..0072999eae --- /dev/null +++ b/1.4.10/docs/examples/example-example113/index-debug.html @@ -0,0 +1,60 @@ + + + + + Example - example-example113-debug + + + + + + + + + + +
+Snippet: + + + + + + + + + + + + + + + + + + + + + +
FilterSourceRendered
linky filter +
<div ng-bind-html="snippet | linky">
</div>
+
+
+
linky target +
<div ng-bind-html="snippetWithTarget | linky:'_blank'">
</div>
+
+
+
no filter
<div ng-bind="snippet">
</div>
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example113/index-jquery.html b/1.4.10/docs/examples/example-example113/index-jquery.html new file mode 100644 index 0000000000..be48390813 --- /dev/null +++ b/1.4.10/docs/examples/example-example113/index-jquery.html @@ -0,0 +1,61 @@ + + + + + Example - example-example113-jquery + + + + + + + + + + + +
+Snippet: + + + + + + + + + + + + + + + + + + + + + +
FilterSourceRendered
linky filter +
<div ng-bind-html="snippet | linky">
</div>
+
+
+
linky target +
<div ng-bind-html="snippetWithTarget | linky:'_blank'">
</div>
+
+
+
no filter
<div ng-bind="snippet">
</div>
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example113/index-production.html b/1.4.10/docs/examples/example-example113/index-production.html new file mode 100644 index 0000000000..e2f70425f8 --- /dev/null +++ b/1.4.10/docs/examples/example-example113/index-production.html @@ -0,0 +1,60 @@ + + + + + Example - example-example113-production + + + + + + + + + + +
+Snippet: + + + + + + + + + + + + + + + + + + + + + +
FilterSourceRendered
linky filter +
<div ng-bind-html="snippet | linky">
</div>
+
+
+
linky target +
<div ng-bind-html="snippetWithTarget | linky:'_blank'">
</div>
+
+
+
no filter
<div ng-bind="snippet">
</div>
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example113/index.html b/1.4.10/docs/examples/example-example113/index.html new file mode 100644 index 0000000000..1fa5a323db --- /dev/null +++ b/1.4.10/docs/examples/example-example113/index.html @@ -0,0 +1,60 @@ + + + + + Example - example-example113 + + + + + + + + + + +
+Snippet: + + + + + + + + + + + + + + + + + + + + + +
FilterSourceRendered
linky filter +
<div ng-bind-html="snippet | linky">
</div>
+
+
+
linky target +
<div ng-bind-html="snippetWithTarget | linky:'_blank'">
</div>
+
+
+
no filter
<div ng-bind="snippet">
</div>
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example113/manifest.json b/1.4.10/docs/examples/example-example113/manifest.json new file mode 100644 index 0000000000..aea2b22001 --- /dev/null +++ b/1.4.10/docs/examples/example-example113/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example113", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example113/protractor.js b/1.4.10/docs/examples/example-example113/protractor.js new file mode 100644 index 0000000000..42712d58b7 --- /dev/null +++ b/1.4.10/docs/examples/example-example113/protractor.js @@ -0,0 +1,30 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example114/index-debug.html b/1.4.10/docs/examples/example-example114/index-debug.html new file mode 100644 index 0000000000..fe48aeaaa8 --- /dev/null +++ b/1.4.10/docs/examples/example-example114/index-debug.html @@ -0,0 +1,60 @@ + + + + + Example - example-example114-debug + + + + + + + + + + +
+ Snippet: + + + + + + + + + + + + + + + + + + + + + + + + + +
DirectiveHowSourceRendered
ng-bind-htmlAutomatically uses $sanitize
<div ng-bind-html="snippet">
</div>
ng-bind-htmlBypass $sanitize by explicitly trusting the dangerous value +
<div ng-bind-html="deliberatelyTrustDangerousSnippet()">
+</div>
+
ng-bindAutomatically escapes
<div ng-bind="snippet">
</div>
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example114/index-jquery.html b/1.4.10/docs/examples/example-example114/index-jquery.html new file mode 100644 index 0000000000..13849c13ed --- /dev/null +++ b/1.4.10/docs/examples/example-example114/index-jquery.html @@ -0,0 +1,61 @@ + + + + + Example - example-example114-jquery + + + + + + + + + + + +
+ Snippet: + + + + + + + + + + + + + + + + + + + + + + + + + +
DirectiveHowSourceRendered
ng-bind-htmlAutomatically uses $sanitize
<div ng-bind-html="snippet">
</div>
ng-bind-htmlBypass $sanitize by explicitly trusting the dangerous value +
<div ng-bind-html="deliberatelyTrustDangerousSnippet()">
+</div>
+
ng-bindAutomatically escapes
<div ng-bind="snippet">
</div>
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example114/index-production.html b/1.4.10/docs/examples/example-example114/index-production.html new file mode 100644 index 0000000000..df22349465 --- /dev/null +++ b/1.4.10/docs/examples/example-example114/index-production.html @@ -0,0 +1,60 @@ + + + + + Example - example-example114-production + + + + + + + + + + +
+ Snippet: + + + + + + + + + + + + + + + + + + + + + + + + + +
DirectiveHowSourceRendered
ng-bind-htmlAutomatically uses $sanitize
<div ng-bind-html="snippet">
</div>
ng-bind-htmlBypass $sanitize by explicitly trusting the dangerous value +
<div ng-bind-html="deliberatelyTrustDangerousSnippet()">
+</div>
+
ng-bindAutomatically escapes
<div ng-bind="snippet">
</div>
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example114/index.html b/1.4.10/docs/examples/example-example114/index.html new file mode 100644 index 0000000000..bc516961b0 --- /dev/null +++ b/1.4.10/docs/examples/example-example114/index.html @@ -0,0 +1,60 @@ + + + + + Example - example-example114 + + + + + + + + + + +
+ Snippet: + + + + + + + + + + + + + + + + + + + + + + + + + +
DirectiveHowSourceRendered
ng-bind-htmlAutomatically uses $sanitize
<div ng-bind-html="snippet">
</div>
ng-bind-htmlBypass $sanitize by explicitly trusting the dangerous value +
<div ng-bind-html="deliberatelyTrustDangerousSnippet()">
+</div>
+
ng-bindAutomatically escapes
<div ng-bind="snippet">
</div>
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example114/manifest.json b/1.4.10/docs/examples/example-example114/manifest.json new file mode 100644 index 0000000000..36adf1b9f3 --- /dev/null +++ b/1.4.10/docs/examples/example-example114/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example114", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example114/protractor.js b/1.4.10/docs/examples/example-example114/protractor.js new file mode 100644 index 0000000000..41a337007e --- /dev/null +++ b/1.4.10/docs/examples/example-example114/protractor.js @@ -0,0 +1,29 @@ +it('should sanitize the html snippet by default', function() { + expect(element(by.css('#bind-html-with-sanitize div')).getInnerHtml()). + toBe('

an html\nclick here\nsnippet

'); +}); + +it('should inline raw snippet if bound to a trusted value', function() { + expect(element(by.css('#bind-html-with-trust div')).getInnerHtml()). + toBe("

an html\n" + + "click here\n" + + "snippet

"); +}); + +it('should escape snippet without any filter', function() { + expect(element(by.css('#bind-default div')).getInnerHtml()). + toBe("<p style=\"color:blue\">an html\n" + + "<em onmouseover=\"this.textContent='PWN3D!'\">click here</em>\n" + + "snippet</p>"); +}); + +it('should update', function() { + element(by.model('snippet')).clear(); + element(by.model('snippet')).sendKeys('new text'); + expect(element(by.css('#bind-html-with-sanitize div')).getInnerHtml()). + toBe('new text'); + expect(element(by.css('#bind-html-with-trust div')).getInnerHtml()).toBe( + 'new text'); + expect(element(by.css('#bind-default div')).getInnerHtml()).toBe( + "new <b onclick=\"alert(1)\">text</b>"); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example115/index-debug.html b/1.4.10/docs/examples/example-example115/index-debug.html new file mode 100644 index 0000000000..ace0c08969 --- /dev/null +++ b/1.4.10/docs/examples/example-example115/index-debug.html @@ -0,0 +1,21 @@ + + + + + Example - example-example115-debug + + + + + + + + + + + +count: {{ count }} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example115/index-jquery.html b/1.4.10/docs/examples/example-example115/index-jquery.html new file mode 100644 index 0000000000..f36d49a068 --- /dev/null +++ b/1.4.10/docs/examples/example-example115/index-jquery.html @@ -0,0 +1,22 @@ + + + + + Example - example-example115-jquery + + + + + + + + + + + + +count: {{ count }} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example115/index-production.html b/1.4.10/docs/examples/example-example115/index-production.html new file mode 100644 index 0000000000..46a99c3779 --- /dev/null +++ b/1.4.10/docs/examples/example-example115/index-production.html @@ -0,0 +1,21 @@ + + + + + Example - example-example115-production + + + + + + + + + + + +count: {{ count }} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example115/index.html b/1.4.10/docs/examples/example-example115/index.html new file mode 100644 index 0000000000..3c4f3851c3 --- /dev/null +++ b/1.4.10/docs/examples/example-example115/index.html @@ -0,0 +1,21 @@ + + + + + Example - example-example115 + + + + + + + + + + + +count: {{ count }} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example115/manifest.json b/1.4.10/docs/examples/example-example115/manifest.json new file mode 100644 index 0000000000..1c6496a219 --- /dev/null +++ b/1.4.10/docs/examples/example-example115/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example115", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example115/script.js b/1.4.10/docs/examples/example-example115/script.js new file mode 100644 index 0000000000..888cebe117 --- /dev/null +++ b/1.4.10/docs/examples/example-example115/script.js @@ -0,0 +1,4 @@ +(function(angular) { + 'use strict'; +angular.module('ngClickExample', ['ngTouch']); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example116/index-debug.html b/1.4.10/docs/examples/example-example116/index-debug.html new file mode 100644 index 0000000000..ce5d4de332 --- /dev/null +++ b/1.4.10/docs/examples/example-example116/index-debug.html @@ -0,0 +1,24 @@ + + + + + Example - example-example116-debug + + + + + + + + + + +
+ Some list content, like an email in the inbox +
+
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example116/index-jquery.html b/1.4.10/docs/examples/example-example116/index-jquery.html new file mode 100644 index 0000000000..cebe79349a --- /dev/null +++ b/1.4.10/docs/examples/example-example116/index-jquery.html @@ -0,0 +1,25 @@ + + + + + Example - example-example116-jquery + + + + + + + + + + + +
+ Some list content, like an email in the inbox +
+
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example116/index-production.html b/1.4.10/docs/examples/example-example116/index-production.html new file mode 100644 index 0000000000..0ec1a6b290 --- /dev/null +++ b/1.4.10/docs/examples/example-example116/index-production.html @@ -0,0 +1,24 @@ + + + + + Example - example-example116-production + + + + + + + + + + +
+ Some list content, like an email in the inbox +
+
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example116/index.html b/1.4.10/docs/examples/example-example116/index.html new file mode 100644 index 0000000000..a8274ec802 --- /dev/null +++ b/1.4.10/docs/examples/example-example116/index.html @@ -0,0 +1,24 @@ + + + + + Example - example-example116 + + + + + + + + + + +
+ Some list content, like an email in the inbox +
+
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example116/manifest.json b/1.4.10/docs/examples/example-example116/manifest.json new file mode 100644 index 0000000000..ac9df8d580 --- /dev/null +++ b/1.4.10/docs/examples/example-example116/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example116", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example116/script.js b/1.4.10/docs/examples/example-example116/script.js new file mode 100644 index 0000000000..0e581c2cc2 --- /dev/null +++ b/1.4.10/docs/examples/example-example116/script.js @@ -0,0 +1,4 @@ +(function(angular) { + 'use strict'; +angular.module('ngSwipeLeftExample', ['ngTouch']); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example117/index-debug.html b/1.4.10/docs/examples/example-example117/index-debug.html new file mode 100644 index 0000000000..4c8a76daf7 --- /dev/null +++ b/1.4.10/docs/examples/example-example117/index-debug.html @@ -0,0 +1,24 @@ + + + + + Example - example-example117-debug + + + + + + + + + + +
+ Some list content, like an email in the inbox +
+
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example117/index-jquery.html b/1.4.10/docs/examples/example-example117/index-jquery.html new file mode 100644 index 0000000000..cab7676d97 --- /dev/null +++ b/1.4.10/docs/examples/example-example117/index-jquery.html @@ -0,0 +1,25 @@ + + + + + Example - example-example117-jquery + + + + + + + + + + + +
+ Some list content, like an email in the inbox +
+
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example117/index-production.html b/1.4.10/docs/examples/example-example117/index-production.html new file mode 100644 index 0000000000..4584282230 --- /dev/null +++ b/1.4.10/docs/examples/example-example117/index-production.html @@ -0,0 +1,24 @@ + + + + + Example - example-example117-production + + + + + + + + + + +
+ Some list content, like an email in the inbox +
+
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example117/index.html b/1.4.10/docs/examples/example-example117/index.html new file mode 100644 index 0000000000..00c85ca2c9 --- /dev/null +++ b/1.4.10/docs/examples/example-example117/index.html @@ -0,0 +1,24 @@ + + + + + Example - example-example117 + + + + + + + + + + +
+ Some list content, like an email in the inbox +
+
+ + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example117/manifest.json b/1.4.10/docs/examples/example-example117/manifest.json new file mode 100644 index 0000000000..2f9f3150cb --- /dev/null +++ b/1.4.10/docs/examples/example-example117/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example117", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example117/script.js b/1.4.10/docs/examples/example-example117/script.js new file mode 100644 index 0000000000..3ad5afc069 --- /dev/null +++ b/1.4.10/docs/examples/example-example117/script.js @@ -0,0 +1,4 @@ +(function(angular) { + 'use strict'; +angular.module('ngSwipeRightExample', ['ngTouch']); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example12/index-debug.html b/1.4.10/docs/examples/example-example12/index-debug.html new file mode 100644 index 0000000000..60e56912c7 --- /dev/null +++ b/1.4.10/docs/examples/example-example12/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example12-debug + + + + + + + + + +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example12/index-jquery.html b/1.4.10/docs/examples/example-example12/index-jquery.html new file mode 100644 index 0000000000..2a63879a3d --- /dev/null +++ b/1.4.10/docs/examples/example-example12/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example12-jquery + + + + + + + + + + +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example12/index-production.html b/1.4.10/docs/examples/example-example12/index-production.html new file mode 100644 index 0000000000..d19f3f8620 --- /dev/null +++ b/1.4.10/docs/examples/example-example12/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example12-production + + + + + + + + + +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example12/index.html b/1.4.10/docs/examples/example-example12/index.html new file mode 100644 index 0000000000..3bf0efe43a --- /dev/null +++ b/1.4.10/docs/examples/example-example12/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example12 + + + + + + + + + +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example12/manifest.json b/1.4.10/docs/examples/example-example12/manifest.json new file mode 100644 index 0000000000..8150e92d09 --- /dev/null +++ b/1.4.10/docs/examples/example-example12/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example12", + "files": [ + "index-production.html", + "script.js", + "my-customer.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example12/my-customer.html b/1.4.10/docs/examples/example-example12/my-customer.html new file mode 100644 index 0000000000..ccf1430dbe --- /dev/null +++ b/1.4.10/docs/examples/example-example12/my-customer.html @@ -0,0 +1 @@ +Name: {{customer.name}} Address: {{customer.address}} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example12/script.js b/1.4.10/docs/examples/example-example12/script.js new file mode 100644 index 0000000000..7b32fb42eb --- /dev/null +++ b/1.4.10/docs/examples/example-example12/script.js @@ -0,0 +1,15 @@ +(function(angular) { + 'use strict'; +angular.module('docsTemplateUrlDirective', []) + .controller('Controller', ['$scope', function($scope) { + $scope.customer = { + name: 'Naomi', + address: '1600 Amphitheatre' + }; + }]) + .directive('myCustomer', function() { + return { + templateUrl: 'my-customer.html' + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example13/customer-address.html b/1.4.10/docs/examples/example-example13/customer-address.html new file mode 100644 index 0000000000..8cd30b971e --- /dev/null +++ b/1.4.10/docs/examples/example-example13/customer-address.html @@ -0,0 +1 @@ +Address: {{customer.address}} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example13/customer-name.html b/1.4.10/docs/examples/example-example13/customer-name.html new file mode 100644 index 0000000000..2ef7e7cafe --- /dev/null +++ b/1.4.10/docs/examples/example-example13/customer-name.html @@ -0,0 +1 @@ +Name: {{customer.name}} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example13/index-debug.html b/1.4.10/docs/examples/example-example13/index-debug.html new file mode 100644 index 0000000000..d5c1bc33a4 --- /dev/null +++ b/1.4.10/docs/examples/example-example13/index-debug.html @@ -0,0 +1,20 @@ + + + + + Example - example-example13-debug + + + + + + + + + +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example13/index-jquery.html b/1.4.10/docs/examples/example-example13/index-jquery.html new file mode 100644 index 0000000000..9edde1efc4 --- /dev/null +++ b/1.4.10/docs/examples/example-example13/index-jquery.html @@ -0,0 +1,21 @@ + + + + + Example - example-example13-jquery + + + + + + + + + + +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example13/index-production.html b/1.4.10/docs/examples/example-example13/index-production.html new file mode 100644 index 0000000000..d8c16c01e6 --- /dev/null +++ b/1.4.10/docs/examples/example-example13/index-production.html @@ -0,0 +1,20 @@ + + + + + Example - example-example13-production + + + + + + + + + +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example13/index.html b/1.4.10/docs/examples/example-example13/index.html new file mode 100644 index 0000000000..de93e49837 --- /dev/null +++ b/1.4.10/docs/examples/example-example13/index.html @@ -0,0 +1,20 @@ + + + + + Example - example-example13 + + + + + + + + + +
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example13/manifest.json b/1.4.10/docs/examples/example-example13/manifest.json new file mode 100644 index 0000000000..574e3c2c7f --- /dev/null +++ b/1.4.10/docs/examples/example-example13/manifest.json @@ -0,0 +1,9 @@ +{ + "name": "example-example13", + "files": [ + "index-production.html", + "script.js", + "customer-name.html", + "customer-address.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example13/script.js b/1.4.10/docs/examples/example-example13/script.js new file mode 100644 index 0000000000..b564c82e46 --- /dev/null +++ b/1.4.10/docs/examples/example-example13/script.js @@ -0,0 +1,17 @@ +(function(angular) { + 'use strict'; +angular.module('docsTemplateUrlDirective', []) + .controller('Controller', ['$scope', function($scope) { + $scope.customer = { + name: 'Naomi', + address: '1600 Amphitheatre' + }; + }]) + .directive('myCustomer', function() { + return { + templateUrl: function(elem, attr){ + return 'customer-'+attr.type+'.html'; + } + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example14/index-debug.html b/1.4.10/docs/examples/example-example14/index-debug.html new file mode 100644 index 0000000000..2fe763d7b2 --- /dev/null +++ b/1.4.10/docs/examples/example-example14/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example14-debug + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example14/index-jquery.html b/1.4.10/docs/examples/example-example14/index-jquery.html new file mode 100644 index 0000000000..7ce83402cd --- /dev/null +++ b/1.4.10/docs/examples/example-example14/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example14-jquery + + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example14/index-production.html b/1.4.10/docs/examples/example-example14/index-production.html new file mode 100644 index 0000000000..4e9c920596 --- /dev/null +++ b/1.4.10/docs/examples/example-example14/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example14-production + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example14/index.html b/1.4.10/docs/examples/example-example14/index.html new file mode 100644 index 0000000000..876c4e2806 --- /dev/null +++ b/1.4.10/docs/examples/example-example14/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example14 + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example14/manifest.json b/1.4.10/docs/examples/example-example14/manifest.json new file mode 100644 index 0000000000..8b948ad9f9 --- /dev/null +++ b/1.4.10/docs/examples/example-example14/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example14", + "files": [ + "index-production.html", + "script.js", + "my-customer.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example14/my-customer.html b/1.4.10/docs/examples/example-example14/my-customer.html new file mode 100644 index 0000000000..ccf1430dbe --- /dev/null +++ b/1.4.10/docs/examples/example-example14/my-customer.html @@ -0,0 +1 @@ +Name: {{customer.name}} Address: {{customer.address}} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example14/script.js b/1.4.10/docs/examples/example-example14/script.js new file mode 100644 index 0000000000..8e601cc919 --- /dev/null +++ b/1.4.10/docs/examples/example-example14/script.js @@ -0,0 +1,16 @@ +(function(angular) { + 'use strict'; +angular.module('docsRestrictDirective', []) + .controller('Controller', ['$scope', function($scope) { + $scope.customer = { + name: 'Naomi', + address: '1600 Amphitheatre' + }; + }]) + .directive('myCustomer', function() { + return { + restrict: 'E', + templateUrl: 'my-customer.html' + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example15/index-debug.html b/1.4.10/docs/examples/example-example15/index-debug.html new file mode 100644 index 0000000000..e9a83a0a8b --- /dev/null +++ b/1.4.10/docs/examples/example-example15/index-debug.html @@ -0,0 +1,23 @@ + + + + + Example - example-example15-debug + + + + + + + + + +
+ +
+
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example15/index-jquery.html b/1.4.10/docs/examples/example-example15/index-jquery.html new file mode 100644 index 0000000000..108aa3bb7b --- /dev/null +++ b/1.4.10/docs/examples/example-example15/index-jquery.html @@ -0,0 +1,24 @@ + + + + + Example - example-example15-jquery + + + + + + + + + + +
+ +
+
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example15/index-production.html b/1.4.10/docs/examples/example-example15/index-production.html new file mode 100644 index 0000000000..95bb67ed55 --- /dev/null +++ b/1.4.10/docs/examples/example-example15/index-production.html @@ -0,0 +1,23 @@ + + + + + Example - example-example15-production + + + + + + + + + +
+ +
+
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example15/index.html b/1.4.10/docs/examples/example-example15/index.html new file mode 100644 index 0000000000..80e4cc9ae7 --- /dev/null +++ b/1.4.10/docs/examples/example-example15/index.html @@ -0,0 +1,23 @@ + + + + + Example - example-example15 + + + + + + + + + +
+ +
+
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example15/manifest.json b/1.4.10/docs/examples/example-example15/manifest.json new file mode 100644 index 0000000000..8e43cef1f3 --- /dev/null +++ b/1.4.10/docs/examples/example-example15/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example15", + "files": [ + "index-production.html", + "script.js", + "my-customer.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example15/my-customer.html b/1.4.10/docs/examples/example-example15/my-customer.html new file mode 100644 index 0000000000..ccf1430dbe --- /dev/null +++ b/1.4.10/docs/examples/example-example15/my-customer.html @@ -0,0 +1 @@ +Name: {{customer.name}} Address: {{customer.address}} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example15/script.js b/1.4.10/docs/examples/example-example15/script.js new file mode 100644 index 0000000000..b7836f0f57 --- /dev/null +++ b/1.4.10/docs/examples/example-example15/script.js @@ -0,0 +1,22 @@ +(function(angular) { + 'use strict'; +angular.module('docsScopeProblemExample', []) + .controller('NaomiController', ['$scope', function($scope) { + $scope.customer = { + name: 'Naomi', + address: '1600 Amphitheatre' + }; + }]) + .controller('IgorController', ['$scope', function($scope) { + $scope.customer = { + name: 'Igor', + address: '123 Somewhere' + }; + }]) + .directive('myCustomer', function() { + return { + restrict: 'E', + templateUrl: 'my-customer.html' + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example16/index-debug.html b/1.4.10/docs/examples/example-example16/index-debug.html new file mode 100644 index 0000000000..91a9b19780 --- /dev/null +++ b/1.4.10/docs/examples/example-example16/index-debug.html @@ -0,0 +1,21 @@ + + + + + Example - example-example16-debug + + + + + + + + + +
+ +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example16/index-jquery.html b/1.4.10/docs/examples/example-example16/index-jquery.html new file mode 100644 index 0000000000..2a7eba0a98 --- /dev/null +++ b/1.4.10/docs/examples/example-example16/index-jquery.html @@ -0,0 +1,22 @@ + + + + + Example - example-example16-jquery + + + + + + + + + + +
+ +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example16/index-production.html b/1.4.10/docs/examples/example-example16/index-production.html new file mode 100644 index 0000000000..2e4f2dd4ce --- /dev/null +++ b/1.4.10/docs/examples/example-example16/index-production.html @@ -0,0 +1,21 @@ + + + + + Example - example-example16-production + + + + + + + + + +
+ +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example16/index.html b/1.4.10/docs/examples/example-example16/index.html new file mode 100644 index 0000000000..0e9cab58b3 --- /dev/null +++ b/1.4.10/docs/examples/example-example16/index.html @@ -0,0 +1,21 @@ + + + + + Example - example-example16 + + + + + + + + + +
+ +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example16/manifest.json b/1.4.10/docs/examples/example-example16/manifest.json new file mode 100644 index 0000000000..37f154617c --- /dev/null +++ b/1.4.10/docs/examples/example-example16/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example16", + "files": [ + "index-production.html", + "script.js", + "my-customer-iso.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example16/my-customer-iso.html b/1.4.10/docs/examples/example-example16/my-customer-iso.html new file mode 100644 index 0000000000..05dbcffaa6 --- /dev/null +++ b/1.4.10/docs/examples/example-example16/my-customer-iso.html @@ -0,0 +1 @@ +Name: {{customerInfo.name}} Address: {{customerInfo.address}} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example16/script.js b/1.4.10/docs/examples/example-example16/script.js new file mode 100644 index 0000000000..32214a33b0 --- /dev/null +++ b/1.4.10/docs/examples/example-example16/script.js @@ -0,0 +1,17 @@ +(function(angular) { + 'use strict'; +angular.module('docsIsolateScopeDirective', []) + .controller('Controller', ['$scope', function($scope) { + $scope.naomi = { name: 'Naomi', address: '1600 Amphitheatre' }; + $scope.igor = { name: 'Igor', address: '123 Somewhere' }; + }]) + .directive('myCustomer', function() { + return { + restrict: 'E', + scope: { + customerInfo: '=info' + }, + templateUrl: 'my-customer-iso.html' + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example17/index-debug.html b/1.4.10/docs/examples/example-example17/index-debug.html new file mode 100644 index 0000000000..a99e1e2a71 --- /dev/null +++ b/1.4.10/docs/examples/example-example17/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example17-debug + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example17/index-jquery.html b/1.4.10/docs/examples/example-example17/index-jquery.html new file mode 100644 index 0000000000..87fbb155cd --- /dev/null +++ b/1.4.10/docs/examples/example-example17/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example17-jquery + + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example17/index-production.html b/1.4.10/docs/examples/example-example17/index-production.html new file mode 100644 index 0000000000..c6bf75d4e6 --- /dev/null +++ b/1.4.10/docs/examples/example-example17/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example17-production + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example17/index.html b/1.4.10/docs/examples/example-example17/index.html new file mode 100644 index 0000000000..1462f3f670 --- /dev/null +++ b/1.4.10/docs/examples/example-example17/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example17 + + + + + + + + + +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example17/manifest.json b/1.4.10/docs/examples/example-example17/manifest.json new file mode 100644 index 0000000000..35d7820211 --- /dev/null +++ b/1.4.10/docs/examples/example-example17/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example17", + "files": [ + "index-production.html", + "script.js", + "my-customer-plus-vojta.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example17/my-customer-plus-vojta.html b/1.4.10/docs/examples/example-example17/my-customer-plus-vojta.html new file mode 100644 index 0000000000..8e357242d3 --- /dev/null +++ b/1.4.10/docs/examples/example-example17/my-customer-plus-vojta.html @@ -0,0 +1,3 @@ +Name: {{customerInfo.name}} Address: {{customerInfo.address}} +
+Name: {{vojta.name}} Address: {{vojta.address}} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example17/script.js b/1.4.10/docs/examples/example-example17/script.js new file mode 100644 index 0000000000..e7dcebd5e8 --- /dev/null +++ b/1.4.10/docs/examples/example-example17/script.js @@ -0,0 +1,17 @@ +(function(angular) { + 'use strict'; +angular.module('docsIsolationExample', []) + .controller('Controller', ['$scope', function($scope) { + $scope.naomi = { name: 'Naomi', address: '1600 Amphitheatre' }; + $scope.vojta = { name: 'Vojta', address: '3456 Somewhere Else' }; + }]) + .directive('myCustomer', function() { + return { + restrict: 'E', + scope: { + customerInfo: '=info' + }, + templateUrl: 'my-customer-plus-vojta.html' + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example18/index-debug.html b/1.4.10/docs/examples/example-example18/index-debug.html new file mode 100644 index 0000000000..6997b7d9cf --- /dev/null +++ b/1.4.10/docs/examples/example-example18/index-debug.html @@ -0,0 +1,20 @@ + + + + + Example - example-example18-debug + + + + + + + + + +
+ Date format:
+ Current time is: +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example18/index-jquery.html b/1.4.10/docs/examples/example-example18/index-jquery.html new file mode 100644 index 0000000000..f033c70a92 --- /dev/null +++ b/1.4.10/docs/examples/example-example18/index-jquery.html @@ -0,0 +1,21 @@ + + + + + Example - example-example18-jquery + + + + + + + + + + +
+ Date format:
+ Current time is: +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example18/index-production.html b/1.4.10/docs/examples/example-example18/index-production.html new file mode 100644 index 0000000000..0d3c984ccf --- /dev/null +++ b/1.4.10/docs/examples/example-example18/index-production.html @@ -0,0 +1,20 @@ + + + + + Example - example-example18-production + + + + + + + + + +
+ Date format:
+ Current time is: +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example18/index.html b/1.4.10/docs/examples/example-example18/index.html new file mode 100644 index 0000000000..449429b2b7 --- /dev/null +++ b/1.4.10/docs/examples/example-example18/index.html @@ -0,0 +1,20 @@ + + + + + Example - example-example18 + + + + + + + + + +
+ Date format:
+ Current time is: +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example18/manifest.json b/1.4.10/docs/examples/example-example18/manifest.json new file mode 100644 index 0000000000..d59252fdec --- /dev/null +++ b/1.4.10/docs/examples/example-example18/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example18", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example18/script.js b/1.4.10/docs/examples/example-example18/script.js new file mode 100644 index 0000000000..e8b422fcb3 --- /dev/null +++ b/1.4.10/docs/examples/example-example18/script.js @@ -0,0 +1,36 @@ +(function(angular) { + 'use strict'; +angular.module('docsTimeDirective', []) + .controller('Controller', ['$scope', function($scope) { + $scope.format = 'M/d/yy h:mm:ss a'; + }]) + .directive('myCurrentTime', ['$interval', 'dateFilter', function($interval, dateFilter) { + + function link(scope, element, attrs) { + var format, + timeoutId; + + function updateTime() { + element.text(dateFilter(new Date(), format)); + } + + scope.$watch(attrs.myCurrentTime, function(value) { + format = value; + updateTime(); + }); + + element.on('$destroy', function() { + $interval.cancel(timeoutId); + }); + + // start the UI update process; save the timeoutId for canceling + timeoutId = $interval(function() { + updateTime(); // update DOM + }, 1000); + } + + return { + link: link + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example19/index-debug.html b/1.4.10/docs/examples/example-example19/index-debug.html new file mode 100644 index 0000000000..564ef62b77 --- /dev/null +++ b/1.4.10/docs/examples/example-example19/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example19-debug + + + + + + + + + +
+ Check out the contents, {{name}}! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example19/index-jquery.html b/1.4.10/docs/examples/example-example19/index-jquery.html new file mode 100644 index 0000000000..0227eddf36 --- /dev/null +++ b/1.4.10/docs/examples/example-example19/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example19-jquery + + + + + + + + + + +
+ Check out the contents, {{name}}! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example19/index-production.html b/1.4.10/docs/examples/example-example19/index-production.html new file mode 100644 index 0000000000..66315a66da --- /dev/null +++ b/1.4.10/docs/examples/example-example19/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example19-production + + + + + + + + + +
+ Check out the contents, {{name}}! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example19/index.html b/1.4.10/docs/examples/example-example19/index.html new file mode 100644 index 0000000000..2acd168d7e --- /dev/null +++ b/1.4.10/docs/examples/example-example19/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example19 + + + + + + + + + +
+ Check out the contents, {{name}}! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example19/manifest.json b/1.4.10/docs/examples/example-example19/manifest.json new file mode 100644 index 0000000000..8371f64210 --- /dev/null +++ b/1.4.10/docs/examples/example-example19/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example19", + "files": [ + "index-production.html", + "script.js", + "my-dialog.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example19/my-dialog.html b/1.4.10/docs/examples/example-example19/my-dialog.html new file mode 100644 index 0000000000..d32089bfe4 --- /dev/null +++ b/1.4.10/docs/examples/example-example19/my-dialog.html @@ -0,0 +1 @@ +
\ No newline at end of file diff --git a/1.4.10/docs/examples/example-example19/script.js b/1.4.10/docs/examples/example-example19/script.js new file mode 100644 index 0000000000..8dac254da8 --- /dev/null +++ b/1.4.10/docs/examples/example-example19/script.js @@ -0,0 +1,15 @@ +(function(angular) { + 'use strict'; +angular.module('docsTransclusionDirective', []) + .controller('Controller', ['$scope', function($scope) { + $scope.name = 'Tobias'; + }]) + .directive('myDialog', function() { + return { + restrict: 'E', + transclude: true, + scope: {}, + templateUrl: 'my-dialog.html' + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example2/index-debug.html b/1.4.10/docs/examples/example-example2/index-debug.html new file mode 100644 index 0000000000..d3b423a84e --- /dev/null +++ b/1.4.10/docs/examples/example-example2/index-debug.html @@ -0,0 +1,86 @@ + + + + + Example - example-example2-debug + + + + + + + + + + +
+
+ + + Custom Checkbox + +
+
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example2/index-jquery.html b/1.4.10/docs/examples/example-example2/index-jquery.html new file mode 100644 index 0000000000..8df1780b3b --- /dev/null +++ b/1.4.10/docs/examples/example-example2/index-jquery.html @@ -0,0 +1,87 @@ + + + + + Example - example-example2-jquery + + + + + + + + + + + +
+
+ + + Custom Checkbox + +
+
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example2/index-production.html b/1.4.10/docs/examples/example-example2/index-production.html new file mode 100644 index 0000000000..b26638eca2 --- /dev/null +++ b/1.4.10/docs/examples/example-example2/index-production.html @@ -0,0 +1,86 @@ + + + + + Example - example-example2-production + + + + + + + + + + +
+
+ + + Custom Checkbox + +
+
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example2/index.html b/1.4.10/docs/examples/example-example2/index.html new file mode 100644 index 0000000000..c14d48cb27 --- /dev/null +++ b/1.4.10/docs/examples/example-example2/index.html @@ -0,0 +1,86 @@ + + + + + Example - example-example2 + + + + + + + + + + +
+
+ + + Custom Checkbox + +
+
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example2/manifest.json b/1.4.10/docs/examples/example-example2/manifest.json new file mode 100644 index 0000000000..11d9dcf70c --- /dev/null +++ b/1.4.10/docs/examples/example-example2/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example2", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example20/index-debug.html b/1.4.10/docs/examples/example-example20/index-debug.html new file mode 100644 index 0000000000..560181e18c --- /dev/null +++ b/1.4.10/docs/examples/example-example20/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example20-debug + + + + + + + + + +
+ Check out the contents, {{name}}! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example20/index-jquery.html b/1.4.10/docs/examples/example-example20/index-jquery.html new file mode 100644 index 0000000000..959ff88420 --- /dev/null +++ b/1.4.10/docs/examples/example-example20/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example20-jquery + + + + + + + + + + +
+ Check out the contents, {{name}}! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example20/index-production.html b/1.4.10/docs/examples/example-example20/index-production.html new file mode 100644 index 0000000000..8ae4583f4a --- /dev/null +++ b/1.4.10/docs/examples/example-example20/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example20-production + + + + + + + + + +
+ Check out the contents, {{name}}! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example20/index.html b/1.4.10/docs/examples/example-example20/index.html new file mode 100644 index 0000000000..a65cfb501b --- /dev/null +++ b/1.4.10/docs/examples/example-example20/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example20 + + + + + + + + + +
+ Check out the contents, {{name}}! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example20/manifest.json b/1.4.10/docs/examples/example-example20/manifest.json new file mode 100644 index 0000000000..11f231c293 --- /dev/null +++ b/1.4.10/docs/examples/example-example20/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example20", + "files": [ + "index-production.html", + "script.js", + "my-dialog.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example20/my-dialog.html b/1.4.10/docs/examples/example-example20/my-dialog.html new file mode 100644 index 0000000000..d32089bfe4 --- /dev/null +++ b/1.4.10/docs/examples/example-example20/my-dialog.html @@ -0,0 +1 @@ +
\ No newline at end of file diff --git a/1.4.10/docs/examples/example-example20/script.js b/1.4.10/docs/examples/example-example20/script.js new file mode 100644 index 0000000000..9fc8396231 --- /dev/null +++ b/1.4.10/docs/examples/example-example20/script.js @@ -0,0 +1,18 @@ +(function(angular) { + 'use strict'; +angular.module('docsTransclusionExample', []) + .controller('Controller', ['$scope', function($scope) { + $scope.name = 'Tobias'; + }]) + .directive('myDialog', function() { + return { + restrict: 'E', + transclude: true, + scope: {}, + templateUrl: 'my-dialog.html', + link: function (scope) { + scope.name = 'Jeff'; + } + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example21/index-debug.html b/1.4.10/docs/examples/example-example21/index-debug.html new file mode 100644 index 0000000000..f517adf326 --- /dev/null +++ b/1.4.10/docs/examples/example-example21/index-debug.html @@ -0,0 +1,22 @@ + + + + + Example - example-example21-debug + + + + + + + + + +
+ {{message}} + + Check out the contents, {{name}}! + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example21/index-jquery.html b/1.4.10/docs/examples/example-example21/index-jquery.html new file mode 100644 index 0000000000..b02f66c61d --- /dev/null +++ b/1.4.10/docs/examples/example-example21/index-jquery.html @@ -0,0 +1,23 @@ + + + + + Example - example-example21-jquery + + + + + + + + + + +
+ {{message}} + + Check out the contents, {{name}}! + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example21/index-production.html b/1.4.10/docs/examples/example-example21/index-production.html new file mode 100644 index 0000000000..0ca3a6b60c --- /dev/null +++ b/1.4.10/docs/examples/example-example21/index-production.html @@ -0,0 +1,22 @@ + + + + + Example - example-example21-production + + + + + + + + + +
+ {{message}} + + Check out the contents, {{name}}! + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example21/index.html b/1.4.10/docs/examples/example-example21/index.html new file mode 100644 index 0000000000..603d0e308e --- /dev/null +++ b/1.4.10/docs/examples/example-example21/index.html @@ -0,0 +1,22 @@ + + + + + Example - example-example21 + + + + + + + + + +
+ {{message}} + + Check out the contents, {{name}}! + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example21/manifest.json b/1.4.10/docs/examples/example-example21/manifest.json new file mode 100644 index 0000000000..a4bdc6e13a --- /dev/null +++ b/1.4.10/docs/examples/example-example21/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example21", + "files": [ + "index-production.html", + "script.js", + "my-dialog-close.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example21/my-dialog-close.html b/1.4.10/docs/examples/example-example21/my-dialog-close.html new file mode 100644 index 0000000000..e38f28f714 --- /dev/null +++ b/1.4.10/docs/examples/example-example21/my-dialog-close.html @@ -0,0 +1,4 @@ +
+ × +
+
\ No newline at end of file diff --git a/1.4.10/docs/examples/example-example21/script.js b/1.4.10/docs/examples/example-example21/script.js new file mode 100644 index 0000000000..28916d458f --- /dev/null +++ b/1.4.10/docs/examples/example-example21/script.js @@ -0,0 +1,26 @@ +(function(angular) { + 'use strict'; +angular.module('docsIsoFnBindExample', []) + .controller('Controller', ['$scope', '$timeout', function($scope, $timeout) { + $scope.name = 'Tobias'; + $scope.message = ''; + $scope.hideDialog = function (message) { + $scope.message = message; + $scope.dialogIsHidden = true; + $timeout(function () { + $scope.message = ''; + $scope.dialogIsHidden = false; + }, 2000); + }; + }]) + .directive('myDialog', function() { + return { + restrict: 'E', + transclude: true, + scope: { + 'close': '&onClose' + }, + templateUrl: 'my-dialog-close.html' + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example22/index-debug.html b/1.4.10/docs/examples/example-example22/index-debug.html new file mode 100644 index 0000000000..54b27651f3 --- /dev/null +++ b/1.4.10/docs/examples/example-example22/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example22-debug + + + + + + + + + + Drag Me + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example22/index-jquery.html b/1.4.10/docs/examples/example-example22/index-jquery.html new file mode 100644 index 0000000000..7fee8b45dd --- /dev/null +++ b/1.4.10/docs/examples/example-example22/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example22-jquery + + + + + + + + + + + Drag Me + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example22/index-production.html b/1.4.10/docs/examples/example-example22/index-production.html new file mode 100644 index 0000000000..4e335ed3b8 --- /dev/null +++ b/1.4.10/docs/examples/example-example22/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example22-production + + + + + + + + + + Drag Me + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example22/index.html b/1.4.10/docs/examples/example-example22/index.html new file mode 100644 index 0000000000..176228b32c --- /dev/null +++ b/1.4.10/docs/examples/example-example22/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example22 + + + + + + + + + + Drag Me + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example22/manifest.json b/1.4.10/docs/examples/example-example22/manifest.json new file mode 100644 index 0000000000..f616d75e66 --- /dev/null +++ b/1.4.10/docs/examples/example-example22/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example22", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example22/script.js b/1.4.10/docs/examples/example-example22/script.js new file mode 100644 index 0000000000..aa55850155 --- /dev/null +++ b/1.4.10/docs/examples/example-example22/script.js @@ -0,0 +1,41 @@ +(function(angular) { + 'use strict'; +angular.module('dragModule', []) + .directive('myDraggable', ['$document', function($document) { + return { + link: function(scope, element, attr) { + var startX = 0, startY = 0, x = 0, y = 0; + + element.css({ + position: 'relative', + border: '1px solid red', + backgroundColor: 'lightgrey', + cursor: 'pointer' + }); + + element.on('mousedown', function(event) { + // Prevent default dragging of selected content + event.preventDefault(); + startX = event.pageX - x; + startY = event.pageY - y; + $document.on('mousemove', mousemove); + $document.on('mouseup', mouseup); + }); + + function mousemove(event) { + y = event.pageY - startY; + x = event.pageX - startX; + element.css({ + top: y + 'px', + left: x + 'px' + }); + } + + function mouseup() { + $document.off('mousemove', mousemove); + $document.off('mouseup', mouseup); + } + } + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example23/index-debug.html b/1.4.10/docs/examples/example-example23/index-debug.html new file mode 100644 index 0000000000..92f26ed566 --- /dev/null +++ b/1.4.10/docs/examples/example-example23/index-debug.html @@ -0,0 +1,25 @@ + + + + + Example - example-example23-debug + + + + + + + + + + + +

Lorem ipsum dolor sit amet

+
+ + Mauris elementum elementum enim at suscipit. +

counter: {{i || 0}}

+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example23/index-jquery.html b/1.4.10/docs/examples/example-example23/index-jquery.html new file mode 100644 index 0000000000..48af4b4f37 --- /dev/null +++ b/1.4.10/docs/examples/example-example23/index-jquery.html @@ -0,0 +1,26 @@ + + + + + Example - example-example23-jquery + + + + + + + + + + + + +

Lorem ipsum dolor sit amet

+
+ + Mauris elementum elementum enim at suscipit. +

counter: {{i || 0}}

+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example23/index-production.html b/1.4.10/docs/examples/example-example23/index-production.html new file mode 100644 index 0000000000..ab8bd7f9de --- /dev/null +++ b/1.4.10/docs/examples/example-example23/index-production.html @@ -0,0 +1,25 @@ + + + + + Example - example-example23-production + + + + + + + + + + + +

Lorem ipsum dolor sit amet

+
+ + Mauris elementum elementum enim at suscipit. +

counter: {{i || 0}}

+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example23/index.html b/1.4.10/docs/examples/example-example23/index.html new file mode 100644 index 0000000000..52b1a6f6e1 --- /dev/null +++ b/1.4.10/docs/examples/example-example23/index.html @@ -0,0 +1,25 @@ + + + + + Example - example-example23 + + + + + + + + + + + +

Lorem ipsum dolor sit amet

+
+ + Mauris elementum elementum enim at suscipit. +

counter: {{i || 0}}

+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example23/manifest.json b/1.4.10/docs/examples/example-example23/manifest.json new file mode 100644 index 0000000000..4a6eaf31e3 --- /dev/null +++ b/1.4.10/docs/examples/example-example23/manifest.json @@ -0,0 +1,9 @@ +{ + "name": "example-example23", + "files": [ + "index-production.html", + "script.js", + "my-tabs.html", + "my-pane.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example23/my-pane.html b/1.4.10/docs/examples/example-example23/my-pane.html new file mode 100644 index 0000000000..71284f7b1d --- /dev/null +++ b/1.4.10/docs/examples/example-example23/my-pane.html @@ -0,0 +1,4 @@ +
+

{{title}}

+
+
\ No newline at end of file diff --git a/1.4.10/docs/examples/example-example23/my-tabs.html b/1.4.10/docs/examples/example-example23/my-tabs.html new file mode 100644 index 0000000000..38c81a6968 --- /dev/null +++ b/1.4.10/docs/examples/example-example23/my-tabs.html @@ -0,0 +1,8 @@ +
+ +
+
\ No newline at end of file diff --git a/1.4.10/docs/examples/example-example23/script.js b/1.4.10/docs/examples/example-example23/script.js new file mode 100644 index 0000000000..7420a2530c --- /dev/null +++ b/1.4.10/docs/examples/example-example23/script.js @@ -0,0 +1,43 @@ +(function(angular) { + 'use strict'; +angular.module('docsTabsExample', []) + .directive('myTabs', function() { + return { + restrict: 'E', + transclude: true, + scope: {}, + controller: ['$scope', function($scope) { + var panes = $scope.panes = []; + + $scope.select = function(pane) { + angular.forEach(panes, function(pane) { + pane.selected = false; + }); + pane.selected = true; + }; + + this.addPane = function(pane) { + if (panes.length === 0) { + $scope.select(pane); + } + panes.push(pane); + }; + }], + templateUrl: 'my-tabs.html' + }; + }) + .directive('myPane', function() { + return { + require: '^^myTabs', + restrict: 'E', + transclude: true, + scope: { + title: '@' + }, + link: function(scope, element, attrs, tabsCtrl) { + tabsCtrl.addPane(scope); + }, + templateUrl: 'my-pane.html' + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example24/index-debug.html b/1.4.10/docs/examples/example-example24/index-debug.html new file mode 100644 index 0000000000..b11c68cd41 --- /dev/null +++ b/1.4.10/docs/examples/example-example24/index-debug.html @@ -0,0 +1,18 @@ + + + + + Example - example-example24-debug + + + + + + + + + + 1+2={{1+2}} + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example24/index-jquery.html b/1.4.10/docs/examples/example-example24/index-jquery.html new file mode 100644 index 0000000000..7f25182228 --- /dev/null +++ b/1.4.10/docs/examples/example-example24/index-jquery.html @@ -0,0 +1,19 @@ + + + + + Example - example-example24-jquery + + + + + + + + + + + 1+2={{1+2}} + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example24/index-production.html b/1.4.10/docs/examples/example-example24/index-production.html new file mode 100644 index 0000000000..574fd9ac49 --- /dev/null +++ b/1.4.10/docs/examples/example-example24/index-production.html @@ -0,0 +1,18 @@ + + + + + Example - example-example24-production + + + + + + + + + + 1+2={{1+2}} + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example24/index.html b/1.4.10/docs/examples/example-example24/index.html new file mode 100644 index 0000000000..4f44eb0923 --- /dev/null +++ b/1.4.10/docs/examples/example-example24/index.html @@ -0,0 +1,18 @@ + + + + + Example - example-example24 + + + + + + + + + + 1+2={{1+2}} + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example24/manifest.json b/1.4.10/docs/examples/example-example24/manifest.json new file mode 100644 index 0000000000..b8055f114e --- /dev/null +++ b/1.4.10/docs/examples/example-example24/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example24", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example24/protractor.js b/1.4.10/docs/examples/example-example24/protractor.js new file mode 100644 index 0000000000..045a6cf6e5 --- /dev/null +++ b/1.4.10/docs/examples/example-example24/protractor.js @@ -0,0 +1,3 @@ +it('should calculate expression in binding', function() { + expect(element(by.binding('1+2')).getText()).toEqual('1+2=3'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example25/index-debug.html b/1.4.10/docs/examples/example-example25/index-debug.html new file mode 100644 index 0000000000..5fe50ae1ed --- /dev/null +++ b/1.4.10/docs/examples/example-example25/index-debug.html @@ -0,0 +1,27 @@ + + + + + Example - example-example25-debug + + + + + + + + + +
+ Expression: + + +
    +
  • + [ X ] + {{expr}} => +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example25/index-jquery.html b/1.4.10/docs/examples/example-example25/index-jquery.html new file mode 100644 index 0000000000..c25e770aab --- /dev/null +++ b/1.4.10/docs/examples/example-example25/index-jquery.html @@ -0,0 +1,28 @@ + + + + + Example - example-example25-jquery + + + + + + + + + + +
+ Expression: + + +
    +
  • + [ X ] + {{expr}} => +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example25/index-production.html b/1.4.10/docs/examples/example-example25/index-production.html new file mode 100644 index 0000000000..dea54ca367 --- /dev/null +++ b/1.4.10/docs/examples/example-example25/index-production.html @@ -0,0 +1,27 @@ + + + + + Example - example-example25-production + + + + + + + + + +
+ Expression: + + +
    +
  • + [ X ] + {{expr}} => +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example25/index.html b/1.4.10/docs/examples/example-example25/index.html new file mode 100644 index 0000000000..a93dccfdef --- /dev/null +++ b/1.4.10/docs/examples/example-example25/index.html @@ -0,0 +1,27 @@ + + + + + Example - example-example25 + + + + + + + + + +
+ Expression: + + +
    +
  • + [ X ] + {{expr}} => +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example25/manifest.json b/1.4.10/docs/examples/example-example25/manifest.json new file mode 100644 index 0000000000..f1f83e8cb2 --- /dev/null +++ b/1.4.10/docs/examples/example-example25/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example25", + "files": [ + "index-production.html", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example25/protractor.js b/1.4.10/docs/examples/example-example25/protractor.js new file mode 100644 index 0000000000..f1a99715fa --- /dev/null +++ b/1.4.10/docs/examples/example-example25/protractor.js @@ -0,0 +1,6 @@ +it('should allow user expression testing', function() { + element(by.css('.expressions button')).click(); + var lis = element(by.css('.expressions ul')).all(by.repeater('expr in exprs')); + expect(lis.count()).toBe(1); + expect(lis.get(0).getText()).toEqual('[ X ] 3*10|currency => $30.00'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example25/script.js b/1.4.10/docs/examples/example-example25/script.js new file mode 100644 index 0000000000..8d7394e82d --- /dev/null +++ b/1.4.10/docs/examples/example-example25/script.js @@ -0,0 +1,15 @@ +(function(angular) { + 'use strict'; +angular.module('expressionExample', []) + .controller('ExampleController', ['$scope', function($scope) { + var exprs = $scope.exprs = []; + $scope.expr = '3*10|currency'; + $scope.addExp = function(expr) { + exprs.push(expr); + }; + + $scope.removeExp = function(index) { + exprs.splice(index, 1); + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example26/index-debug.html b/1.4.10/docs/examples/example-example26/index-debug.html new file mode 100644 index 0000000000..d9753da5be --- /dev/null +++ b/1.4.10/docs/examples/example-example26/index-debug.html @@ -0,0 +1,21 @@ + + + + + Example - example-example26-debug + + + + + + + + + +
+ Name: + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example26/index-jquery.html b/1.4.10/docs/examples/example-example26/index-jquery.html new file mode 100644 index 0000000000..f4320ec472 --- /dev/null +++ b/1.4.10/docs/examples/example-example26/index-jquery.html @@ -0,0 +1,22 @@ + + + + + Example - example-example26-jquery + + + + + + + + + + +
+ Name: + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example26/index-production.html b/1.4.10/docs/examples/example-example26/index-production.html new file mode 100644 index 0000000000..20a01285d4 --- /dev/null +++ b/1.4.10/docs/examples/example-example26/index-production.html @@ -0,0 +1,21 @@ + + + + + Example - example-example26-production + + + + + + + + + +
+ Name: + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example26/index.html b/1.4.10/docs/examples/example-example26/index.html new file mode 100644 index 0000000000..49acbabd8f --- /dev/null +++ b/1.4.10/docs/examples/example-example26/index.html @@ -0,0 +1,21 @@ + + + + + Example - example-example26 + + + + + + + + + +
+ Name: + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example26/manifest.json b/1.4.10/docs/examples/example-example26/manifest.json new file mode 100644 index 0000000000..3ecaf2432c --- /dev/null +++ b/1.4.10/docs/examples/example-example26/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example26", + "files": [ + "index-production.html", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example26/protractor.js b/1.4.10/docs/examples/example-example26/protractor.js new file mode 100644 index 0000000000..a90ba10209 --- /dev/null +++ b/1.4.10/docs/examples/example-example26/protractor.js @@ -0,0 +1,16 @@ +it('should calculate expression in binding', function() { + if (browser.params.browser == 'safari') { + // Safari can't handle dialogs. + return; + } + element(by.css('[ng-click="greet()"]')).click(); + + // We need to give the browser time to display the alert + browser.wait(protractor.ExpectedConditions.alertIsPresent(), 1000); + + var alertDialog = browser.switchTo().alert(); + + expect(alertDialog.getText()).toEqual('Hello World'); + + alertDialog.accept(); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example26/script.js b/1.4.10/docs/examples/example-example26/script.js new file mode 100644 index 0000000000..3b0967a058 --- /dev/null +++ b/1.4.10/docs/examples/example-example26/script.js @@ -0,0 +1,11 @@ +(function(angular) { + 'use strict'; +angular.module('expressionExample', []) + .controller('ExampleController', ['$window', '$scope', function($window, $scope) { + $scope.name = 'World'; + + $scope.greet = function() { + $window.alert('Hello ' + $scope.name); + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example27/index-debug.html b/1.4.10/docs/examples/example-example27/index-debug.html new file mode 100644 index 0000000000..8e70d96c83 --- /dev/null +++ b/1.4.10/docs/examples/example-example27/index-debug.html @@ -0,0 +1,21 @@ + + + + + Example - example-example27-debug + + + + + + + + + +
+ +

$event:

 {{$event | json}}

+

clickEvent:

{{clickEvent | json}}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example27/index-jquery.html b/1.4.10/docs/examples/example-example27/index-jquery.html new file mode 100644 index 0000000000..cd67386825 --- /dev/null +++ b/1.4.10/docs/examples/example-example27/index-jquery.html @@ -0,0 +1,22 @@ + + + + + Example - example-example27-jquery + + + + + + + + + + +
+ +

$event:

 {{$event | json}}

+

clickEvent:

{{clickEvent | json}}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example27/index-production.html b/1.4.10/docs/examples/example-example27/index-production.html new file mode 100644 index 0000000000..7abf742cc9 --- /dev/null +++ b/1.4.10/docs/examples/example-example27/index-production.html @@ -0,0 +1,21 @@ + + + + + Example - example-example27-production + + + + + + + + + +
+ +

$event:

 {{$event | json}}

+

clickEvent:

{{clickEvent | json}}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example27/index.html b/1.4.10/docs/examples/example-example27/index.html new file mode 100644 index 0000000000..4f629493a0 --- /dev/null +++ b/1.4.10/docs/examples/example-example27/index.html @@ -0,0 +1,21 @@ + + + + + Example - example-example27 + + + + + + + + + +
+ +

$event:

 {{$event | json}}

+

clickEvent:

{{clickEvent | json}}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example27/manifest.json b/1.4.10/docs/examples/example-example27/manifest.json new file mode 100644 index 0000000000..b81632aa17 --- /dev/null +++ b/1.4.10/docs/examples/example-example27/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example27", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example27/script.js b/1.4.10/docs/examples/example-example27/script.js new file mode 100644 index 0000000000..02caac1791 --- /dev/null +++ b/1.4.10/docs/examples/example-example27/script.js @@ -0,0 +1,24 @@ +(function(angular) { + 'use strict'; +angular.module('eventExampleApp', []). + controller('EventController', ['$scope', function($scope) { + /* + * expose the event object to the scope + */ + $scope.clickMe = function(clickEvent) { + $scope.clickEvent = simpleKeys(clickEvent); + console.log(clickEvent); + }; + + /* + * return a copy of an object with only non-object keys + * we need this to avoid circular references + */ + function simpleKeys (original) { + return Object.keys(original).reduce(function (obj, key) { + obj[key] = typeof original[key] === 'object' ? '{ ... }' : original[key]; + return obj; + }, {}); + } + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example28/index-debug.html b/1.4.10/docs/examples/example-example28/index-debug.html new file mode 100644 index 0000000000..a2065ab6f7 --- /dev/null +++ b/1.4.10/docs/examples/example-example28/index-debug.html @@ -0,0 +1,21 @@ + + + + + Example - example-example28-debug + + + + + + + + + +
+ +

One time binding: {{::name}}

+

Normal binding: {{name}}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example28/index-jquery.html b/1.4.10/docs/examples/example-example28/index-jquery.html new file mode 100644 index 0000000000..fddbeee6bb --- /dev/null +++ b/1.4.10/docs/examples/example-example28/index-jquery.html @@ -0,0 +1,22 @@ + + + + + Example - example-example28-jquery + + + + + + + + + + +
+ +

One time binding: {{::name}}

+

Normal binding: {{name}}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example28/index-production.html b/1.4.10/docs/examples/example-example28/index-production.html new file mode 100644 index 0000000000..562709d0ac --- /dev/null +++ b/1.4.10/docs/examples/example-example28/index-production.html @@ -0,0 +1,21 @@ + + + + + Example - example-example28-production + + + + + + + + + +
+ +

One time binding: {{::name}}

+

Normal binding: {{name}}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example28/index.html b/1.4.10/docs/examples/example-example28/index.html new file mode 100644 index 0000000000..e5c466635e --- /dev/null +++ b/1.4.10/docs/examples/example-example28/index.html @@ -0,0 +1,21 @@ + + + + + Example - example-example28 + + + + + + + + + +
+ +

One time binding: {{::name}}

+

Normal binding: {{name}}

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example28/manifest.json b/1.4.10/docs/examples/example-example28/manifest.json new file mode 100644 index 0000000000..01b527137e --- /dev/null +++ b/1.4.10/docs/examples/example-example28/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example28", + "files": [ + "index-production.html", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example28/protractor.js b/1.4.10/docs/examples/example-example28/protractor.js new file mode 100644 index 0000000000..639ff0b339 --- /dev/null +++ b/1.4.10/docs/examples/example-example28/protractor.js @@ -0,0 +1,21 @@ +it('should freeze binding after its value has stabilized', function() { + var oneTimeBiding = element(by.id('one-time-binding-example')); + var normalBinding = element(by.id('normal-binding-example')); + + expect(oneTimeBiding.getText()).toEqual('One time binding:'); + expect(normalBinding.getText()).toEqual('Normal binding:'); + element(by.buttonText('Click Me')).click(); + + expect(oneTimeBiding.getText()).toEqual('One time binding: Igor'); + expect(normalBinding.getText()).toEqual('Normal binding: Igor'); + element(by.buttonText('Click Me')).click(); + + expect(oneTimeBiding.getText()).toEqual('One time binding: Igor'); + expect(normalBinding.getText()).toEqual('Normal binding: Misko'); + + element(by.buttonText('Click Me')).click(); + element(by.buttonText('Click Me')).click(); + + expect(oneTimeBiding.getText()).toEqual('One time binding: Igor'); + expect(normalBinding.getText()).toEqual('Normal binding: Lucas'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example28/script.js b/1.4.10/docs/examples/example-example28/script.js new file mode 100644 index 0000000000..07e17a7649 --- /dev/null +++ b/1.4.10/docs/examples/example-example28/script.js @@ -0,0 +1,15 @@ +(function(angular) { + 'use strict'; +angular.module('oneTimeBidingExampleApp', []). + controller('EventController', ['$scope', function($scope) { + var counter = 0; + var names = ['Igor', 'Misko', 'Chirayu', 'Lucas']; + /* + * expose the event object to the scope + */ + $scope.clickMe = function(clickEvent) { + $scope.name = names[counter % names.length]; + counter++; + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example29/index-debug.html b/1.4.10/docs/examples/example-example29/index-debug.html new file mode 100644 index 0000000000..f9f88f8bfa --- /dev/null +++ b/1.4.10/docs/examples/example-example29/index-debug.html @@ -0,0 +1,26 @@ + + + + + Example - example-example29-debug + + + + + + + + + +
+
+ All entries: + {{entry.name}} +
+
+ Entries that contain an "a": + {{entry.name}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example29/index-jquery.html b/1.4.10/docs/examples/example-example29/index-jquery.html new file mode 100644 index 0000000000..7db91a0fab --- /dev/null +++ b/1.4.10/docs/examples/example-example29/index-jquery.html @@ -0,0 +1,27 @@ + + + + + Example - example-example29-jquery + + + + + + + + + + +
+
+ All entries: + {{entry.name}} +
+
+ Entries that contain an "a": + {{entry.name}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example29/index-production.html b/1.4.10/docs/examples/example-example29/index-production.html new file mode 100644 index 0000000000..a9b5cfcf15 --- /dev/null +++ b/1.4.10/docs/examples/example-example29/index-production.html @@ -0,0 +1,26 @@ + + + + + Example - example-example29-production + + + + + + + + + +
+
+ All entries: + {{entry.name}} +
+
+ Entries that contain an "a": + {{entry.name}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example29/index.html b/1.4.10/docs/examples/example-example29/index.html new file mode 100644 index 0000000000..cd9edc2d8a --- /dev/null +++ b/1.4.10/docs/examples/example-example29/index.html @@ -0,0 +1,26 @@ + + + + + Example - example-example29 + + + + + + + + + +
+
+ All entries: + {{entry.name}} +
+
+ Entries that contain an "a": + {{entry.name}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example29/manifest.json b/1.4.10/docs/examples/example-example29/manifest.json new file mode 100644 index 0000000000..ded23b1dba --- /dev/null +++ b/1.4.10/docs/examples/example-example29/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example29", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example29/script.js b/1.4.10/docs/examples/example-example29/script.js new file mode 100644 index 0000000000..233f44accd --- /dev/null +++ b/1.4.10/docs/examples/example-example29/script.js @@ -0,0 +1,15 @@ +(function(angular) { + 'use strict'; +angular.module('FilterInControllerModule', []). + controller('FilterController', ['filterFilter', function(filterFilter) { + this.array = [ + {name: 'Tobias'}, + {name: 'Jeff'}, + {name: 'Brian'}, + {name: 'Igor'}, + {name: 'James'}, + {name: 'Brad'} + ]; + this.filteredArray = filterFilter(this.array, 'a'); + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example3/index-debug.html b/1.4.10/docs/examples/example-example3/index-debug.html new file mode 100644 index 0000000000..7d38ecef22 --- /dev/null +++ b/1.4.10/docs/examples/example-example3/index-debug.html @@ -0,0 +1,44 @@ + + + + + Example - example-example3-debug + + + + + + + + + +
+ <div> with ng-click and bindRoleForClick, tabindex set to false +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example3/index-jquery.html b/1.4.10/docs/examples/example-example3/index-jquery.html new file mode 100644 index 0000000000..1dab317621 --- /dev/null +++ b/1.4.10/docs/examples/example-example3/index-jquery.html @@ -0,0 +1,45 @@ + + + + + Example - example-example3-jquery + + + + + + + + + + +
+ <div> with ng-click and bindRoleForClick, tabindex set to false +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example3/index-production.html b/1.4.10/docs/examples/example-example3/index-production.html new file mode 100644 index 0000000000..0a6358965f --- /dev/null +++ b/1.4.10/docs/examples/example-example3/index-production.html @@ -0,0 +1,44 @@ + + + + + Example - example-example3-production + + + + + + + + + +
+ <div> with ng-click and bindRoleForClick, tabindex set to false +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example3/index.html b/1.4.10/docs/examples/example-example3/index.html new file mode 100644 index 0000000000..37907c9398 --- /dev/null +++ b/1.4.10/docs/examples/example-example3/index.html @@ -0,0 +1,44 @@ + + + + + Example - example-example3 + + + + + + + + + +
+ <div> with ng-click and bindRoleForClick, tabindex set to false +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example3/manifest.json b/1.4.10/docs/examples/example-example3/manifest.json new file mode 100644 index 0000000000..6247aa8e99 --- /dev/null +++ b/1.4.10/docs/examples/example-example3/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example3", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example30/index-debug.html b/1.4.10/docs/examples/example-example30/index-debug.html new file mode 100644 index 0000000000..70e5b025e9 --- /dev/null +++ b/1.4.10/docs/examples/example-example30/index-debug.html @@ -0,0 +1,23 @@ + + + + + Example - example-example30-debug + + + + + + + + + +
+
+ No filter: {{greeting}}
+ Reverse: {{greeting|reverse}}
+ Reverse + uppercase: {{greeting|reverse:true}}
+ Reverse, filtered in controller: {{filteredGreeting}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example30/index-jquery.html b/1.4.10/docs/examples/example-example30/index-jquery.html new file mode 100644 index 0000000000..31c15b988d --- /dev/null +++ b/1.4.10/docs/examples/example-example30/index-jquery.html @@ -0,0 +1,24 @@ + + + + + Example - example-example30-jquery + + + + + + + + + + +
+
+ No filter: {{greeting}}
+ Reverse: {{greeting|reverse}}
+ Reverse + uppercase: {{greeting|reverse:true}}
+ Reverse, filtered in controller: {{filteredGreeting}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example30/index-production.html b/1.4.10/docs/examples/example-example30/index-production.html new file mode 100644 index 0000000000..d46e2be937 --- /dev/null +++ b/1.4.10/docs/examples/example-example30/index-production.html @@ -0,0 +1,23 @@ + + + + + Example - example-example30-production + + + + + + + + + +
+
+ No filter: {{greeting}}
+ Reverse: {{greeting|reverse}}
+ Reverse + uppercase: {{greeting|reverse:true}}
+ Reverse, filtered in controller: {{filteredGreeting}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example30/index.html b/1.4.10/docs/examples/example-example30/index.html new file mode 100644 index 0000000000..f751ebe877 --- /dev/null +++ b/1.4.10/docs/examples/example-example30/index.html @@ -0,0 +1,23 @@ + + + + + Example - example-example30 + + + + + + + + + +
+
+ No filter: {{greeting}}
+ Reverse: {{greeting|reverse}}
+ Reverse + uppercase: {{greeting|reverse:true}}
+ Reverse, filtered in controller: {{filteredGreeting}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example30/manifest.json b/1.4.10/docs/examples/example-example30/manifest.json new file mode 100644 index 0000000000..af950406b2 --- /dev/null +++ b/1.4.10/docs/examples/example-example30/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example30", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example30/script.js b/1.4.10/docs/examples/example-example30/script.js new file mode 100644 index 0000000000..11a155a7ed --- /dev/null +++ b/1.4.10/docs/examples/example-example30/script.js @@ -0,0 +1,22 @@ +(function(angular) { + 'use strict'; +angular.module('myReverseFilterApp', []) + .filter('reverse', function() { + return function(input, uppercase) { + input = input || ''; + var out = ""; + for (var i = 0; i < input.length; i++) { + out = input.charAt(i) + out; + } + // conditional based on optional argument + if (uppercase) { + out = out.toUpperCase(); + } + return out; + }; + }) + .controller('MyController', ['$scope', 'reverseFilter', function($scope, reverseFilter) { + $scope.greeting = 'hello'; + $scope.filteredGreeting = reverseFilter($scope.greeting); + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example31/index-debug.html b/1.4.10/docs/examples/example-example31/index-debug.html new file mode 100644 index 0000000000..43dfbda1f5 --- /dev/null +++ b/1.4.10/docs/examples/example-example31/index-debug.html @@ -0,0 +1,22 @@ + + + + + Example - example-example31-debug + + + + + + + + + +
+ Input:
+ Decoration:
+ No filter: {{greeting}}
+ Decorated: {{greeting | decorate}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example31/index-jquery.html b/1.4.10/docs/examples/example-example31/index-jquery.html new file mode 100644 index 0000000000..532c6cf6ea --- /dev/null +++ b/1.4.10/docs/examples/example-example31/index-jquery.html @@ -0,0 +1,23 @@ + + + + + Example - example-example31-jquery + + + + + + + + + + +
+ Input:
+ Decoration:
+ No filter: {{greeting}}
+ Decorated: {{greeting | decorate}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example31/index-production.html b/1.4.10/docs/examples/example-example31/index-production.html new file mode 100644 index 0000000000..f1c8bd778b --- /dev/null +++ b/1.4.10/docs/examples/example-example31/index-production.html @@ -0,0 +1,22 @@ + + + + + Example - example-example31-production + + + + + + + + + +
+ Input:
+ Decoration:
+ No filter: {{greeting}}
+ Decorated: {{greeting | decorate}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example31/index.html b/1.4.10/docs/examples/example-example31/index.html new file mode 100644 index 0000000000..b34043f259 --- /dev/null +++ b/1.4.10/docs/examples/example-example31/index.html @@ -0,0 +1,22 @@ + + + + + Example - example-example31 + + + + + + + + + +
+ Input:
+ Decoration:
+ No filter: {{greeting}}
+ Decorated: {{greeting | decorate}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example31/manifest.json b/1.4.10/docs/examples/example-example31/manifest.json new file mode 100644 index 0000000000..ad4dd988b9 --- /dev/null +++ b/1.4.10/docs/examples/example-example31/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example31", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example31/script.js b/1.4.10/docs/examples/example-example31/script.js new file mode 100644 index 0000000000..664b51b688 --- /dev/null +++ b/1.4.10/docs/examples/example-example31/script.js @@ -0,0 +1,18 @@ +(function(angular) { + 'use strict'; +angular.module('myStatefulFilterApp', []) + .filter('decorate', ['decoration', function(decoration) { + + function decorateFilter(input) { + return decoration.symbol + input + decoration.symbol; + } + decorateFilter.$stateful = true; + + return decorateFilter; + }]) + .controller('MyController', ['$scope', 'decoration', function($scope, decoration) { + $scope.greeting = 'hello'; + $scope.decoration = decoration; + }]) + .value('decoration', {symbol: '*'}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example32/index-debug.html b/1.4.10/docs/examples/example-example32/index-debug.html new file mode 100644 index 0000000000..03279e8302 --- /dev/null +++ b/1.4.10/docs/examples/example-example32/index-debug.html @@ -0,0 +1,44 @@ + + + + + Example - example-example32-debug + + + + + + + + +
+
+ Name:
+ E-mail:
+ Gender: male + female
+ + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example32/index-jquery.html b/1.4.10/docs/examples/example-example32/index-jquery.html new file mode 100644 index 0000000000..38b5592e61 --- /dev/null +++ b/1.4.10/docs/examples/example-example32/index-jquery.html @@ -0,0 +1,45 @@ + + + + + Example - example-example32-jquery + + + + + + + + + +
+
+ Name:
+ E-mail:
+ Gender: male + female
+ + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example32/index-production.html b/1.4.10/docs/examples/example-example32/index-production.html new file mode 100644 index 0000000000..544bc98e48 --- /dev/null +++ b/1.4.10/docs/examples/example-example32/index-production.html @@ -0,0 +1,44 @@ + + + + + Example - example-example32-production + + + + + + + + +
+
+ Name:
+ E-mail:
+ Gender: male + female
+ + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example32/index.html b/1.4.10/docs/examples/example-example32/index.html new file mode 100644 index 0000000000..3cf6292a65 --- /dev/null +++ b/1.4.10/docs/examples/example-example32/index.html @@ -0,0 +1,44 @@ + + + + + Example - example-example32 + + + + + + + + +
+
+ Name:
+ E-mail:
+ Gender: male + female
+ + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example32/manifest.json b/1.4.10/docs/examples/example-example32/manifest.json new file mode 100644 index 0000000000..e0e567b386 --- /dev/null +++ b/1.4.10/docs/examples/example-example32/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example32", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example33/index-debug.html b/1.4.10/docs/examples/example-example33/index-debug.html new file mode 100644 index 0000000000..4adca7f825 --- /dev/null +++ b/1.4.10/docs/examples/example-example33/index-debug.html @@ -0,0 +1,54 @@ + + + + + Example - example-example33-debug + + + + + + + + +
+
+ Name:
+ E-mail:
+ Gender: male + female
+ + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + + + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example33/index-jquery.html b/1.4.10/docs/examples/example-example33/index-jquery.html new file mode 100644 index 0000000000..b32a0370a1 --- /dev/null +++ b/1.4.10/docs/examples/example-example33/index-jquery.html @@ -0,0 +1,55 @@ + + + + + Example - example-example33-jquery + + + + + + + + + +
+
+ Name:
+ E-mail:
+ Gender: male + female
+ + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + + + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example33/index-production.html b/1.4.10/docs/examples/example-example33/index-production.html new file mode 100644 index 0000000000..0f62c94df4 --- /dev/null +++ b/1.4.10/docs/examples/example-example33/index-production.html @@ -0,0 +1,54 @@ + + + + + Example - example-example33-production + + + + + + + + +
+
+ Name:
+ E-mail:
+ Gender: male + female
+ + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + + + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example33/index.html b/1.4.10/docs/examples/example-example33/index.html new file mode 100644 index 0000000000..956573d6e0 --- /dev/null +++ b/1.4.10/docs/examples/example-example33/index.html @@ -0,0 +1,54 @@ + + + + + Example - example-example33 + + + + + + + + +
+
+ Name:
+ E-mail:
+ Gender: male + female
+ + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + + + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example33/manifest.json b/1.4.10/docs/examples/example-example33/manifest.json new file mode 100644 index 0000000000..be9db818f4 --- /dev/null +++ b/1.4.10/docs/examples/example-example33/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example33", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example34/index-debug.html b/1.4.10/docs/examples/example-example34/index-debug.html new file mode 100644 index 0000000000..ffaaa3dadb --- /dev/null +++ b/1.4.10/docs/examples/example-example34/index-debug.html @@ -0,0 +1,52 @@ + + + + + Example - example-example34-debug + + + + + + + + + +
+
+ Name: + +
+
+
Tell us your name.
+
+ + E-mail: + +
+
+ Tell us your email. + This is not a valid email. +
+ + Gender: + male + female +
+ + + I agree: + +
+
+
Please agree and sign.
+
+ + + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example34/index-jquery.html b/1.4.10/docs/examples/example-example34/index-jquery.html new file mode 100644 index 0000000000..50f3896acd --- /dev/null +++ b/1.4.10/docs/examples/example-example34/index-jquery.html @@ -0,0 +1,53 @@ + + + + + Example - example-example34-jquery + + + + + + + + + + +
+
+ Name: + +
+
+
Tell us your name.
+
+ + E-mail: + +
+
+ Tell us your email. + This is not a valid email. +
+ + Gender: + male + female +
+ + + I agree: + +
+
+
Please agree and sign.
+
+ + + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example34/index-production.html b/1.4.10/docs/examples/example-example34/index-production.html new file mode 100644 index 0000000000..3ea6c20d0e --- /dev/null +++ b/1.4.10/docs/examples/example-example34/index-production.html @@ -0,0 +1,52 @@ + + + + + Example - example-example34-production + + + + + + + + + +
+
+ Name: + +
+
+
Tell us your name.
+
+ + E-mail: + +
+
+ Tell us your email. + This is not a valid email. +
+ + Gender: + male + female +
+ + + I agree: + +
+
+
Please agree and sign.
+
+ + + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example34/index.html b/1.4.10/docs/examples/example-example34/index.html new file mode 100644 index 0000000000..3182f7d129 --- /dev/null +++ b/1.4.10/docs/examples/example-example34/index.html @@ -0,0 +1,52 @@ + + + + + Example - example-example34 + + + + + + + + + +
+
+ Name: + +
+
+
Tell us your name.
+
+ + E-mail: + +
+
+ Tell us your email. + This is not a valid email. +
+ + Gender: + male + female +
+ + + I agree: + +
+
+
Please agree and sign.
+
+ + + +
+
user = {{user | json}}
+
master = {{master | json}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example34/manifest.json b/1.4.10/docs/examples/example-example34/manifest.json new file mode 100644 index 0000000000..44fe328554 --- /dev/null +++ b/1.4.10/docs/examples/example-example34/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example34", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example34/script.js b/1.4.10/docs/examples/example-example34/script.js new file mode 100644 index 0000000000..dcbb0a5605 --- /dev/null +++ b/1.4.10/docs/examples/example-example34/script.js @@ -0,0 +1,21 @@ +(function(angular) { + 'use strict'; +angular.module('formExample', []) + .controller('ExampleController', ['$scope', function($scope) { + $scope.master = {}; + + $scope.update = function(user) { + $scope.master = angular.copy(user); + }; + + $scope.reset = function(form) { + if (form) { + form.$setPristine(); + form.$setUntouched(); + } + $scope.user = angular.copy($scope.master); + }; + + $scope.reset(); + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example35/index-debug.html b/1.4.10/docs/examples/example-example35/index-debug.html new file mode 100644 index 0000000000..7969b6d86d --- /dev/null +++ b/1.4.10/docs/examples/example-example35/index-debug.html @@ -0,0 +1,26 @@ + + + + + Example - example-example35-debug + + + + + + + + + +
+
+ Name: +
+ Other data: +
+
+
username = "{{user.name}}"
+
userdata = "{{user.data}}"
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example35/index-jquery.html b/1.4.10/docs/examples/example-example35/index-jquery.html new file mode 100644 index 0000000000..ced5a65b53 --- /dev/null +++ b/1.4.10/docs/examples/example-example35/index-jquery.html @@ -0,0 +1,27 @@ + + + + + Example - example-example35-jquery + + + + + + + + + + +
+
+ Name: +
+ Other data: +
+
+
username = "{{user.name}}"
+
userdata = "{{user.data}}"
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example35/index-production.html b/1.4.10/docs/examples/example-example35/index-production.html new file mode 100644 index 0000000000..fa08985f14 --- /dev/null +++ b/1.4.10/docs/examples/example-example35/index-production.html @@ -0,0 +1,26 @@ + + + + + Example - example-example35-production + + + + + + + + + +
+
+ Name: +
+ Other data: +
+
+
username = "{{user.name}}"
+
userdata = "{{user.data}}"
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example35/index.html b/1.4.10/docs/examples/example-example35/index.html new file mode 100644 index 0000000000..3f655cc098 --- /dev/null +++ b/1.4.10/docs/examples/example-example35/index.html @@ -0,0 +1,26 @@ + + + + + Example - example-example35 + + + + + + + + + +
+
+ Name: +
+ Other data: +
+
+
username = "{{user.name}}"
+
userdata = "{{user.data}}"
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example35/manifest.json b/1.4.10/docs/examples/example-example35/manifest.json new file mode 100644 index 0000000000..1b72f83f81 --- /dev/null +++ b/1.4.10/docs/examples/example-example35/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example35", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example35/script.js b/1.4.10/docs/examples/example-example35/script.js new file mode 100644 index 0000000000..4663d8f070 --- /dev/null +++ b/1.4.10/docs/examples/example-example35/script.js @@ -0,0 +1,7 @@ +(function(angular) { + 'use strict'; +angular.module('customTriggerExample', []) + .controller('ExampleController', ['$scope', function($scope) { + $scope.user = {}; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example36/index-debug.html b/1.4.10/docs/examples/example-example36/index-debug.html new file mode 100644 index 0000000000..15b1e98307 --- /dev/null +++ b/1.4.10/docs/examples/example-example36/index-debug.html @@ -0,0 +1,23 @@ + + + + + Example - example-example36-debug + + + + + + + + + +
+
+ Name: +
+
+
username = "{{user.name}}"
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example36/index-jquery.html b/1.4.10/docs/examples/example-example36/index-jquery.html new file mode 100644 index 0000000000..e9be7d51d4 --- /dev/null +++ b/1.4.10/docs/examples/example-example36/index-jquery.html @@ -0,0 +1,24 @@ + + + + + Example - example-example36-jquery + + + + + + + + + + +
+
+ Name: +
+
+
username = "{{user.name}}"
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example36/index-production.html b/1.4.10/docs/examples/example-example36/index-production.html new file mode 100644 index 0000000000..f49726edca --- /dev/null +++ b/1.4.10/docs/examples/example-example36/index-production.html @@ -0,0 +1,23 @@ + + + + + Example - example-example36-production + + + + + + + + + +
+
+ Name: +
+
+
username = "{{user.name}}"
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example36/index.html b/1.4.10/docs/examples/example-example36/index.html new file mode 100644 index 0000000000..7cc0311ca4 --- /dev/null +++ b/1.4.10/docs/examples/example-example36/index.html @@ -0,0 +1,23 @@ + + + + + Example - example-example36 + + + + + + + + + +
+
+ Name: +
+
+
username = "{{user.name}}"
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example36/manifest.json b/1.4.10/docs/examples/example-example36/manifest.json new file mode 100644 index 0000000000..d249fa6c68 --- /dev/null +++ b/1.4.10/docs/examples/example-example36/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example36", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example36/script.js b/1.4.10/docs/examples/example-example36/script.js new file mode 100644 index 0000000000..1ba36d63b5 --- /dev/null +++ b/1.4.10/docs/examples/example-example36/script.js @@ -0,0 +1,7 @@ +(function(angular) { + 'use strict'; +angular.module('debounceExample', []) + .controller('ExampleController', ['$scope', function($scope) { + $scope.user = {}; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example37/index-debug.html b/1.4.10/docs/examples/example-example37/index-debug.html new file mode 100644 index 0000000000..f96a7da473 --- /dev/null +++ b/1.4.10/docs/examples/example-example37/index-debug.html @@ -0,0 +1,34 @@ + + + + + Example - example-example37-debug + + + + + + + + + +
+
+ Size (integer 0 - 10): + {{size}}
+ The value is not a valid integer! + + The value must be in range 0 to 10! +
+ +
+ Username: + {{name}}
+ Checking if this name is available... + This username is already taken! +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example37/index-jquery.html b/1.4.10/docs/examples/example-example37/index-jquery.html new file mode 100644 index 0000000000..651beddfad --- /dev/null +++ b/1.4.10/docs/examples/example-example37/index-jquery.html @@ -0,0 +1,35 @@ + + + + + Example - example-example37-jquery + + + + + + + + + + +
+
+ Size (integer 0 - 10): + {{size}}
+ The value is not a valid integer! + + The value must be in range 0 to 10! +
+ +
+ Username: + {{name}}
+ Checking if this name is available... + This username is already taken! +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example37/index-production.html b/1.4.10/docs/examples/example-example37/index-production.html new file mode 100644 index 0000000000..dbf01d234b --- /dev/null +++ b/1.4.10/docs/examples/example-example37/index-production.html @@ -0,0 +1,34 @@ + + + + + Example - example-example37-production + + + + + + + + + +
+
+ Size (integer 0 - 10): + {{size}}
+ The value is not a valid integer! + + The value must be in range 0 to 10! +
+ +
+ Username: + {{name}}
+ Checking if this name is available... + This username is already taken! +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example37/index.html b/1.4.10/docs/examples/example-example37/index.html new file mode 100644 index 0000000000..c784cd2346 --- /dev/null +++ b/1.4.10/docs/examples/example-example37/index.html @@ -0,0 +1,34 @@ + + + + + Example - example-example37 + + + + + + + + + +
+
+ Size (integer 0 - 10): + {{size}}
+ The value is not a valid integer! + + The value must be in range 0 to 10! +
+ +
+ Username: + {{name}}
+ Checking if this name is available... + This username is already taken! +
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example37/manifest.json b/1.4.10/docs/examples/example-example37/manifest.json new file mode 100644 index 0000000000..d3a3b54441 --- /dev/null +++ b/1.4.10/docs/examples/example-example37/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example37", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example37/script.js b/1.4.10/docs/examples/example-example37/script.js new file mode 100644 index 0000000000..603cf08197 --- /dev/null +++ b/1.4.10/docs/examples/example-example37/script.js @@ -0,0 +1,59 @@ +(function(angular) { + 'use strict'; +var app = angular.module('form-example1', []); + +var INTEGER_REGEXP = /^\-?\d+$/; +app.directive('integer', function() { + return { + require: 'ngModel', + link: function(scope, elm, attrs, ctrl) { + ctrl.$validators.integer = function(modelValue, viewValue) { + if (ctrl.$isEmpty(modelValue)) { + // consider empty models to be valid + return true; + } + + if (INTEGER_REGEXP.test(viewValue)) { + // it is valid + return true; + } + + // it is invalid + return false; + }; + } + }; +}); + +app.directive('username', function($q, $timeout) { + return { + require: 'ngModel', + link: function(scope, elm, attrs, ctrl) { + var usernames = ['Jim', 'John', 'Jill', 'Jackie']; + + ctrl.$asyncValidators.username = function(modelValue, viewValue) { + + if (ctrl.$isEmpty(modelValue)) { + // consider empty model valid + return $q.when(); + } + + var def = $q.defer(); + + $timeout(function() { + // Mock a delayed response + if (usernames.indexOf(modelValue) === -1) { + // The username is available + def.resolve(); + } else { + def.reject(); + } + + }, 2000); + + return def.promise; + }; + } + }; +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example38/index-debug.html b/1.4.10/docs/examples/example-example38/index-debug.html new file mode 100644 index 0000000000..a583cc5b70 --- /dev/null +++ b/1.4.10/docs/examples/example-example38/index-debug.html @@ -0,0 +1,24 @@ + + + + + Example - example-example38-debug + + + + + + + + + +
+
+ Overwritten Email: + + This email format is invalid!
+ Model: {{myEmail}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example38/index-jquery.html b/1.4.10/docs/examples/example-example38/index-jquery.html new file mode 100644 index 0000000000..fd4e6295ae --- /dev/null +++ b/1.4.10/docs/examples/example-example38/index-jquery.html @@ -0,0 +1,25 @@ + + + + + Example - example-example38-jquery + + + + + + + + + + +
+
+ Overwritten Email: + + This email format is invalid!
+ Model: {{myEmail}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example38/index-production.html b/1.4.10/docs/examples/example-example38/index-production.html new file mode 100644 index 0000000000..7d51b7ec7d --- /dev/null +++ b/1.4.10/docs/examples/example-example38/index-production.html @@ -0,0 +1,24 @@ + + + + + Example - example-example38-production + + + + + + + + + +
+
+ Overwritten Email: + + This email format is invalid!
+ Model: {{myEmail}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example38/index.html b/1.4.10/docs/examples/example-example38/index.html new file mode 100644 index 0000000000..d06bedc63a --- /dev/null +++ b/1.4.10/docs/examples/example-example38/index.html @@ -0,0 +1,24 @@ + + + + + Example - example-example38 + + + + + + + + + +
+
+ Overwritten Email: + + This email format is invalid!
+ Model: {{myEmail}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example38/manifest.json b/1.4.10/docs/examples/example-example38/manifest.json new file mode 100644 index 0000000000..881bd28c3a --- /dev/null +++ b/1.4.10/docs/examples/example-example38/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example38", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example38/script.js b/1.4.10/docs/examples/example-example38/script.js new file mode 100644 index 0000000000..3926a5b6ea --- /dev/null +++ b/1.4.10/docs/examples/example-example38/script.js @@ -0,0 +1,22 @@ +(function(angular) { + 'use strict'; +var app = angular.module('form-example-modify-validators', []); + +app.directive('overwriteEmail', function() { + var EMAIL_REGEXP = /^[a-z0-9!#$%&'*+/=?^_`{|}~.-]+@example\.com$/i; + + return { + require: '?ngModel', + link: function(scope, elm, attrs, ctrl) { + // only apply the validator if ngModel is present and Angular has added the email validator + if (ctrl && ctrl.$validators.email) { + + // this will overwrite the default Angular email validator + ctrl.$validators.email = function(modelValue) { + return ctrl.$isEmpty(modelValue) || EMAIL_REGEXP.test(modelValue); + }; + } + } + }; +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example39/index-debug.html b/1.4.10/docs/examples/example-example39/index-debug.html new file mode 100644 index 0000000000..2aa7cdb655 --- /dev/null +++ b/1.4.10/docs/examples/example-example39/index-debug.html @@ -0,0 +1,25 @@ + + + + + Example - example-example39-debug + + + + + + + + + +
Some
+
model = {{content}}
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example39/index-jquery.html b/1.4.10/docs/examples/example-example39/index-jquery.html new file mode 100644 index 0000000000..917f3319df --- /dev/null +++ b/1.4.10/docs/examples/example-example39/index-jquery.html @@ -0,0 +1,26 @@ + + + + + Example - example-example39-jquery + + + + + + + + + + +
Some
+
model = {{content}}
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example39/index-production.html b/1.4.10/docs/examples/example-example39/index-production.html new file mode 100644 index 0000000000..3dcae145c6 --- /dev/null +++ b/1.4.10/docs/examples/example-example39/index-production.html @@ -0,0 +1,25 @@ + + + + + Example - example-example39-production + + + + + + + + + +
Some
+
model = {{content}}
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example39/index.html b/1.4.10/docs/examples/example-example39/index.html new file mode 100644 index 0000000000..3538f41851 --- /dev/null +++ b/1.4.10/docs/examples/example-example39/index.html @@ -0,0 +1,25 @@ + + + + + Example - example-example39 + + + + + + + + + +
Some
+
model = {{content}}
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example39/manifest.json b/1.4.10/docs/examples/example-example39/manifest.json new file mode 100644 index 0000000000..46e0f6fa0f --- /dev/null +++ b/1.4.10/docs/examples/example-example39/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example39", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example39/script.js b/1.4.10/docs/examples/example-example39/script.js new file mode 100644 index 0000000000..ff3ecbbba2 --- /dev/null +++ b/1.4.10/docs/examples/example-example39/script.js @@ -0,0 +1,22 @@ +(function(angular) { + 'use strict'; +angular.module('form-example2', []).directive('contenteditable', function() { + return { + require: 'ngModel', + link: function(scope, elm, attrs, ctrl) { + // view -> model + elm.on('blur', function() { + ctrl.$setViewValue(elm.html()); + }); + + // model -> view + ctrl.$render = function() { + elm.html(ctrl.$viewValue); + }; + + // load init value from DOM + ctrl.$setViewValue(elm.html()); + } + }; +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example4/animations.css b/1.4.10/docs/examples/example-example4/animations.css new file mode 100644 index 0000000000..3e14309548 --- /dev/null +++ b/1.4.10/docs/examples/example-example4/animations.css @@ -0,0 +1,14 @@ +.sample-show-hide { + padding:10px; + border:1px solid black; + background:white; +} + +.sample-show-hide { + -webkit-transition:all linear 0.5s; + transition:all linear 0.5s; +} + +.sample-show-hide.ng-hide { + opacity:0; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example4/index-debug.html b/1.4.10/docs/examples/example-example4/index-debug.html new file mode 100644 index 0000000000..f3814d61cd --- /dev/null +++ b/1.4.10/docs/examples/example-example4/index-debug.html @@ -0,0 +1,25 @@ + + + + + Example - example-example4-debug + + + + + + + + + + +
+ +
+ Visible... +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example4/index-jquery.html b/1.4.10/docs/examples/example-example4/index-jquery.html new file mode 100644 index 0000000000..4305debfa5 --- /dev/null +++ b/1.4.10/docs/examples/example-example4/index-jquery.html @@ -0,0 +1,26 @@ + + + + + Example - example-example4-jquery + + + + + + + + + + + +
+ +
+ Visible... +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example4/index-production.html b/1.4.10/docs/examples/example-example4/index-production.html new file mode 100644 index 0000000000..e6f05ff2b5 --- /dev/null +++ b/1.4.10/docs/examples/example-example4/index-production.html @@ -0,0 +1,25 @@ + + + + + Example - example-example4-production + + + + + + + + + + +
+ +
+ Visible... +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example4/index.html b/1.4.10/docs/examples/example-example4/index.html new file mode 100644 index 0000000000..9b0dd11a74 --- /dev/null +++ b/1.4.10/docs/examples/example-example4/index.html @@ -0,0 +1,25 @@ + + + + + Example - example-example4 + + + + + + + + + + +
+ +
+ Visible... +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example4/manifest.json b/1.4.10/docs/examples/example-example4/manifest.json new file mode 100644 index 0000000000..db169af0fd --- /dev/null +++ b/1.4.10/docs/examples/example-example4/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example4", + "files": [ + "index-production.html", + "animations.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example40/index-debug.html b/1.4.10/docs/examples/example-example40/index-debug.html new file mode 100644 index 0000000000..4086786373 --- /dev/null +++ b/1.4.10/docs/examples/example-example40/index-debug.html @@ -0,0 +1,21 @@ + + + + + Example - example-example40-debug + + + + + + + + + +
+
+ {{ 'World' | greet }} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example40/index-jquery.html b/1.4.10/docs/examples/example-example40/index-jquery.html new file mode 100644 index 0000000000..75e4aaa6c8 --- /dev/null +++ b/1.4.10/docs/examples/example-example40/index-jquery.html @@ -0,0 +1,22 @@ + + + + + Example - example-example40-jquery + + + + + + + + + + +
+
+ {{ 'World' | greet }} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example40/index-production.html b/1.4.10/docs/examples/example-example40/index-production.html new file mode 100644 index 0000000000..f39920ef54 --- /dev/null +++ b/1.4.10/docs/examples/example-example40/index-production.html @@ -0,0 +1,21 @@ + + + + + Example - example-example40-production + + + + + + + + + +
+
+ {{ 'World' | greet }} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example40/index.html b/1.4.10/docs/examples/example-example40/index.html new file mode 100644 index 0000000000..29fc357264 --- /dev/null +++ b/1.4.10/docs/examples/example-example40/index.html @@ -0,0 +1,21 @@ + + + + + Example - example-example40 + + + + + + + + + +
+
+ {{ 'World' | greet }} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example40/manifest.json b/1.4.10/docs/examples/example-example40/manifest.json new file mode 100644 index 0000000000..1bddb7354a --- /dev/null +++ b/1.4.10/docs/examples/example-example40/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example40", + "files": [ + "index-production.html", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example40/protractor.js b/1.4.10/docs/examples/example-example40/protractor.js new file mode 100644 index 0000000000..27a29da60a --- /dev/null +++ b/1.4.10/docs/examples/example-example40/protractor.js @@ -0,0 +1,3 @@ +it('should add Hello to the name', function() { + expect(element(by.binding("'World' | greet")).getText()).toEqual('Hello, World!'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example40/script.js b/1.4.10/docs/examples/example-example40/script.js new file mode 100644 index 0000000000..bf80af60be --- /dev/null +++ b/1.4.10/docs/examples/example-example40/script.js @@ -0,0 +1,13 @@ +(function(angular) { + 'use strict'; +// declare a module +var myAppModule = angular.module('myApp', []); + +// configure the module. +// in this example we will create a greeting filter +myAppModule.filter('greet', function() { + return function(name) { + return 'Hello, ' + name + '!'; + }; +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example41/index-debug.html b/1.4.10/docs/examples/example-example41/index-debug.html new file mode 100644 index 0000000000..d057f3488a --- /dev/null +++ b/1.4.10/docs/examples/example-example41/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example41-debug + + + + + + + + + +
+ {{ greeting }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example41/index-jquery.html b/1.4.10/docs/examples/example-example41/index-jquery.html new file mode 100644 index 0000000000..d24519702b --- /dev/null +++ b/1.4.10/docs/examples/example-example41/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example41-jquery + + + + + + + + + + +
+ {{ greeting }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example41/index-production.html b/1.4.10/docs/examples/example-example41/index-production.html new file mode 100644 index 0000000000..380783634c --- /dev/null +++ b/1.4.10/docs/examples/example-example41/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example41-production + + + + + + + + + +
+ {{ greeting }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example41/index.html b/1.4.10/docs/examples/example-example41/index.html new file mode 100644 index 0000000000..46485e9095 --- /dev/null +++ b/1.4.10/docs/examples/example-example41/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example41 + + + + + + + + + +
+ {{ greeting }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example41/manifest.json b/1.4.10/docs/examples/example-example41/manifest.json new file mode 100644 index 0000000000..6f3d41105d --- /dev/null +++ b/1.4.10/docs/examples/example-example41/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example41", + "files": [ + "index-production.html", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example41/protractor.js b/1.4.10/docs/examples/example-example41/protractor.js new file mode 100644 index 0000000000..c19da84411 --- /dev/null +++ b/1.4.10/docs/examples/example-example41/protractor.js @@ -0,0 +1,3 @@ +it('should add Hello to the name', function() { + expect(element(by.binding("greeting")).getText()).toEqual('Bonjour World!'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example41/script.js b/1.4.10/docs/examples/example-example41/script.js new file mode 100644 index 0000000000..83336b2489 --- /dev/null +++ b/1.4.10/docs/examples/example-example41/script.js @@ -0,0 +1,38 @@ +(function(angular) { + 'use strict'; +angular.module('xmpl.service', []) + + .value('greeter', { + salutation: 'Hello', + localize: function(localization) { + this.salutation = localization.salutation; + }, + greet: function(name) { + return this.salutation + ' ' + name + '!'; + } + }) + + .value('user', { + load: function(name) { + this.name = name; + } + }); + +angular.module('xmpl.directive', []); + +angular.module('xmpl.filter', []); + +angular.module('xmpl', ['xmpl.service', 'xmpl.directive', 'xmpl.filter']) + + .run(function(greeter, user) { + // This is effectively part of the main method initialization code + greeter.localize({ + salutation: 'Bonjour' + }); + user.load('World'); + }) + + .controller('XmplController', function($scope, greeter, user){ + $scope.greeting = greeter.greet(user.name); + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example42/index-debug.html b/1.4.10/docs/examples/example-example42/index-debug.html new file mode 100644 index 0000000000..82ed4c9971 --- /dev/null +++ b/1.4.10/docs/examples/example-example42/index-debug.html @@ -0,0 +1,23 @@ + + + + + Example - example-example42-debug + + + + + + + + + +
+ Your name: + + +
+ {{greeting}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example42/index-jquery.html b/1.4.10/docs/examples/example-example42/index-jquery.html new file mode 100644 index 0000000000..62459d755e --- /dev/null +++ b/1.4.10/docs/examples/example-example42/index-jquery.html @@ -0,0 +1,24 @@ + + + + + Example - example-example42-jquery + + + + + + + + + + +
+ Your name: + + +
+ {{greeting}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example42/index-production.html b/1.4.10/docs/examples/example-example42/index-production.html new file mode 100644 index 0000000000..57c5e5415e --- /dev/null +++ b/1.4.10/docs/examples/example-example42/index-production.html @@ -0,0 +1,23 @@ + + + + + Example - example-example42-production + + + + + + + + + +
+ Your name: + + +
+ {{greeting}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example42/index.html b/1.4.10/docs/examples/example-example42/index.html new file mode 100644 index 0000000000..c446f3f90d --- /dev/null +++ b/1.4.10/docs/examples/example-example42/index.html @@ -0,0 +1,23 @@ + + + + + Example - example-example42 + + + + + + + + + +
+ Your name: + + +
+ {{greeting}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example42/manifest.json b/1.4.10/docs/examples/example-example42/manifest.json new file mode 100644 index 0000000000..e7b5938760 --- /dev/null +++ b/1.4.10/docs/examples/example-example42/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example42", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example42/script.js b/1.4.10/docs/examples/example-example42/script.js new file mode 100644 index 0000000000..7d16a71d65 --- /dev/null +++ b/1.4.10/docs/examples/example-example42/script.js @@ -0,0 +1,11 @@ +(function(angular) { + 'use strict'; +angular.module('scopeExample', []) + .controller('MyController', ['$scope', function($scope) { + $scope.username = 'World'; + + $scope.sayHello = function() { + $scope.greeting = 'Hello ' + $scope.username + '!'; + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example43/index-debug.html b/1.4.10/docs/examples/example-example43/index-debug.html new file mode 100644 index 0000000000..2ab55592ab --- /dev/null +++ b/1.4.10/docs/examples/example-example43/index-debug.html @@ -0,0 +1,27 @@ + + + + + Example - example-example43-debug + + + + + + + + + + +
+
+ Hello {{name}}! +
+
+
    +
  1. {{name}} from {{department}}
  2. +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example43/index-jquery.html b/1.4.10/docs/examples/example-example43/index-jquery.html new file mode 100644 index 0000000000..bda1e38e4a --- /dev/null +++ b/1.4.10/docs/examples/example-example43/index-jquery.html @@ -0,0 +1,28 @@ + + + + + Example - example-example43-jquery + + + + + + + + + + + +
+
+ Hello {{name}}! +
+
+
    +
  1. {{name}} from {{department}}
  2. +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example43/index-production.html b/1.4.10/docs/examples/example-example43/index-production.html new file mode 100644 index 0000000000..b2eb183dd4 --- /dev/null +++ b/1.4.10/docs/examples/example-example43/index-production.html @@ -0,0 +1,27 @@ + + + + + Example - example-example43-production + + + + + + + + + + +
+
+ Hello {{name}}! +
+
+
    +
  1. {{name}} from {{department}}
  2. +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example43/index.html b/1.4.10/docs/examples/example-example43/index.html new file mode 100644 index 0000000000..f365ae9c4e --- /dev/null +++ b/1.4.10/docs/examples/example-example43/index.html @@ -0,0 +1,27 @@ + + + + + Example - example-example43 + + + + + + + + + + +
+
+ Hello {{name}}! +
+
+
    +
  1. {{name}} from {{department}}
  2. +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example43/manifest.json b/1.4.10/docs/examples/example-example43/manifest.json new file mode 100644 index 0000000000..e0c02a50bd --- /dev/null +++ b/1.4.10/docs/examples/example-example43/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example43", + "files": [ + "index-production.html", + "script.js", + "style.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example43/script.js b/1.4.10/docs/examples/example-example43/script.js new file mode 100644 index 0000000000..e98864b1fa --- /dev/null +++ b/1.4.10/docs/examples/example-example43/script.js @@ -0,0 +1,11 @@ +(function(angular) { + 'use strict'; +angular.module('scopeExample', []) + .controller('GreetController', ['$scope', '$rootScope', function($scope, $rootScope) { + $scope.name = 'World'; + $rootScope.department = 'Angular'; + }]) + .controller('ListController', ['$scope', function($scope) { + $scope.names = ['Igor', 'Misko', 'Vojta']; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example43/style.css b/1.4.10/docs/examples/example-example43/style.css new file mode 100644 index 0000000000..fb9168410a --- /dev/null +++ b/1.4.10/docs/examples/example-example43/style.css @@ -0,0 +1,5 @@ +.show-scope-demo.ng-scope, +.show-scope-demo .ng-scope { + border: 1px solid red; + margin: 3px; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example44/index-debug.html b/1.4.10/docs/examples/example-example44/index-debug.html new file mode 100644 index 0000000000..859136649a --- /dev/null +++ b/1.4.10/docs/examples/example-example44/index-debug.html @@ -0,0 +1,32 @@ + + + + + Example - example-example44-debug + + + + + + + + + +
+ Root scope MyEvent count: {{count}} +
    +
  • + + +
    + Middle scope MyEvent count: {{count}} +
      +
    • + Leaf scope MyEvent count: {{count}} +
    • +
    +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example44/index-jquery.html b/1.4.10/docs/examples/example-example44/index-jquery.html new file mode 100644 index 0000000000..d1c472f1bf --- /dev/null +++ b/1.4.10/docs/examples/example-example44/index-jquery.html @@ -0,0 +1,33 @@ + + + + + Example - example-example44-jquery + + + + + + + + + + +
+ Root scope MyEvent count: {{count}} +
    +
  • + + +
    + Middle scope MyEvent count: {{count}} +
      +
    • + Leaf scope MyEvent count: {{count}} +
    • +
    +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example44/index-production.html b/1.4.10/docs/examples/example-example44/index-production.html new file mode 100644 index 0000000000..f244a493a5 --- /dev/null +++ b/1.4.10/docs/examples/example-example44/index-production.html @@ -0,0 +1,32 @@ + + + + + Example - example-example44-production + + + + + + + + + +
+ Root scope MyEvent count: {{count}} +
    +
  • + + +
    + Middle scope MyEvent count: {{count}} +
      +
    • + Leaf scope MyEvent count: {{count}} +
    • +
    +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example44/index.html b/1.4.10/docs/examples/example-example44/index.html new file mode 100644 index 0000000000..f9ce59b918 --- /dev/null +++ b/1.4.10/docs/examples/example-example44/index.html @@ -0,0 +1,32 @@ + + + + + Example - example-example44 + + + + + + + + + +
+ Root scope MyEvent count: {{count}} +
    +
  • + + +
    + Middle scope MyEvent count: {{count}} +
      +
    • + Leaf scope MyEvent count: {{count}} +
    • +
    +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example44/manifest.json b/1.4.10/docs/examples/example-example44/manifest.json new file mode 100644 index 0000000000..1c60c6303a --- /dev/null +++ b/1.4.10/docs/examples/example-example44/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example44", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example44/script.js b/1.4.10/docs/examples/example-example44/script.js new file mode 100644 index 0000000000..bef5f7f7bb --- /dev/null +++ b/1.4.10/docs/examples/example-example44/script.js @@ -0,0 +1,10 @@ +(function(angular) { + 'use strict'; +angular.module('eventExample', []) + .controller('EventController', ['$scope', function($scope) { + $scope.count = 0; + $scope.$on('MyEvent', function() { + $scope.count++; + }); + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example45/index-debug.html b/1.4.10/docs/examples/example-example45/index-debug.html new file mode 100644 index 0000000000..9f4ececb32 --- /dev/null +++ b/1.4.10/docs/examples/example-example45/index-debug.html @@ -0,0 +1,22 @@ + + + + + Example - example-example45-debug + + + + + + + + + +
+

Let's try this simple notify service, injected into the controller...

+ + +

(you have to click 3 times to see an alert)

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example45/index-jquery.html b/1.4.10/docs/examples/example-example45/index-jquery.html new file mode 100644 index 0000000000..06dd220404 --- /dev/null +++ b/1.4.10/docs/examples/example-example45/index-jquery.html @@ -0,0 +1,23 @@ + + + + + Example - example-example45-jquery + + + + + + + + + + +
+

Let's try this simple notify service, injected into the controller...

+ + +

(you have to click 3 times to see an alert)

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example45/index-production.html b/1.4.10/docs/examples/example-example45/index-production.html new file mode 100644 index 0000000000..8c5e1ed998 --- /dev/null +++ b/1.4.10/docs/examples/example-example45/index-production.html @@ -0,0 +1,22 @@ + + + + + Example - example-example45-production + + + + + + + + + +
+

Let's try this simple notify service, injected into the controller...

+ + +

(you have to click 3 times to see an alert)

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example45/index.html b/1.4.10/docs/examples/example-example45/index.html new file mode 100644 index 0000000000..86b05fedbb --- /dev/null +++ b/1.4.10/docs/examples/example-example45/index.html @@ -0,0 +1,22 @@ + + + + + Example - example-example45 + + + + + + + + + +
+

Let's try this simple notify service, injected into the controller...

+ + +

(you have to click 3 times to see an alert)

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example45/manifest.json b/1.4.10/docs/examples/example-example45/manifest.json new file mode 100644 index 0000000000..64241fb1cb --- /dev/null +++ b/1.4.10/docs/examples/example-example45/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example45", + "files": [ + "index-production.html", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example45/protractor.js b/1.4.10/docs/examples/example-example45/protractor.js new file mode 100644 index 0000000000..67fed4598a --- /dev/null +++ b/1.4.10/docs/examples/example-example45/protractor.js @@ -0,0 +1,4 @@ +it('should test service', function() { + expect(element(by.id('simple')).element(by.model('message')).getAttribute('value')) + .toEqual('test'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example45/script.js b/1.4.10/docs/examples/example-example45/script.js new file mode 100644 index 0000000000..4bb23fe08b --- /dev/null +++ b/1.4.10/docs/examples/example-example45/script.js @@ -0,0 +1,20 @@ +(function(angular) { + 'use strict'; +angular. + module('myServiceModule', []). + controller('MyController', ['$scope','notify', function ($scope, notify) { + $scope.callNotify = function(msg) { + notify(msg); + }; + }]). + factory('notify', ['$window', function(win) { + var msgs = []; + return function(msg) { + msgs.push(msg); + if (msgs.length == 3) { + win.alert(msgs.join("\n")); + msgs = []; + } + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example46/index-debug.html b/1.4.10/docs/examples/example-example46/index-debug.html new file mode 100644 index 0000000000..8ecf06edcd --- /dev/null +++ b/1.4.10/docs/examples/example-example46/index-debug.html @@ -0,0 +1,46 @@ + + + + + Example - example-example46-debug + + + + + + + + +
+
+Name:
+E-mail:
+Gender: male +female
+ + +
+
form = {{user | json}}
+
master = {{master | json}}
+
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example46/index-jquery.html b/1.4.10/docs/examples/example-example46/index-jquery.html new file mode 100644 index 0000000000..3737d7bf9b --- /dev/null +++ b/1.4.10/docs/examples/example-example46/index-jquery.html @@ -0,0 +1,47 @@ + + + + + Example - example-example46-jquery + + + + + + + + + +
+
+Name:
+E-mail:
+Gender: male +female
+ + +
+
form = {{user | json}}
+
master = {{master | json}}
+
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example46/index-production.html b/1.4.10/docs/examples/example-example46/index-production.html new file mode 100644 index 0000000000..2f2d3a3268 --- /dev/null +++ b/1.4.10/docs/examples/example-example46/index-production.html @@ -0,0 +1,46 @@ + + + + + Example - example-example46-production + + + + + + + + +
+
+Name:
+E-mail:
+Gender: male +female
+ + +
+
form = {{user | json}}
+
master = {{master | json}}
+
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example46/index.html b/1.4.10/docs/examples/example-example46/index.html new file mode 100644 index 0000000000..11e80a8d0f --- /dev/null +++ b/1.4.10/docs/examples/example-example46/index.html @@ -0,0 +1,46 @@ + + + + + Example - example-example46 + + + + + + + + +
+
+Name:
+E-mail:
+Gender: male +female
+ + +
+
form = {{user | json}}
+
master = {{master | json}}
+
+ + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example46/manifest.json b/1.4.10/docs/examples/example-example46/manifest.json new file mode 100644 index 0000000000..9f46a4fcea --- /dev/null +++ b/1.4.10/docs/examples/example-example46/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example46", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example47/index-debug.html b/1.4.10/docs/examples/example-example47/index-debug.html new file mode 100644 index 0000000000..37ee78555e --- /dev/null +++ b/1.4.10/docs/examples/example-example47/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example47-debug + + + + + + + + + +
+ I can add: {{a}} + {{b}} = {{ a+b }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example47/index-jquery.html b/1.4.10/docs/examples/example-example47/index-jquery.html new file mode 100644 index 0000000000..0f56398190 --- /dev/null +++ b/1.4.10/docs/examples/example-example47/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example47-jquery + + + + + + + + + + +
+ I can add: {{a}} + {{b}} = {{ a+b }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example47/index-production.html b/1.4.10/docs/examples/example-example47/index-production.html new file mode 100644 index 0000000000..29968c6bd3 --- /dev/null +++ b/1.4.10/docs/examples/example-example47/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example47-production + + + + + + + + + +
+ I can add: {{a}} + {{b}} = {{ a+b }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example47/index.html b/1.4.10/docs/examples/example-example47/index.html new file mode 100644 index 0000000000..3420119aa1 --- /dev/null +++ b/1.4.10/docs/examples/example-example47/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example47 + + + + + + + + + +
+ I can add: {{a}} + {{b}} = {{ a+b }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example47/manifest.json b/1.4.10/docs/examples/example-example47/manifest.json new file mode 100644 index 0000000000..c9d0f53343 --- /dev/null +++ b/1.4.10/docs/examples/example-example47/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example47", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example47/script.js b/1.4.10/docs/examples/example-example47/script.js new file mode 100644 index 0000000000..116b24d4a6 --- /dev/null +++ b/1.4.10/docs/examples/example-example47/script.js @@ -0,0 +1,7 @@ +(function(angular) { + 'use strict'; +angular.module('ngAppDemo', []).controller('ngAppDemoController', function($scope) { + $scope.a = 1; + $scope.b = 2; +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example48/index-debug.html b/1.4.10/docs/examples/example-example48/index-debug.html new file mode 100644 index 0000000000..c964d63b25 --- /dev/null +++ b/1.4.10/docs/examples/example-example48/index-debug.html @@ -0,0 +1,47 @@ + + + + + Example - example-example48-debug + + + + + + + + + + +
+
+ I can add: {{a}} + {{b}} = {{ a+b }} + +

This renders because the controller does not fail to + instantiate, by using explicit annotation style (see + script.js for details) +

+
+ +
+ Name:
+ Hello, {{name}}! + +

This renders because the controller does not fail to + instantiate, by using explicit annotation style + (see script.js for details) +

+
+ +
+ I can add: {{a}} + {{b}} = {{ a+b }} + +

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. +

+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example48/index-jquery.html b/1.4.10/docs/examples/example-example48/index-jquery.html new file mode 100644 index 0000000000..2644cf5ce7 --- /dev/null +++ b/1.4.10/docs/examples/example-example48/index-jquery.html @@ -0,0 +1,48 @@ + + + + + Example - example-example48-jquery + + + + + + + + + + + +
+
+ I can add: {{a}} + {{b}} = {{ a+b }} + +

This renders because the controller does not fail to + instantiate, by using explicit annotation style (see + script.js for details) +

+
+ +
+ Name:
+ Hello, {{name}}! + +

This renders because the controller does not fail to + instantiate, by using explicit annotation style + (see script.js for details) +

+
+ +
+ I can add: {{a}} + {{b}} = {{ a+b }} + +

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. +

+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example48/index-production.html b/1.4.10/docs/examples/example-example48/index-production.html new file mode 100644 index 0000000000..8d81544b08 --- /dev/null +++ b/1.4.10/docs/examples/example-example48/index-production.html @@ -0,0 +1,47 @@ + + + + + Example - example-example48-production + + + + + + + + + + +
+
+ I can add: {{a}} + {{b}} = {{ a+b }} + +

This renders because the controller does not fail to + instantiate, by using explicit annotation style (see + script.js for details) +

+
+ +
+ Name:
+ Hello, {{name}}! + +

This renders because the controller does not fail to + instantiate, by using explicit annotation style + (see script.js for details) +

+
+ +
+ I can add: {{a}} + {{b}} = {{ a+b }} + +

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. +

+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example48/index.html b/1.4.10/docs/examples/example-example48/index.html new file mode 100644 index 0000000000..c77625485f --- /dev/null +++ b/1.4.10/docs/examples/example-example48/index.html @@ -0,0 +1,47 @@ + + + + + Example - example-example48 + + + + + + + + + + +
+
+ I can add: {{a}} + {{b}} = {{ a+b }} + +

This renders because the controller does not fail to + instantiate, by using explicit annotation style (see + script.js for details) +

+
+ +
+ Name:
+ Hello, {{name}}! + +

This renders because the controller does not fail to + instantiate, by using explicit annotation style + (see script.js for details) +

+
+ +
+ I can add: {{a}} + {{b}} = {{ a+b }} + +

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. +

+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example48/manifest.json b/1.4.10/docs/examples/example-example48/manifest.json new file mode 100644 index 0000000000..168f25e251 --- /dev/null +++ b/1.4.10/docs/examples/example-example48/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example48", + "files": [ + "index-production.html", + "script.js", + "style.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example48/script.js b/1.4.10/docs/examples/example-example48/script.js new file mode 100644 index 0000000000..8394853bc6 --- /dev/null +++ b/1.4.10/docs/examples/example-example48/script.js @@ -0,0 +1,21 @@ +(function(angular) { + 'use strict'; +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']; +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example48/style.css b/1.4.10/docs/examples/example-example48/style.css new file mode 100644 index 0000000000..e8379bfdf7 --- /dev/null +++ b/1.4.10/docs/examples/example-example48/style.css @@ -0,0 +1,18 @@ +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; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example49/index-debug.html b/1.4.10/docs/examples/example-example49/index-debug.html new file mode 100644 index 0000000000..698a596de2 --- /dev/null +++ b/1.4.10/docs/examples/example-example49/index-debug.html @@ -0,0 +1,21 @@ + + + + + Example - example-example49-debug + + + + + + + + + + +
+ Go to bottom + You're at the bottom! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example49/index-jquery.html b/1.4.10/docs/examples/example-example49/index-jquery.html new file mode 100644 index 0000000000..905eebb6ae --- /dev/null +++ b/1.4.10/docs/examples/example-example49/index-jquery.html @@ -0,0 +1,22 @@ + + + + + Example - example-example49-jquery + + + + + + + + + + + +
+ Go to bottom + You're at the bottom! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example49/index-production.html b/1.4.10/docs/examples/example-example49/index-production.html new file mode 100644 index 0000000000..c9e15fa1f2 --- /dev/null +++ b/1.4.10/docs/examples/example-example49/index-production.html @@ -0,0 +1,21 @@ + + + + + Example - example-example49-production + + + + + + + + + + +
+ Go to bottom + You're at the bottom! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example49/index.html b/1.4.10/docs/examples/example-example49/index.html new file mode 100644 index 0000000000..997b9eb96d --- /dev/null +++ b/1.4.10/docs/examples/example-example49/index.html @@ -0,0 +1,21 @@ + + + + + Example - example-example49 + + + + + + + + + + +
+ Go to bottom + You're at the bottom! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example49/manifest.json b/1.4.10/docs/examples/example-example49/manifest.json new file mode 100644 index 0000000000..38712b02fd --- /dev/null +++ b/1.4.10/docs/examples/example-example49/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example49", + "files": [ + "index-production.html", + "script.js", + "style.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example49/script.js b/1.4.10/docs/examples/example-example49/script.js new file mode 100644 index 0000000000..c93a7a6e24 --- /dev/null +++ b/1.4.10/docs/examples/example-example49/script.js @@ -0,0 +1,15 @@ +(function(angular) { + 'use strict'; +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(); + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example49/style.css b/1.4.10/docs/examples/example-example49/style.css new file mode 100644 index 0000000000..608700b07e --- /dev/null +++ b/1.4.10/docs/examples/example-example49/style.css @@ -0,0 +1,9 @@ +#scrollArea { + height: 280px; + overflow: auto; +} + +#bottom { + display: block; + margin-top: 2000px; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example5/index-debug.html b/1.4.10/docs/examples/example-example5/index-debug.html new file mode 100644 index 0000000000..d35d1468a0 --- /dev/null +++ b/1.4.10/docs/examples/example-example5/index-debug.html @@ -0,0 +1,23 @@ + + + + + Example - example-example5-debug + + + + + + + + + + +

+ + +
+ CSS-Animated Text +

+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example5/index-jquery.html b/1.4.10/docs/examples/example-example5/index-jquery.html new file mode 100644 index 0000000000..968afa866e --- /dev/null +++ b/1.4.10/docs/examples/example-example5/index-jquery.html @@ -0,0 +1,24 @@ + + + + + Example - example-example5-jquery + + + + + + + + + + + +

+ + +
+ CSS-Animated Text +

+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example5/index-production.html b/1.4.10/docs/examples/example-example5/index-production.html new file mode 100644 index 0000000000..273755051f --- /dev/null +++ b/1.4.10/docs/examples/example-example5/index-production.html @@ -0,0 +1,23 @@ + + + + + Example - example-example5-production + + + + + + + + + + +

+ + +
+ CSS-Animated Text +

+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example5/index.html b/1.4.10/docs/examples/example-example5/index.html new file mode 100644 index 0000000000..32ac5b7f2a --- /dev/null +++ b/1.4.10/docs/examples/example-example5/index.html @@ -0,0 +1,23 @@ + + + + + Example - example-example5 + + + + + + + + + + +

+ + +
+ CSS-Animated Text +

+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example5/manifest.json b/1.4.10/docs/examples/example-example5/manifest.json new file mode 100644 index 0000000000..6161e79d84 --- /dev/null +++ b/1.4.10/docs/examples/example-example5/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example5", + "files": [ + "index-production.html", + "style.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example5/style.css b/1.4.10/docs/examples/example-example5/style.css new file mode 100644 index 0000000000..d833326f5a --- /dev/null +++ b/1.4.10/docs/examples/example-example5/style.css @@ -0,0 +1,17 @@ +.css-class-add, .css-class-remove { + -webkit-transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 0.5s; + -moz-transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 0.5s; + -o-transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 0.5s; + transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 0.5s; +} + +.css-class, +.css-class-add.css-class-add-active { + color: red; + font-size:3em; +} + +.css-class-remove.css-class-remove-active { + font-size:1.0em; + color:black; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example50/index-debug.html b/1.4.10/docs/examples/example-example50/index-debug.html new file mode 100644 index 0000000000..f2f0ab4228 --- /dev/null +++ b/1.4.10/docs/examples/example-example50/index-debug.html @@ -0,0 +1,25 @@ + + + + + Example - example-example50-debug + + + + + + + + + + + +
+ Anchor {{x}} of 5 +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example50/index-jquery.html b/1.4.10/docs/examples/example-example50/index-jquery.html new file mode 100644 index 0000000000..d529eb0d70 --- /dev/null +++ b/1.4.10/docs/examples/example-example50/index-jquery.html @@ -0,0 +1,26 @@ + + + + + Example - example-example50-jquery + + + + + + + + + + + + +
+ Anchor {{x}} of 5 +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example50/index-production.html b/1.4.10/docs/examples/example-example50/index-production.html new file mode 100644 index 0000000000..e34b5e8218 --- /dev/null +++ b/1.4.10/docs/examples/example-example50/index-production.html @@ -0,0 +1,25 @@ + + + + + Example - example-example50-production + + + + + + + + + + + +
+ Anchor {{x}} of 5 +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example50/index.html b/1.4.10/docs/examples/example-example50/index.html new file mode 100644 index 0000000000..63343b57d1 --- /dev/null +++ b/1.4.10/docs/examples/example-example50/index.html @@ -0,0 +1,25 @@ + + + + + Example - example-example50 + + + + + + + + + + + +
+ Anchor {{x}} of 5 +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example50/manifest.json b/1.4.10/docs/examples/example-example50/manifest.json new file mode 100644 index 0000000000..7c971e84f4 --- /dev/null +++ b/1.4.10/docs/examples/example-example50/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example50", + "files": [ + "index-production.html", + "script.js", + "style.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example50/script.js b/1.4.10/docs/examples/example-example50/script.js new file mode 100644 index 0000000000..ea7b5eec3a --- /dev/null +++ b/1.4.10/docs/examples/example-example50/script.js @@ -0,0 +1,23 @@ +(function(angular) { + 'use strict'; +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(); + } + }; + } + ]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example50/style.css b/1.4.10/docs/examples/example-example50/style.css new file mode 100644 index 0000000000..9fa05a2ad2 --- /dev/null +++ b/1.4.10/docs/examples/example-example50/style.css @@ -0,0 +1,20 @@ +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; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example51/index-debug.html b/1.4.10/docs/examples/example-example51/index-debug.html new file mode 100644 index 0000000000..3e0aa984f8 --- /dev/null +++ b/1.4.10/docs/examples/example-example51/index-debug.html @@ -0,0 +1,36 @@ + + + + + Example - example-example51-debug + + + + + + + + + + +
+ + + + +

Cached Values

+
+ + : + +
+ +

Cache Info

+
+ + : + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example51/index-jquery.html b/1.4.10/docs/examples/example-example51/index-jquery.html new file mode 100644 index 0000000000..65688ee0f4 --- /dev/null +++ b/1.4.10/docs/examples/example-example51/index-jquery.html @@ -0,0 +1,37 @@ + + + + + Example - example-example51-jquery + + + + + + + + + + + +
+ + + + +

Cached Values

+
+ + : + +
+ +

Cache Info

+
+ + : + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example51/index-production.html b/1.4.10/docs/examples/example-example51/index-production.html new file mode 100644 index 0000000000..3dfa470779 --- /dev/null +++ b/1.4.10/docs/examples/example-example51/index-production.html @@ -0,0 +1,36 @@ + + + + + Example - example-example51-production + + + + + + + + + + +
+ + + + +

Cached Values

+
+ + : + +
+ +

Cache Info

+
+ + : + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example51/index.html b/1.4.10/docs/examples/example-example51/index.html new file mode 100644 index 0000000000..66cea29212 --- /dev/null +++ b/1.4.10/docs/examples/example-example51/index.html @@ -0,0 +1,36 @@ + + + + + Example - example-example51 + + + + + + + + + + +
+ + + + +

Cached Values

+
+ + : + +
+ +

Cache Info

+
+ + : + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example51/manifest.json b/1.4.10/docs/examples/example-example51/manifest.json new file mode 100644 index 0000000000..77dd74af71 --- /dev/null +++ b/1.4.10/docs/examples/example-example51/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example51", + "files": [ + "index-production.html", + "script.js", + "style.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example51/script.js b/1.4.10/docs/examples/example-example51/script.js new file mode 100644 index 0000000000..bd51b0c2e7 --- /dev/null +++ b/1.4.10/docs/examples/example-example51/script.js @@ -0,0 +1,14 @@ +(function(angular) { + 'use strict'; +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); + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example51/style.css b/1.4.10/docs/examples/example-example51/style.css new file mode 100644 index 0000000000..ea64f1cb8c --- /dev/null +++ b/1.4.10/docs/examples/example-example51/style.css @@ -0,0 +1,3 @@ +p { + margin: 10px 0 3px; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example52/index-debug.html b/1.4.10/docs/examples/example-example52/index-debug.html new file mode 100644 index 0000000000..35369ff50c --- /dev/null +++ b/1.4.10/docs/examples/example-example52/index-debug.html @@ -0,0 +1,52 @@ + + + + + Example - example-example52-debug + + + + + + + + + +
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example52/index-jquery.html b/1.4.10/docs/examples/example-example52/index-jquery.html new file mode 100644 index 0000000000..8c57d971d1 --- /dev/null +++ b/1.4.10/docs/examples/example-example52/index-jquery.html @@ -0,0 +1,53 @@ + + + + + Example - example-example52-jquery + + + + + + + + + + +
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example52/index-production.html b/1.4.10/docs/examples/example-example52/index-production.html new file mode 100644 index 0000000000..b0c79296e8 --- /dev/null +++ b/1.4.10/docs/examples/example-example52/index-production.html @@ -0,0 +1,52 @@ + + + + + Example - example-example52-production + + + + + + + + + +
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example52/index.html b/1.4.10/docs/examples/example-example52/index.html new file mode 100644 index 0000000000..ea1c083c97 --- /dev/null +++ b/1.4.10/docs/examples/example-example52/index.html @@ -0,0 +1,52 @@ + + + + + Example - example-example52 + + + + + + + + + +
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example52/manifest.json b/1.4.10/docs/examples/example-example52/manifest.json new file mode 100644 index 0000000000..b63d0c84e6 --- /dev/null +++ b/1.4.10/docs/examples/example-example52/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example52", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example52/protractor.js b/1.4.10/docs/examples/example-example52/protractor.js new file mode 100644 index 0000000000..7bed5a5b2a --- /dev/null +++ b/1.4.10/docs/examples/example-example52/protractor.js @@ -0,0 +1,9 @@ +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!'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example53/index-debug.html b/1.4.10/docs/examples/example-example53/index-debug.html new file mode 100644 index 0000000000..e72c1f2b59 --- /dev/null +++ b/1.4.10/docs/examples/example-example53/index-debug.html @@ -0,0 +1,22 @@ + + + + + Example - example-example53-debug + + + + + + + + +
+link 1 (link, don't reload)
+link 2 (link, don't reload)
+link 3 (link, reload!)
+anchor (link, don't reload)
+anchor (no link)
+link (link, change location) + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example53/index-jquery.html b/1.4.10/docs/examples/example-example53/index-jquery.html new file mode 100644 index 0000000000..10ad8c345c --- /dev/null +++ b/1.4.10/docs/examples/example-example53/index-jquery.html @@ -0,0 +1,23 @@ + + + + + Example - example-example53-jquery + + + + + + + + + +
+link 1 (link, don't reload)
+link 2 (link, don't reload)
+link 3 (link, reload!)
+anchor (link, don't reload)
+anchor (no link)
+link (link, change location) + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example53/index-production.html b/1.4.10/docs/examples/example-example53/index-production.html new file mode 100644 index 0000000000..16c7d5ac64 --- /dev/null +++ b/1.4.10/docs/examples/example-example53/index-production.html @@ -0,0 +1,22 @@ + + + + + Example - example-example53-production + + + + + + + + +
+link 1 (link, don't reload)
+link 2 (link, don't reload)
+link 3 (link, reload!)
+anchor (link, don't reload)
+anchor (no link)
+link (link, change location) + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example53/index.html b/1.4.10/docs/examples/example-example53/index.html new file mode 100644 index 0000000000..4ea787a252 --- /dev/null +++ b/1.4.10/docs/examples/example-example53/index.html @@ -0,0 +1,22 @@ + + + + + Example - example-example53 + + + + + + + + +
+link 1 (link, don't reload)
+link 2 (link, don't reload)
+link 3 (link, reload!)
+anchor (link, don't reload)
+anchor (no link)
+link (link, change location) + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example53/manifest.json b/1.4.10/docs/examples/example-example53/manifest.json new file mode 100644 index 0000000000..f8f469763c --- /dev/null +++ b/1.4.10/docs/examples/example-example53/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example53", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example53/protractor.js b/1.4.10/docs/examples/example-example53/protractor.js new file mode 100644 index 0000000000..88c57571f1 --- /dev/null +++ b/1.4.10/docs/examples/example-example53/protractor.js @@ -0,0 +1,54 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example54/index-debug.html b/1.4.10/docs/examples/example-example54/index-debug.html new file mode 100644 index 0000000000..995261749a --- /dev/null +++ b/1.4.10/docs/examples/example-example54/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example54-debug + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example54/index-jquery.html b/1.4.10/docs/examples/example-example54/index-jquery.html new file mode 100644 index 0000000000..df22f92d68 --- /dev/null +++ b/1.4.10/docs/examples/example-example54/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example54-jquery + + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example54/index-production.html b/1.4.10/docs/examples/example-example54/index-production.html new file mode 100644 index 0000000000..d960fdaa0a --- /dev/null +++ b/1.4.10/docs/examples/example-example54/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example54-production + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example54/index.html b/1.4.10/docs/examples/example-example54/index.html new file mode 100644 index 0000000000..47eae1f5d6 --- /dev/null +++ b/1.4.10/docs/examples/example-example54/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example54 + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example54/manifest.json b/1.4.10/docs/examples/example-example54/manifest.json new file mode 100644 index 0000000000..622adc4325 --- /dev/null +++ b/1.4.10/docs/examples/example-example54/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example54", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example54/protractor.js b/1.4.10/docs/examples/example-example54/protractor.js new file mode 100644 index 0000000000..f3e2d3a8e3 --- /dev/null +++ b/1.4.10/docs/examples/example-example54/protractor.js @@ -0,0 +1,5 @@ +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(); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example55/index-debug.html b/1.4.10/docs/examples/example-example55/index-debug.html new file mode 100644 index 0000000000..4277f6f644 --- /dev/null +++ b/1.4.10/docs/examples/example-example55/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example55-debug + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example55/index-jquery.html b/1.4.10/docs/examples/example-example55/index-jquery.html new file mode 100644 index 0000000000..dbe4cc8cce --- /dev/null +++ b/1.4.10/docs/examples/example-example55/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example55-jquery + + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example55/index-production.html b/1.4.10/docs/examples/example-example55/index-production.html new file mode 100644 index 0000000000..eb9e59faaf --- /dev/null +++ b/1.4.10/docs/examples/example-example55/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example55-production + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example55/index.html b/1.4.10/docs/examples/example-example55/index.html new file mode 100644 index 0000000000..3076664e97 --- /dev/null +++ b/1.4.10/docs/examples/example-example55/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example55 + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example55/manifest.json b/1.4.10/docs/examples/example-example55/manifest.json new file mode 100644 index 0000000000..faf2f32099 --- /dev/null +++ b/1.4.10/docs/examples/example-example55/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example55", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example55/protractor.js b/1.4.10/docs/examples/example-example55/protractor.js new file mode 100644 index 0000000000..a2febc73c6 --- /dev/null +++ b/1.4.10/docs/examples/example-example55/protractor.js @@ -0,0 +1,5 @@ +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(); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example56/index-debug.html b/1.4.10/docs/examples/example-example56/index-debug.html new file mode 100644 index 0000000000..7aa85b7c18 --- /dev/null +++ b/1.4.10/docs/examples/example-example56/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example56-debug + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example56/index-jquery.html b/1.4.10/docs/examples/example-example56/index-jquery.html new file mode 100644 index 0000000000..697d74b512 --- /dev/null +++ b/1.4.10/docs/examples/example-example56/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example56-jquery + + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example56/index-production.html b/1.4.10/docs/examples/example-example56/index-production.html new file mode 100644 index 0000000000..a0c6abdc9e --- /dev/null +++ b/1.4.10/docs/examples/example-example56/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example56-production + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example56/index.html b/1.4.10/docs/examples/example-example56/index.html new file mode 100644 index 0000000000..654eb02ac7 --- /dev/null +++ b/1.4.10/docs/examples/example-example56/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example56 + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example56/manifest.json b/1.4.10/docs/examples/example-example56/manifest.json new file mode 100644 index 0000000000..883ac7d8f3 --- /dev/null +++ b/1.4.10/docs/examples/example-example56/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example56", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example56/protractor.js b/1.4.10/docs/examples/example-example56/protractor.js new file mode 100644 index 0000000000..814c59aa8d --- /dev/null +++ b/1.4.10/docs/examples/example-example56/protractor.js @@ -0,0 +1,5 @@ +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(); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example57/index-debug.html b/1.4.10/docs/examples/example-example57/index-debug.html new file mode 100644 index 0000000000..14a7c1efed --- /dev/null +++ b/1.4.10/docs/examples/example-example57/index-debug.html @@ -0,0 +1,20 @@ + + + + + Example - example-example57-debug + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example57/index-jquery.html b/1.4.10/docs/examples/example-example57/index-jquery.html new file mode 100644 index 0000000000..b211a34acf --- /dev/null +++ b/1.4.10/docs/examples/example-example57/index-jquery.html @@ -0,0 +1,21 @@ + + + + + Example - example-example57-jquery + + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example57/index-production.html b/1.4.10/docs/examples/example-example57/index-production.html new file mode 100644 index 0000000000..11ddfa644e --- /dev/null +++ b/1.4.10/docs/examples/example-example57/index-production.html @@ -0,0 +1,20 @@ + + + + + Example - example-example57-production + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example57/index.html b/1.4.10/docs/examples/example-example57/index.html new file mode 100644 index 0000000000..2b2a9bdb0e --- /dev/null +++ b/1.4.10/docs/examples/example-example57/index.html @@ -0,0 +1,20 @@ + + + + + Example - example-example57 + + + + + + + + +
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example57/manifest.json b/1.4.10/docs/examples/example-example57/manifest.json new file mode 100644 index 0000000000..45bd91965f --- /dev/null +++ b/1.4.10/docs/examples/example-example57/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example57", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example57/protractor.js b/1.4.10/docs/examples/example-example57/protractor.js new file mode 100644 index 0000000000..8e9ecffb96 --- /dev/null +++ b/1.4.10/docs/examples/example-example57/protractor.js @@ -0,0 +1,5 @@ +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(); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example58/index-debug.html b/1.4.10/docs/examples/example-example58/index-debug.html new file mode 100644 index 0000000000..c3d370b4ac --- /dev/null +++ b/1.4.10/docs/examples/example-example58/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example58-debug + + + + + + + + +
+
+ Show/Hide me +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example58/index-jquery.html b/1.4.10/docs/examples/example-example58/index-jquery.html new file mode 100644 index 0000000000..266916359a --- /dev/null +++ b/1.4.10/docs/examples/example-example58/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example58-jquery + + + + + + + + + +
+
+ Show/Hide me +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example58/index-production.html b/1.4.10/docs/examples/example-example58/index-production.html new file mode 100644 index 0000000000..d2aebf265f --- /dev/null +++ b/1.4.10/docs/examples/example-example58/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example58-production + + + + + + + + +
+
+ Show/Hide me +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example58/index.html b/1.4.10/docs/examples/example-example58/index.html new file mode 100644 index 0000000000..9cbf2dbca3 --- /dev/null +++ b/1.4.10/docs/examples/example-example58/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example58 + + + + + + + + +
+
+ Show/Hide me +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example58/manifest.json b/1.4.10/docs/examples/example-example58/manifest.json new file mode 100644 index 0000000000..9a438d4659 --- /dev/null +++ b/1.4.10/docs/examples/example-example58/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example58", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example58/protractor.js b/1.4.10/docs/examples/example-example58/protractor.js new file mode 100644 index 0000000000..1a95888f08 --- /dev/null +++ b/1.4.10/docs/examples/example-example58/protractor.js @@ -0,0 +1,5 @@ +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(); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example59/index-debug.html b/1.4.10/docs/examples/example-example59/index-debug.html new file mode 100644 index 0000000000..6f7940c9c7 --- /dev/null +++ b/1.4.10/docs/examples/example-example59/index-debug.html @@ -0,0 +1,42 @@ + + + + + Example - example-example59-debug + + + + + + + + + + + +
+ userType: + Required!
+ userType = {{userType}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example59/index-jquery.html b/1.4.10/docs/examples/example-example59/index-jquery.html new file mode 100644 index 0000000000..373d72be4e --- /dev/null +++ b/1.4.10/docs/examples/example-example59/index-jquery.html @@ -0,0 +1,43 @@ + + + + + Example - example-example59-jquery + + + + + + + + + + + + +
+ userType: + Required!
+ userType = {{userType}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example59/index-production.html b/1.4.10/docs/examples/example-example59/index-production.html new file mode 100644 index 0000000000..915f4b8e6b --- /dev/null +++ b/1.4.10/docs/examples/example-example59/index-production.html @@ -0,0 +1,42 @@ + + + + + Example - example-example59-production + + + + + + + + + + + +
+ userType: + Required!
+ userType = {{userType}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example59/index.html b/1.4.10/docs/examples/example-example59/index.html new file mode 100644 index 0000000000..d7d20b01d4 --- /dev/null +++ b/1.4.10/docs/examples/example-example59/index.html @@ -0,0 +1,42 @@ + + + + + Example - example-example59 + + + + + + + + + + + +
+ userType: + Required!
+ userType = {{userType}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example59/manifest.json b/1.4.10/docs/examples/example-example59/manifest.json new file mode 100644 index 0000000000..fe12b81e79 --- /dev/null +++ b/1.4.10/docs/examples/example-example59/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example59", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example59/protractor.js b/1.4.10/docs/examples/example-example59/protractor.js new file mode 100644 index 0000000000..09668bb3e4 --- /dev/null +++ b/1.4.10/docs/examples/example-example59/protractor.js @@ -0,0 +1,19 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example6/index-debug.html b/1.4.10/docs/examples/example-example6/index-debug.html new file mode 100644 index 0000000000..4614b99cc3 --- /dev/null +++ b/1.4.10/docs/examples/example-example6/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example6-debug + + + + + + + + + + Drag ME + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example6/index-jquery.html b/1.4.10/docs/examples/example-example6/index-jquery.html new file mode 100644 index 0000000000..25ef292039 --- /dev/null +++ b/1.4.10/docs/examples/example-example6/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example6-jquery + + + + + + + + + + + Drag ME + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example6/index-production.html b/1.4.10/docs/examples/example-example6/index-production.html new file mode 100644 index 0000000000..eae18019c8 --- /dev/null +++ b/1.4.10/docs/examples/example-example6/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example6-production + + + + + + + + + + Drag ME + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example6/index.html b/1.4.10/docs/examples/example-example6/index.html new file mode 100644 index 0000000000..875deda588 --- /dev/null +++ b/1.4.10/docs/examples/example-example6/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example6 + + + + + + + + + + Drag ME + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example6/manifest.json b/1.4.10/docs/examples/example-example6/manifest.json new file mode 100644 index 0000000000..2f1e90d08e --- /dev/null +++ b/1.4.10/docs/examples/example-example6/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example6", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example6/script.js b/1.4.10/docs/examples/example-example6/script.js new file mode 100644 index 0000000000..aba458959c --- /dev/null +++ b/1.4.10/docs/examples/example-example6/script.js @@ -0,0 +1,39 @@ +(function(angular) { + 'use strict'; +angular.module('drag', []). + directive('draggable', function($document) { + return function(scope, element, attr) { + var startX = 0, startY = 0, x = 0, y = 0; + element.css({ + position: 'relative', + border: '1px solid red', + backgroundColor: 'lightgrey', + cursor: 'pointer', + display: 'block', + width: '65px' + }); + element.on('mousedown', function(event) { + // Prevent default dragging of selected content + event.preventDefault(); + startX = event.screenX - x; + startY = event.screenY - y; + $document.on('mousemove', mousemove); + $document.on('mouseup', mouseup); + }); + + function mousemove(event) { + y = event.screenY - startY; + x = event.screenX - startX; + element.css({ + top: y + 'px', + left: x + 'px' + }); + } + + function mouseup() { + $document.off('mousemove', mousemove); + $document.off('mouseup', mouseup); + } + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example60/index-debug.html b/1.4.10/docs/examples/example-example60/index-debug.html new file mode 100644 index 0000000000..5bb21e0081 --- /dev/null +++ b/1.4.10/docs/examples/example-example60/index-debug.html @@ -0,0 +1,25 @@ + + + + + Example - example-example60-debug + + + + + + + + + +
+
+ Hello ! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example60/index-jquery.html b/1.4.10/docs/examples/example-example60/index-jquery.html new file mode 100644 index 0000000000..224efba8f4 --- /dev/null +++ b/1.4.10/docs/examples/example-example60/index-jquery.html @@ -0,0 +1,26 @@ + + + + + Example - example-example60-jquery + + + + + + + + + + +
+
+ Hello ! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example60/index-production.html b/1.4.10/docs/examples/example-example60/index-production.html new file mode 100644 index 0000000000..752ea6f028 --- /dev/null +++ b/1.4.10/docs/examples/example-example60/index-production.html @@ -0,0 +1,25 @@ + + + + + Example - example-example60-production + + + + + + + + + +
+
+ Hello ! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example60/index.html b/1.4.10/docs/examples/example-example60/index.html new file mode 100644 index 0000000000..6bf46b3430 --- /dev/null +++ b/1.4.10/docs/examples/example-example60/index.html @@ -0,0 +1,25 @@ + + + + + Example - example-example60 + + + + + + + + + +
+
+ Hello ! +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example60/manifest.json b/1.4.10/docs/examples/example-example60/manifest.json new file mode 100644 index 0000000000..c41e550b73 --- /dev/null +++ b/1.4.10/docs/examples/example-example60/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example60", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example60/protractor.js b/1.4.10/docs/examples/example-example60/protractor.js new file mode 100644 index 0000000000..0ef43a9b48 --- /dev/null +++ b/1.4.10/docs/examples/example-example60/protractor.js @@ -0,0 +1,8 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example61/index-debug.html b/1.4.10/docs/examples/example-example61/index-debug.html new file mode 100644 index 0000000000..62eef55941 --- /dev/null +++ b/1.4.10/docs/examples/example-example61/index-debug.html @@ -0,0 +1,27 @@ + + + + + Example - example-example61-debug + + + + + + + + + +
+
+
+

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example61/index-jquery.html b/1.4.10/docs/examples/example-example61/index-jquery.html new file mode 100644 index 0000000000..499b363815 --- /dev/null +++ b/1.4.10/docs/examples/example-example61/index-jquery.html @@ -0,0 +1,28 @@ + + + + + Example - example-example61-jquery + + + + + + + + + + +
+
+
+

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example61/index-production.html b/1.4.10/docs/examples/example-example61/index-production.html new file mode 100644 index 0000000000..0049029e4b --- /dev/null +++ b/1.4.10/docs/examples/example-example61/index-production.html @@ -0,0 +1,27 @@ + + + + + Example - example-example61-production + + + + + + + + + +
+
+
+

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example61/index.html b/1.4.10/docs/examples/example-example61/index.html new file mode 100644 index 0000000000..29e9eabe48 --- /dev/null +++ b/1.4.10/docs/examples/example-example61/index.html @@ -0,0 +1,27 @@ + + + + + Example - example-example61 + + + + + + + + + +
+
+
+

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example61/manifest.json b/1.4.10/docs/examples/example-example61/manifest.json new file mode 100644 index 0000000000..a23f0ddd32 --- /dev/null +++ b/1.4.10/docs/examples/example-example61/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example61", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example61/protractor.js b/1.4.10/docs/examples/example-example61/protractor.js new file mode 100644 index 0000000000..3cbeae24fc --- /dev/null +++ b/1.4.10/docs/examples/example-example61/protractor.js @@ -0,0 +1,14 @@ +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!'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example62/index-debug.html b/1.4.10/docs/examples/example-example62/index-debug.html new file mode 100644 index 0000000000..de0c83780e --- /dev/null +++ b/1.4.10/docs/examples/example-example62/index-debug.html @@ -0,0 +1,20 @@ + + + + + Example - example-example62-debug + + + + + + + + + + +
+

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example62/index-jquery.html b/1.4.10/docs/examples/example-example62/index-jquery.html new file mode 100644 index 0000000000..6bb7f2b0c8 --- /dev/null +++ b/1.4.10/docs/examples/example-example62/index-jquery.html @@ -0,0 +1,21 @@ + + + + + Example - example-example62-jquery + + + + + + + + + + + +
+

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example62/index-production.html b/1.4.10/docs/examples/example-example62/index-production.html new file mode 100644 index 0000000000..e37f0fb9ce --- /dev/null +++ b/1.4.10/docs/examples/example-example62/index-production.html @@ -0,0 +1,20 @@ + + + + + Example - example-example62-production + + + + + + + + + + +
+

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example62/index.html b/1.4.10/docs/examples/example-example62/index.html new file mode 100644 index 0000000000..dd1186312e --- /dev/null +++ b/1.4.10/docs/examples/example-example62/index.html @@ -0,0 +1,20 @@ + + + + + Example - example-example62 + + + + + + + + + + +
+

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example62/manifest.json b/1.4.10/docs/examples/example-example62/manifest.json new file mode 100644 index 0000000000..b094ab70a0 --- /dev/null +++ b/1.4.10/docs/examples/example-example62/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example62", + "files": [ + "index-production.html", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example62/protractor.js b/1.4.10/docs/examples/example-example62/protractor.js new file mode 100644 index 0000000000..b236449821 --- /dev/null +++ b/1.4.10/docs/examples/example-example62/protractor.js @@ -0,0 +1,4 @@ +it('should check ng-bind-html', function() { + expect(element(by.binding('myHTML')).getText()).toBe( + 'I am an HTMLstring with links! and other stuff'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example62/script.js b/1.4.10/docs/examples/example-example62/script.js new file mode 100644 index 0000000000..5f24b74ae1 --- /dev/null +++ b/1.4.10/docs/examples/example-example62/script.js @@ -0,0 +1,9 @@ +(function(angular) { + 'use strict'; +angular.module('bindHtmlExample', ['ngSanitize']) + .controller('ExampleController', ['$scope', function($scope) { + $scope.myHTML = + 'I am an HTMLstring with ' + + 'links! and other stuff'; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example63/index-debug.html b/1.4.10/docs/examples/example-example63/index-debug.html new file mode 100644 index 0000000000..ca6f31c9cc --- /dev/null +++ b/1.4.10/docs/examples/example-example63/index-debug.html @@ -0,0 +1,45 @@ + + + + + Example - example-example63-debug + + + + + + + + + +

Map Syntax Example

+
+
+ +
+

Using String Syntax

+ +
+

Using Array Syntax

+
+
+
+
+

Using Array and Map Syntax

+
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example63/index-jquery.html b/1.4.10/docs/examples/example-example63/index-jquery.html new file mode 100644 index 0000000000..e10c19ee82 --- /dev/null +++ b/1.4.10/docs/examples/example-example63/index-jquery.html @@ -0,0 +1,46 @@ + + + + + Example - example-example63-jquery + + + + + + + + + + +

Map Syntax Example

+
+
+ +
+

Using String Syntax

+ +
+

Using Array Syntax

+
+
+
+
+

Using Array and Map Syntax

+
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example63/index-production.html b/1.4.10/docs/examples/example-example63/index-production.html new file mode 100644 index 0000000000..b957419ae0 --- /dev/null +++ b/1.4.10/docs/examples/example-example63/index-production.html @@ -0,0 +1,45 @@ + + + + + Example - example-example63-production + + + + + + + + + +

Map Syntax Example

+
+
+ +
+

Using String Syntax

+ +
+

Using Array Syntax

+
+
+
+
+

Using Array and Map Syntax

+
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example63/index.html b/1.4.10/docs/examples/example-example63/index.html new file mode 100644 index 0000000000..6e2ff497e0 --- /dev/null +++ b/1.4.10/docs/examples/example-example63/index.html @@ -0,0 +1,45 @@ + + + + + Example - example-example63 + + + + + + + + + +

Map Syntax Example

+
+
+ +
+

Using String Syntax

+ +
+

Using Array Syntax

+
+
+
+
+

Using Array and Map Syntax

+
+ + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example63/manifest.json b/1.4.10/docs/examples/example-example63/manifest.json new file mode 100644 index 0000000000..31595f6191 --- /dev/null +++ b/1.4.10/docs/examples/example-example63/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example63", + "files": [ + "index-production.html", + "style.css", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example63/protractor.js b/1.4.10/docs/examples/example-example63/protractor.js new file mode 100644 index 0000000000..de06aa8f40 --- /dev/null +++ b/1.4.10/docs/examples/example-example63/protractor.js @@ -0,0 +1,35 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example63/style.css b/1.4.10/docs/examples/example-example63/style.css new file mode 100644 index 0000000000..3084ddfb3a --- /dev/null +++ b/1.4.10/docs/examples/example-example63/style.css @@ -0,0 +1,16 @@ +.strike { + text-decoration: line-through; +} +.bold { + font-weight: bold; +} +.red { + color: red; +} +.has-error { + color: red; + background-color: yellow; +} +.orange { + color: orange; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example64/index-debug.html b/1.4.10/docs/examples/example-example64/index-debug.html new file mode 100644 index 0000000000..eb6c284449 --- /dev/null +++ b/1.4.10/docs/examples/example-example64/index-debug.html @@ -0,0 +1,21 @@ + + + + + Example - example-example64-debug + + + + + + + + + + + + +
+Sample Text + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example64/index-jquery.html b/1.4.10/docs/examples/example-example64/index-jquery.html new file mode 100644 index 0000000000..361777af0b --- /dev/null +++ b/1.4.10/docs/examples/example-example64/index-jquery.html @@ -0,0 +1,22 @@ + + + + + Example - example-example64-jquery + + + + + + + + + + + + + +
+Sample Text + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example64/index-production.html b/1.4.10/docs/examples/example-example64/index-production.html new file mode 100644 index 0000000000..67dfdd3465 --- /dev/null +++ b/1.4.10/docs/examples/example-example64/index-production.html @@ -0,0 +1,21 @@ + + + + + Example - example-example64-production + + + + + + + + + + + + +
+Sample Text + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example64/index.html b/1.4.10/docs/examples/example-example64/index.html new file mode 100644 index 0000000000..9e49531cf1 --- /dev/null +++ b/1.4.10/docs/examples/example-example64/index.html @@ -0,0 +1,21 @@ + + + + + Example - example-example64 + + + + + + + + + + + + +
+Sample Text + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example64/manifest.json b/1.4.10/docs/examples/example-example64/manifest.json new file mode 100644 index 0000000000..c5f2d7fe11 --- /dev/null +++ b/1.4.10/docs/examples/example-example64/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example64", + "files": [ + "index-production.html", + "style.css", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example64/protractor.js b/1.4.10/docs/examples/example-example64/protractor.js new file mode 100644 index 0000000000..93e79f1b1d --- /dev/null +++ b/1.4.10/docs/examples/example-example64/protractor.js @@ -0,0 +1,14 @@ +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/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example64/style.css b/1.4.10/docs/examples/example-example64/style.css new file mode 100644 index 0000000000..922e358c3e --- /dev/null +++ b/1.4.10/docs/examples/example-example64/style.css @@ -0,0 +1,8 @@ +.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; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example65/index-debug.html b/1.4.10/docs/examples/example-example65/index-debug.html new file mode 100644 index 0000000000..b4c9817bd2 --- /dev/null +++ b/1.4.10/docs/examples/example-example65/index-debug.html @@ -0,0 +1,23 @@ + + + + + Example - example-example65-debug + + + + + + + + + +
    +
  1. + + {{name}} + +
  2. +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example65/index-jquery.html b/1.4.10/docs/examples/example-example65/index-jquery.html new file mode 100644 index 0000000000..a5e8282b65 --- /dev/null +++ b/1.4.10/docs/examples/example-example65/index-jquery.html @@ -0,0 +1,24 @@ + + + + + Example - example-example65-jquery + + + + + + + + + + +
    +
  1. + + {{name}} + +
  2. +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example65/index-production.html b/1.4.10/docs/examples/example-example65/index-production.html new file mode 100644 index 0000000000..3f9eb12efb --- /dev/null +++ b/1.4.10/docs/examples/example-example65/index-production.html @@ -0,0 +1,23 @@ + + + + + Example - example-example65-production + + + + + + + + + +
    +
  1. + + {{name}} + +
  2. +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example65/index.html b/1.4.10/docs/examples/example-example65/index.html new file mode 100644 index 0000000000..b47c619815 --- /dev/null +++ b/1.4.10/docs/examples/example-example65/index.html @@ -0,0 +1,23 @@ + + + + + Example - example-example65 + + + + + + + + + +
    +
  1. + + {{name}} + +
  2. +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example65/manifest.json b/1.4.10/docs/examples/example-example65/manifest.json new file mode 100644 index 0000000000..08c97e2a1a --- /dev/null +++ b/1.4.10/docs/examples/example-example65/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example65", + "files": [ + "index-production.html", + "style.css", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example65/protractor.js b/1.4.10/docs/examples/example-example65/protractor.js new file mode 100644 index 0000000000..896a4ced33 --- /dev/null +++ b/1.4.10/docs/examples/example-example65/protractor.js @@ -0,0 +1,6 @@ +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/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example65/style.css b/1.4.10/docs/examples/example-example65/style.css new file mode 100644 index 0000000000..b71c758760 --- /dev/null +++ b/1.4.10/docs/examples/example-example65/style.css @@ -0,0 +1,6 @@ +.odd { + color: red; +} +.even { + color: blue; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example66/index-debug.html b/1.4.10/docs/examples/example-example66/index-debug.html new file mode 100644 index 0000000000..758da29e40 --- /dev/null +++ b/1.4.10/docs/examples/example-example66/index-debug.html @@ -0,0 +1,23 @@ + + + + + Example - example-example66-debug + + + + + + + + + +
    +
  1. + + {{name}}       + +
  2. +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example66/index-jquery.html b/1.4.10/docs/examples/example-example66/index-jquery.html new file mode 100644 index 0000000000..26ba85dd0b --- /dev/null +++ b/1.4.10/docs/examples/example-example66/index-jquery.html @@ -0,0 +1,24 @@ + + + + + Example - example-example66-jquery + + + + + + + + + + +
    +
  1. + + {{name}}       + +
  2. +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example66/index-production.html b/1.4.10/docs/examples/example-example66/index-production.html new file mode 100644 index 0000000000..c4d7ede69d --- /dev/null +++ b/1.4.10/docs/examples/example-example66/index-production.html @@ -0,0 +1,23 @@ + + + + + Example - example-example66-production + + + + + + + + + +
    +
  1. + + {{name}}       + +
  2. +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example66/index.html b/1.4.10/docs/examples/example-example66/index.html new file mode 100644 index 0000000000..1fb0f5419d --- /dev/null +++ b/1.4.10/docs/examples/example-example66/index.html @@ -0,0 +1,23 @@ + + + + + Example - example-example66 + + + + + + + + + +
    +
  1. + + {{name}}       + +
  2. +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example66/manifest.json b/1.4.10/docs/examples/example-example66/manifest.json new file mode 100644 index 0000000000..cafde0343a --- /dev/null +++ b/1.4.10/docs/examples/example-example66/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example66", + "files": [ + "index-production.html", + "style.css", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example66/protractor.js b/1.4.10/docs/examples/example-example66/protractor.js new file mode 100644 index 0000000000..896a4ced33 --- /dev/null +++ b/1.4.10/docs/examples/example-example66/protractor.js @@ -0,0 +1,6 @@ +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/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example66/style.css b/1.4.10/docs/examples/example-example66/style.css new file mode 100644 index 0000000000..b71c758760 --- /dev/null +++ b/1.4.10/docs/examples/example-example66/style.css @@ -0,0 +1,6 @@ +.odd { + color: red; +} +.even { + color: blue; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example67/index-debug.html b/1.4.10/docs/examples/example-example67/index-debug.html new file mode 100644 index 0000000000..404a23def0 --- /dev/null +++ b/1.4.10/docs/examples/example-example67/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example67-debug + + + + + + + + +
{{ 'hello' }}
+
{{ 'world' }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example67/index-jquery.html b/1.4.10/docs/examples/example-example67/index-jquery.html new file mode 100644 index 0000000000..fe21c68324 --- /dev/null +++ b/1.4.10/docs/examples/example-example67/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example67-jquery + + + + + + + + + +
{{ 'hello' }}
+
{{ 'world' }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example67/index-production.html b/1.4.10/docs/examples/example-example67/index-production.html new file mode 100644 index 0000000000..5935e9c059 --- /dev/null +++ b/1.4.10/docs/examples/example-example67/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example67-production + + + + + + + + +
{{ 'hello' }}
+
{{ 'world' }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example67/index.html b/1.4.10/docs/examples/example-example67/index.html new file mode 100644 index 0000000000..9152a7b228 --- /dev/null +++ b/1.4.10/docs/examples/example-example67/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example67 + + + + + + + + +
{{ 'hello' }}
+
{{ 'world' }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example67/manifest.json b/1.4.10/docs/examples/example-example67/manifest.json new file mode 100644 index 0000000000..0ded652b12 --- /dev/null +++ b/1.4.10/docs/examples/example-example67/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example67", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example67/protractor.js b/1.4.10/docs/examples/example-example67/protractor.js new file mode 100644 index 0000000000..eabf82f390 --- /dev/null +++ b/1.4.10/docs/examples/example-example67/protractor.js @@ -0,0 +1,6 @@ +it('should remove the template directive and css class', function() { + expect($('#template1').getAttribute('ng-cloak')). + toBeNull(); + expect($('#template2').getAttribute('ng-cloak')). + toBeNull(); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example68/index-debug.html b/1.4.10/docs/examples/example-example68/index-debug.html new file mode 100644 index 0000000000..e1302d3b8f --- /dev/null +++ b/1.4.10/docs/examples/example-example68/index-debug.html @@ -0,0 +1,21 @@ + + + + + Example - example-example68-debug + + + + + + + + + + + count: {{count}} + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example68/index-jquery.html b/1.4.10/docs/examples/example-example68/index-jquery.html new file mode 100644 index 0000000000..b02014150a --- /dev/null +++ b/1.4.10/docs/examples/example-example68/index-jquery.html @@ -0,0 +1,22 @@ + + + + + Example - example-example68-jquery + + + + + + + + + + + + count: {{count}} + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example68/index-production.html b/1.4.10/docs/examples/example-example68/index-production.html new file mode 100644 index 0000000000..c88cf29ed7 --- /dev/null +++ b/1.4.10/docs/examples/example-example68/index-production.html @@ -0,0 +1,21 @@ + + + + + Example - example-example68-production + + + + + + + + + + + count: {{count}} + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example68/index.html b/1.4.10/docs/examples/example-example68/index.html new file mode 100644 index 0000000000..a1c0b63d85 --- /dev/null +++ b/1.4.10/docs/examples/example-example68/index.html @@ -0,0 +1,21 @@ + + + + + Example - example-example68 + + + + + + + + + + + count: {{count}} + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example68/manifest.json b/1.4.10/docs/examples/example-example68/manifest.json new file mode 100644 index 0000000000..3c9e7f14ac --- /dev/null +++ b/1.4.10/docs/examples/example-example68/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example68", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example68/protractor.js b/1.4.10/docs/examples/example-example68/protractor.js new file mode 100644 index 0000000000..50954578b8 --- /dev/null +++ b/1.4.10/docs/examples/example-example68/protractor.js @@ -0,0 +1,5 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example69/index-debug.html b/1.4.10/docs/examples/example-example69/index-debug.html new file mode 100644 index 0000000000..d983f9567b --- /dev/null +++ b/1.4.10/docs/examples/example-example69/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example69-debug + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example69/index-jquery.html b/1.4.10/docs/examples/example-example69/index-jquery.html new file mode 100644 index 0000000000..4269adccc6 --- /dev/null +++ b/1.4.10/docs/examples/example-example69/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example69-jquery + + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example69/index-production.html b/1.4.10/docs/examples/example-example69/index-production.html new file mode 100644 index 0000000000..c2e7df0238 --- /dev/null +++ b/1.4.10/docs/examples/example-example69/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example69-production + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example69/index.html b/1.4.10/docs/examples/example-example69/index.html new file mode 100644 index 0000000000..8b12bd57f6 --- /dev/null +++ b/1.4.10/docs/examples/example-example69/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example69 + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example69/manifest.json b/1.4.10/docs/examples/example-example69/manifest.json new file mode 100644 index 0000000000..f8fe603c39 --- /dev/null +++ b/1.4.10/docs/examples/example-example69/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example69", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example7/app.js b/1.4.10/docs/examples/example-example7/app.js new file mode 100644 index 0000000000..17efd46b2c --- /dev/null +++ b/1.4.10/docs/examples/example-example7/app.js @@ -0,0 +1,16 @@ +(function(angular) { + 'use strict'; +var myApp = angular.module('spicyApp1', []); + +myApp.controller('SpicyController', ['$scope', function($scope) { + $scope.spice = 'very'; + + $scope.chiliSpicy = function() { + $scope.spice = 'chili'; + }; + + $scope.jalapenoSpicy = function() { + $scope.spice = 'jalapeño'; + }; +}]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example7/index-debug.html b/1.4.10/docs/examples/example-example7/index-debug.html new file mode 100644 index 0000000000..7604e29df2 --- /dev/null +++ b/1.4.10/docs/examples/example-example7/index-debug.html @@ -0,0 +1,21 @@ + + + + + Example - example-example7-debug + + + + + + + + + +
+ + +

The food is {{spice}} spicy!

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example7/index-jquery.html b/1.4.10/docs/examples/example-example7/index-jquery.html new file mode 100644 index 0000000000..76c4e95eed --- /dev/null +++ b/1.4.10/docs/examples/example-example7/index-jquery.html @@ -0,0 +1,22 @@ + + + + + Example - example-example7-jquery + + + + + + + + + + +
+ + +

The food is {{spice}} spicy!

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example7/index-production.html b/1.4.10/docs/examples/example-example7/index-production.html new file mode 100644 index 0000000000..08b51e271a --- /dev/null +++ b/1.4.10/docs/examples/example-example7/index-production.html @@ -0,0 +1,21 @@ + + + + + Example - example-example7-production + + + + + + + + + +
+ + +

The food is {{spice}} spicy!

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example7/index.html b/1.4.10/docs/examples/example-example7/index.html new file mode 100644 index 0000000000..898518572d --- /dev/null +++ b/1.4.10/docs/examples/example-example7/index.html @@ -0,0 +1,21 @@ + + + + + Example - example-example7 + + + + + + + + + +
+ + +

The food is {{spice}} spicy!

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example7/manifest.json b/1.4.10/docs/examples/example-example7/manifest.json new file mode 100644 index 0000000000..eeef02d434 --- /dev/null +++ b/1.4.10/docs/examples/example-example7/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example7", + "files": [ + "index-production.html", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example70/index-debug.html b/1.4.10/docs/examples/example-example70/index-debug.html new file mode 100644 index 0000000000..24f7c4ebea --- /dev/null +++ b/1.4.10/docs/examples/example-example70/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example70-debug + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example70/index-jquery.html b/1.4.10/docs/examples/example-example70/index-jquery.html new file mode 100644 index 0000000000..95af2191c7 --- /dev/null +++ b/1.4.10/docs/examples/example-example70/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example70-jquery + + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example70/index-production.html b/1.4.10/docs/examples/example-example70/index-production.html new file mode 100644 index 0000000000..3ef39e2e68 --- /dev/null +++ b/1.4.10/docs/examples/example-example70/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example70-production + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example70/index.html b/1.4.10/docs/examples/example-example70/index.html new file mode 100644 index 0000000000..507f34cb06 --- /dev/null +++ b/1.4.10/docs/examples/example-example70/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example70 + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example70/manifest.json b/1.4.10/docs/examples/example-example70/manifest.json new file mode 100644 index 0000000000..655f73f2b3 --- /dev/null +++ b/1.4.10/docs/examples/example-example70/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example70", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example71/index-debug.html b/1.4.10/docs/examples/example-example71/index-debug.html new file mode 100644 index 0000000000..f2d1fcfd55 --- /dev/null +++ b/1.4.10/docs/examples/example-example71/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example71-debug + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example71/index-jquery.html b/1.4.10/docs/examples/example-example71/index-jquery.html new file mode 100644 index 0000000000..b033952e3d --- /dev/null +++ b/1.4.10/docs/examples/example-example71/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example71-jquery + + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example71/index-production.html b/1.4.10/docs/examples/example-example71/index-production.html new file mode 100644 index 0000000000..703ad28f4e --- /dev/null +++ b/1.4.10/docs/examples/example-example71/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example71-production + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example71/index.html b/1.4.10/docs/examples/example-example71/index.html new file mode 100644 index 0000000000..2ab6bc731b --- /dev/null +++ b/1.4.10/docs/examples/example-example71/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example71 + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example71/manifest.json b/1.4.10/docs/examples/example-example71/manifest.json new file mode 100644 index 0000000000..90e8d86e5c --- /dev/null +++ b/1.4.10/docs/examples/example-example71/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example71", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example72/index-debug.html b/1.4.10/docs/examples/example-example72/index-debug.html new file mode 100644 index 0000000000..64aadc0e5e --- /dev/null +++ b/1.4.10/docs/examples/example-example72/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example72-debug + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example72/index-jquery.html b/1.4.10/docs/examples/example-example72/index-jquery.html new file mode 100644 index 0000000000..ca7d64717b --- /dev/null +++ b/1.4.10/docs/examples/example-example72/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example72-jquery + + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example72/index-production.html b/1.4.10/docs/examples/example-example72/index-production.html new file mode 100644 index 0000000000..caaae52e18 --- /dev/null +++ b/1.4.10/docs/examples/example-example72/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example72-production + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example72/index.html b/1.4.10/docs/examples/example-example72/index.html new file mode 100644 index 0000000000..4edf27dd72 --- /dev/null +++ b/1.4.10/docs/examples/example-example72/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example72 + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example72/manifest.json b/1.4.10/docs/examples/example-example72/manifest.json new file mode 100644 index 0000000000..043a930603 --- /dev/null +++ b/1.4.10/docs/examples/example-example72/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example72", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example73/index-debug.html b/1.4.10/docs/examples/example-example73/index-debug.html new file mode 100644 index 0000000000..1fb674f404 --- /dev/null +++ b/1.4.10/docs/examples/example-example73/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example73-debug + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example73/index-jquery.html b/1.4.10/docs/examples/example-example73/index-jquery.html new file mode 100644 index 0000000000..abc2233450 --- /dev/null +++ b/1.4.10/docs/examples/example-example73/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example73-jquery + + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example73/index-production.html b/1.4.10/docs/examples/example-example73/index-production.html new file mode 100644 index 0000000000..bb46550293 --- /dev/null +++ b/1.4.10/docs/examples/example-example73/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example73-production + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example73/index.html b/1.4.10/docs/examples/example-example73/index.html new file mode 100644 index 0000000000..4d33732df1 --- /dev/null +++ b/1.4.10/docs/examples/example-example73/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example73 + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example73/manifest.json b/1.4.10/docs/examples/example-example73/manifest.json new file mode 100644 index 0000000000..e2fbbe3fe9 --- /dev/null +++ b/1.4.10/docs/examples/example-example73/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example73", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example74/index-debug.html b/1.4.10/docs/examples/example-example74/index-debug.html new file mode 100644 index 0000000000..880e8c2dba --- /dev/null +++ b/1.4.10/docs/examples/example-example74/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example74-debug + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example74/index-jquery.html b/1.4.10/docs/examples/example-example74/index-jquery.html new file mode 100644 index 0000000000..186386ea7f --- /dev/null +++ b/1.4.10/docs/examples/example-example74/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example74-jquery + + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example74/index-production.html b/1.4.10/docs/examples/example-example74/index-production.html new file mode 100644 index 0000000000..fc903ba574 --- /dev/null +++ b/1.4.10/docs/examples/example-example74/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example74-production + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example74/index.html b/1.4.10/docs/examples/example-example74/index.html new file mode 100644 index 0000000000..6042f33b0d --- /dev/null +++ b/1.4.10/docs/examples/example-example74/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example74 + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example74/manifest.json b/1.4.10/docs/examples/example-example74/manifest.json new file mode 100644 index 0000000000..6cf6697bf9 --- /dev/null +++ b/1.4.10/docs/examples/example-example74/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example74", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example75/index-debug.html b/1.4.10/docs/examples/example-example75/index-debug.html new file mode 100644 index 0000000000..7ff4e6bf4d --- /dev/null +++ b/1.4.10/docs/examples/example-example75/index-debug.html @@ -0,0 +1,19 @@ + + + + + Example - example-example75-debug + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example75/index-jquery.html b/1.4.10/docs/examples/example-example75/index-jquery.html new file mode 100644 index 0000000000..3a8ac866f4 --- /dev/null +++ b/1.4.10/docs/examples/example-example75/index-jquery.html @@ -0,0 +1,20 @@ + + + + + Example - example-example75-jquery + + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example75/index-production.html b/1.4.10/docs/examples/example-example75/index-production.html new file mode 100644 index 0000000000..76af613cf8 --- /dev/null +++ b/1.4.10/docs/examples/example-example75/index-production.html @@ -0,0 +1,19 @@ + + + + + Example - example-example75-production + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example75/index.html b/1.4.10/docs/examples/example-example75/index.html new file mode 100644 index 0000000000..0336da38b9 --- /dev/null +++ b/1.4.10/docs/examples/example-example75/index.html @@ -0,0 +1,19 @@ + + + + + Example - example-example75 + + + + + + + + + +count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example75/manifest.json b/1.4.10/docs/examples/example-example75/manifest.json new file mode 100644 index 0000000000..b6e9669f71 --- /dev/null +++ b/1.4.10/docs/examples/example-example75/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example75", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example76/index-debug.html b/1.4.10/docs/examples/example-example76/index-debug.html new file mode 100644 index 0000000000..a952f6455f --- /dev/null +++ b/1.4.10/docs/examples/example-example76/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example76-debug + + + + + + + + + +key down count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example76/index-jquery.html b/1.4.10/docs/examples/example-example76/index-jquery.html new file mode 100644 index 0000000000..6bd7a54f4e --- /dev/null +++ b/1.4.10/docs/examples/example-example76/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example76-jquery + + + + + + + + + + +key down count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example76/index-production.html b/1.4.10/docs/examples/example-example76/index-production.html new file mode 100644 index 0000000000..19735eb650 --- /dev/null +++ b/1.4.10/docs/examples/example-example76/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example76-production + + + + + + + + + +key down count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example76/index.html b/1.4.10/docs/examples/example-example76/index.html new file mode 100644 index 0000000000..9e9e7869fb --- /dev/null +++ b/1.4.10/docs/examples/example-example76/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example76 + + + + + + + + + +key down count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example76/manifest.json b/1.4.10/docs/examples/example-example76/manifest.json new file mode 100644 index 0000000000..603a58919c --- /dev/null +++ b/1.4.10/docs/examples/example-example76/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example76", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example77/index-debug.html b/1.4.10/docs/examples/example-example77/index-debug.html new file mode 100644 index 0000000000..59b61e7799 --- /dev/null +++ b/1.4.10/docs/examples/example-example77/index-debug.html @@ -0,0 +1,22 @@ + + + + + Example - example-example77-debug + + + + + + + + +

Typing in the input box below updates the key count

+ key up count: {{count}} + +

Typing in the input box below updates the keycode

+ +

event keyCode: {{ event.keyCode }}

+

event altKey: {{ event.altKey }}

+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example77/index-jquery.html b/1.4.10/docs/examples/example-example77/index-jquery.html new file mode 100644 index 0000000000..fa1ab8e31e --- /dev/null +++ b/1.4.10/docs/examples/example-example77/index-jquery.html @@ -0,0 +1,23 @@ + + + + + Example - example-example77-jquery + + + + + + + + + +

Typing in the input box below updates the key count

+ key up count: {{count}} + +

Typing in the input box below updates the keycode

+ +

event keyCode: {{ event.keyCode }}

+

event altKey: {{ event.altKey }}

+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example77/index-production.html b/1.4.10/docs/examples/example-example77/index-production.html new file mode 100644 index 0000000000..d5a8d16700 --- /dev/null +++ b/1.4.10/docs/examples/example-example77/index-production.html @@ -0,0 +1,22 @@ + + + + + Example - example-example77-production + + + + + + + + +

Typing in the input box below updates the key count

+ key up count: {{count}} + +

Typing in the input box below updates the keycode

+ +

event keyCode: {{ event.keyCode }}

+

event altKey: {{ event.altKey }}

+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example77/index.html b/1.4.10/docs/examples/example-example77/index.html new file mode 100644 index 0000000000..a93694f058 --- /dev/null +++ b/1.4.10/docs/examples/example-example77/index.html @@ -0,0 +1,22 @@ + + + + + Example - example-example77 + + + + + + + + +

Typing in the input box below updates the key count

+ key up count: {{count}} + +

Typing in the input box below updates the keycode

+ +

event keyCode: {{ event.keyCode }}

+

event altKey: {{ event.altKey }}

+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example77/manifest.json b/1.4.10/docs/examples/example-example77/manifest.json new file mode 100644 index 0000000000..fa6864c758 --- /dev/null +++ b/1.4.10/docs/examples/example-example77/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example77", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example78/index-debug.html b/1.4.10/docs/examples/example-example78/index-debug.html new file mode 100644 index 0000000000..c3e032cd56 --- /dev/null +++ b/1.4.10/docs/examples/example-example78/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example78-debug + + + + + + + + + +key press count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example78/index-jquery.html b/1.4.10/docs/examples/example-example78/index-jquery.html new file mode 100644 index 0000000000..95a3853bdb --- /dev/null +++ b/1.4.10/docs/examples/example-example78/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example78-jquery + + + + + + + + + + +key press count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example78/index-production.html b/1.4.10/docs/examples/example-example78/index-production.html new file mode 100644 index 0000000000..5618b1b993 --- /dev/null +++ b/1.4.10/docs/examples/example-example78/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example78-production + + + + + + + + + +key press count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example78/index.html b/1.4.10/docs/examples/example-example78/index.html new file mode 100644 index 0000000000..0792822bbe --- /dev/null +++ b/1.4.10/docs/examples/example-example78/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example78 + + + + + + + + + +key press count: {{count}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example78/manifest.json b/1.4.10/docs/examples/example-example78/manifest.json new file mode 100644 index 0000000000..19dd18d8ca --- /dev/null +++ b/1.4.10/docs/examples/example-example78/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example78", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example79/index-debug.html b/1.4.10/docs/examples/example-example79/index-debug.html new file mode 100644 index 0000000000..2058fa05a7 --- /dev/null +++ b/1.4.10/docs/examples/example-example79/index-debug.html @@ -0,0 +1,34 @@ + + + + + Example - example-example79-debug + + + + + + + + + +
+ Enter text and hit enter: + + +
list={{list}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example79/index-jquery.html b/1.4.10/docs/examples/example-example79/index-jquery.html new file mode 100644 index 0000000000..968dbd3d90 --- /dev/null +++ b/1.4.10/docs/examples/example-example79/index-jquery.html @@ -0,0 +1,35 @@ + + + + + Example - example-example79-jquery + + + + + + + + + + +
+ Enter text and hit enter: + + +
list={{list}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example79/index-production.html b/1.4.10/docs/examples/example-example79/index-production.html new file mode 100644 index 0000000000..1fabf3054e --- /dev/null +++ b/1.4.10/docs/examples/example-example79/index-production.html @@ -0,0 +1,34 @@ + + + + + Example - example-example79-production + + + + + + + + + +
+ Enter text and hit enter: + + +
list={{list}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example79/index.html b/1.4.10/docs/examples/example-example79/index.html new file mode 100644 index 0000000000..63f6db5c3e --- /dev/null +++ b/1.4.10/docs/examples/example-example79/index.html @@ -0,0 +1,34 @@ + + + + + Example - example-example79 + + + + + + + + + +
+ Enter text and hit enter: + + +
list={{list}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example79/manifest.json b/1.4.10/docs/examples/example-example79/manifest.json new file mode 100644 index 0000000000..040870b3bb --- /dev/null +++ b/1.4.10/docs/examples/example-example79/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example79", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example79/protractor.js b/1.4.10/docs/examples/example-example79/protractor.js new file mode 100644 index 0000000000..e9920defb0 --- /dev/null +++ b/1.4.10/docs/examples/example-example79/protractor.js @@ -0,0 +1,12 @@ +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'); + }); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example8/app.js b/1.4.10/docs/examples/example-example8/app.js new file mode 100644 index 0000000000..d62b032da9 --- /dev/null +++ b/1.4.10/docs/examples/example-example8/app.js @@ -0,0 +1,13 @@ +(function(angular) { + 'use strict'; +var myApp = angular.module('spicyApp2', []); + +myApp.controller('SpicyController', ['$scope', function($scope) { + $scope.customSpice = "wasabi"; + $scope.spice = 'very'; + + $scope.spicy = function(spice) { + $scope.spice = spice; + }; +}]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example8/index-debug.html b/1.4.10/docs/examples/example-example8/index-debug.html new file mode 100644 index 0000000000..637eeb148d --- /dev/null +++ b/1.4.10/docs/examples/example-example8/index-debug.html @@ -0,0 +1,22 @@ + + + + + Example - example-example8-debug + + + + + + + + + +
+ + + +

The food is {{spice}} spicy!

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example8/index-jquery.html b/1.4.10/docs/examples/example-example8/index-jquery.html new file mode 100644 index 0000000000..d36c80e763 --- /dev/null +++ b/1.4.10/docs/examples/example-example8/index-jquery.html @@ -0,0 +1,23 @@ + + + + + Example - example-example8-jquery + + + + + + + + + + +
+ + + +

The food is {{spice}} spicy!

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example8/index-production.html b/1.4.10/docs/examples/example-example8/index-production.html new file mode 100644 index 0000000000..2387792c7e --- /dev/null +++ b/1.4.10/docs/examples/example-example8/index-production.html @@ -0,0 +1,22 @@ + + + + + Example - example-example8-production + + + + + + + + + +
+ + + +

The food is {{spice}} spicy!

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example8/index.html b/1.4.10/docs/examples/example-example8/index.html new file mode 100644 index 0000000000..0110a146fe --- /dev/null +++ b/1.4.10/docs/examples/example-example8/index.html @@ -0,0 +1,22 @@ + + + + + Example - example-example8 + + + + + + + + + +
+ + + +

The food is {{spice}} spicy!

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example8/manifest.json b/1.4.10/docs/examples/example-example8/manifest.json new file mode 100644 index 0000000000..e285068dc2 --- /dev/null +++ b/1.4.10/docs/examples/example-example8/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example8", + "files": [ + "index-production.html", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example80/index-debug.html b/1.4.10/docs/examples/example-example80/index-debug.html new file mode 100644 index 0000000000..ad771b8963 --- /dev/null +++ b/1.4.10/docs/examples/example-example80/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example80-debug + + + + + + + + + +copied: {{copied}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example80/index-jquery.html b/1.4.10/docs/examples/example-example80/index-jquery.html new file mode 100644 index 0000000000..e9b450d2d4 --- /dev/null +++ b/1.4.10/docs/examples/example-example80/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example80-jquery + + + + + + + + + + +copied: {{copied}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example80/index-production.html b/1.4.10/docs/examples/example-example80/index-production.html new file mode 100644 index 0000000000..d07537c088 --- /dev/null +++ b/1.4.10/docs/examples/example-example80/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example80-production + + + + + + + + + +copied: {{copied}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example80/index.html b/1.4.10/docs/examples/example-example80/index.html new file mode 100644 index 0000000000..b06c500138 --- /dev/null +++ b/1.4.10/docs/examples/example-example80/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example80 + + + + + + + + + +copied: {{copied}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example80/manifest.json b/1.4.10/docs/examples/example-example80/manifest.json new file mode 100644 index 0000000000..cc682fd4bb --- /dev/null +++ b/1.4.10/docs/examples/example-example80/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example80", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example81/index-debug.html b/1.4.10/docs/examples/example-example81/index-debug.html new file mode 100644 index 0000000000..44b1ffd7bc --- /dev/null +++ b/1.4.10/docs/examples/example-example81/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example81-debug + + + + + + + + + +cut: {{cut}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example81/index-jquery.html b/1.4.10/docs/examples/example-example81/index-jquery.html new file mode 100644 index 0000000000..492736287d --- /dev/null +++ b/1.4.10/docs/examples/example-example81/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example81-jquery + + + + + + + + + + +cut: {{cut}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example81/index-production.html b/1.4.10/docs/examples/example-example81/index-production.html new file mode 100644 index 0000000000..1f404798f3 --- /dev/null +++ b/1.4.10/docs/examples/example-example81/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example81-production + + + + + + + + + +cut: {{cut}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example81/index.html b/1.4.10/docs/examples/example-example81/index.html new file mode 100644 index 0000000000..9db235f64d --- /dev/null +++ b/1.4.10/docs/examples/example-example81/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example81 + + + + + + + + + +cut: {{cut}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example81/manifest.json b/1.4.10/docs/examples/example-example81/manifest.json new file mode 100644 index 0000000000..ae98b40f90 --- /dev/null +++ b/1.4.10/docs/examples/example-example81/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example81", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example82/index-debug.html b/1.4.10/docs/examples/example-example82/index-debug.html new file mode 100644 index 0000000000..551d43ba5c --- /dev/null +++ b/1.4.10/docs/examples/example-example82/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example82-debug + + + + + + + + + +pasted: {{paste}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example82/index-jquery.html b/1.4.10/docs/examples/example-example82/index-jquery.html new file mode 100644 index 0000000000..9a74d1a483 --- /dev/null +++ b/1.4.10/docs/examples/example-example82/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example82-jquery + + + + + + + + + + +pasted: {{paste}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example82/index-production.html b/1.4.10/docs/examples/example-example82/index-production.html new file mode 100644 index 0000000000..c248b0846b --- /dev/null +++ b/1.4.10/docs/examples/example-example82/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example82-production + + + + + + + + + +pasted: {{paste}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example82/index.html b/1.4.10/docs/examples/example-example82/index.html new file mode 100644 index 0000000000..199e196e5b --- /dev/null +++ b/1.4.10/docs/examples/example-example82/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example82 + + + + + + + + + +pasted: {{paste}} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example82/manifest.json b/1.4.10/docs/examples/example-example82/manifest.json new file mode 100644 index 0000000000..f284dc8063 --- /dev/null +++ b/1.4.10/docs/examples/example-example82/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example82", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example83/animations.css b/1.4.10/docs/examples/example-example83/animations.css new file mode 100644 index 0000000000..b1f6c2237e --- /dev/null +++ b/1.4.10/docs/examples/example-example83/animations.css @@ -0,0 +1,19 @@ +.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; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example83/index-debug.html b/1.4.10/docs/examples/example-example83/index-debug.html new file mode 100644 index 0000000000..913daffa02 --- /dev/null +++ b/1.4.10/docs/examples/example-example83/index-debug.html @@ -0,0 +1,22 @@ + + + + + Example - example-example83-debug + + + + + + + + + + +
+Show when checked: + + This is removed when the checkbox is unchecked. + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example83/index-jquery.html b/1.4.10/docs/examples/example-example83/index-jquery.html new file mode 100644 index 0000000000..53e2dbd6d1 --- /dev/null +++ b/1.4.10/docs/examples/example-example83/index-jquery.html @@ -0,0 +1,23 @@ + + + + + Example - example-example83-jquery + + + + + + + + + + + +
+Show when checked: + + This is removed when the checkbox is unchecked. + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example83/index-production.html b/1.4.10/docs/examples/example-example83/index-production.html new file mode 100644 index 0000000000..2eef2d477a --- /dev/null +++ b/1.4.10/docs/examples/example-example83/index-production.html @@ -0,0 +1,22 @@ + + + + + Example - example-example83-production + + + + + + + + + + +
+Show when checked: + + This is removed when the checkbox is unchecked. + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example83/index.html b/1.4.10/docs/examples/example-example83/index.html new file mode 100644 index 0000000000..8160999c0a --- /dev/null +++ b/1.4.10/docs/examples/example-example83/index.html @@ -0,0 +1,22 @@ + + + + + Example - example-example83 + + + + + + + + + + +
+Show when checked: + + This is removed when the checkbox is unchecked. + + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example83/manifest.json b/1.4.10/docs/examples/example-example83/manifest.json new file mode 100644 index 0000000000..09421deb3e --- /dev/null +++ b/1.4.10/docs/examples/example-example83/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example83", + "files": [ + "index-production.html", + "animations.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example84/animations.css b/1.4.10/docs/examples/example-example84/animations.css new file mode 100644 index 0000000000..1f92938d7c --- /dev/null +++ b/1.4.10/docs/examples/example-example84/animations.css @@ -0,0 +1,37 @@ +.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; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example84/index-debug.html b/1.4.10/docs/examples/example-example84/index-debug.html new file mode 100644 index 0000000000..b0094c73ef --- /dev/null +++ b/1.4.10/docs/examples/example-example84/index-debug.html @@ -0,0 +1,28 @@ + + + + + Example - example-example84-debug + + + + + + + + + + + +
+ + url of the template: {{template.url}} +
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example84/index-jquery.html b/1.4.10/docs/examples/example-example84/index-jquery.html new file mode 100644 index 0000000000..ddac4c22da --- /dev/null +++ b/1.4.10/docs/examples/example-example84/index-jquery.html @@ -0,0 +1,29 @@ + + + + + Example - example-example84-jquery + + + + + + + + + + + + +
+ + url of the template: {{template.url}} +
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example84/index-production.html b/1.4.10/docs/examples/example-example84/index-production.html new file mode 100644 index 0000000000..6fa00cf914 --- /dev/null +++ b/1.4.10/docs/examples/example-example84/index-production.html @@ -0,0 +1,28 @@ + + + + + Example - example-example84-production + + + + + + + + + + + +
+ + url of the template: {{template.url}} +
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example84/index.html b/1.4.10/docs/examples/example-example84/index.html new file mode 100644 index 0000000000..fcfa26d73c --- /dev/null +++ b/1.4.10/docs/examples/example-example84/index.html @@ -0,0 +1,28 @@ + + + + + Example - example-example84 + + + + + + + + + + + +
+ + url of the template: {{template.url}} +
+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example84/manifest.json b/1.4.10/docs/examples/example-example84/manifest.json new file mode 100644 index 0000000000..34e428739c --- /dev/null +++ b/1.4.10/docs/examples/example-example84/manifest.json @@ -0,0 +1,11 @@ +{ + "name": "example-example84", + "files": [ + "index-production.html", + "script.js", + "template1.html", + "template2.html", + "animations.css", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example84/protractor.js b/1.4.10/docs/examples/example-example84/protractor.js new file mode 100644 index 0000000000..e0b43f77bd --- /dev/null +++ b/1.4.10/docs/examples/example-example84/protractor.js @@ -0,0 +1,27 @@ +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); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example84/script.js b/1.4.10/docs/examples/example-example84/script.js new file mode 100644 index 0000000000..de24dc8fc7 --- /dev/null +++ b/1.4.10/docs/examples/example-example84/script.js @@ -0,0 +1,10 @@ +(function(angular) { + 'use strict'; +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]; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example84/template1.html b/1.4.10/docs/examples/example-example84/template1.html new file mode 100644 index 0000000000..a77c76a4a1 --- /dev/null +++ b/1.4.10/docs/examples/example-example84/template1.html @@ -0,0 +1 @@ +Content of template1.html \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example84/template2.html b/1.4.10/docs/examples/example-example84/template2.html new file mode 100644 index 0000000000..2d108be8b7 --- /dev/null +++ b/1.4.10/docs/examples/example-example84/template2.html @@ -0,0 +1 @@ +Content of template2.html \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example85/index-debug.html b/1.4.10/docs/examples/example-example85/index-debug.html new file mode 100644 index 0000000000..94bb94b472 --- /dev/null +++ b/1.4.10/docs/examples/example-example85/index-debug.html @@ -0,0 +1,28 @@ + + + + + Example - example-example85-debug + + + + + + + + + +
+
+
+ list[ {{outerIndex}} ][ {{innerIndex}} ] = {{value}}; +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example85/index-jquery.html b/1.4.10/docs/examples/example-example85/index-jquery.html new file mode 100644 index 0000000000..b9f0220901 --- /dev/null +++ b/1.4.10/docs/examples/example-example85/index-jquery.html @@ -0,0 +1,29 @@ + + + + + Example - example-example85-jquery + + + + + + + + + + +
+
+
+ list[ {{outerIndex}} ][ {{innerIndex}} ] = {{value}}; +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example85/index-production.html b/1.4.10/docs/examples/example-example85/index-production.html new file mode 100644 index 0000000000..f6cee30e8a --- /dev/null +++ b/1.4.10/docs/examples/example-example85/index-production.html @@ -0,0 +1,28 @@ + + + + + Example - example-example85-production + + + + + + + + + +
+
+
+ list[ {{outerIndex}} ][ {{innerIndex}} ] = {{value}}; +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example85/index.html b/1.4.10/docs/examples/example-example85/index.html new file mode 100644 index 0000000000..adf8e60649 --- /dev/null +++ b/1.4.10/docs/examples/example-example85/index.html @@ -0,0 +1,28 @@ + + + + + Example - example-example85 + + + + + + + + + +
+
+
+ list[ {{outerIndex}} ][ {{innerIndex}} ] = {{value}}; +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example85/manifest.json b/1.4.10/docs/examples/example-example85/manifest.json new file mode 100644 index 0000000000..f835b19b0f --- /dev/null +++ b/1.4.10/docs/examples/example-example85/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example85", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example85/protractor.js b/1.4.10/docs/examples/example-example85/protractor.js new file mode 100644 index 0000000000..5c48a97c5a --- /dev/null +++ b/1.4.10/docs/examples/example-example85/protractor.js @@ -0,0 +1,7 @@ +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;'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example86/index-debug.html b/1.4.10/docs/examples/example-example86/index-debug.html new file mode 100644 index 0000000000..c27043f803 --- /dev/null +++ b/1.4.10/docs/examples/example-example86/index-debug.html @@ -0,0 +1,42 @@ + + + + + Example - example-example86-debug + + + + + + + + + + + +

+ Update input to see transitions when valid/invalid. + Integer is a valid value. +

+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example86/index-jquery.html b/1.4.10/docs/examples/example-example86/index-jquery.html new file mode 100644 index 0000000000..4fe46e414d --- /dev/null +++ b/1.4.10/docs/examples/example-example86/index-jquery.html @@ -0,0 +1,43 @@ + + + + + Example - example-example86-jquery + + + + + + + + + + + + +

+ Update input to see transitions when valid/invalid. + Integer is a valid value. +

+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example86/index-production.html b/1.4.10/docs/examples/example-example86/index-production.html new file mode 100644 index 0000000000..73459fe53c --- /dev/null +++ b/1.4.10/docs/examples/example-example86/index-production.html @@ -0,0 +1,42 @@ + + + + + Example - example-example86-production + + + + + + + + + + + +

+ Update input to see transitions when valid/invalid. + Integer is a valid value. +

+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example86/index.html b/1.4.10/docs/examples/example-example86/index.html new file mode 100644 index 0000000000..2eeb8a8e27 --- /dev/null +++ b/1.4.10/docs/examples/example-example86/index.html @@ -0,0 +1,42 @@ + + + + + Example - example-example86 + + + + + + + + + + + +

+ Update input to see transitions when valid/invalid. + Integer is a valid value. +

+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example86/manifest.json b/1.4.10/docs/examples/example-example86/manifest.json new file mode 100644 index 0000000000..7c007f3b5e --- /dev/null +++ b/1.4.10/docs/examples/example-example86/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-example86", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example87/index-debug.html b/1.4.10/docs/examples/example-example87/index-debug.html new file mode 100644 index 0000000000..ad9ee0771b --- /dev/null +++ b/1.4.10/docs/examples/example-example87/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-example87-debug + + + + + + + + +
Normal: {{1 + 2}}
+
Ignored: {{1 + 2}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example87/index-jquery.html b/1.4.10/docs/examples/example-example87/index-jquery.html new file mode 100644 index 0000000000..aa2b81a4f9 --- /dev/null +++ b/1.4.10/docs/examples/example-example87/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-example87-jquery + + + + + + + + + +
Normal: {{1 + 2}}
+
Ignored: {{1 + 2}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example87/index-production.html b/1.4.10/docs/examples/example-example87/index-production.html new file mode 100644 index 0000000000..0b16f5839b --- /dev/null +++ b/1.4.10/docs/examples/example-example87/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-example87-production + + + + + + + + +
Normal: {{1 + 2}}
+
Ignored: {{1 + 2}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example87/index.html b/1.4.10/docs/examples/example-example87/index.html new file mode 100644 index 0000000000..c9ea5c6dc0 --- /dev/null +++ b/1.4.10/docs/examples/example-example87/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-example87 + + + + + + + + +
Normal: {{1 + 2}}
+
Ignored: {{1 + 2}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example87/manifest.json b/1.4.10/docs/examples/example-example87/manifest.json new file mode 100644 index 0000000000..17e0442c11 --- /dev/null +++ b/1.4.10/docs/examples/example-example87/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example87", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example87/protractor.js b/1.4.10/docs/examples/example-example87/protractor.js new file mode 100644 index 0000000000..1acda3c144 --- /dev/null +++ b/1.4.10/docs/examples/example-example87/protractor.js @@ -0,0 +1,4 @@ +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/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example88/index-debug.html b/1.4.10/docs/examples/example-example88/index-debug.html new file mode 100644 index 0000000000..a40ddcc060 --- /dev/null +++ b/1.4.10/docs/examples/example-example88/index-debug.html @@ -0,0 +1,71 @@ + + + + + Example - example-example88-debug + + + + + + + + + +
+
    +
  • + + + +
  • +
  • + +
  • +
+
+
+
+ +
+ +
+ + + + Select . +
+
+ Currently selected: {{ {selected_color:myColor} }} +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example88/index-jquery.html b/1.4.10/docs/examples/example-example88/index-jquery.html new file mode 100644 index 0000000000..18728d85c3 --- /dev/null +++ b/1.4.10/docs/examples/example-example88/index-jquery.html @@ -0,0 +1,72 @@ + + + + + Example - example-example88-jquery + + + + + + + + + + +
+
    +
  • + + + +
  • +
  • + +
  • +
+
+
+
+ +
+ +
+ + + + Select . +
+
+ Currently selected: {{ {selected_color:myColor} }} +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example88/index-production.html b/1.4.10/docs/examples/example-example88/index-production.html new file mode 100644 index 0000000000..572dd86dd3 --- /dev/null +++ b/1.4.10/docs/examples/example-example88/index-production.html @@ -0,0 +1,71 @@ + + + + + Example - example-example88-production + + + + + + + + + +
+
    +
  • + + + +
  • +
  • + +
  • +
+
+
+
+ +
+ +
+ + + + Select . +
+
+ Currently selected: {{ {selected_color:myColor} }} +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example88/index.html b/1.4.10/docs/examples/example-example88/index.html new file mode 100644 index 0000000000..77a4ff1eaa --- /dev/null +++ b/1.4.10/docs/examples/example-example88/index.html @@ -0,0 +1,71 @@ + + + + + Example - example-example88 + + + + + + + + + +
+
    +
  • + + + +
  • +
  • + +
  • +
+
+
+
+ +
+ +
+ + + + Select . +
+
+ Currently selected: {{ {selected_color:myColor} }} +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example88/manifest.json b/1.4.10/docs/examples/example-example88/manifest.json new file mode 100644 index 0000000000..64c4ccfa99 --- /dev/null +++ b/1.4.10/docs/examples/example-example88/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example88", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example88/protractor.js b/1.4.10/docs/examples/example-example88/protractor.js new file mode 100644 index 0000000000..8fe6d040b8 --- /dev/null +++ b/1.4.10/docs/examples/example-example88/protractor.js @@ -0,0 +1,9 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example89/index-debug.html b/1.4.10/docs/examples/example-example89/index-debug.html new file mode 100644 index 0000000000..96a75e124d --- /dev/null +++ b/1.4.10/docs/examples/example-example89/index-debug.html @@ -0,0 +1,46 @@ + + + + + Example - example-example89-debug + + + + + + + + + +
+
+
+
+ + + Without Offset: + +
+ + + With Offset(2): + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example89/index-jquery.html b/1.4.10/docs/examples/example-example89/index-jquery.html new file mode 100644 index 0000000000..2d35cf13dd --- /dev/null +++ b/1.4.10/docs/examples/example-example89/index-jquery.html @@ -0,0 +1,47 @@ + + + + + Example - example-example89-jquery + + + + + + + + + + +
+
+
+
+ + + Without Offset: + +
+ + + With Offset(2): + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example89/index-production.html b/1.4.10/docs/examples/example-example89/index-production.html new file mode 100644 index 0000000000..588e8f3726 --- /dev/null +++ b/1.4.10/docs/examples/example-example89/index-production.html @@ -0,0 +1,46 @@ + + + + + Example - example-example89-production + + + + + + + + + +
+
+
+
+ + + Without Offset: + +
+ + + With Offset(2): + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example89/index.html b/1.4.10/docs/examples/example-example89/index.html new file mode 100644 index 0000000000..6856f0ffaa --- /dev/null +++ b/1.4.10/docs/examples/example-example89/index.html @@ -0,0 +1,46 @@ + + + + + Example - example-example89 + + + + + + + + + +
+
+
+
+ + + Without Offset: + +
+ + + With Offset(2): + + +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example89/manifest.json b/1.4.10/docs/examples/example-example89/manifest.json new file mode 100644 index 0000000000..2bfd5ff4b5 --- /dev/null +++ b/1.4.10/docs/examples/example-example89/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example89", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example89/protractor.js b/1.4.10/docs/examples/example-example89/protractor.js new file mode 100644 index 0000000000..3261ab63c0 --- /dev/null +++ b/1.4.10/docs/examples/example-example89/protractor.js @@ -0,0 +1,45 @@ +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.'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example9/app.css b/1.4.10/docs/examples/example-example9/app.css new file mode 100644 index 0000000000..a99bd2254b --- /dev/null +++ b/1.4.10/docs/examples/example-example9/app.css @@ -0,0 +1,4 @@ +div.spicy div { + padding: 10px; + border: solid 2px blue; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example9/app.js b/1.4.10/docs/examples/example-example9/app.js new file mode 100644 index 0000000000..58e964a909 --- /dev/null +++ b/1.4.10/docs/examples/example-example9/app.js @@ -0,0 +1,15 @@ +(function(angular) { + 'use strict'; +var myApp = angular.module('scopeInheritance', []); +myApp.controller('MainController', ['$scope', function($scope) { + $scope.timeOfDay = 'morning'; + $scope.name = 'Nikki'; +}]); +myApp.controller('ChildController', ['$scope', function($scope) { + $scope.name = 'Mattie'; +}]); +myApp.controller('GrandChildController', ['$scope', function($scope) { + $scope.timeOfDay = 'evening'; + $scope.name = 'Gingerbread Baby'; +}]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example9/index-debug.html b/1.4.10/docs/examples/example-example9/index-debug.html new file mode 100644 index 0000000000..513cb8bd76 --- /dev/null +++ b/1.4.10/docs/examples/example-example9/index-debug.html @@ -0,0 +1,30 @@ + + + + + Example - example-example9-debug + + + + + + + + + + +
+
+

Good {{timeOfDay}}, {{name}}!

+ +
+

Good {{timeOfDay}}, {{name}}!

+ +
+

Good {{timeOfDay}}, {{name}}!

+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example9/index-jquery.html b/1.4.10/docs/examples/example-example9/index-jquery.html new file mode 100644 index 0000000000..d9f1afcc0b --- /dev/null +++ b/1.4.10/docs/examples/example-example9/index-jquery.html @@ -0,0 +1,31 @@ + + + + + Example - example-example9-jquery + + + + + + + + + + + +
+
+

Good {{timeOfDay}}, {{name}}!

+ +
+

Good {{timeOfDay}}, {{name}}!

+ +
+

Good {{timeOfDay}}, {{name}}!

+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example9/index-production.html b/1.4.10/docs/examples/example-example9/index-production.html new file mode 100644 index 0000000000..d77c82c6c1 --- /dev/null +++ b/1.4.10/docs/examples/example-example9/index-production.html @@ -0,0 +1,30 @@ + + + + + Example - example-example9-production + + + + + + + + + + +
+
+

Good {{timeOfDay}}, {{name}}!

+ +
+

Good {{timeOfDay}}, {{name}}!

+ +
+

Good {{timeOfDay}}, {{name}}!

+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example9/index.html b/1.4.10/docs/examples/example-example9/index.html new file mode 100644 index 0000000000..667b2484dd --- /dev/null +++ b/1.4.10/docs/examples/example-example9/index.html @@ -0,0 +1,30 @@ + + + + + Example - example-example9 + + + + + + + + + + +
+
+

Good {{timeOfDay}}, {{name}}!

+ +
+

Good {{timeOfDay}}, {{name}}!

+ +
+

Good {{timeOfDay}}, {{name}}!

+
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example9/manifest.json b/1.4.10/docs/examples/example-example9/manifest.json new file mode 100644 index 0000000000..6074ee5738 --- /dev/null +++ b/1.4.10/docs/examples/example-example9/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example9", + "files": [ + "index-production.html", + "app.css", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example90/animations.css b/1.4.10/docs/examples/example-example90/animations.css new file mode 100644 index 0000000000..2da539cf35 --- /dev/null +++ b/1.4.10/docs/examples/example-example90/animations.css @@ -0,0 +1,33 @@ +.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; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example90/index-debug.html b/1.4.10/docs/examples/example-example90/index-debug.html new file mode 100644 index 0000000000..91ae5626c6 --- /dev/null +++ b/1.4.10/docs/examples/example-example90/index-debug.html @@ -0,0 +1,40 @@ + + + + + Example - example-example90-debug + + + + + + + + + + +
+ I have {{friends.length}} friends. They are: + +
    +
  • + [{{$index + 1}}] {{friend.name}} who is {{friend.age}} years old. +
  • +
  • + No results found... +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example90/index-jquery.html b/1.4.10/docs/examples/example-example90/index-jquery.html new file mode 100644 index 0000000000..25bef3783b --- /dev/null +++ b/1.4.10/docs/examples/example-example90/index-jquery.html @@ -0,0 +1,41 @@ + + + + + Example - example-example90-jquery + + + + + + + + + + + +
+ I have {{friends.length}} friends. They are: + +
    +
  • + [{{$index + 1}}] {{friend.name}} who is {{friend.age}} years old. +
  • +
  • + No results found... +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example90/index-production.html b/1.4.10/docs/examples/example-example90/index-production.html new file mode 100644 index 0000000000..84d420dd40 --- /dev/null +++ b/1.4.10/docs/examples/example-example90/index-production.html @@ -0,0 +1,40 @@ + + + + + Example - example-example90-production + + + + + + + + + + +
+ I have {{friends.length}} friends. They are: + +
    +
  • + [{{$index + 1}}] {{friend.name}} who is {{friend.age}} years old. +
  • +
  • + No results found... +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example90/index.html b/1.4.10/docs/examples/example-example90/index.html new file mode 100644 index 0000000000..378cc7bce1 --- /dev/null +++ b/1.4.10/docs/examples/example-example90/index.html @@ -0,0 +1,40 @@ + + + + + Example - example-example90 + + + + + + + + + + +
+ I have {{friends.length}} friends. They are: + +
    +
  • + [{{$index + 1}}] {{friend.name}} who is {{friend.age}} years old. +
  • +
  • + No results found... +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example90/manifest.json b/1.4.10/docs/examples/example-example90/manifest.json new file mode 100644 index 0000000000..a0fb8f44fb --- /dev/null +++ b/1.4.10/docs/examples/example-example90/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example90", + "files": [ + "index-production.html", + "animations.css", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example90/protractor.js b/1.4.10/docs/examples/example-example90/protractor.js new file mode 100644 index 0000000000..b90d6fb61a --- /dev/null +++ b/1.4.10/docs/examples/example-example90/protractor.js @@ -0,0 +1,20 @@ +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.'); + }); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example91/animations.css b/1.4.10/docs/examples/example-example91/animations.css new file mode 100644 index 0000000000..bcfc7e0dfb --- /dev/null +++ b/1.4.10/docs/examples/example-example91/animations.css @@ -0,0 +1,23 @@ +.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; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example91/glyphicons.css b/1.4.10/docs/examples/example-example91/glyphicons.css new file mode 100644 index 0000000000..bf48d0f45c --- /dev/null +++ b/1.4.10/docs/examples/example-example91/glyphicons.css @@ -0,0 +1 @@ +@import url(../../components/bootstrap-3.1.1/css/bootstrap.css); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example91/index-debug.html b/1.4.10/docs/examples/example-example91/index-debug.html new file mode 100644 index 0000000000..15d46bd0be --- /dev/null +++ b/1.4.10/docs/examples/example-example91/index-debug.html @@ -0,0 +1,31 @@ + + + + + Example - example-example91-debug + + + + + + + + + + + + Click me:
+
+ Show: +
+ I show up when your checkbox is checked. +
+
+
+ Hide: +
+ I hide when your checkbox is checked. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example91/index-jquery.html b/1.4.10/docs/examples/example-example91/index-jquery.html new file mode 100644 index 0000000000..e5d4501907 --- /dev/null +++ b/1.4.10/docs/examples/example-example91/index-jquery.html @@ -0,0 +1,32 @@ + + + + + Example - example-example91-jquery + + + + + + + + + + + + + Click me:
+
+ Show: +
+ I show up when your checkbox is checked. +
+
+
+ Hide: +
+ I hide when your checkbox is checked. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example91/index-production.html b/1.4.10/docs/examples/example-example91/index-production.html new file mode 100644 index 0000000000..20047c54cd --- /dev/null +++ b/1.4.10/docs/examples/example-example91/index-production.html @@ -0,0 +1,31 @@ + + + + + Example - example-example91-production + + + + + + + + + + + + Click me:
+
+ Show: +
+ I show up when your checkbox is checked. +
+
+
+ Hide: +
+ I hide when your checkbox is checked. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example91/index.html b/1.4.10/docs/examples/example-example91/index.html new file mode 100644 index 0000000000..920592b02d --- /dev/null +++ b/1.4.10/docs/examples/example-example91/index.html @@ -0,0 +1,31 @@ + + + + + Example - example-example91 + + + + + + + + + + + + Click me:
+
+ Show: +
+ I show up when your checkbox is checked. +
+
+
+ Hide: +
+ I hide when your checkbox is checked. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example91/manifest.json b/1.4.10/docs/examples/example-example91/manifest.json new file mode 100644 index 0000000000..59721a6e08 --- /dev/null +++ b/1.4.10/docs/examples/example-example91/manifest.json @@ -0,0 +1,9 @@ +{ + "name": "example-example91", + "files": [ + "index-production.html", + "glyphicons.css", + "animations.css", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example91/protractor.js b/1.4.10/docs/examples/example-example91/protractor.js new file mode 100644 index 0000000000..da70674f30 --- /dev/null +++ b/1.4.10/docs/examples/example-example91/protractor.js @@ -0,0 +1,12 @@ +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(); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example92/animations.css b/1.4.10/docs/examples/example-example92/animations.css new file mode 100644 index 0000000000..c1d270e7f5 --- /dev/null +++ b/1.4.10/docs/examples/example-example92/animations.css @@ -0,0 +1,20 @@ +.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; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example92/glyphicons.css b/1.4.10/docs/examples/example-example92/glyphicons.css new file mode 100644 index 0000000000..bf48d0f45c --- /dev/null +++ b/1.4.10/docs/examples/example-example92/glyphicons.css @@ -0,0 +1 @@ +@import url(../../components/bootstrap-3.1.1/css/bootstrap.css); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example92/index-debug.html b/1.4.10/docs/examples/example-example92/index-debug.html new file mode 100644 index 0000000000..cb24c8f5d9 --- /dev/null +++ b/1.4.10/docs/examples/example-example92/index-debug.html @@ -0,0 +1,31 @@ + + + + + Example - example-example92-debug + + + + + + + + + + + + Click me:
+
+ Show: +
+ I show up when your checkbox is checked. +
+
+
+ Hide: +
+ I hide when your checkbox is checked. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example92/index-jquery.html b/1.4.10/docs/examples/example-example92/index-jquery.html new file mode 100644 index 0000000000..2c1a426f52 --- /dev/null +++ b/1.4.10/docs/examples/example-example92/index-jquery.html @@ -0,0 +1,32 @@ + + + + + Example - example-example92-jquery + + + + + + + + + + + + + Click me:
+
+ Show: +
+ I show up when your checkbox is checked. +
+
+
+ Hide: +
+ I hide when your checkbox is checked. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example92/index-production.html b/1.4.10/docs/examples/example-example92/index-production.html new file mode 100644 index 0000000000..a10e71cc70 --- /dev/null +++ b/1.4.10/docs/examples/example-example92/index-production.html @@ -0,0 +1,31 @@ + + + + + Example - example-example92-production + + + + + + + + + + + + Click me:
+
+ Show: +
+ I show up when your checkbox is checked. +
+
+
+ Hide: +
+ I hide when your checkbox is checked. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example92/index.html b/1.4.10/docs/examples/example-example92/index.html new file mode 100644 index 0000000000..dc19eb7cf0 --- /dev/null +++ b/1.4.10/docs/examples/example-example92/index.html @@ -0,0 +1,31 @@ + + + + + Example - example-example92 + + + + + + + + + + + + Click me:
+
+ Show: +
+ I show up when your checkbox is checked. +
+
+
+ Hide: +
+ I hide when your checkbox is checked. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example92/manifest.json b/1.4.10/docs/examples/example-example92/manifest.json new file mode 100644 index 0000000000..858787f197 --- /dev/null +++ b/1.4.10/docs/examples/example-example92/manifest.json @@ -0,0 +1,9 @@ +{ + "name": "example-example92", + "files": [ + "index-production.html", + "glyphicons.css", + "animations.css", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example92/protractor.js b/1.4.10/docs/examples/example-example92/protractor.js new file mode 100644 index 0000000000..da70674f30 --- /dev/null +++ b/1.4.10/docs/examples/example-example92/protractor.js @@ -0,0 +1,12 @@ +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(); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example93/index-debug.html b/1.4.10/docs/examples/example-example93/index-debug.html new file mode 100644 index 0000000000..bfd021dc39 --- /dev/null +++ b/1.4.10/docs/examples/example-example93/index-debug.html @@ -0,0 +1,22 @@ + + + + + Example - example-example93-debug + + + + + + + + + + + + +
+Sample Text +
myStyle={{myStyle}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example93/index-jquery.html b/1.4.10/docs/examples/example-example93/index-jquery.html new file mode 100644 index 0000000000..f6c02f9a10 --- /dev/null +++ b/1.4.10/docs/examples/example-example93/index-jquery.html @@ -0,0 +1,23 @@ + + + + + Example - example-example93-jquery + + + + + + + + + + + + + +
+Sample Text +
myStyle={{myStyle}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example93/index-production.html b/1.4.10/docs/examples/example-example93/index-production.html new file mode 100644 index 0000000000..083d2a97a3 --- /dev/null +++ b/1.4.10/docs/examples/example-example93/index-production.html @@ -0,0 +1,22 @@ + + + + + Example - example-example93-production + + + + + + + + + + + + +
+Sample Text +
myStyle={{myStyle}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example93/index.html b/1.4.10/docs/examples/example-example93/index.html new file mode 100644 index 0000000000..21fce887f9 --- /dev/null +++ b/1.4.10/docs/examples/example-example93/index.html @@ -0,0 +1,22 @@ + + + + + Example - example-example93 + + + + + + + + + + + + +
+Sample Text +
myStyle={{myStyle}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example93/manifest.json b/1.4.10/docs/examples/example-example93/manifest.json new file mode 100644 index 0000000000..c3b45f3729 --- /dev/null +++ b/1.4.10/docs/examples/example-example93/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-example93", + "files": [ + "index-production.html", + "style.css", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example93/protractor.js b/1.4.10/docs/examples/example-example93/protractor.js new file mode 100644 index 0000000000..06b8690f5f --- /dev/null +++ b/1.4.10/docs/examples/example-example93/protractor.js @@ -0,0 +1,9 @@ +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)'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example93/style.css b/1.4.10/docs/examples/example-example93/style.css new file mode 100644 index 0000000000..c635ff6078 --- /dev/null +++ b/1.4.10/docs/examples/example-example93/style.css @@ -0,0 +1,3 @@ +span { + color: black; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example94/animations.css b/1.4.10/docs/examples/example-example94/animations.css new file mode 100644 index 0000000000..0769553769 --- /dev/null +++ b/1.4.10/docs/examples/example-example94/animations.css @@ -0,0 +1,30 @@ +.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; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example94/index-debug.html b/1.4.10/docs/examples/example-example94/index-debug.html new file mode 100644 index 0000000000..7bf94b90d1 --- /dev/null +++ b/1.4.10/docs/examples/example-example94/index-debug.html @@ -0,0 +1,30 @@ + + + + + Example - example-example94-debug + + + + + + + + + + + +
+ + selection={{selection}} +
+
+
Settings Div
+
Home Span
+
default
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example94/index-jquery.html b/1.4.10/docs/examples/example-example94/index-jquery.html new file mode 100644 index 0000000000..6f944ec721 --- /dev/null +++ b/1.4.10/docs/examples/example-example94/index-jquery.html @@ -0,0 +1,31 @@ + + + + + Example - example-example94-jquery + + + + + + + + + + + + +
+ + selection={{selection}} +
+
+
Settings Div
+
Home Span
+
default
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example94/index-production.html b/1.4.10/docs/examples/example-example94/index-production.html new file mode 100644 index 0000000000..309b08edeb --- /dev/null +++ b/1.4.10/docs/examples/example-example94/index-production.html @@ -0,0 +1,30 @@ + + + + + Example - example-example94-production + + + + + + + + + + + +
+ + selection={{selection}} +
+
+
Settings Div
+
Home Span
+
default
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example94/index.html b/1.4.10/docs/examples/example-example94/index.html new file mode 100644 index 0000000000..cd3389f7d2 --- /dev/null +++ b/1.4.10/docs/examples/example-example94/index.html @@ -0,0 +1,30 @@ + + + + + Example - example-example94 + + + + + + + + + + + +
+ + selection={{selection}} +
+
+
Settings Div
+
Home Span
+
default
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example94/manifest.json b/1.4.10/docs/examples/example-example94/manifest.json new file mode 100644 index 0000000000..380f5fb30e --- /dev/null +++ b/1.4.10/docs/examples/example-example94/manifest.json @@ -0,0 +1,9 @@ +{ + "name": "example-example94", + "files": [ + "index-production.html", + "script.js", + "animations.css", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example94/protractor.js b/1.4.10/docs/examples/example-example94/protractor.js new file mode 100644 index 0000000000..4cc792035c --- /dev/null +++ b/1.4.10/docs/examples/example-example94/protractor.js @@ -0,0 +1,14 @@ +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/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example94/script.js b/1.4.10/docs/examples/example-example94/script.js new file mode 100644 index 0000000000..328f85f191 --- /dev/null +++ b/1.4.10/docs/examples/example-example94/script.js @@ -0,0 +1,8 @@ +(function(angular) { + 'use strict'; +angular.module('switchExample', ['ngAnimate']) + .controller('ExampleController', ['$scope', function($scope) { + $scope.items = ['settings', 'home', 'other']; + $scope.selection = $scope.items[0]; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example95/index-debug.html b/1.4.10/docs/examples/example-example95/index-debug.html new file mode 100644 index 0000000000..f005ceb7c5 --- /dev/null +++ b/1.4.10/docs/examples/example-example95/index-debug.html @@ -0,0 +1,38 @@ + + + + + Example - example-example95-debug + + + + + + + + + +
+
+
+ {{text}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example95/index-jquery.html b/1.4.10/docs/examples/example-example95/index-jquery.html new file mode 100644 index 0000000000..bc63d4af13 --- /dev/null +++ b/1.4.10/docs/examples/example-example95/index-jquery.html @@ -0,0 +1,39 @@ + + + + + Example - example-example95-jquery + + + + + + + + + + +
+
+
+ {{text}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example95/index-production.html b/1.4.10/docs/examples/example-example95/index-production.html new file mode 100644 index 0000000000..e828aa8fb7 --- /dev/null +++ b/1.4.10/docs/examples/example-example95/index-production.html @@ -0,0 +1,38 @@ + + + + + Example - example-example95-production + + + + + + + + + +
+
+
+ {{text}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example95/index.html b/1.4.10/docs/examples/example-example95/index.html new file mode 100644 index 0000000000..0310f1a62f --- /dev/null +++ b/1.4.10/docs/examples/example-example95/index.html @@ -0,0 +1,38 @@ + + + + + Example - example-example95 + + + + + + + + + +
+
+
+ {{text}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example95/manifest.json b/1.4.10/docs/examples/example-example95/manifest.json new file mode 100644 index 0000000000..cfc342a964 --- /dev/null +++ b/1.4.10/docs/examples/example-example95/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example95", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example95/protractor.js b/1.4.10/docs/examples/example-example95/protractor.js new file mode 100644 index 0000000000..fb43905b52 --- /dev/null +++ b/1.4.10/docs/examples/example-example95/protractor.js @@ -0,0 +1,10 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example96/index-debug.html b/1.4.10/docs/examples/example-example96/index-debug.html new file mode 100644 index 0000000000..623c134999 --- /dev/null +++ b/1.4.10/docs/examples/example-example96/index-debug.html @@ -0,0 +1,21 @@ + + + + + Example - example-example96-debug + + + + + + + + + + +Load inlined template +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example96/index-jquery.html b/1.4.10/docs/examples/example-example96/index-jquery.html new file mode 100644 index 0000000000..aebb1c01f3 --- /dev/null +++ b/1.4.10/docs/examples/example-example96/index-jquery.html @@ -0,0 +1,22 @@ + + + + + Example - example-example96-jquery + + + + + + + + + + + +Load inlined template +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example96/index-production.html b/1.4.10/docs/examples/example-example96/index-production.html new file mode 100644 index 0000000000..6ec59fe6dd --- /dev/null +++ b/1.4.10/docs/examples/example-example96/index-production.html @@ -0,0 +1,21 @@ + + + + + Example - example-example96-production + + + + + + + + + + +Load inlined template +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example96/index.html b/1.4.10/docs/examples/example-example96/index.html new file mode 100644 index 0000000000..0072278b41 --- /dev/null +++ b/1.4.10/docs/examples/example-example96/index.html @@ -0,0 +1,21 @@ + + + + + Example - example-example96 + + + + + + + + + + +Load inlined template +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example96/manifest.json b/1.4.10/docs/examples/example-example96/manifest.json new file mode 100644 index 0000000000..4a2b1fb9f5 --- /dev/null +++ b/1.4.10/docs/examples/example-example96/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example96", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example96/protractor.js b/1.4.10/docs/examples/example-example96/protractor.js new file mode 100644 index 0000000000..4daaff50b2 --- /dev/null +++ b/1.4.10/docs/examples/example-example96/protractor.js @@ -0,0 +1,4 @@ +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/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example97/index-debug.html b/1.4.10/docs/examples/example-example97/index-debug.html new file mode 100644 index 0000000000..de4b8297fb --- /dev/null +++ b/1.4.10/docs/examples/example-example97/index-debug.html @@ -0,0 +1,20 @@ + + + + + Example - example-example97-debug + + + + + + + + + +
+

$document title:

+

window.document title:

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example97/index-jquery.html b/1.4.10/docs/examples/example-example97/index-jquery.html new file mode 100644 index 0000000000..51ec0d16df --- /dev/null +++ b/1.4.10/docs/examples/example-example97/index-jquery.html @@ -0,0 +1,21 @@ + + + + + Example - example-example97-jquery + + + + + + + + + + +
+

$document title:

+

window.document title:

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example97/index-production.html b/1.4.10/docs/examples/example-example97/index-production.html new file mode 100644 index 0000000000..3b9d97ecf4 --- /dev/null +++ b/1.4.10/docs/examples/example-example97/index-production.html @@ -0,0 +1,20 @@ + + + + + Example - example-example97-production + + + + + + + + + +
+

$document title:

+

window.document title:

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example97/index.html b/1.4.10/docs/examples/example-example97/index.html new file mode 100644 index 0000000000..1c51e7322e --- /dev/null +++ b/1.4.10/docs/examples/example-example97/index.html @@ -0,0 +1,20 @@ + + + + + Example - example-example97 + + + + + + + + + +
+

$document title:

+

window.document title:

+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example97/manifest.json b/1.4.10/docs/examples/example-example97/manifest.json new file mode 100644 index 0000000000..e546d0bb42 --- /dev/null +++ b/1.4.10/docs/examples/example-example97/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example97", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example97/script.js b/1.4.10/docs/examples/example-example97/script.js new file mode 100644 index 0000000000..6be16a387f --- /dev/null +++ b/1.4.10/docs/examples/example-example97/script.js @@ -0,0 +1,8 @@ +(function(angular) { + 'use strict'; +angular.module('documentExample', []) + .controller('ExampleController', ['$scope', '$document', function($scope, $document) { + $scope.title = $document[0].title; + $scope.windowTitle = angular.element(window.document)[0].title; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example98/index-debug.html b/1.4.10/docs/examples/example-example98/index-debug.html new file mode 100644 index 0000000000..7452be2f9a --- /dev/null +++ b/1.4.10/docs/examples/example-example98/index-debug.html @@ -0,0 +1,42 @@ + + + + + Example - example-example98-debug + + + + + + + + +
+ + + + + + + + +
NamePhone
{{friend.name}}{{friend.phone}}
+
+
+
+
+
+ + + + + + +
NamePhone
{{friendObj.name}}{{friendObj.phone}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example98/index-jquery.html b/1.4.10/docs/examples/example-example98/index-jquery.html new file mode 100644 index 0000000000..46f7662eba --- /dev/null +++ b/1.4.10/docs/examples/example-example98/index-jquery.html @@ -0,0 +1,43 @@ + + + + + Example - example-example98-jquery + + + + + + + + + +
+ + + + + + + + +
NamePhone
{{friend.name}}{{friend.phone}}
+
+
+
+
+
+ + + + + + +
NamePhone
{{friendObj.name}}{{friendObj.phone}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example98/index-production.html b/1.4.10/docs/examples/example-example98/index-production.html new file mode 100644 index 0000000000..82922c64a1 --- /dev/null +++ b/1.4.10/docs/examples/example-example98/index-production.html @@ -0,0 +1,42 @@ + + + + + Example - example-example98-production + + + + + + + + +
+ + + + + + + + +
NamePhone
{{friend.name}}{{friend.phone}}
+
+
+
+
+
+ + + + + + +
NamePhone
{{friendObj.name}}{{friendObj.phone}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example98/index.html b/1.4.10/docs/examples/example-example98/index.html new file mode 100644 index 0000000000..34a63c0e6d --- /dev/null +++ b/1.4.10/docs/examples/example-example98/index.html @@ -0,0 +1,42 @@ + + + + + Example - example-example98 + + + + + + + + +
+ + + + + + + + +
NamePhone
{{friend.name}}{{friend.phone}}
+
+
+
+
+
+ + + + + + +
NamePhone
{{friendObj.name}}{{friendObj.phone}}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example98/manifest.json b/1.4.10/docs/examples/example-example98/manifest.json new file mode 100644 index 0000000000..3cb76926ce --- /dev/null +++ b/1.4.10/docs/examples/example-example98/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example98", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example98/protractor.js b/1.4.10/docs/examples/example-example98/protractor.js new file mode 100644 index 0000000000..bb50ec08bd --- /dev/null +++ b/1.4.10/docs/examples/example-example98/protractor.js @@ -0,0 +1,33 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example99/index-debug.html b/1.4.10/docs/examples/example-example99/index-debug.html new file mode 100644 index 0000000000..2065caac8d --- /dev/null +++ b/1.4.10/docs/examples/example-example99/index-debug.html @@ -0,0 +1,27 @@ + + + + + Example - example-example99-debug + + + + + + + + + +
+
+ default currency symbol ($): {{amount | currency}}
+ custom currency identifier (USD$): {{amount | currency:"USD$"}} + no fractions (0): {{amount | currency:"USD$":0}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example99/index-jquery.html b/1.4.10/docs/examples/example-example99/index-jquery.html new file mode 100644 index 0000000000..75acab7029 --- /dev/null +++ b/1.4.10/docs/examples/example-example99/index-jquery.html @@ -0,0 +1,28 @@ + + + + + Example - example-example99-jquery + + + + + + + + + + +
+
+ default currency symbol ($): {{amount | currency}}
+ custom currency identifier (USD$): {{amount | currency:"USD$"}} + no fractions (0): {{amount | currency:"USD$":0}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example99/index-production.html b/1.4.10/docs/examples/example-example99/index-production.html new file mode 100644 index 0000000000..7015fe228a --- /dev/null +++ b/1.4.10/docs/examples/example-example99/index-production.html @@ -0,0 +1,27 @@ + + + + + Example - example-example99-production + + + + + + + + + +
+
+ default currency symbol ($): {{amount | currency}}
+ custom currency identifier (USD$): {{amount | currency:"USD$"}} + no fractions (0): {{amount | currency:"USD$":0}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example99/index.html b/1.4.10/docs/examples/example-example99/index.html new file mode 100644 index 0000000000..d94c27c51a --- /dev/null +++ b/1.4.10/docs/examples/example-example99/index.html @@ -0,0 +1,27 @@ + + + + + Example - example-example99 + + + + + + + + + +
+
+ default currency symbol ($): {{amount | currency}}
+ custom currency identifier (USD$): {{amount | currency:"USD$"}} + no fractions (0): {{amount | currency:"USD$":0}} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example99/manifest.json b/1.4.10/docs/examples/example-example99/manifest.json new file mode 100644 index 0000000000..48b4ee7b80 --- /dev/null +++ b/1.4.10/docs/examples/example-example99/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-example99", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-example99/protractor.js b/1.4.10/docs/examples/example-example99/protractor.js new file mode 100644 index 0000000000..a9b3ea7e0f --- /dev/null +++ b/1.4.10/docs/examples/example-example99/protractor.js @@ -0,0 +1,17 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-1/index-debug.html b/1.4.10/docs/examples/example-guide-concepts-1/index-debug.html new file mode 100644 index 0000000000..2577dc62e6 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-1/index-debug.html @@ -0,0 +1,27 @@ + + + + + Example - example-guide-concepts-1-debug + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: +
+
+ Total: {{qty * cost | currency}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-1/index-jquery.html b/1.4.10/docs/examples/example-guide-concepts-1/index-jquery.html new file mode 100644 index 0000000000..2f0086036d --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-1/index-jquery.html @@ -0,0 +1,28 @@ + + + + + Example - example-guide-concepts-1-jquery + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: +
+
+ Total: {{qty * cost | currency}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-1/index-production.html b/1.4.10/docs/examples/example-guide-concepts-1/index-production.html new file mode 100644 index 0000000000..87587091f5 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-1/index-production.html @@ -0,0 +1,27 @@ + + + + + Example - example-guide-concepts-1-production + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: +
+
+ Total: {{qty * cost | currency}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-1/index.html b/1.4.10/docs/examples/example-guide-concepts-1/index.html new file mode 100644 index 0000000000..23635f0a30 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-1/index.html @@ -0,0 +1,27 @@ + + + + + Example - example-guide-concepts-1 + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: +
+
+ Total: {{qty * cost | currency}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-1/manifest.json b/1.4.10/docs/examples/example-guide-concepts-1/manifest.json new file mode 100644 index 0000000000..f4c83905a1 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-1/manifest.json @@ -0,0 +1,6 @@ +{ + "name": "example-guide-concepts-1", + "files": [ + "index-production.html" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-2/index-debug.html b/1.4.10/docs/examples/example-guide-concepts-2/index-debug.html new file mode 100644 index 0000000000..ff16ec4831 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-2/index-debug.html @@ -0,0 +1,35 @@ + + + + + Example - example-guide-concepts-2-debug + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-2/index-jquery.html b/1.4.10/docs/examples/example-guide-concepts-2/index-jquery.html new file mode 100644 index 0000000000..d684079204 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-2/index-jquery.html @@ -0,0 +1,36 @@ + + + + + Example - example-guide-concepts-2-jquery + + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-2/index-production.html b/1.4.10/docs/examples/example-guide-concepts-2/index-production.html new file mode 100644 index 0000000000..b4990523e4 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-2/index-production.html @@ -0,0 +1,35 @@ + + + + + Example - example-guide-concepts-2-production + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-2/index.html b/1.4.10/docs/examples/example-guide-concepts-2/index.html new file mode 100644 index 0000000000..f36c0f21db --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-2/index.html @@ -0,0 +1,35 @@ + + + + + Example - example-guide-concepts-2 + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-2/invoice1.js b/1.4.10/docs/examples/example-guide-concepts-2/invoice1.js new file mode 100644 index 0000000000..dd958d1f25 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-2/invoice1.js @@ -0,0 +1,25 @@ +(function(angular) { + 'use strict'; +angular.module('invoice1', []) + .controller('InvoiceController', function() { + this.qty = 1; + this.cost = 2; + this.inCurr = 'EUR'; + this.currencies = ['USD', 'EUR', 'CNY']; + this.usdToForeignRates = { + USD: 1, + EUR: 0.74, + CNY: 6.09 + }; + + this.total = function total(outCurr) { + return this.convertCurrency(this.qty * this.cost, this.inCurr, outCurr); + }; + this.convertCurrency = function convertCurrency(amount, inCurr, outCurr) { + return amount * this.usdToForeignRates[outCurr] / this.usdToForeignRates[inCurr]; + }; + this.pay = function pay() { + window.alert("Thanks!"); + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-2/manifest.json b/1.4.10/docs/examples/example-guide-concepts-2/manifest.json new file mode 100644 index 0000000000..9c6af114cc --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-2/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-guide-concepts-2", + "files": [ + "index-production.html", + "invoice1.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-21/finance2.js b/1.4.10/docs/examples/example-guide-concepts-21/finance2.js new file mode 100644 index 0000000000..ef35399f9a --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-21/finance2.js @@ -0,0 +1,20 @@ +(function(angular) { + 'use strict'; +angular.module('finance2', []) + .factory('currencyConverter', function() { + var currencies = ['USD', 'EUR', 'CNY']; + var usdToForeignRates = { + USD: 1, + EUR: 0.74, + CNY: 6.09 + }; + var convert = function (amount, inCurr, outCurr) { + return amount * usdToForeignRates[outCurr] / usdToForeignRates[inCurr]; + }; + + return { + currencies: currencies, + convert: convert + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-21/index-debug.html b/1.4.10/docs/examples/example-guide-concepts-21/index-debug.html new file mode 100644 index 0000000000..b0c718c319 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-21/index-debug.html @@ -0,0 +1,36 @@ + + + + + Example - example-guide-concepts-21-debug + + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-21/index-jquery.html b/1.4.10/docs/examples/example-guide-concepts-21/index-jquery.html new file mode 100644 index 0000000000..4b7cf07ac6 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-21/index-jquery.html @@ -0,0 +1,37 @@ + + + + + Example - example-guide-concepts-21-jquery + + + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-21/index-production.html b/1.4.10/docs/examples/example-guide-concepts-21/index-production.html new file mode 100644 index 0000000000..0af334512f --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-21/index-production.html @@ -0,0 +1,36 @@ + + + + + Example - example-guide-concepts-21-production + + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-21/index.html b/1.4.10/docs/examples/example-guide-concepts-21/index.html new file mode 100644 index 0000000000..42538a37d6 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-21/index.html @@ -0,0 +1,36 @@ + + + + + Example - example-guide-concepts-21 + + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-21/invoice2.js b/1.4.10/docs/examples/example-guide-concepts-21/invoice2.js new file mode 100644 index 0000000000..f35c3793d3 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-21/invoice2.js @@ -0,0 +1,17 @@ +(function(angular) { + 'use strict'; +angular.module('invoice2', ['finance2']) + .controller('InvoiceController', ['currencyConverter', function(currencyConverter) { + this.qty = 1; + this.cost = 2; + this.inCurr = 'EUR'; + this.currencies = currencyConverter.currencies; + + this.total = function total(outCurr) { + return currencyConverter.convert(this.qty * this.cost, this.inCurr, outCurr); + }; + this.pay = function pay() { + window.alert("Thanks!"); + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-21/manifest.json b/1.4.10/docs/examples/example-guide-concepts-21/manifest.json new file mode 100644 index 0000000000..036dcd520a --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-21/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-guide-concepts-21", + "files": [ + "index-production.html", + "finance2.js", + "invoice2.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-3/finance3.js b/1.4.10/docs/examples/example-guide-concepts-3/finance3.js new file mode 100644 index 0000000000..102e5633e1 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-3/finance3.js @@ -0,0 +1,36 @@ +(function(angular) { + 'use strict'; +angular.module('finance3', []) + .factory('currencyConverter', ['$http', function($http) { + var YAHOO_FINANCE_URL_PATTERN = + '//query.yahooapis.com/v1/public/yql?q=select * from '+ + 'yahoo.finance.xchange where pair in ("PAIRS")&format=json&'+ + 'env=store://datatables.org/alltableswithkeys&callback=JSON_CALLBACK'; + var currencies = ['USD', 'EUR', 'CNY']; + var usdToForeignRates = {}; + + var convert = function (amount, inCurr, outCurr) { + return amount * usdToForeignRates[outCurr] / usdToForeignRates[inCurr]; + }; + + var refresh = function() { + var url = YAHOO_FINANCE_URL_PATTERN. + replace('PAIRS', 'USD' + currencies.join('","USD')); + return $http.jsonp(url).success(function(data) { + var newUsdToForeignRates = {}; + angular.forEach(data.query.results.rate, function(rate) { + var currency = rate.id.substring(3,6); + newUsdToForeignRates[currency] = window.parseFloat(rate.Rate); + }); + usdToForeignRates = newUsdToForeignRates; + }); + }; + + refresh(); + + return { + currencies: currencies, + convert: convert + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-3/index-debug.html b/1.4.10/docs/examples/example-guide-concepts-3/index-debug.html new file mode 100644 index 0000000000..40f25665fb --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-3/index-debug.html @@ -0,0 +1,36 @@ + + + + + Example - example-guide-concepts-3-debug + + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-3/index-jquery.html b/1.4.10/docs/examples/example-guide-concepts-3/index-jquery.html new file mode 100644 index 0000000000..f7d676cb86 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-3/index-jquery.html @@ -0,0 +1,37 @@ + + + + + Example - example-guide-concepts-3-jquery + + + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-3/index-production.html b/1.4.10/docs/examples/example-guide-concepts-3/index-production.html new file mode 100644 index 0000000000..f319f3b54e --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-3/index-production.html @@ -0,0 +1,36 @@ + + + + + Example - example-guide-concepts-3-production + + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-3/index.html b/1.4.10/docs/examples/example-guide-concepts-3/index.html new file mode 100644 index 0000000000..b9b834b29d --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-3/index.html @@ -0,0 +1,36 @@ + + + + + Example - example-guide-concepts-3 + + + + + + + + + + +
+ Invoice: +
+ Quantity: +
+
+ Costs: + +
+
+ Total: + + {{invoice.total(c) | currency:c}} + + +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-3/invoice3.js b/1.4.10/docs/examples/example-guide-concepts-3/invoice3.js new file mode 100644 index 0000000000..3fd044dac4 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-3/invoice3.js @@ -0,0 +1,17 @@ +(function(angular) { + 'use strict'; +angular.module('invoice3', ['finance3']) + .controller('InvoiceController', ['currencyConverter', function(currencyConverter) { + this.qty = 1; + this.cost = 2; + this.inCurr = 'EUR'; + this.currencies = currencyConverter.currencies; + + this.total = function total(outCurr) { + return currencyConverter.convert(this.qty * this.cost, this.inCurr, outCurr); + }; + this.pay = function pay() { + window.alert("Thanks!"); + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-guide-concepts-3/manifest.json b/1.4.10/docs/examples/example-guide-concepts-3/manifest.json new file mode 100644 index 0000000000..4581d36108 --- /dev/null +++ b/1.4.10/docs/examples/example-guide-concepts-3/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-guide-concepts-3", + "files": [ + "index-production.html", + "invoice3.js", + "finance3.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-input-directive/index-debug.html b/1.4.10/docs/examples/example-input-directive/index-debug.html new file mode 100644 index 0000000000..c94e361ca5 --- /dev/null +++ b/1.4.10/docs/examples/example-input-directive/index-debug.html @@ -0,0 +1,54 @@ + + + + + Example - example-input-directive-debug + + + + + + + + + +
+
+ +
+ + Required! +
+ +
+ + Too short! + + Too long! +
+
+
+ user = {{user}}
+ myForm.userName.$valid = {{myForm.userName.$valid}}
+ myForm.userName.$error = {{myForm.userName.$error}}
+ myForm.lastName.$valid = {{myForm.lastName.$valid}}
+ myForm.lastName.$error = {{myForm.lastName.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+ myForm.$error.minlength = {{!!myForm.$error.minlength}}
+ myForm.$error.maxlength = {{!!myForm.$error.maxlength}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-input-directive/index-jquery.html b/1.4.10/docs/examples/example-input-directive/index-jquery.html new file mode 100644 index 0000000000..9a837b3817 --- /dev/null +++ b/1.4.10/docs/examples/example-input-directive/index-jquery.html @@ -0,0 +1,55 @@ + + + + + Example - example-input-directive-jquery + + + + + + + + + + +
+
+ +
+ + Required! +
+ +
+ + Too short! + + Too long! +
+
+
+ user = {{user}}
+ myForm.userName.$valid = {{myForm.userName.$valid}}
+ myForm.userName.$error = {{myForm.userName.$error}}
+ myForm.lastName.$valid = {{myForm.lastName.$valid}}
+ myForm.lastName.$error = {{myForm.lastName.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+ myForm.$error.minlength = {{!!myForm.$error.minlength}}
+ myForm.$error.maxlength = {{!!myForm.$error.maxlength}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-input-directive/index-production.html b/1.4.10/docs/examples/example-input-directive/index-production.html new file mode 100644 index 0000000000..b3f869e35c --- /dev/null +++ b/1.4.10/docs/examples/example-input-directive/index-production.html @@ -0,0 +1,54 @@ + + + + + Example - example-input-directive-production + + + + + + + + + +
+
+ +
+ + Required! +
+ +
+ + Too short! + + Too long! +
+
+
+ user = {{user}}
+ myForm.userName.$valid = {{myForm.userName.$valid}}
+ myForm.userName.$error = {{myForm.userName.$error}}
+ myForm.lastName.$valid = {{myForm.lastName.$valid}}
+ myForm.lastName.$error = {{myForm.lastName.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+ myForm.$error.minlength = {{!!myForm.$error.minlength}}
+ myForm.$error.maxlength = {{!!myForm.$error.maxlength}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-input-directive/index.html b/1.4.10/docs/examples/example-input-directive/index.html new file mode 100644 index 0000000000..a04bba8029 --- /dev/null +++ b/1.4.10/docs/examples/example-input-directive/index.html @@ -0,0 +1,54 @@ + + + + + Example - example-input-directive + + + + + + + + + +
+
+ +
+ + Required! +
+ +
+ + Too short! + + Too long! +
+
+
+ user = {{user}}
+ myForm.userName.$valid = {{myForm.userName.$valid}}
+ myForm.userName.$error = {{myForm.userName.$error}}
+ myForm.lastName.$valid = {{myForm.lastName.$valid}}
+ myForm.lastName.$error = {{myForm.lastName.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+ myForm.$error.minlength = {{!!myForm.$error.minlength}}
+ myForm.$error.maxlength = {{!!myForm.$error.maxlength}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-input-directive/manifest.json b/1.4.10/docs/examples/example-input-directive/manifest.json new file mode 100644 index 0000000000..b4c6da9122 --- /dev/null +++ b/1.4.10/docs/examples/example-input-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-input-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-input-directive/protractor.js b/1.4.10/docs/examples/example-input-directive/protractor.js new file mode 100644 index 0000000000..f07600feb2 --- /dev/null +++ b/1.4.10/docs/examples/example-input-directive/protractor.js @@ -0,0 +1,51 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-hashbang-mode/addressBar.js b/1.4.10/docs/examples/example-location-hashbang-mode/addressBar.js new file mode 100644 index 0000000000..af5562ca52 --- /dev/null +++ b/1.4.10/docs/examples/example-location-hashbang-mode/addressBar.js @@ -0,0 +1,27 @@ +(function(angular) { + 'use strict'; +angular.module('address-bar', []) +.directive('ngAddressBar', function($browser, $timeout) { + return { + template: 'Address: ', + link: function(scope, element, attrs){ + var input = element.children("input"), delay; + + input.on('keypress keyup keydown', function(event) { + delay = (!delay ? $timeout(fireUrlChange, 250) : null); + event.stopPropagation(); + }) + .val($browser.url()); + + $browser.url = function(url) { + return url ? input.val(url) : input.val(); + }; + + function fireUrlChange() { + delay = null; + $browser.urlChange(input.val()); + } + } + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-hashbang-mode/app.js b/1.4.10/docs/examples/example-location-hashbang-mode/app.js new file mode 100644 index 0000000000..2fb9b17808 --- /dev/null +++ b/1.4.10/docs/examples/example-location-hashbang-mode/app.js @@ -0,0 +1,29 @@ +(function(angular) { + 'use strict'; +angular.module('hashbang-mode', ['fake-browser', 'address-bar']) + +// Configure the fakeBrowser. Do not set these values in actual projects. +.constant('initUrl', 'http://www.example.com/base/index.html#!/path?a=b#h') +.constant('baseHref', '/base/index.html') +.value('$sniffer', { history: false }) + +.config(function($locationProvider) { + $locationProvider.html5Mode(true).hashPrefix('!'); +}) + +.controller("LocationController", function($scope, $location) { + $scope.$location = {}; + angular.forEach("protocol host port path search hash".split(" "), function(method){ + $scope.$location[method] = function(){ + var result = $location[method].call($location); + return angular.isObject(result) ? angular.toJson(result) : result; + }; + }); +}) + +.run(function($rootElement) { + $rootElement.on('click', function(e) { + e.stopPropagation(); + }); +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-hashbang-mode/fakeBrowser.js b/1.4.10/docs/examples/example-location-hashbang-mode/fakeBrowser.js new file mode 100644 index 0000000000..f4170e2c40 --- /dev/null +++ b/1.4.10/docs/examples/example-location-hashbang-mode/fakeBrowser.js @@ -0,0 +1,27 @@ +(function(angular) { + 'use strict'; +angular.module('fake-browser', []) + +.config(function($provide) { + $provide.decorator('$browser', function($delegate, baseHref, initUrl) { + + $delegate.onUrlChange = function(fn) { + this.urlChange = fn; + }; + + $delegate.url = function() { + return initUrl; + }; + + $delegate.defer = function(fn, delay) { + setTimeout(function() { fn(); }, delay || 0); + }; + + $delegate.baseHref = function() { + return baseHref; + }; + + return $delegate; + }); +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-hashbang-mode/index-debug.html b/1.4.10/docs/examples/example-location-hashbang-mode/index-debug.html new file mode 100644 index 0000000000..a01f7c4a38 --- /dev/null +++ b/1.4.10/docs/examples/example-location-hashbang-mode/index-debug.html @@ -0,0 +1,34 @@ + + + + + Example - example-location-hashbang-mode-debug + + + + + + + + + + + +
+


+
+ $location.protocol() =
+ $location.host() =
+ $location.port() =
+ $location.path() =
+ $location.search() =
+ $location.hash() =
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-hashbang-mode/index-jquery.html b/1.4.10/docs/examples/example-location-hashbang-mode/index-jquery.html new file mode 100644 index 0000000000..8d7fa0e730 --- /dev/null +++ b/1.4.10/docs/examples/example-location-hashbang-mode/index-jquery.html @@ -0,0 +1,35 @@ + + + + + Example - example-location-hashbang-mode-jquery + + + + + + + + + + + + +
+


+
+ $location.protocol() =
+ $location.host() =
+ $location.port() =
+ $location.path() =
+ $location.search() =
+ $location.hash() =
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-hashbang-mode/index-production.html b/1.4.10/docs/examples/example-location-hashbang-mode/index-production.html new file mode 100644 index 0000000000..44469bccbb --- /dev/null +++ b/1.4.10/docs/examples/example-location-hashbang-mode/index-production.html @@ -0,0 +1,34 @@ + + + + + Example - example-location-hashbang-mode-production + + + + + + + + + + + +
+


+
+ $location.protocol() =
+ $location.host() =
+ $location.port() =
+ $location.path() =
+ $location.search() =
+ $location.hash() =
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-hashbang-mode/index.html b/1.4.10/docs/examples/example-location-hashbang-mode/index.html new file mode 100644 index 0000000000..64cf5d0a2f --- /dev/null +++ b/1.4.10/docs/examples/example-location-hashbang-mode/index.html @@ -0,0 +1,34 @@ + + + + + Example - example-location-hashbang-mode + + + + + + + + + + + +
+


+
+ $location.protocol() =
+ $location.host() =
+ $location.port() =
+ $location.path() =
+ $location.search() =
+ $location.hash() =
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-hashbang-mode/manifest.json b/1.4.10/docs/examples/example-location-hashbang-mode/manifest.json new file mode 100644 index 0000000000..504da50af8 --- /dev/null +++ b/1.4.10/docs/examples/example-location-hashbang-mode/manifest.json @@ -0,0 +1,10 @@ +{ + "name": "example-location-hashbang-mode", + "files": [ + "index-production.html", + "app.js", + "fakeBrowser.js", + "addressBar.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-hashbang-mode/protractor.js b/1.4.10/docs/examples/example-location-hashbang-mode/protractor.js new file mode 100644 index 0000000000..8151330371 --- /dev/null +++ b/1.4.10/docs/examples/example-location-hashbang-mode/protractor.js @@ -0,0 +1,42 @@ +var addressBar = element(by.css("#addressBar")), + url = 'http://www.example.com/base/index.html#!/path?a=b#h'; + +it("should show fake browser info on load", function(){ + expect(addressBar.getAttribute('value')).toBe(url); + + expect(element(by.binding('$location.protocol()')).getText()).toBe('http'); + expect(element(by.binding('$location.host()')).getText()).toBe('www.example.com'); + expect(element(by.binding('$location.port()')).getText()).toBe('80'); + expect(element(by.binding('$location.path()')).getText()).toBe('/path'); + expect(element(by.binding('$location.search()')).getText()).toBe('{"a":"b"}'); + expect(element(by.binding('$location.hash()')).getText()).toBe('h'); + +}); + +it("should change $location accordingly", function(){ + var navigation = element.all(by.css("#navigation a")); + + navigation.get(0).click(); + + expect(addressBar.getAttribute('value')).toBe("http://www.example.com/base/index.html#!/first?a=b"); + + expect(element(by.binding('$location.protocol()')).getText()).toBe('http'); + expect(element(by.binding('$location.host()')).getText()).toBe('www.example.com'); + expect(element(by.binding('$location.port()')).getText()).toBe('80'); + expect(element(by.binding('$location.path()')).getText()).toBe('/first'); + expect(element(by.binding('$location.search()')).getText()).toBe('{"a":"b"}'); + expect(element(by.binding('$location.hash()')).getText()).toBe(''); + + + navigation.get(1).click(); + + expect(addressBar.getAttribute('value')).toBe("http://www.example.com/base/index.html#!/sec/ond?flag#hash"); + + expect(element(by.binding('$location.protocol()')).getText()).toBe('http'); + expect(element(by.binding('$location.host()')).getText()).toBe('www.example.com'); + expect(element(by.binding('$location.port()')).getText()).toBe('80'); + expect(element(by.binding('$location.path()')).getText()).toBe('/sec/ond'); + expect(element(by.binding('$location.search()')).getText()).toBe('{"flag":true}'); + expect(element(by.binding('$location.hash()')).getText()).toBe('hash'); + +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-html5-mode/addressBar.js b/1.4.10/docs/examples/example-location-html5-mode/addressBar.js new file mode 100644 index 0000000000..af5562ca52 --- /dev/null +++ b/1.4.10/docs/examples/example-location-html5-mode/addressBar.js @@ -0,0 +1,27 @@ +(function(angular) { + 'use strict'; +angular.module('address-bar', []) +.directive('ngAddressBar', function($browser, $timeout) { + return { + template: 'Address: ', + link: function(scope, element, attrs){ + var input = element.children("input"), delay; + + input.on('keypress keyup keydown', function(event) { + delay = (!delay ? $timeout(fireUrlChange, 250) : null); + event.stopPropagation(); + }) + .val($browser.url()); + + $browser.url = function(url) { + return url ? input.val(url) : input.val(); + }; + + function fireUrlChange() { + delay = null; + $browser.urlChange(input.val()); + } + } + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-html5-mode/app.js b/1.4.10/docs/examples/example-location-html5-mode/app.js new file mode 100644 index 0000000000..bd9989a6e7 --- /dev/null +++ b/1.4.10/docs/examples/example-location-html5-mode/app.js @@ -0,0 +1,27 @@ +(function(angular) { + 'use strict'; +angular.module('html5-mode', ['fake-browser', 'address-bar']) + +// Configure the fakeBrowser. Do not set these values in actual projects. +.constant('initUrl', 'http://www.example.com/base/path?a=b#h') +.constant('baseHref', '/base/index.html') +.value('$sniffer', { history: true }) + +.controller("LocationController", function($scope, $location) { + $scope.$location = {}; + angular.forEach("protocol host port path search hash".split(" "), function(method){ + $scope.$location[method] = function(){ + var result = $location[method].call($location); + return angular.isObject(result) ? angular.toJson(result) : result; + }; + }); +}) + +.config(function($locationProvider) { + $locationProvider.html5Mode(true).hashPrefix('!'); +}) + +.run(function($rootElement) { + $rootElement.on('click', function(e) { e.stopPropagation(); }); +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-html5-mode/fakeBrowser.js b/1.4.10/docs/examples/example-location-html5-mode/fakeBrowser.js new file mode 100644 index 0000000000..f4170e2c40 --- /dev/null +++ b/1.4.10/docs/examples/example-location-html5-mode/fakeBrowser.js @@ -0,0 +1,27 @@ +(function(angular) { + 'use strict'; +angular.module('fake-browser', []) + +.config(function($provide) { + $provide.decorator('$browser', function($delegate, baseHref, initUrl) { + + $delegate.onUrlChange = function(fn) { + this.urlChange = fn; + }; + + $delegate.url = function() { + return initUrl; + }; + + $delegate.defer = function(fn, delay) { + setTimeout(function() { fn(); }, delay || 0); + }; + + $delegate.baseHref = function() { + return baseHref; + }; + + return $delegate; + }); +}); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-html5-mode/index-debug.html b/1.4.10/docs/examples/example-location-html5-mode/index-debug.html new file mode 100644 index 0000000000..1291dac6a0 --- /dev/null +++ b/1.4.10/docs/examples/example-location-html5-mode/index-debug.html @@ -0,0 +1,34 @@ + + + + + Example - example-location-html5-mode-debug + + + + + + + + + + + +
+


+
+ $location.protocol() =
+ $location.host() =
+ $location.port() =
+ $location.path() =
+ $location.search() =
+ $location.hash() =
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-html5-mode/index-jquery.html b/1.4.10/docs/examples/example-location-html5-mode/index-jquery.html new file mode 100644 index 0000000000..f08a802765 --- /dev/null +++ b/1.4.10/docs/examples/example-location-html5-mode/index-jquery.html @@ -0,0 +1,35 @@ + + + + + Example - example-location-html5-mode-jquery + + + + + + + + + + + + +
+


+
+ $location.protocol() =
+ $location.host() =
+ $location.port() =
+ $location.path() =
+ $location.search() =
+ $location.hash() =
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-html5-mode/index-production.html b/1.4.10/docs/examples/example-location-html5-mode/index-production.html new file mode 100644 index 0000000000..2a1655ee0d --- /dev/null +++ b/1.4.10/docs/examples/example-location-html5-mode/index-production.html @@ -0,0 +1,34 @@ + + + + + Example - example-location-html5-mode-production + + + + + + + + + + + +
+


+
+ $location.protocol() =
+ $location.host() =
+ $location.port() =
+ $location.path() =
+ $location.search() =
+ $location.hash() =
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-html5-mode/index.html b/1.4.10/docs/examples/example-location-html5-mode/index.html new file mode 100644 index 0000000000..8306e56d5d --- /dev/null +++ b/1.4.10/docs/examples/example-location-html5-mode/index.html @@ -0,0 +1,34 @@ + + + + + Example - example-location-html5-mode + + + + + + + + + + + +
+


+
+ $location.protocol() =
+ $location.host() =
+ $location.port() =
+ $location.path() =
+ $location.search() =
+ $location.hash() =
+
+ +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-html5-mode/manifest.json b/1.4.10/docs/examples/example-location-html5-mode/manifest.json new file mode 100644 index 0000000000..d8e7e30e19 --- /dev/null +++ b/1.4.10/docs/examples/example-location-html5-mode/manifest.json @@ -0,0 +1,10 @@ +{ + "name": "example-location-html5-mode", + "files": [ + "index-production.html", + "app.js", + "fakeBrowser.js", + "addressBar.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-location-html5-mode/protractor.js b/1.4.10/docs/examples/example-location-html5-mode/protractor.js new file mode 100644 index 0000000000..e7dd43f045 --- /dev/null +++ b/1.4.10/docs/examples/example-location-html5-mode/protractor.js @@ -0,0 +1,42 @@ +var addressBar = element(by.css("#addressBar")), + url = 'http://www.example.com/base/path?a=b#h'; + + +it("should show fake browser info on load", function(){ + expect(addressBar.getAttribute('value')).toBe(url); + + expect(element(by.binding('$location.protocol()')).getText()).toBe('http'); + expect(element(by.binding('$location.host()')).getText()).toBe('www.example.com'); + expect(element(by.binding('$location.port()')).getText()).toBe('80'); + expect(element(by.binding('$location.path()')).getText()).toBe('/path'); + expect(element(by.binding('$location.search()')).getText()).toBe('{"a":"b"}'); + expect(element(by.binding('$location.hash()')).getText()).toBe('h'); + +}); + +it("should change $location accordingly", function(){ + var navigation = element.all(by.css("#navigation a")); + + navigation.get(0).click(); + + expect(addressBar.getAttribute('value')).toBe("http://www.example.com/base/first?a=b"); + + expect(element(by.binding('$location.protocol()')).getText()).toBe('http'); + expect(element(by.binding('$location.host()')).getText()).toBe('www.example.com'); + expect(element(by.binding('$location.port()')).getText()).toBe('80'); + expect(element(by.binding('$location.path()')).getText()).toBe('/first'); + expect(element(by.binding('$location.search()')).getText()).toBe('{"a":"b"}'); + expect(element(by.binding('$location.hash()')).getText()).toBe(''); + + + navigation.get(1).click(); + + expect(addressBar.getAttribute('value')).toBe("http://www.example.com/base/sec/ond?flag#hash"); + + expect(element(by.binding('$location.protocol()')).getText()).toBe('http'); + expect(element(by.binding('$location.host()')).getText()).toBe('www.example.com'); + expect(element(by.binding('$location.port()')).getText()).toBe('80'); + expect(element(by.binding('$location.path()')).getText()).toBe('/sec/ond'); + expect(element(by.binding('$location.search()')).getText()).toBe('{"flag":true}'); + expect(element(by.binding('$location.hash()')).getText()).toBe('hash'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-message-format-example/app.js b/1.4.10/docs/examples/example-message-format-example/app.js new file mode 100644 index 0000000000..ace5b007af --- /dev/null +++ b/1.4.10/docs/examples/example-message-format-example/app.js @@ -0,0 +1,26 @@ +(function(angular) { + 'use strict'; +function Person(name, gender) { + this.name = name; + this.gender = gender; +} + +angular.module('messageFormatExample', ['ngMessageFormat']) + .controller('ckCtrl', function ($scope, $injector, $parse) { + var people = [ new Person("Alice", "female"), + new Person("Bob", "male"), + new Person("Charlie", "male") ]; + + $scope.sender = new Person("Harry Potter", "male"); + $scope.recipients = people.slice(); + + $scope.setNumRecipients = function(n) { + n = n > people.length ? people.length : n; + $scope.recipients = people.slice(0, n); + }; + + $scope.setGender = function(person, gender) { + person.gender = gender; + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-message-format-example/index-debug.html b/1.4.10/docs/examples/example-message-format-example/index-debug.html new file mode 100644 index 0000000000..b758c3d248 --- /dev/null +++ b/1.4.10/docs/examples/example-message-format-example/index-debug.html @@ -0,0 +1,74 @@ + + + + + Example - example-message-format-example-debug + + + + + + + + + + +
+ Set number of recipients + + + + + + +

+ Sender's name:    + +

Recipients
+
+ Name:    + Gender: + + +
+ +

Message
+ {{recipients.length, plural, offset:1 + =0 {You ({{sender.name}}) gave no gifts} + =1 { {{ recipients[0].gender, select, + male {You ({{sender.name}}) gave him ({{recipients[0].name}}) a gift.} + female {You ({{sender.name}}) gave her ({{recipients[0].name}}) a gift.} + other {You ({{sender.name}}) gave them ({{recipients[0].name}}) a gift.} + }} + } + one { {{ recipients[0].gender, select, + male {You ({{sender.name}}) gave him ({{recipients[0].name}}) and one other person a gift.} + female {You ({{sender.name}}) gave her ({{recipients[0].name}}) and one other person a gift.} + other {You ({{sender.name}}) gave them ({{recipients[0].name}}) and one other person a gift.} + }} + } + other {You ({{sender.name}}) gave {{recipients.length}} people gifts. } + }} + +

In an attribute
+
+ This div has an attribute interpolated with messageformat. Use the DOM inspector to check it out. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-message-format-example/index-jquery.html b/1.4.10/docs/examples/example-message-format-example/index-jquery.html new file mode 100644 index 0000000000..14c1695381 --- /dev/null +++ b/1.4.10/docs/examples/example-message-format-example/index-jquery.html @@ -0,0 +1,75 @@ + + + + + Example - example-message-format-example-jquery + + + + + + + + + + + +
+ Set number of recipients + + + + + + +

+ Sender's name:    + +

Recipients
+
+ Name:    + Gender: + + +
+ +

Message
+ {{recipients.length, plural, offset:1 + =0 {You ({{sender.name}}) gave no gifts} + =1 { {{ recipients[0].gender, select, + male {You ({{sender.name}}) gave him ({{recipients[0].name}}) a gift.} + female {You ({{sender.name}}) gave her ({{recipients[0].name}}) a gift.} + other {You ({{sender.name}}) gave them ({{recipients[0].name}}) a gift.} + }} + } + one { {{ recipients[0].gender, select, + male {You ({{sender.name}}) gave him ({{recipients[0].name}}) and one other person a gift.} + female {You ({{sender.name}}) gave her ({{recipients[0].name}}) and one other person a gift.} + other {You ({{sender.name}}) gave them ({{recipients[0].name}}) and one other person a gift.} + }} + } + other {You ({{sender.name}}) gave {{recipients.length}} people gifts. } + }} + +

In an attribute
+
+ This div has an attribute interpolated with messageformat. Use the DOM inspector to check it out. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-message-format-example/index-production.html b/1.4.10/docs/examples/example-message-format-example/index-production.html new file mode 100644 index 0000000000..a4d4fcca9e --- /dev/null +++ b/1.4.10/docs/examples/example-message-format-example/index-production.html @@ -0,0 +1,74 @@ + + + + + Example - example-message-format-example-production + + + + + + + + + + +
+ Set number of recipients + + + + + + +

+ Sender's name:    + +

Recipients
+
+ Name:    + Gender: + + +
+ +

Message
+ {{recipients.length, plural, offset:1 + =0 {You ({{sender.name}}) gave no gifts} + =1 { {{ recipients[0].gender, select, + male {You ({{sender.name}}) gave him ({{recipients[0].name}}) a gift.} + female {You ({{sender.name}}) gave her ({{recipients[0].name}}) a gift.} + other {You ({{sender.name}}) gave them ({{recipients[0].name}}) a gift.} + }} + } + one { {{ recipients[0].gender, select, + male {You ({{sender.name}}) gave him ({{recipients[0].name}}) and one other person a gift.} + female {You ({{sender.name}}) gave her ({{recipients[0].name}}) and one other person a gift.} + other {You ({{sender.name}}) gave them ({{recipients[0].name}}) and one other person a gift.} + }} + } + other {You ({{sender.name}}) gave {{recipients.length}} people gifts. } + }} + +

In an attribute
+
+ This div has an attribute interpolated with messageformat. Use the DOM inspector to check it out. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-message-format-example/index.html b/1.4.10/docs/examples/example-message-format-example/index.html new file mode 100644 index 0000000000..1f075336fc --- /dev/null +++ b/1.4.10/docs/examples/example-message-format-example/index.html @@ -0,0 +1,74 @@ + + + + + Example - example-message-format-example + + + + + + + + + + +
+ Set number of recipients + + + + + + +

+ Sender's name:    + +

Recipients
+
+ Name:    + Gender: + + +
+ +

Message
+ {{recipients.length, plural, offset:1 + =0 {You ({{sender.name}}) gave no gifts} + =1 { {{ recipients[0].gender, select, + male {You ({{sender.name}}) gave him ({{recipients[0].name}}) a gift.} + female {You ({{sender.name}}) gave her ({{recipients[0].name}}) a gift.} + other {You ({{sender.name}}) gave them ({{recipients[0].name}}) a gift.} + }} + } + one { {{ recipients[0].gender, select, + male {You ({{sender.name}}) gave him ({{recipients[0].name}}) and one other person a gift.} + female {You ({{sender.name}}) gave her ({{recipients[0].name}}) and one other person a gift.} + other {You ({{sender.name}}) gave them ({{recipients[0].name}}) and one other person a gift.} + }} + } + other {You ({{sender.name}}) gave {{recipients.length}} people gifts. } + }} + +

In an attribute
+
+ This div has an attribute interpolated with messageformat. Use the DOM inspector to check it out. +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-message-format-example/manifest.json b/1.4.10/docs/examples/example-message-format-example/manifest.json new file mode 100644 index 0000000000..7e00391553 --- /dev/null +++ b/1.4.10/docs/examples/example-message-format-example/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-message-format-example", + "files": [ + "index-production.html", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-month-input-directive/index-debug.html b/1.4.10/docs/examples/example-month-input-directive/index-debug.html new file mode 100644 index 0000000000..99a38c8bec --- /dev/null +++ b/1.4.10/docs/examples/example-month-input-directive/index-debug.html @@ -0,0 +1,39 @@ + + + + + Example - example-month-input-directive-debug + + + + + + + + + +
+ + +
+ + Required! + + Not a valid month! +
+ value = {{example.value | date: "yyyy-MM"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-month-input-directive/index-jquery.html b/1.4.10/docs/examples/example-month-input-directive/index-jquery.html new file mode 100644 index 0000000000..1353e846ee --- /dev/null +++ b/1.4.10/docs/examples/example-month-input-directive/index-jquery.html @@ -0,0 +1,40 @@ + + + + + Example - example-month-input-directive-jquery + + + + + + + + + + +
+ + +
+ + Required! + + Not a valid month! +
+ value = {{example.value | date: "yyyy-MM"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-month-input-directive/index-production.html b/1.4.10/docs/examples/example-month-input-directive/index-production.html new file mode 100644 index 0000000000..5b7af1b849 --- /dev/null +++ b/1.4.10/docs/examples/example-month-input-directive/index-production.html @@ -0,0 +1,39 @@ + + + + + Example - example-month-input-directive-production + + + + + + + + + +
+ + +
+ + Required! + + Not a valid month! +
+ value = {{example.value | date: "yyyy-MM"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-month-input-directive/index.html b/1.4.10/docs/examples/example-month-input-directive/index.html new file mode 100644 index 0000000000..f918ab75d0 --- /dev/null +++ b/1.4.10/docs/examples/example-month-input-directive/index.html @@ -0,0 +1,39 @@ + + + + + Example - example-month-input-directive + + + + + + + + + +
+ + +
+ + Required! + + Not a valid month! +
+ value = {{example.value | date: "yyyy-MM"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-month-input-directive/manifest.json b/1.4.10/docs/examples/example-month-input-directive/manifest.json new file mode 100644 index 0000000000..ba45a6069e --- /dev/null +++ b/1.4.10/docs/examples/example-month-input-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-month-input-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-month-input-directive/protractor.js b/1.4.10/docs/examples/example-month-input-directive/protractor.js new file mode 100644 index 0000000000..b5b20b9d3a --- /dev/null +++ b/1.4.10/docs/examples/example-month-input-directive/protractor.js @@ -0,0 +1,31 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ng-model-cancel-update/app.js b/1.4.10/docs/examples/example-ng-model-cancel-update/app.js new file mode 100644 index 0000000000..1f4eca6310 --- /dev/null +++ b/1.4.10/docs/examples/example-ng-model-cancel-update/app.js @@ -0,0 +1,18 @@ +(function(angular) { + 'use strict'; +angular.module('cancel-update-example', []) + +.controller('CancelUpdateController', ['$scope', function($scope) { + $scope.model = {}; + + $scope.setEmpty = function(e, value, rollback) { + if (e.keyCode == 27) { + e.preventDefault(); + if (rollback) { + $scope.myForm[value].$rollbackViewValue(); + } + $scope.model[value] = ''; + } + }; +}]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ng-model-cancel-update/index-debug.html b/1.4.10/docs/examples/example-ng-model-cancel-update/index-debug.html new file mode 100644 index 0000000000..afcc8cc9c2 --- /dev/null +++ b/1.4.10/docs/examples/example-ng-model-cancel-update/index-debug.html @@ -0,0 +1,50 @@ + + + + + Example - example-ng-model-cancel-update-debug + + + + + + + + + + +
+

Both of these inputs are only updated if they are blurred. Hitting escape should + empty them. Follow these steps and observe the difference:

+
    +
  1. Type something in the input. You will see that the model is not yet updated
  2. +
  3. Press the Escape key. +
      +
    1. In the first example, nothing happens, because the model is already '', and no + update is detected. If you blur the input, the model will be set to the current view. +
    2. +
    3. In the second example, the pending update is cancelled, and the input is set back + to the last committed view value (''). Blurring the input does nothing. +
    4. +
    +
  4. +
+ +
+
+

Without $rollbackViewValue():

+ + value1: "{{ model.value1 }}" +
+ +
+

With $rollbackViewValue():

+ + value2: "{{ model.value2 }}" +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ng-model-cancel-update/index-jquery.html b/1.4.10/docs/examples/example-ng-model-cancel-update/index-jquery.html new file mode 100644 index 0000000000..6467f57e72 --- /dev/null +++ b/1.4.10/docs/examples/example-ng-model-cancel-update/index-jquery.html @@ -0,0 +1,51 @@ + + + + + Example - example-ng-model-cancel-update-jquery + + + + + + + + + + + +
+

Both of these inputs are only updated if they are blurred. Hitting escape should + empty them. Follow these steps and observe the difference:

+
    +
  1. Type something in the input. You will see that the model is not yet updated
  2. +
  3. Press the Escape key. +
      +
    1. In the first example, nothing happens, because the model is already '', and no + update is detected. If you blur the input, the model will be set to the current view. +
    2. +
    3. In the second example, the pending update is cancelled, and the input is set back + to the last committed view value (''). Blurring the input does nothing. +
    4. +
    +
  4. +
+ +
+
+

Without $rollbackViewValue():

+ + value1: "{{ model.value1 }}" +
+ +
+

With $rollbackViewValue():

+ + value2: "{{ model.value2 }}" +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ng-model-cancel-update/index-production.html b/1.4.10/docs/examples/example-ng-model-cancel-update/index-production.html new file mode 100644 index 0000000000..448f542cf3 --- /dev/null +++ b/1.4.10/docs/examples/example-ng-model-cancel-update/index-production.html @@ -0,0 +1,50 @@ + + + + + Example - example-ng-model-cancel-update-production + + + + + + + + + + +
+

Both of these inputs are only updated if they are blurred. Hitting escape should + empty them. Follow these steps and observe the difference:

+
    +
  1. Type something in the input. You will see that the model is not yet updated
  2. +
  3. Press the Escape key. +
      +
    1. In the first example, nothing happens, because the model is already '', and no + update is detected. If you blur the input, the model will be set to the current view. +
    2. +
    3. In the second example, the pending update is cancelled, and the input is set back + to the last committed view value (''). Blurring the input does nothing. +
    4. +
    +
  4. +
+ +
+
+

Without $rollbackViewValue():

+ + value1: "{{ model.value1 }}" +
+ +
+

With $rollbackViewValue():

+ + value2: "{{ model.value2 }}" +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ng-model-cancel-update/index.html b/1.4.10/docs/examples/example-ng-model-cancel-update/index.html new file mode 100644 index 0000000000..db284ca5f9 --- /dev/null +++ b/1.4.10/docs/examples/example-ng-model-cancel-update/index.html @@ -0,0 +1,50 @@ + + + + + Example - example-ng-model-cancel-update + + + + + + + + + + +
+

Both of these inputs are only updated if they are blurred. Hitting escape should + empty them. Follow these steps and observe the difference:

+
    +
  1. Type something in the input. You will see that the model is not yet updated
  2. +
  3. Press the Escape key. +
      +
    1. In the first example, nothing happens, because the model is already '', and no + update is detected. If you blur the input, the model will be set to the current view. +
    2. +
    3. In the second example, the pending update is cancelled, and the input is set back + to the last committed view value (''). Blurring the input does nothing. +
    4. +
    +
  4. +
+ +
+
+

Without $rollbackViewValue():

+ + value1: "{{ model.value1 }}" +
+ +
+

With $rollbackViewValue():

+ + value2: "{{ model.value2 }}" +
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ng-model-cancel-update/manifest.json b/1.4.10/docs/examples/example-ng-model-cancel-update/manifest.json new file mode 100644 index 0000000000..583e381e69 --- /dev/null +++ b/1.4.10/docs/examples/example-ng-model-cancel-update/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-ng-model-cancel-update", + "files": [ + "index-production.html", + "app.js", + "style.css" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ng-model-cancel-update/style.css b/1.4.10/docs/examples/example-ng-model-cancel-update/style.css new file mode 100644 index 0000000000..691bc99445 --- /dev/null +++ b/1.4.10/docs/examples/example-ng-model-cancel-update/style.css @@ -0,0 +1,6 @@ +div { + display: table-cell; +} +div:nth-child(1) { + padding-right: 30px; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngAnimateChildren/animations.css b/1.4.10/docs/examples/example-ngAnimateChildren/animations.css new file mode 100644 index 0000000000..ac09d863bc --- /dev/null +++ b/1.4.10/docs/examples/example-ngAnimateChildren/animations.css @@ -0,0 +1,33 @@ +.container.ng-enter, +.container.ng-leave { + transition: all ease 1.5s; +} + +.container.ng-enter, +.container.ng-leave-active { + opacity: 0; +} + +.container.ng-leave, +.container.ng-enter-active { + opacity: 1; +} + +.item { + background: firebrick; + color: #FFF; + margin-bottom: 10px; +} + +.item.ng-enter, +.item.ng-leave { + transition: transform 1.5s ease; +} + +.item.ng-enter { + transform: translateX(50px); +} + +.item.ng-enter-active { + transform: translateX(0); +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngAnimateChildren/index-debug.html b/1.4.10/docs/examples/example-ngAnimateChildren/index-debug.html new file mode 100644 index 0000000000..7f6ed98659 --- /dev/null +++ b/1.4.10/docs/examples/example-ngAnimateChildren/index-debug.html @@ -0,0 +1,29 @@ + + + + + Example - example-ngAnimateChildren-debug + + + + + + + + + + + +
+ + +
+
+
+ List of items: +
Item {{item}}
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngAnimateChildren/index-jquery.html b/1.4.10/docs/examples/example-ngAnimateChildren/index-jquery.html new file mode 100644 index 0000000000..bef901f2a2 --- /dev/null +++ b/1.4.10/docs/examples/example-ngAnimateChildren/index-jquery.html @@ -0,0 +1,30 @@ + + + + + Example - example-ngAnimateChildren-jquery + + + + + + + + + + + + +
+ + +
+
+
+ List of items: +
Item {{item}}
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngAnimateChildren/index-production.html b/1.4.10/docs/examples/example-ngAnimateChildren/index-production.html new file mode 100644 index 0000000000..582e146e84 --- /dev/null +++ b/1.4.10/docs/examples/example-ngAnimateChildren/index-production.html @@ -0,0 +1,29 @@ + + + + + Example - example-ngAnimateChildren-production + + + + + + + + + + + +
+ + +
+
+
+ List of items: +
Item {{item}}
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngAnimateChildren/index.html b/1.4.10/docs/examples/example-ngAnimateChildren/index.html new file mode 100644 index 0000000000..a949d0ac77 --- /dev/null +++ b/1.4.10/docs/examples/example-ngAnimateChildren/index.html @@ -0,0 +1,29 @@ + + + + + Example - example-ngAnimateChildren + + + + + + + + + + + +
+ + +
+
+
+ List of items: +
Item {{item}}
+
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngAnimateChildren/manifest.json b/1.4.10/docs/examples/example-ngAnimateChildren/manifest.json new file mode 100644 index 0000000000..80161246c7 --- /dev/null +++ b/1.4.10/docs/examples/example-ngAnimateChildren/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-ngAnimateChildren", + "files": [ + "index-production.html", + "animations.css", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngAnimateChildren/script.js b/1.4.10/docs/examples/example-ngAnimateChildren/script.js new file mode 100644 index 0000000000..d8c2391009 --- /dev/null +++ b/1.4.10/docs/examples/example-ngAnimateChildren/script.js @@ -0,0 +1,8 @@ +(function(angular) { + 'use strict'; +angular.module('ngAnimateChildren', ['ngAnimate']) + .controller('mainController', function() { + this.animateChildren = false; + this.enterElement = false; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngChange-directive/index-debug.html b/1.4.10/docs/examples/example-ngChange-directive/index-debug.html new file mode 100644 index 0000000000..1fc8dc7cf5 --- /dev/null +++ b/1.4.10/docs/examples/example-ngChange-directive/index-debug.html @@ -0,0 +1,31 @@ + + + + + Example - example-ngChange-directive-debug + + + + + + + + + +
+ + +
+ debug = {{confirmed}}
+ counter = {{counter}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngChange-directive/index-jquery.html b/1.4.10/docs/examples/example-ngChange-directive/index-jquery.html new file mode 100644 index 0000000000..1bdd6f0dce --- /dev/null +++ b/1.4.10/docs/examples/example-ngChange-directive/index-jquery.html @@ -0,0 +1,32 @@ + + + + + Example - example-ngChange-directive-jquery + + + + + + + + + + +
+ + +
+ debug = {{confirmed}}
+ counter = {{counter}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngChange-directive/index-production.html b/1.4.10/docs/examples/example-ngChange-directive/index-production.html new file mode 100644 index 0000000000..2c1aa06d37 --- /dev/null +++ b/1.4.10/docs/examples/example-ngChange-directive/index-production.html @@ -0,0 +1,31 @@ + + + + + Example - example-ngChange-directive-production + + + + + + + + + +
+ + +
+ debug = {{confirmed}}
+ counter = {{counter}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngChange-directive/index.html b/1.4.10/docs/examples/example-ngChange-directive/index.html new file mode 100644 index 0000000000..8f050d68ea --- /dev/null +++ b/1.4.10/docs/examples/example-ngChange-directive/index.html @@ -0,0 +1,31 @@ + + + + + Example - example-ngChange-directive + + + + + + + + + +
+ + +
+ debug = {{confirmed}}
+ counter = {{counter}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngChange-directive/manifest.json b/1.4.10/docs/examples/example-ngChange-directive/manifest.json new file mode 100644 index 0000000000..738d489cdc --- /dev/null +++ b/1.4.10/docs/examples/example-ngChange-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngChange-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngChange-directive/protractor.js b/1.4.10/docs/examples/example-ngChange-directive/protractor.js new file mode 100644 index 0000000000..84691d3a6d --- /dev/null +++ b/1.4.10/docs/examples/example-ngChange-directive/protractor.js @@ -0,0 +1,18 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngController/app.js b/1.4.10/docs/examples/example-ngController/app.js new file mode 100644 index 0000000000..00ad138371 --- /dev/null +++ b/1.4.10/docs/examples/example-ngController/app.js @@ -0,0 +1,30 @@ +(function(angular) { + 'use strict'; +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 = ''; + }; +} +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngController/index-debug.html b/1.4.10/docs/examples/example-ngController/index-debug.html new file mode 100644 index 0000000000..bb57bbfd94 --- /dev/null +++ b/1.4.10/docs/examples/example-ngController/index-debug.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngController-debug + + + + + + + + + +
+ +
+ Contact: +
    +
  • + + + + +
  • +
  • [ ]
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngController/index-jquery.html b/1.4.10/docs/examples/example-ngController/index-jquery.html new file mode 100644 index 0000000000..cb9c89dcc9 --- /dev/null +++ b/1.4.10/docs/examples/example-ngController/index-jquery.html @@ -0,0 +1,34 @@ + + + + + Example - example-ngController-jquery + + + + + + + + + + +
+ +
+ Contact: +
    +
  • + + + + +
  • +
  • [ ]
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngController/index-production.html b/1.4.10/docs/examples/example-ngController/index-production.html new file mode 100644 index 0000000000..01f0b09fdb --- /dev/null +++ b/1.4.10/docs/examples/example-ngController/index-production.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngController-production + + + + + + + + + +
+ +
+ Contact: +
    +
  • + + + + +
  • +
  • [ ]
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngController/index.html b/1.4.10/docs/examples/example-ngController/index.html new file mode 100644 index 0000000000..6aa4ab4ad7 --- /dev/null +++ b/1.4.10/docs/examples/example-ngController/index.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngController + + + + + + + + + +
+ +
+ Contact: +
    +
  • + + + + +
  • +
  • [ ]
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngController/manifest.json b/1.4.10/docs/examples/example-ngController/manifest.json new file mode 100644 index 0000000000..ae92ea9e23 --- /dev/null +++ b/1.4.10/docs/examples/example-ngController/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-ngController", + "files": [ + "index-production.html", + "app.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngController/protractor.js b/1.4.10/docs/examples/example-ngController/protractor.js new file mode 100644 index 0000000000..0e3dafbd61 --- /dev/null +++ b/1.4.10/docs/examples/example-ngController/protractor.js @@ -0,0 +1,28 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngControllerAs/app.js b/1.4.10/docs/examples/example-ngControllerAs/app.js new file mode 100644 index 0000000000..04aba49153 --- /dev/null +++ b/1.4.10/docs/examples/example-ngControllerAs/app.js @@ -0,0 +1,30 @@ +(function(angular) { + 'use strict'; +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 = ''; +}; +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngControllerAs/index-debug.html b/1.4.10/docs/examples/example-ngControllerAs/index-debug.html new file mode 100644 index 0000000000..9ffb9aa1ed --- /dev/null +++ b/1.4.10/docs/examples/example-ngControllerAs/index-debug.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngControllerAs-debug + + + + + + + + + +
+ +
+ Contact: +
    +
  • + + + + +
  • +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngControllerAs/index-jquery.html b/1.4.10/docs/examples/example-ngControllerAs/index-jquery.html new file mode 100644 index 0000000000..33b5f4564e --- /dev/null +++ b/1.4.10/docs/examples/example-ngControllerAs/index-jquery.html @@ -0,0 +1,34 @@ + + + + + Example - example-ngControllerAs-jquery + + + + + + + + + + +
+ +
+ Contact: +
    +
  • + + + + +
  • +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngControllerAs/index-production.html b/1.4.10/docs/examples/example-ngControllerAs/index-production.html new file mode 100644 index 0000000000..a183bb9a5f --- /dev/null +++ b/1.4.10/docs/examples/example-ngControllerAs/index-production.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngControllerAs-production + + + + + + + + + +
+ +
+ Contact: +
    +
  • + + + + +
  • +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngControllerAs/index.html b/1.4.10/docs/examples/example-ngControllerAs/index.html new file mode 100644 index 0000000000..93b2d52373 --- /dev/null +++ b/1.4.10/docs/examples/example-ngControllerAs/index.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngControllerAs + + + + + + + + + +
+ +
+ Contact: +
    +
  • + + + + +
  • +
  • +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngControllerAs/manifest.json b/1.4.10/docs/examples/example-ngControllerAs/manifest.json new file mode 100644 index 0000000000..b54b3acd59 --- /dev/null +++ b/1.4.10/docs/examples/example-ngControllerAs/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-ngControllerAs", + "files": [ + "index-production.html", + "app.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngControllerAs/protractor.js b/1.4.10/docs/examples/example-ngControllerAs/protractor.js new file mode 100644 index 0000000000..995ed4c830 --- /dev/null +++ b/1.4.10/docs/examples/example-ngControllerAs/protractor.js @@ -0,0 +1,28 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive-newlines/index-debug.html b/1.4.10/docs/examples/example-ngList-directive-newlines/index-debug.html new file mode 100644 index 0000000000..4cfaa118d8 --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive-newlines/index-debug.html @@ -0,0 +1,17 @@ + + + + + Example - example-ngList-directive-newlines-debug + + + + + + + + + +
{{ list | json }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive-newlines/index-jquery.html b/1.4.10/docs/examples/example-ngList-directive-newlines/index-jquery.html new file mode 100644 index 0000000000..fa28e08c1c --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive-newlines/index-jquery.html @@ -0,0 +1,18 @@ + + + + + Example - example-ngList-directive-newlines-jquery + + + + + + + + + + +
{{ list | json }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive-newlines/index-production.html b/1.4.10/docs/examples/example-ngList-directive-newlines/index-production.html new file mode 100644 index 0000000000..574592c168 --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive-newlines/index-production.html @@ -0,0 +1,17 @@ + + + + + Example - example-ngList-directive-newlines-production + + + + + + + + + +
{{ list | json }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive-newlines/index.html b/1.4.10/docs/examples/example-ngList-directive-newlines/index.html new file mode 100644 index 0000000000..fbd7af9fd0 --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive-newlines/index.html @@ -0,0 +1,17 @@ + + + + + Example - example-ngList-directive-newlines + + + + + + + + + +
{{ list | json }}
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive-newlines/manifest.json b/1.4.10/docs/examples/example-ngList-directive-newlines/manifest.json new file mode 100644 index 0000000000..887abf2e03 --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive-newlines/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngList-directive-newlines", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive-newlines/protractor.js b/1.4.10/docs/examples/example-ngList-directive-newlines/protractor.js new file mode 100644 index 0000000000..341788f243 --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive-newlines/protractor.js @@ -0,0 +1,6 @@ +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]'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive/app.js b/1.4.10/docs/examples/example-ngList-directive/app.js new file mode 100644 index 0000000000..12a486c591 --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive/app.js @@ -0,0 +1,7 @@ +(function(angular) { + 'use strict'; +angular.module('listExample', []) + .controller('ExampleController', ['$scope', function($scope) { + $scope.names = ['morpheus', 'neo', 'trinity']; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive/index-debug.html b/1.4.10/docs/examples/example-ngList-directive/index-debug.html new file mode 100644 index 0000000000..86cc594790 --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive/index-debug.html @@ -0,0 +1,29 @@ + + + + + Example - example-ngList-directive-debug + + + + + + + + + +
+ + + + Required! + +
+ names = {{names}}
+ myForm.namesInput.$valid = {{myForm.namesInput.$valid}}
+ myForm.namesInput.$error = {{myForm.namesInput.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive/index-jquery.html b/1.4.10/docs/examples/example-ngList-directive/index-jquery.html new file mode 100644 index 0000000000..a42d7b4623 --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive/index-jquery.html @@ -0,0 +1,30 @@ + + + + + Example - example-ngList-directive-jquery + + + + + + + + + + +
+ + + + Required! + +
+ names = {{names}}
+ myForm.namesInput.$valid = {{myForm.namesInput.$valid}}
+ myForm.namesInput.$error = {{myForm.namesInput.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive/index-production.html b/1.4.10/docs/examples/example-ngList-directive/index-production.html new file mode 100644 index 0000000000..70983b6979 --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive/index-production.html @@ -0,0 +1,29 @@ + + + + + Example - example-ngList-directive-production + + + + + + + + + +
+ + + + Required! + +
+ names = {{names}}
+ myForm.namesInput.$valid = {{myForm.namesInput.$valid}}
+ myForm.namesInput.$error = {{myForm.namesInput.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive/index.html b/1.4.10/docs/examples/example-ngList-directive/index.html new file mode 100644 index 0000000000..0a2b41e97e --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive/index.html @@ -0,0 +1,29 @@ + + + + + Example - example-ngList-directive + + + + + + + + + +
+ + + + Required! + +
+ names = {{names}}
+ myForm.namesInput.$valid = {{myForm.namesInput.$valid}}
+ myForm.namesInput.$error = {{myForm.namesInput.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive/manifest.json b/1.4.10/docs/examples/example-ngList-directive/manifest.json new file mode 100644 index 0000000000..46ff1d990b --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-ngList-directive", + "files": [ + "index-production.html", + "app.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngList-directive/protractor.js b/1.4.10/docs/examples/example-ngList-directive/protractor.js new file mode 100644 index 0000000000..f210531efc --- /dev/null +++ b/1.4.10/docs/examples/example-ngList-directive/protractor.js @@ -0,0 +1,19 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMaxlengthDirective/index-debug.html b/1.4.10/docs/examples/example-ngMaxlengthDirective/index-debug.html new file mode 100644 index 0000000000..2606865b32 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMaxlengthDirective/index-debug.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngMaxlengthDirective-debug + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMaxlengthDirective/index-jquery.html b/1.4.10/docs/examples/example-ngMaxlengthDirective/index-jquery.html new file mode 100644 index 0000000000..84be95dc60 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMaxlengthDirective/index-jquery.html @@ -0,0 +1,34 @@ + + + + + Example - example-ngMaxlengthDirective-jquery + + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMaxlengthDirective/index-production.html b/1.4.10/docs/examples/example-ngMaxlengthDirective/index-production.html new file mode 100644 index 0000000000..e8bffc067a --- /dev/null +++ b/1.4.10/docs/examples/example-ngMaxlengthDirective/index-production.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngMaxlengthDirective-production + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMaxlengthDirective/index.html b/1.4.10/docs/examples/example-ngMaxlengthDirective/index.html new file mode 100644 index 0000000000..93711539b7 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMaxlengthDirective/index.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngMaxlengthDirective + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMaxlengthDirective/manifest.json b/1.4.10/docs/examples/example-ngMaxlengthDirective/manifest.json new file mode 100644 index 0000000000..a2a16be597 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMaxlengthDirective/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngMaxlengthDirective", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMaxlengthDirective/protractor.js b/1.4.10/docs/examples/example-ngMaxlengthDirective/protractor.js new file mode 100644 index 0000000000..62a1e696b5 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMaxlengthDirective/protractor.js @@ -0,0 +1,12 @@ +var model = element(by.binding('model')); +var input = element(by.id('input')); + +it('should validate the input with the default maxlength', function() { + input.sendKeys('abcdef'); + expect(model.getText()).not.toContain('abcdef'); + + input.clear().then(function() { + input.sendKeys('abcde'); + expect(model.getText()).toContain('abcde'); + }); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessageFormat-example/index-debug.html b/1.4.10/docs/examples/example-ngMessageFormat-example/index-debug.html new file mode 100644 index 0000000000..6e1321f58c --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessageFormat-example/index-debug.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngMessageFormat-example-debug + + + + + + + + + + +
+
+ {{recipients.length, plural, offset:1 + =0 {{{sender.name}} gave no gifts (\#=#)} + =1 {{{sender.name}} gave one gift to {{recipients[0].name}} (\#=#)} + one {{{sender.name}} gave {{recipients[0].name}} and one other person a gift (\#=#)} + other {{{sender.name}} gave {{recipients[0].name}} and # other people a gift (\#=#)} + }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessageFormat-example/index-jquery.html b/1.4.10/docs/examples/example-ngMessageFormat-example/index-jquery.html new file mode 100644 index 0000000000..6d0bac69f8 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessageFormat-example/index-jquery.html @@ -0,0 +1,27 @@ + + + + + Example - example-ngMessageFormat-example-jquery + + + + + + + + + + + +
+
+ {{recipients.length, plural, offset:1 + =0 {{{sender.name}} gave no gifts (\#=#)} + =1 {{{sender.name}} gave one gift to {{recipients[0].name}} (\#=#)} + one {{{sender.name}} gave {{recipients[0].name}} and one other person a gift (\#=#)} + other {{{sender.name}} gave {{recipients[0].name}} and # other people a gift (\#=#)} + }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessageFormat-example/index-production.html b/1.4.10/docs/examples/example-ngMessageFormat-example/index-production.html new file mode 100644 index 0000000000..4f63df35c2 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessageFormat-example/index-production.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngMessageFormat-example-production + + + + + + + + + + +
+
+ {{recipients.length, plural, offset:1 + =0 {{{sender.name}} gave no gifts (\#=#)} + =1 {{{sender.name}} gave one gift to {{recipients[0].name}} (\#=#)} + one {{{sender.name}} gave {{recipients[0].name}} and one other person a gift (\#=#)} + other {{{sender.name}} gave {{recipients[0].name}} and # other people a gift (\#=#)} + }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessageFormat-example/index.html b/1.4.10/docs/examples/example-ngMessageFormat-example/index.html new file mode 100644 index 0000000000..e4be331efd --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessageFormat-example/index.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngMessageFormat-example + + + + + + + + + + +
+
+ {{recipients.length, plural, offset:1 + =0 {{{sender.name}} gave no gifts (\#=#)} + =1 {{{sender.name}} gave one gift to {{recipients[0].name}} (\#=#)} + one {{{sender.name}} gave {{recipients[0].name}} and one other person a gift (\#=#)} + other {{{sender.name}} gave {{recipients[0].name}} and # other people a gift (\#=#)} + }} +
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessageFormat-example/manifest.json b/1.4.10/docs/examples/example-ngMessageFormat-example/manifest.json new file mode 100644 index 0000000000..7ba6f5c788 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessageFormat-example/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-ngMessageFormat-example", + "files": [ + "index-production.html", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessageFormat-example/protractor.js b/1.4.10/docs/examples/example-ngMessageFormat-example/protractor.js new file mode 100644 index 0000000000..3d3cc93f22 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessageFormat-example/protractor.js @@ -0,0 +1,12 @@ +describe('MessageFormat plural', function() { + it('should pluralize initial values', function() { + var messageElem = element(by.binding('recipients.length')), decreaseRecipientsBtn = element(by.id('decreaseRecipients')); + expect(messageElem.getText()).toEqual('Harry Potter gave Alice and 2 other people a gift (#=2)'); + decreaseRecipientsBtn.click(); + expect(messageElem.getText()).toEqual('Harry Potter gave Alice and one other person a gift (#=1)'); + decreaseRecipientsBtn.click(); + expect(messageElem.getText()).toEqual('Harry Potter gave one gift to Alice (#=0)'); + decreaseRecipientsBtn.click(); + expect(messageElem.getText()).toEqual('Harry Potter gave no gifts (#=-1)'); + }); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessageFormat-example/script.js b/1.4.10/docs/examples/example-ngMessageFormat-example/script.js new file mode 100644 index 0000000000..0c37b9d572 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessageFormat-example/script.js @@ -0,0 +1,21 @@ +(function(angular) { + 'use strict'; +function Person(name, gender) { + this.name = name; + this.gender = gender; +} + +var alice = new Person("Alice", "female"), + bob = new Person("Bob", "male"), + charlie = new Person("Charlie", "male"), + harry = new Person("Harry Potter", "male"); + +angular.module('msgFmtExample', ['ngMessageFormat']) + .controller('AppController', ['$scope', function($scope) { + $scope.recipients = [alice, bob, charlie]; + $scope.sender = harry; + $scope.decreaseRecipients = function() { + --$scope.recipients.length; + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessages-directive/index-debug.html b/1.4.10/docs/examples/example-ngMessages-directive/index-debug.html new file mode 100644 index 0000000000..dbeef7cd4a --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessages-directive/index-debug.html @@ -0,0 +1,37 @@ + + + + + Example - example-ngMessages-directive-debug + + + + + + + + + + +
+ +
myForm.myName.$error = {{ myForm.myName.$error | json }}
+ +
+
You did not enter a field
+
Your field is too short
+
Your field is too long
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessages-directive/index-jquery.html b/1.4.10/docs/examples/example-ngMessages-directive/index-jquery.html new file mode 100644 index 0000000000..bee999025c --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessages-directive/index-jquery.html @@ -0,0 +1,38 @@ + + + + + Example - example-ngMessages-directive-jquery + + + + + + + + + + + +
+ +
myForm.myName.$error = {{ myForm.myName.$error | json }}
+ +
+
You did not enter a field
+
Your field is too short
+
Your field is too long
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessages-directive/index-production.html b/1.4.10/docs/examples/example-ngMessages-directive/index-production.html new file mode 100644 index 0000000000..ad7f8ba0f9 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessages-directive/index-production.html @@ -0,0 +1,37 @@ + + + + + Example - example-ngMessages-directive-production + + + + + + + + + + +
+ +
myForm.myName.$error = {{ myForm.myName.$error | json }}
+ +
+
You did not enter a field
+
Your field is too short
+
Your field is too long
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessages-directive/index.html b/1.4.10/docs/examples/example-ngMessages-directive/index.html new file mode 100644 index 0000000000..e8b0ae2aaa --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessages-directive/index.html @@ -0,0 +1,37 @@ + + + + + Example - example-ngMessages-directive + + + + + + + + + + +
+ +
myForm.myName.$error = {{ myForm.myName.$error | json }}
+ +
+
You did not enter a field
+
Your field is too short
+
Your field is too long
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessages-directive/manifest.json b/1.4.10/docs/examples/example-ngMessages-directive/manifest.json new file mode 100644 index 0000000000..1918dce064 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessages-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngMessages-directive", + "files": [ + "index-production.html", + "script.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMessages-directive/script.js b/1.4.10/docs/examples/example-ngMessages-directive/script.js new file mode 100644 index 0000000000..c4ec81628b --- /dev/null +++ b/1.4.10/docs/examples/example-ngMessages-directive/script.js @@ -0,0 +1,4 @@ +(function(angular) { + 'use strict'; +angular.module('ngMessagesExample', ['ngMessages']); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMinlengthDirective/index-debug.html b/1.4.10/docs/examples/example-ngMinlengthDirective/index-debug.html new file mode 100644 index 0000000000..fdfc2828e2 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMinlengthDirective/index-debug.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngMinlengthDirective-debug + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMinlengthDirective/index-jquery.html b/1.4.10/docs/examples/example-ngMinlengthDirective/index-jquery.html new file mode 100644 index 0000000000..6c4041c346 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMinlengthDirective/index-jquery.html @@ -0,0 +1,34 @@ + + + + + Example - example-ngMinlengthDirective-jquery + + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMinlengthDirective/index-production.html b/1.4.10/docs/examples/example-ngMinlengthDirective/index-production.html new file mode 100644 index 0000000000..e38c04b7c9 --- /dev/null +++ b/1.4.10/docs/examples/example-ngMinlengthDirective/index-production.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngMinlengthDirective-production + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMinlengthDirective/index.html b/1.4.10/docs/examples/example-ngMinlengthDirective/index.html new file mode 100644 index 0000000000..6cbe7a282d --- /dev/null +++ b/1.4.10/docs/examples/example-ngMinlengthDirective/index.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngMinlengthDirective + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMinlengthDirective/manifest.json b/1.4.10/docs/examples/example-ngMinlengthDirective/manifest.json new file mode 100644 index 0000000000..5acc9f9a5e --- /dev/null +++ b/1.4.10/docs/examples/example-ngMinlengthDirective/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngMinlengthDirective", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngMinlengthDirective/protractor.js b/1.4.10/docs/examples/example-ngMinlengthDirective/protractor.js new file mode 100644 index 0000000000..4edb59559d --- /dev/null +++ b/1.4.10/docs/examples/example-ngMinlengthDirective/protractor.js @@ -0,0 +1,10 @@ +var model = element(by.binding('model')); +var input = element(by.id('input')); + +it('should validate the input with the default minlength', function() { + input.sendKeys('ab'); + expect(model.getText()).not.toContain('ab'); + + input.sendKeys('abc'); + expect(model.getText()).toContain('abc'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModel-getter-setter/app.js b/1.4.10/docs/examples/example-ngModel-getter-setter/app.js new file mode 100644 index 0000000000..5042eb4970 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModel-getter-setter/app.js @@ -0,0 +1,16 @@ +(function(angular) { + 'use strict'; +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; + } + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModel-getter-setter/index-debug.html b/1.4.10/docs/examples/example-ngModel-getter-setter/index-debug.html new file mode 100644 index 0000000000..974e2ddf37 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModel-getter-setter/index-debug.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngModel-getter-setter-debug + + + + + + + + + +
+
+ +
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModel-getter-setter/index-jquery.html b/1.4.10/docs/examples/example-ngModel-getter-setter/index-jquery.html new file mode 100644 index 0000000000..5260e089f5 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModel-getter-setter/index-jquery.html @@ -0,0 +1,27 @@ + + + + + Example - example-ngModel-getter-setter-jquery + + + + + + + + + + +
+
+ +
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModel-getter-setter/index-production.html b/1.4.10/docs/examples/example-ngModel-getter-setter/index-production.html new file mode 100644 index 0000000000..244388a131 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModel-getter-setter/index-production.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngModel-getter-setter-production + + + + + + + + + +
+
+ +
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModel-getter-setter/index.html b/1.4.10/docs/examples/example-ngModel-getter-setter/index.html new file mode 100644 index 0000000000..ed91d4225a --- /dev/null +++ b/1.4.10/docs/examples/example-ngModel-getter-setter/index.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngModel-getter-setter + + + + + + + + + +
+
+ +
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModel-getter-setter/manifest.json b/1.4.10/docs/examples/example-ngModel-getter-setter/manifest.json new file mode 100644 index 0000000000..2d8bb1e7b8 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModel-getter-setter/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngModel-getter-setter", + "files": [ + "index-production.html", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-blur/app.js b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/app.js new file mode 100644 index 0000000000..a32a7ebfc3 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/app.js @@ -0,0 +1,13 @@ +(function(angular) { + 'use strict'; +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(); + } + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index-debug.html b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index-debug.html new file mode 100644 index 0000000000..1845b5f94c --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index-debug.html @@ -0,0 +1,31 @@ + + + + + Example - example-ngModelOptions-directive-blur-debug + + + + + + + + + +
+
+
+
+
+
user.name = 
+
user.data = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index-jquery.html b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index-jquery.html new file mode 100644 index 0000000000..b566332a30 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index-jquery.html @@ -0,0 +1,32 @@ + + + + + Example - example-ngModelOptions-directive-blur-jquery + + + + + + + + + + +
+
+
+
+
+
user.name = 
+
user.data = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index-production.html b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index-production.html new file mode 100644 index 0000000000..1bb4ca178b --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index-production.html @@ -0,0 +1,31 @@ + + + + + Example - example-ngModelOptions-directive-blur-production + + + + + + + + + +
+
+
+
+
+
user.name = 
+
user.data = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index.html b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index.html new file mode 100644 index 0000000000..de5b955405 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/index.html @@ -0,0 +1,31 @@ + + + + + Example - example-ngModelOptions-directive-blur + + + + + + + + + +
+
+
+
+
+
user.name = 
+
user.data = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-blur/manifest.json b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/manifest.json new file mode 100644 index 0000000000..84481b31ea --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-ngModelOptions-directive-blur", + "files": [ + "index-production.html", + "app.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-blur/protractor.js b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/protractor.js new file mode 100644 index 0000000000..25335f37d2 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-blur/protractor.js @@ -0,0 +1,20 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/app.js b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/app.js new file mode 100644 index 0000000000..2fadf3a8b8 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/app.js @@ -0,0 +1,7 @@ +(function(angular) { + 'use strict'; +angular.module('optionsExample', []) + .controller('ExampleController', ['$scope', function($scope) { + $scope.user = { name: 'Igor' }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index-debug.html b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index-debug.html new file mode 100644 index 0000000000..516aa6a7dc --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index-debug.html @@ -0,0 +1,28 @@ + + + + + Example - example-ngModelOptions-directive-debounce-debug + + + + + + + + + +
+
+ + +
+
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index-jquery.html b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index-jquery.html new file mode 100644 index 0000000000..2de12df82f --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index-jquery.html @@ -0,0 +1,29 @@ + + + + + Example - example-ngModelOptions-directive-debounce-jquery + + + + + + + + + + +
+
+ + +
+
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index-production.html b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index-production.html new file mode 100644 index 0000000000..23c0d37aef --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index-production.html @@ -0,0 +1,28 @@ + + + + + Example - example-ngModelOptions-directive-debounce-production + + + + + + + + + +
+
+ + +
+
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index.html b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index.html new file mode 100644 index 0000000000..af68659000 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/index.html @@ -0,0 +1,28 @@ + + + + + Example - example-ngModelOptions-directive-debounce + + + + + + + + + +
+
+ + +
+
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/manifest.json b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/manifest.json new file mode 100644 index 0000000000..c1836815bf --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-debounce/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngModelOptions-directive-debounce", + "files": [ + "index-production.html", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/app.js b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/app.js new file mode 100644 index 0000000000..7a374000cb --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/app.js @@ -0,0 +1,16 @@ +(function(angular) { + 'use strict'; +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; + } + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index-debug.html b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index-debug.html new file mode 100644 index 0000000000..22659decec --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index-debug.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngModelOptions-directive-getter-setter-debug + + + + + + + + + +
+
+ +
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index-jquery.html b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index-jquery.html new file mode 100644 index 0000000000..5df9981479 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index-jquery.html @@ -0,0 +1,27 @@ + + + + + Example - example-ngModelOptions-directive-getter-setter-jquery + + + + + + + + + + +
+
+ +
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index-production.html b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index-production.html new file mode 100644 index 0000000000..7394aa9f9c --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index-production.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngModelOptions-directive-getter-setter-production + + + + + + + + + +
+
+ +
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index.html b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index.html new file mode 100644 index 0000000000..f2b3d9dc86 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/index.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngModelOptions-directive-getter-setter + + + + + + + + + +
+
+ +
+
user.name = 
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/manifest.json b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/manifest.json new file mode 100644 index 0000000000..3fbf3d7b08 --- /dev/null +++ b/1.4.10/docs/examples/example-ngModelOptions-directive-getter-setter/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngModelOptions-directive-getter-setter", + "files": [ + "index-production.html", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngPatternDirective/index-debug.html b/1.4.10/docs/examples/example-ngPatternDirective/index-debug.html new file mode 100644 index 0000000000..7ac11a4c1d --- /dev/null +++ b/1.4.10/docs/examples/example-ngPatternDirective/index-debug.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngPatternDirective-debug + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngPatternDirective/index-jquery.html b/1.4.10/docs/examples/example-ngPatternDirective/index-jquery.html new file mode 100644 index 0000000000..f1fc2ae16a --- /dev/null +++ b/1.4.10/docs/examples/example-ngPatternDirective/index-jquery.html @@ -0,0 +1,34 @@ + + + + + Example - example-ngPatternDirective-jquery + + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngPatternDirective/index-production.html b/1.4.10/docs/examples/example-ngPatternDirective/index-production.html new file mode 100644 index 0000000000..50c3e7c7d3 --- /dev/null +++ b/1.4.10/docs/examples/example-ngPatternDirective/index-production.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngPatternDirective-production + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngPatternDirective/index.html b/1.4.10/docs/examples/example-ngPatternDirective/index.html new file mode 100644 index 0000000000..47daeb5dbe --- /dev/null +++ b/1.4.10/docs/examples/example-ngPatternDirective/index.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngPatternDirective + + + + + + + + + +
+
+ + +
+ +
+
+ input valid? = {{form.input.$valid}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngPatternDirective/manifest.json b/1.4.10/docs/examples/example-ngPatternDirective/manifest.json new file mode 100644 index 0000000000..257a4f2a96 --- /dev/null +++ b/1.4.10/docs/examples/example-ngPatternDirective/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngPatternDirective", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngPatternDirective/protractor.js b/1.4.10/docs/examples/example-ngPatternDirective/protractor.js new file mode 100644 index 0000000000..9b807ef05c --- /dev/null +++ b/1.4.10/docs/examples/example-ngPatternDirective/protractor.js @@ -0,0 +1,12 @@ +var model = element(by.binding('model')); +var input = element(by.id('input')); + +it('should validate the input with the default pattern', function() { + input.sendKeys('aaa'); + expect(model.getText()).not.toContain('aaa'); + + input.clear().then(function() { + input.sendKeys('123'); + expect(model.getText()).toContain('123'); + }); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngRequiredDirective/index-debug.html b/1.4.10/docs/examples/example-ngRequiredDirective/index-debug.html new file mode 100644 index 0000000000..66b5f2b186 --- /dev/null +++ b/1.4.10/docs/examples/example-ngRequiredDirective/index-debug.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngRequiredDirective-debug + + + + + + + + + +
+
+ + +
+ +
+
+ required error set? = {{form.input.$error.required}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngRequiredDirective/index-jquery.html b/1.4.10/docs/examples/example-ngRequiredDirective/index-jquery.html new file mode 100644 index 0000000000..e243709f89 --- /dev/null +++ b/1.4.10/docs/examples/example-ngRequiredDirective/index-jquery.html @@ -0,0 +1,34 @@ + + + + + Example - example-ngRequiredDirective-jquery + + + + + + + + + + +
+
+ + +
+ +
+
+ required error set? = {{form.input.$error.required}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngRequiredDirective/index-production.html b/1.4.10/docs/examples/example-ngRequiredDirective/index-production.html new file mode 100644 index 0000000000..574f2c6c55 --- /dev/null +++ b/1.4.10/docs/examples/example-ngRequiredDirective/index-production.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngRequiredDirective-production + + + + + + + + + +
+
+ + +
+ +
+
+ required error set? = {{form.input.$error.required}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngRequiredDirective/index.html b/1.4.10/docs/examples/example-ngRequiredDirective/index.html new file mode 100644 index 0000000000..9841549784 --- /dev/null +++ b/1.4.10/docs/examples/example-ngRequiredDirective/index.html @@ -0,0 +1,33 @@ + + + + + Example - example-ngRequiredDirective + + + + + + + + + +
+
+ + +
+ +
+
+ required error set? = {{form.input.$error.required}}
+ model = {{model}} +
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngRequiredDirective/manifest.json b/1.4.10/docs/examples/example-ngRequiredDirective/manifest.json new file mode 100644 index 0000000000..5fd71273cf --- /dev/null +++ b/1.4.10/docs/examples/example-ngRequiredDirective/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngRequiredDirective", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngRequiredDirective/protractor.js b/1.4.10/docs/examples/example-ngRequiredDirective/protractor.js new file mode 100644 index 0000000000..4b1328641e --- /dev/null +++ b/1.4.10/docs/examples/example-ngRequiredDirective/protractor.js @@ -0,0 +1,11 @@ +var required = element(by.binding('form.input.$error.required')); +var model = element(by.binding('model')); +var input = element(by.id('input')); + +it('should set the required error', function() { + expect(required.getText()).toContain('true'); + + input.sendKeys('123'); + expect(required.getText()).not.toContain('true'); + expect(model.getText()).toContain('123'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngValue-directive/index-debug.html b/1.4.10/docs/examples/example-ngValue-directive/index-debug.html new file mode 100644 index 0000000000..534eb70be0 --- /dev/null +++ b/1.4.10/docs/examples/example-ngValue-directive/index-debug.html @@ -0,0 +1,34 @@ + + + + + Example - example-ngValue-directive-debug + + + + + + + + + +
+

Which is your favorite?

+ +
You chose {{my.favorite}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngValue-directive/index-jquery.html b/1.4.10/docs/examples/example-ngValue-directive/index-jquery.html new file mode 100644 index 0000000000..344b1ce12b --- /dev/null +++ b/1.4.10/docs/examples/example-ngValue-directive/index-jquery.html @@ -0,0 +1,35 @@ + + + + + Example - example-ngValue-directive-jquery + + + + + + + + + + +
+

Which is your favorite?

+ +
You chose {{my.favorite}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngValue-directive/index-production.html b/1.4.10/docs/examples/example-ngValue-directive/index-production.html new file mode 100644 index 0000000000..09182afb4c --- /dev/null +++ b/1.4.10/docs/examples/example-ngValue-directive/index-production.html @@ -0,0 +1,34 @@ + + + + + Example - example-ngValue-directive-production + + + + + + + + + +
+

Which is your favorite?

+ +
You chose {{my.favorite}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngValue-directive/index.html b/1.4.10/docs/examples/example-ngValue-directive/index.html new file mode 100644 index 0000000000..b434e8ef11 --- /dev/null +++ b/1.4.10/docs/examples/example-ngValue-directive/index.html @@ -0,0 +1,34 @@ + + + + + Example - example-ngValue-directive + + + + + + + + + +
+

Which is your favorite?

+ +
You chose {{my.favorite}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngValue-directive/manifest.json b/1.4.10/docs/examples/example-ngValue-directive/manifest.json new file mode 100644 index 0000000000..1e8f897466 --- /dev/null +++ b/1.4.10/docs/examples/example-ngValue-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngValue-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngValue-directive/protractor.js b/1.4.10/docs/examples/example-ngValue-directive/protractor.js new file mode 100644 index 0000000000..fd6ff1273e --- /dev/null +++ b/1.4.10/docs/examples/example-ngValue-directive/protractor.js @@ -0,0 +1,9 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngView-directive/animations.css b/1.4.10/docs/examples/example-ngView-directive/animations.css new file mode 100644 index 0000000000..63356d9441 --- /dev/null +++ b/1.4.10/docs/examples/example-ngView-directive/animations.css @@ -0,0 +1,37 @@ +.view-animate-container { + position:relative; + height:100px!important; + background:white; + border:1px solid black; + height:40px; + overflow:hidden; +} + +.view-animate { + padding:10px; +} + +.view-animate.ng-enter, .view-animate.ng-leave { + transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 1.5s; + + display:block; + width:100%; + border-left:1px solid black; + + position:absolute; + top:0; + left:0; + right:0; + bottom:0; + padding:10px; +} + +.view-animate.ng-enter { + left:100%; +} +.view-animate.ng-enter.ng-enter-active { + left:0; +} +.view-animate.ng-leave.ng-leave-active { + left:-100%; +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngView-directive/book.html b/1.4.10/docs/examples/example-ngView-directive/book.html new file mode 100644 index 0000000000..df473cc324 --- /dev/null +++ b/1.4.10/docs/examples/example-ngView-directive/book.html @@ -0,0 +1,4 @@ +
+ controller: {{book.name}}
+ Book Id: {{book.params.bookId}}
+
\ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngView-directive/chapter.html b/1.4.10/docs/examples/example-ngView-directive/chapter.html new file mode 100644 index 0000000000..390a0d8e1d --- /dev/null +++ b/1.4.10/docs/examples/example-ngView-directive/chapter.html @@ -0,0 +1,5 @@ +
+ controller: {{chapter.name}}
+ Book Id: {{chapter.params.bookId}}
+ Chapter Id: {{chapter.params.chapterId}} +
\ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngView-directive/index-debug.html b/1.4.10/docs/examples/example-ngView-directive/index-debug.html new file mode 100644 index 0000000000..dd60d3bd64 --- /dev/null +++ b/1.4.10/docs/examples/example-ngView-directive/index-debug.html @@ -0,0 +1,39 @@ + + + + + Example - example-ngView-directive-debug + + + + + + + + + + + + +
+ Choose: + Moby | + Moby: Ch1 | + Gatsby | + Gatsby: Ch4 | + Scarlet Letter
+ +
+
+
+
+ +
$location.path() = {{main.$location.path()}}
+
$route.current.templateUrl = {{main.$route.current.templateUrl}}
+
$route.current.params = {{main.$route.current.params}}
+
$routeParams = {{main.$routeParams}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngView-directive/index-jquery.html b/1.4.10/docs/examples/example-ngView-directive/index-jquery.html new file mode 100644 index 0000000000..47ac99ff20 --- /dev/null +++ b/1.4.10/docs/examples/example-ngView-directive/index-jquery.html @@ -0,0 +1,40 @@ + + + + + Example - example-ngView-directive-jquery + + + + + + + + + + + + + +
+ Choose: + Moby | + Moby: Ch1 | + Gatsby | + Gatsby: Ch4 | + Scarlet Letter
+ +
+
+
+
+ +
$location.path() = {{main.$location.path()}}
+
$route.current.templateUrl = {{main.$route.current.templateUrl}}
+
$route.current.params = {{main.$route.current.params}}
+
$routeParams = {{main.$routeParams}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngView-directive/index-production.html b/1.4.10/docs/examples/example-ngView-directive/index-production.html new file mode 100644 index 0000000000..0155110022 --- /dev/null +++ b/1.4.10/docs/examples/example-ngView-directive/index-production.html @@ -0,0 +1,39 @@ + + + + + Example - example-ngView-directive-production + + + + + + + + + + + + +
+ Choose: + Moby | + Moby: Ch1 | + Gatsby | + Gatsby: Ch4 | + Scarlet Letter
+ +
+
+
+
+ +
$location.path() = {{main.$location.path()}}
+
$route.current.templateUrl = {{main.$route.current.templateUrl}}
+
$route.current.params = {{main.$route.current.params}}
+
$routeParams = {{main.$routeParams}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngView-directive/index.html b/1.4.10/docs/examples/example-ngView-directive/index.html new file mode 100644 index 0000000000..8e71e5903c --- /dev/null +++ b/1.4.10/docs/examples/example-ngView-directive/index.html @@ -0,0 +1,39 @@ + + + + + Example - example-ngView-directive + + + + + + + + + + + + +
+ Choose: + Moby | + Moby: Ch1 | + Gatsby | + Gatsby: Ch4 | + Scarlet Letter
+ +
+
+
+
+ +
$location.path() = {{main.$location.path()}}
+
$route.current.templateUrl = {{main.$route.current.templateUrl}}
+
$route.current.params = {{main.$route.current.params}}
+
$routeParams = {{main.$routeParams}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngView-directive/manifest.json b/1.4.10/docs/examples/example-ngView-directive/manifest.json new file mode 100644 index 0000000000..5ebc47d05d --- /dev/null +++ b/1.4.10/docs/examples/example-ngView-directive/manifest.json @@ -0,0 +1,11 @@ +{ + "name": "example-ngView-directive", + "files": [ + "index-production.html", + "book.html", + "chapter.html", + "animations.css", + "script.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngView-directive/protractor.js b/1.4.10/docs/examples/example-ngView-directive/protractor.js new file mode 100644 index 0000000000..afd460a982 --- /dev/null +++ b/1.4.10/docs/examples/example-ngView-directive/protractor.js @@ -0,0 +1,13 @@ +it('should load and compile correct template', function() { + element(by.linkText('Moby: Ch1')).click(); + var content = element(by.css('[ng-view]')).getText(); + expect(content).toMatch(/controller\: ChapterCtrl/); + expect(content).toMatch(/Book Id\: Moby/); + expect(content).toMatch(/Chapter Id\: 1/); + + element(by.partialLinkText('Scarlet')).click(); + + content = element(by.css('[ng-view]')).getText(); + expect(content).toMatch(/controller\: BookCtrl/); + expect(content).toMatch(/Book Id\: Scarlet/); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngView-directive/script.js b/1.4.10/docs/examples/example-ngView-directive/script.js new file mode 100644 index 0000000000..370722efdc --- /dev/null +++ b/1.4.10/docs/examples/example-ngView-directive/script.js @@ -0,0 +1,34 @@ +(function(angular) { + 'use strict'; +angular.module('ngViewExample', ['ngRoute', 'ngAnimate']) + .config(['$routeProvider', '$locationProvider', + function($routeProvider, $locationProvider) { + $routeProvider + .when('/Book/:bookId', { + templateUrl: 'book.html', + controller: 'BookCtrl', + controllerAs: 'book' + }) + .when('/Book/:bookId/ch/:chapterId', { + templateUrl: 'chapter.html', + controller: 'ChapterCtrl', + controllerAs: 'chapter' + }); + + $locationProvider.html5Mode(true); + }]) + .controller('MainCtrl', ['$route', '$routeParams', '$location', + function($route, $routeParams, $location) { + this.$route = $route; + this.$location = $location; + this.$routeParams = $routeParams; + }]) + .controller('BookCtrl', ['$routeParams', function($routeParams) { + this.name = "BookCtrl"; + this.params = $routeParams; + }]) + .controller('ChapterCtrl', ['$routeParams', function($routeParams) { + this.name = "ChapterCtrl"; + this.params = $routeParams; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngrepeat-select/app.js b/1.4.10/docs/examples/example-ngrepeat-select/app.js new file mode 100644 index 0000000000..7805f0aed5 --- /dev/null +++ b/1.4.10/docs/examples/example-ngrepeat-select/app.js @@ -0,0 +1,14 @@ +(function(angular) { + 'use strict'; +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'} + ], + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngrepeat-select/index-debug.html b/1.4.10/docs/examples/example-ngrepeat-select/index-debug.html new file mode 100644 index 0000000000..a4b885d1c8 --- /dev/null +++ b/1.4.10/docs/examples/example-ngrepeat-select/index-debug.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngrepeat-select-debug + + + + + + + + + +
+
+ + +
+
+ repeatSelect = {{data.repeatSelect}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngrepeat-select/index-jquery.html b/1.4.10/docs/examples/example-ngrepeat-select/index-jquery.html new file mode 100644 index 0000000000..b7e190318e --- /dev/null +++ b/1.4.10/docs/examples/example-ngrepeat-select/index-jquery.html @@ -0,0 +1,27 @@ + + + + + Example - example-ngrepeat-select-jquery + + + + + + + + + + +
+
+ + +
+
+ repeatSelect = {{data.repeatSelect}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngrepeat-select/index-production.html b/1.4.10/docs/examples/example-ngrepeat-select/index-production.html new file mode 100644 index 0000000000..3d61aae828 --- /dev/null +++ b/1.4.10/docs/examples/example-ngrepeat-select/index-production.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngrepeat-select-production + + + + + + + + + +
+
+ + +
+
+ repeatSelect = {{data.repeatSelect}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngrepeat-select/index.html b/1.4.10/docs/examples/example-ngrepeat-select/index.html new file mode 100644 index 0000000000..37270240e8 --- /dev/null +++ b/1.4.10/docs/examples/example-ngrepeat-select/index.html @@ -0,0 +1,26 @@ + + + + + Example - example-ngrepeat-select + + + + + + + + + +
+
+ + +
+
+ repeatSelect = {{data.repeatSelect}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-ngrepeat-select/manifest.json b/1.4.10/docs/examples/example-ngrepeat-select/manifest.json new file mode 100644 index 0000000000..b148b3bd47 --- /dev/null +++ b/1.4.10/docs/examples/example-ngrepeat-select/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-ngrepeat-select", + "files": [ + "index-production.html", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-number-input-directive/index-debug.html b/1.4.10/docs/examples/example-number-input-directive/index-debug.html new file mode 100644 index 0000000000..fd97b073fb --- /dev/null +++ b/1.4.10/docs/examples/example-number-input-directive/index-debug.html @@ -0,0 +1,40 @@ + + + + + Example - example-number-input-directive-debug + + + + + + + + + +
+ +
+ + Required! + + Not valid number! +
+ value = {{example.value}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-number-input-directive/index-jquery.html b/1.4.10/docs/examples/example-number-input-directive/index-jquery.html new file mode 100644 index 0000000000..beb19a6df6 --- /dev/null +++ b/1.4.10/docs/examples/example-number-input-directive/index-jquery.html @@ -0,0 +1,41 @@ + + + + + Example - example-number-input-directive-jquery + + + + + + + + + + +
+ +
+ + Required! + + Not valid number! +
+ value = {{example.value}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-number-input-directive/index-production.html b/1.4.10/docs/examples/example-number-input-directive/index-production.html new file mode 100644 index 0000000000..edb8f8bfe8 --- /dev/null +++ b/1.4.10/docs/examples/example-number-input-directive/index-production.html @@ -0,0 +1,40 @@ + + + + + Example - example-number-input-directive-production + + + + + + + + + +
+ +
+ + Required! + + Not valid number! +
+ value = {{example.value}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-number-input-directive/index.html b/1.4.10/docs/examples/example-number-input-directive/index.html new file mode 100644 index 0000000000..707e669cff --- /dev/null +++ b/1.4.10/docs/examples/example-number-input-directive/index.html @@ -0,0 +1,40 @@ + + + + + Example - example-number-input-directive + + + + + + + + + +
+ +
+ + Required! + + Not valid number! +
+ value = {{example.value}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-number-input-directive/manifest.json b/1.4.10/docs/examples/example-number-input-directive/manifest.json new file mode 100644 index 0000000000..514d04f3b7 --- /dev/null +++ b/1.4.10/docs/examples/example-number-input-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-number-input-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-number-input-directive/protractor.js b/1.4.10/docs/examples/example-number-input-directive/protractor.js new file mode 100644 index 0000000000..623877e325 --- /dev/null +++ b/1.4.10/docs/examples/example-number-input-directive/protractor.js @@ -0,0 +1,22 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-radio-input-directive/index-debug.html b/1.4.10/docs/examples/example-radio-input-directive/index-debug.html new file mode 100644 index 0000000000..e358a05823 --- /dev/null +++ b/1.4.10/docs/examples/example-radio-input-directive/index-debug.html @@ -0,0 +1,43 @@ + + + + + Example - example-radio-input-directive-debug + + + + + + + + + +
+
+
+
+ color = {{color.name | json}}
+
+ Note that `ng-value="specialValue"` sets radio item's value to be the value of `$scope.specialValue`. + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-radio-input-directive/index-jquery.html b/1.4.10/docs/examples/example-radio-input-directive/index-jquery.html new file mode 100644 index 0000000000..651e1f4630 --- /dev/null +++ b/1.4.10/docs/examples/example-radio-input-directive/index-jquery.html @@ -0,0 +1,44 @@ + + + + + Example - example-radio-input-directive-jquery + + + + + + + + + + +
+
+
+
+ color = {{color.name | json}}
+
+ Note that `ng-value="specialValue"` sets radio item's value to be the value of `$scope.specialValue`. + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-radio-input-directive/index-production.html b/1.4.10/docs/examples/example-radio-input-directive/index-production.html new file mode 100644 index 0000000000..69aeb911b8 --- /dev/null +++ b/1.4.10/docs/examples/example-radio-input-directive/index-production.html @@ -0,0 +1,43 @@ + + + + + Example - example-radio-input-directive-production + + + + + + + + + +
+
+
+
+ color = {{color.name | json}}
+
+ Note that `ng-value="specialValue"` sets radio item's value to be the value of `$scope.specialValue`. + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-radio-input-directive/index.html b/1.4.10/docs/examples/example-radio-input-directive/index.html new file mode 100644 index 0000000000..11b8a03ac6 --- /dev/null +++ b/1.4.10/docs/examples/example-radio-input-directive/index.html @@ -0,0 +1,43 @@ + + + + + Example - example-radio-input-directive + + + + + + + + + +
+
+
+
+ color = {{color.name | json}}
+
+ Note that `ng-value="specialValue"` sets radio item's value to be the value of `$scope.specialValue`. + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-radio-input-directive/manifest.json b/1.4.10/docs/examples/example-radio-input-directive/manifest.json new file mode 100644 index 0000000000..863bef42e2 --- /dev/null +++ b/1.4.10/docs/examples/example-radio-input-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-radio-input-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-radio-input-directive/protractor.js b/1.4.10/docs/examples/example-radio-input-directive/protractor.js new file mode 100644 index 0000000000..ef0ebbe209 --- /dev/null +++ b/1.4.10/docs/examples/example-radio-input-directive/protractor.js @@ -0,0 +1,9 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-default-values/app.js b/1.4.10/docs/examples/example-select-with-default-values/app.js new file mode 100644 index 0000000000..2c9d774249 --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-default-values/app.js @@ -0,0 +1,14 @@ +(function(angular) { + 'use strict'; +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 + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-default-values/index-debug.html b/1.4.10/docs/examples/example-select-with-default-values/index-debug.html new file mode 100644 index 0000000000..c2e28c50d7 --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-default-values/index-debug.html @@ -0,0 +1,26 @@ + + + + + Example - example-select-with-default-values-debug + + + + + + + + + +
+
+ + +
+
+ option = {{data.selectedOption}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-default-values/index-jquery.html b/1.4.10/docs/examples/example-select-with-default-values/index-jquery.html new file mode 100644 index 0000000000..6f962ba624 --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-default-values/index-jquery.html @@ -0,0 +1,27 @@ + + + + + Example - example-select-with-default-values-jquery + + + + + + + + + + +
+
+ + +
+
+ option = {{data.selectedOption}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-default-values/index-production.html b/1.4.10/docs/examples/example-select-with-default-values/index-production.html new file mode 100644 index 0000000000..83bdd1bb42 --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-default-values/index-production.html @@ -0,0 +1,26 @@ + + + + + Example - example-select-with-default-values-production + + + + + + + + + +
+
+ + +
+
+ option = {{data.selectedOption}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-default-values/index.html b/1.4.10/docs/examples/example-select-with-default-values/index.html new file mode 100644 index 0000000000..1fe3529ed6 --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-default-values/index.html @@ -0,0 +1,26 @@ + + + + + Example - example-select-with-default-values + + + + + + + + + +
+
+ + +
+
+ option = {{data.selectedOption}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-default-values/manifest.json b/1.4.10/docs/examples/example-select-with-default-values/manifest.json new file mode 100644 index 0000000000..99cca6df7d --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-default-values/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-select-with-default-values", + "files": [ + "index-production.html", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-non-string-options/app.js b/1.4.10/docs/examples/example-select-with-non-string-options/app.js new file mode 100644 index 0000000000..bcfb16761e --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-non-string-options/app.js @@ -0,0 +1,20 @@ +(function(angular) { + 'use strict'; +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; + }); + } + }; + }); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-non-string-options/index-debug.html b/1.4.10/docs/examples/example-select-with-non-string-options/index-debug.html new file mode 100644 index 0000000000..890606d5da --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-non-string-options/index-debug.html @@ -0,0 +1,22 @@ + + + + + Example - example-select-with-non-string-options-debug + + + + + + + + + + +{{ model }} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-non-string-options/index-jquery.html b/1.4.10/docs/examples/example-select-with-non-string-options/index-jquery.html new file mode 100644 index 0000000000..fd24bd45aa --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-non-string-options/index-jquery.html @@ -0,0 +1,23 @@ + + + + + Example - example-select-with-non-string-options-jquery + + + + + + + + + + + +{{ model }} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-non-string-options/index-production.html b/1.4.10/docs/examples/example-select-with-non-string-options/index-production.html new file mode 100644 index 0000000000..c37e7be4e8 --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-non-string-options/index-production.html @@ -0,0 +1,22 @@ + + + + + Example - example-select-with-non-string-options-production + + + + + + + + + + +{{ model }} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-non-string-options/index.html b/1.4.10/docs/examples/example-select-with-non-string-options/index.html new file mode 100644 index 0000000000..4737afb422 --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-non-string-options/index.html @@ -0,0 +1,22 @@ + + + + + Example - example-select-with-non-string-options + + + + + + + + + + +{{ model }} + + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-non-string-options/manifest.json b/1.4.10/docs/examples/example-select-with-non-string-options/manifest.json new file mode 100644 index 0000000000..baf90744b4 --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-non-string-options/manifest.json @@ -0,0 +1,8 @@ +{ + "name": "example-select-with-non-string-options", + "files": [ + "index-production.html", + "app.js", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-select-with-non-string-options/protractor.js b/1.4.10/docs/examples/example-select-with-non-string-options/protractor.js new file mode 100644 index 0000000000..29bf01f68b --- /dev/null +++ b/1.4.10/docs/examples/example-select-with-non-string-options/protractor.js @@ -0,0 +1,4 @@ +it('should initialize to model', function() { + var select = element(by.css('select')); + expect(element(by.model('model.id')).$('option:checked').getText()).toEqual('Two'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-static-select/app.js b/1.4.10/docs/examples/example-static-select/app.js new file mode 100644 index 0000000000..d75110154d --- /dev/null +++ b/1.4.10/docs/examples/example-static-select/app.js @@ -0,0 +1,15 @@ +(function(angular) { + 'use strict'; +angular.module('staticSelect', []) + .controller('ExampleController', ['$scope', function($scope) { + $scope.data = { + singleSelect: null, + multipleSelect: [], + option1: 'option-1', + }; + + $scope.forceUnknownOption = function() { + $scope.data.singleSelect = 'nonsense'; + }; + }]); +})(window.angular); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-static-select/index-debug.html b/1.4.10/docs/examples/example-static-select/index-debug.html new file mode 100644 index 0000000000..e6947b9d44 --- /dev/null +++ b/1.4.10/docs/examples/example-static-select/index-debug.html @@ -0,0 +1,43 @@ + + + + + Example - example-static-select-debug + + + + + + + + + +
+
+
+
+ +
+
+
+ singleSelect = {{data.singleSelect}} + +
+
+
+ multipleSelect = {{data.multipleSelect}}
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-static-select/index-jquery.html b/1.4.10/docs/examples/example-static-select/index-jquery.html new file mode 100644 index 0000000000..aaf40daa6a --- /dev/null +++ b/1.4.10/docs/examples/example-static-select/index-jquery.html @@ -0,0 +1,44 @@ + + + + + Example - example-static-select-jquery + + + + + + + + + + +
+
+
+
+ +
+
+
+ singleSelect = {{data.singleSelect}} + +
+
+
+ multipleSelect = {{data.multipleSelect}}
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-static-select/index-production.html b/1.4.10/docs/examples/example-static-select/index-production.html new file mode 100644 index 0000000000..f50fa36b31 --- /dev/null +++ b/1.4.10/docs/examples/example-static-select/index-production.html @@ -0,0 +1,43 @@ + + + + + Example - example-static-select-production + + + + + + + + + +
+
+
+
+ +
+
+
+ singleSelect = {{data.singleSelect}} + +
+
+
+ multipleSelect = {{data.multipleSelect}}
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-static-select/index.html b/1.4.10/docs/examples/example-static-select/index.html new file mode 100644 index 0000000000..6e7a1fc576 --- /dev/null +++ b/1.4.10/docs/examples/example-static-select/index.html @@ -0,0 +1,43 @@ + + + + + Example - example-static-select + + + + + + + + + +
+
+
+
+ +
+
+
+ singleSelect = {{data.singleSelect}} + +
+
+
+ multipleSelect = {{data.multipleSelect}}
+
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-static-select/manifest.json b/1.4.10/docs/examples/example-static-select/manifest.json new file mode 100644 index 0000000000..3b098d1bfd --- /dev/null +++ b/1.4.10/docs/examples/example-static-select/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-static-select", + "files": [ + "index-production.html", + "app.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-text-input-directive/index-debug.html b/1.4.10/docs/examples/example-text-input-directive/index-debug.html new file mode 100644 index 0000000000..a7c39a15e0 --- /dev/null +++ b/1.4.10/docs/examples/example-text-input-directive/index-debug.html @@ -0,0 +1,41 @@ + + + + + Example - example-text-input-directive-debug + + + + + + + + + +
+ +
+ + Required! + + Single word only! +
+ text = {{example.text}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-text-input-directive/index-jquery.html b/1.4.10/docs/examples/example-text-input-directive/index-jquery.html new file mode 100644 index 0000000000..33f46699be --- /dev/null +++ b/1.4.10/docs/examples/example-text-input-directive/index-jquery.html @@ -0,0 +1,42 @@ + + + + + Example - example-text-input-directive-jquery + + + + + + + + + + +
+ +
+ + Required! + + Single word only! +
+ text = {{example.text}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-text-input-directive/index-production.html b/1.4.10/docs/examples/example-text-input-directive/index-production.html new file mode 100644 index 0000000000..be314a62fc --- /dev/null +++ b/1.4.10/docs/examples/example-text-input-directive/index-production.html @@ -0,0 +1,41 @@ + + + + + Example - example-text-input-directive-production + + + + + + + + + +
+ +
+ + Required! + + Single word only! +
+ text = {{example.text}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-text-input-directive/index.html b/1.4.10/docs/examples/example-text-input-directive/index.html new file mode 100644 index 0000000000..7bdd6d0f24 --- /dev/null +++ b/1.4.10/docs/examples/example-text-input-directive/index.html @@ -0,0 +1,41 @@ + + + + + Example - example-text-input-directive + + + + + + + + + +
+ +
+ + Required! + + Single word only! +
+ text = {{example.text}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-text-input-directive/manifest.json b/1.4.10/docs/examples/example-text-input-directive/manifest.json new file mode 100644 index 0000000000..2612b8ac3a --- /dev/null +++ b/1.4.10/docs/examples/example-text-input-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-text-input-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-text-input-directive/protractor.js b/1.4.10/docs/examples/example-text-input-directive/protractor.js new file mode 100644 index 0000000000..dcf124d802 --- /dev/null +++ b/1.4.10/docs/examples/example-text-input-directive/protractor.js @@ -0,0 +1,23 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-time-input-directive/index-debug.html b/1.4.10/docs/examples/example-time-input-directive/index-debug.html new file mode 100644 index 0000000000..d9b1bf97f0 --- /dev/null +++ b/1.4.10/docs/examples/example-time-input-directive/index-debug.html @@ -0,0 +1,39 @@ + + + + + Example - example-time-input-directive-debug + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "HH:mm:ss"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-time-input-directive/index-jquery.html b/1.4.10/docs/examples/example-time-input-directive/index-jquery.html new file mode 100644 index 0000000000..c23be5ff59 --- /dev/null +++ b/1.4.10/docs/examples/example-time-input-directive/index-jquery.html @@ -0,0 +1,40 @@ + + + + + Example - example-time-input-directive-jquery + + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "HH:mm:ss"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-time-input-directive/index-production.html b/1.4.10/docs/examples/example-time-input-directive/index-production.html new file mode 100644 index 0000000000..caea811f81 --- /dev/null +++ b/1.4.10/docs/examples/example-time-input-directive/index-production.html @@ -0,0 +1,39 @@ + + + + + Example - example-time-input-directive-production + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "HH:mm:ss"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-time-input-directive/index.html b/1.4.10/docs/examples/example-time-input-directive/index.html new file mode 100644 index 0000000000..9c5767471f --- /dev/null +++ b/1.4.10/docs/examples/example-time-input-directive/index.html @@ -0,0 +1,39 @@ + + + + + Example - example-time-input-directive + + + + + + + + + +
+ + +
+ + Required! + + Not a valid date! +
+ value = {{example.value | date: "HH:mm:ss"}}
+ myForm.input.$valid = {{myForm.input.$valid}}
+ myForm.input.$error = {{myForm.input.$error}}
+ myForm.$valid = {{myForm.$valid}}
+ myForm.$error.required = {{!!myForm.$error.required}}
+
+ + \ No newline at end of file diff --git a/1.4.10/docs/examples/example-time-input-directive/manifest.json b/1.4.10/docs/examples/example-time-input-directive/manifest.json new file mode 100644 index 0000000000..fdf771023f --- /dev/null +++ b/1.4.10/docs/examples/example-time-input-directive/manifest.json @@ -0,0 +1,7 @@ +{ + "name": "example-time-input-directive", + "files": [ + "index-production.html", + "protractor.js" + ] +} \ No newline at end of file diff --git a/1.4.10/docs/examples/example-time-input-directive/protractor.js b/1.4.10/docs/examples/example-time-input-directive/protractor.js new file mode 100644 index 0000000000..dc5d940cf4 --- /dev/null +++ b/1.4.10/docs/examples/example-time-input-directive/protractor.js @@ -0,0 +1,31 @@ +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'); +}); \ No newline at end of file diff --git a/1.4.10/docs/examples/example-url-input-directive/index-debug.html b/1.4.10/docs/examples/example-url-input-directive/index-debug.html new file mode 100644 index 0000000000..e38ce2d29d --- /dev/null +++ b/1.4.10/docs/examples/example-url-input-directive/index-debug.html @@ -0,0 +1,40 @@ + + + + + Example - example-url-input-directive-debug + + + + + + + + + +
+