diff --git a/.travis.yml b/.travis.yml index db65c2d1bcef..98ed1445173b 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,7 +1,7 @@ language: node_js sudo: false node_js: - - '0.10' + - '4.4' cache: directories: @@ -15,28 +15,26 @@ branches: env: matrix: + - JOB=ci-checks - JOB=unit BROWSER_PROVIDER=saucelabs - JOB=docs-e2e BROWSER_PROVIDER=saucelabs - JOB=e2e TEST_TARGET=jqlite BROWSER_PROVIDER=saucelabs - JOB=e2e TEST_TARGET=jquery BROWSER_PROVIDER=saucelabs - - JOB=unit BROWSER_PROVIDER=browserstack - - JOB=docs-e2e BROWSER_PROVIDER=browserstack - - JOB=e2e TEST_TARGET=jqlite BROWSER_PROVIDER=browserstack - - JOB=e2e TEST_TARGET=jquery BROWSER_PROVIDER=browserstack global: + - CXX=g++-4.8 # node 4 likes the G++ v4.8 compiler - SAUCE_USERNAME=angular-ci - SAUCE_ACCESS_KEY=9b988f434ff8-fbca-8aa4-4ae3-35442987 - - BROWSER_STACK_USERNAME=VojtaJina - - BROWSER_STACK_ACCESS_KEY=QCQJ1ZpWXpBkSwEdD8ev - LOGS_DIR=/tmp/angular-build/logs - BROWSER_PROVIDER_READY_FILE=/tmp/browsersprovider-tunnel-ready -matrix: - allow_failures: - - env: "JOB=unit BROWSER_PROVIDER=browserstack" - - env: "JOB=docs-e2e BROWSER_PROVIDER=browserstack" - - env: "JOB=e2e TEST_TARGET=jqlite BROWSER_PROVIDER=browserstack" - - env: "JOB=e2e TEST_TARGET=jquery BROWSER_PROVIDER=browserstack" +# node 4 likes the G++ v4.8 compiler +# see https://docs.travis-ci.com/user/languages/javascript-with-nodejs#Node.js-v4-(or-io.js-v3)-compiler-requirements +addons: + apt: + sources: + - ubuntu-toolchain-r-test + packages: + - g++-4.8 install: # Check the size of caches @@ -46,21 +44,18 @@ install: - npm config set spin false # Log HTTP requests - npm config set loglevel http - - npm install -g npm@2.5 - # Instal npm dependecies and ensure that npm cache is not stale - - scripts/npm/install-dependencies.sh + #- npm install -g npm@2.5 + # Install npm dependencies and ensure that npm cache is not stale + - npm install before_script: - - mkdir -p $LOGS_DIR - - ./scripts/travis/start_browser_provider.sh - - npm install -g grunt-cli - - grunt package - - ./scripts/travis/wait_for_browser_provider.sh + - ./scripts/travis/before_build.sh script: - ./scripts/travis/build.sh after_script: + - ./scripts/travis/tear_down_browser_provider.sh - ./scripts/travis/print_logs.sh notifications: diff --git a/CHANGELOG.md b/CHANGELOG.md index 6144c3a0965b..c82b5b6a4d50 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,361 @@ + +# 1.4.10 benignant-oscillation (2016-03-16) + + +## Bug Fixes + +- **core:** only call `console.log` when `window.console` exists + ([beb00e44](https://github.com/angular/angular.js/commit/beb00e44de947981dbe35d5cf7a116e10ea8dc67), + [#14006](https://github.com/angular/angular.js/issues/14006), [#14007](https://github.com/angular/angular.js/issues/14007), [#14047](https://github.com/angular/angular.js/issues/14047)) +- **$animateCss:** cancel fallback timeout when animation ends normally + ([a60bbc12](https://github.com/angular/angular.js/commit/a60bbc12e8c5170e70d95f1b2c3e309b3b95cb84), + [#13787](https://github.com/angular/angular.js/issues/13787)) +- **$compile:** + - allow directives to have decorators + ([77cdc37c](https://github.com/angular/angular.js/commit/77cdc37c65491b551fcf01a18ab848a693c293d7)) + - properly denormalize templates when only one of the start/end symbols is different + ([2d44a681](https://github.com/angular/angular.js/commit/2d44a681eb912a81a8bc8e16a278c45dae91fa24), + [#13848](https://github.com/angular/angular.js/issues/13848)) + - handle boolean attributes in `@` bindings + ([2ffbfb0a](https://github.com/angular/angular.js/commit/2ffbfb0ad0647d103ff339ee4b772b62d4823bf3), + [#13767](https://github.com/angular/angular.js/issues/13767), [#13769](https://github.com/angular/angular.js/issues/13769)) +- **$parse:** + - prevent assignment on constructor properties + ([f47e2180](https://github.com/angular/angular.js/commit/f47e218006029f39b4785d820b430de3a0eebcb0), + [#13417](https://github.com/angular/angular.js/issues/13417)) + - preserve expensive checks when runnning `$eval` inside an expression + ([96d62cc0](https://github.com/angular/angular.js/commit/96d62cc0fc77248d7e3ec4aa458bac0d3e072629)) + - copy `inputs` for expressions with expensive checks + ([0b7fff30](https://github.com/angular/angular.js/commit/0b7fff303f46202bbae1ff3ca9d0e5fa76e0fc9a)) +- **$rootScope:** set no context when calling helper functions for `$watch` + ([ab5c7698](https://github.com/angular/angular.js/commit/ab5c7698bb106669ca31b5f79a95afa54d65c5f1)) +- **$route:** allow preventing a route reload + ([4bc30314](https://github.com/angular/angular.js/commit/4bc3031497447ad527356f12bd0ceee1d7d09db5), + [#9824](https://github.com/angular/angular.js/issues/9824), [#13894](https://github.com/angular/angular.js/issues/13894)) +- **$routeProvider:** properly handle optional eager path named groups + ([6a4403a1](https://github.com/angular/angular.js/commit/6a4403a11845173d6a96232f77d73aa544b182af), + [#14011](https://github.com/angular/angular.js/issues/14011)) +- **copy:** add support for copying `Blob` objects + ([863a4232](https://github.com/angular/angular.js/commit/863a4232a6faa92428df45cd54d5a519be2434de), + [#9669](https://github.com/angular/angular.js/issues/9669), [#14064](https://github.com/angular/angular.js/issues/14064)) +- **dateFilter:** follow the CLDR on pattern escape sequences + ([f476060d](https://github.com/angular/angular.js/commit/f476060de6cc016380c0343490a184543f853652), + [#12839](https://github.com/angular/angular.js/issues/12839)) +- **dateFilter, input:** fix Date parsing in IE/Edge when timezone offset contains `:` + ([571afd65](https://github.com/angular/angular.js/commit/571afd6558786d7b99e2aebd307b4a94c9f2bb87), + [#13880](https://github.com/angular/angular.js/issues/13880), [#13887](https://github.com/angular/angular.js/issues/13887)) +- **input:** re-validate when partially editing date-family inputs + ([02929f82](https://github.com/angular/angular.js/commit/02929f82f30449301ff18fea84a6396a017683b1), + [#12207](https://github.com/angular/angular.js/issues/12207), [#13886](https://github.com/angular/angular.js/issues/13886)) +- **select:** handle corner case of adding options via a custom directive + ([df6e7315](https://github.com/angular/angular.js/commit/df6e731506831a3dc7f44c9a90abe17515450b3e), + [#13874](https://github.com/angular/angular.js/issues/13874), [#13878](https://github.com/angular/angular.js/issues/13878)) +- **ngOptions:** always set the 'selected' attribute for selected options + ([f87e8288](https://github.com/angular/angular.js/commit/f87e8288fb69526fd240a66a046f5de52ed204de), + [#14115](https://github.com/angular/angular.js/issues/14115)) +- **ngAnimate:** properly cancel previously running class-based animations + ([3b27dd37](https://github.com/angular/angular.js/commit/3b27dd37a2cc8a52992784ece6b371023dadf792), + [#10156](https://github.com/angular/angular.js/issues/10156), [#13822](https://github.com/angular/angular.js/issues/13822)) +- **ngAnimateChildren:** make it compatible with `ngIf` + ([dc158e7e](https://github.com/angular/angular.js/commit/dc158e7e40624ef94c66560386522ef7e991a9ce), + [#13865](https://github.com/angular/angular.js/issues/13865), [#13876](https://github.com/angular/angular.js/issues/13876)) +- **ngMockE2E:** pass `responseType` to `$delegate` when using `passThrough` + ([947cb4d1](https://github.com/angular/angular.js/commit/947cb4d1451afa4f5090a693df5b1968dd0df70c), + [#5415](https://github.com/angular/angular.js/issues/5415), [#5783](https://github.com/angular/angular.js/issues/5783)) + + +## Features + +- **$locale:** Include original locale ID in $locale + ([e69f3550](https://github.com/angular/angular.js/commit/e69f35507e10c994708ce4f1efba7573951d1acd), + [#13390](https://github.com/angular/angular.js/issues/13390)) +- **ngAnimate:** provide ng-[event]-prepare class for structural animations + ([796f7ab4](https://github.com/angular/angular.js/commit/796f7ab41487e124b5b0c02dbf0a03bd581bf073)) + + +## Performance Improvements + +- **$compile:** avoid needless overhead when wrapping text nodes + ([946d9ae9](https://github.com/angular/angular.js/commit/946d9ae90bb31fe911ebbe1b80cd4c8af5a665c6)) +- **ngRepeat:** avoid duplicate jqLite wrappers + ([d04c38c4](https://github.com/angular/angular.js/commit/d04c38c48968db777c3ea6a177ce2ff0116df7b4)) +- **ngAnimate:** + - avoid jqLite/jQuery for upward DOM traversal + ([ab95ba65](https://github.com/angular/angular.js/commit/ab95ba65c08b38cace83de6717b7681079182b45)) + - avoid `$.fn.data` overhead with jQuery + ([86416bcb](https://github.com/angular/angular.js/commit/86416bcbee2192fa31c017163c5d856763182ade)) + + + +# 1.4.9 implicit-superannuation (2016-01-21) + + +## Bug Fixes + +- **Animation** + - ensure that animate promises resolve when the document is hidden + ([9a60408c](https://github.com/angular/angular.js/commit/9a60408c804a62a9517857bdb9a42182ab6769e3)) + - do not trigger animations if the document is hidden + ([09f6061a](https://github.com/angular/angular.js/commit/09f6061a8ee41cae4268e8d44d727d3bf52e22a9), + [#12842](https://github.com/angular/angular.js/issues/12842), [#13776](https://github.com/angular/angular.js/issues/13776)) + - only copy over the animation options once + ([2fc954d3](https://github.com/angular/angular.js/commit/2fc954d33a3a4c5d4f355be1e15a381664e02f1b), + [#13722](https://github.com/angular/angular.js/issues/13722), [#13578](https://github.com/angular/angular.js/issues/13578)) + - allow event listeners on document in IE + ([5ba4419e](https://github.com/angular/angular.js/commit/5ba4419e265ff34c6c23bf3533a3332c99c5f014), + [#13548](https://github.com/angular/angular.js/issues/13548), [#13696](https://github.com/angular/angular.js/issues/13696)) + - allow removing classes that are added by a running animation + ([6c4581fc](https://github.com/angular/angular.js/commit/6c4581fcb692b17295a41b8918c6038333e7bc3d), + [#13339](https://github.com/angular/angular.js/issues/13339), [#13380](https://github.com/angular/angular.js/issues/13380), [#13414](https://github.com/angular/angular.js/issues/13414), [#13472](https://github.com/angular/angular.js/issues/13472), [#13678](https://github.com/angular/angular.js/issues/13678)) + - do not use `event.timeStamp` anymore for time tracking + ([620a20d1](https://github.com/angular/angular.js/commit/620a20d1b3376d95f85004ffa494e36bb19a2e4d), + [#13494](https://github.com/angular/angular.js/issues/13494), [#13495](https://github.com/angular/angular.js/issues/13495)) + - ignore children without animation data when closing them + ([be01cebf](https://github.com/angular/angular.js/commit/be01cebfae9ca2383105e535820442b39a96b240), + [#11992](https://github.com/angular/angular.js/issues/11992), [#13424](https://github.com/angular/angular.js/issues/13424)) + - do not alter the provided options data + ([7a81e6fe](https://github.com/angular/angular.js/commit/7a81e6fe2db084172e34d509f0baad2b33a8722c), + [#13040](https://github.com/angular/angular.js/issues/13040), [#13175](https://github.com/angular/angular.js/issues/13175)) + - correctly handle `$animate.pin()` host elements + ([a985adfd](https://github.com/angular/angular.js/commit/a985adfdabd871f3f3f3ee59f371da50cd9611d9), + [#13783](https://github.com/angular/angular.js/issues/13783)) + - allow animations when pinned element is parent element + ([4cb8ac61](https://github.com/angular/angular.js/commit/4cb8ac61c7574ab4039852c358dd5946268b69fb), + [#13466](https://github.com/angular/angular.js/issues/13466)) + - allow enabled children to animate on disabled parents + ([6d85f24e](https://github.com/angular/angular.js/commit/6d85f24e2081d2a69c80697d90ebd45f228d9682), + [#13179](https://github.com/angular/angular.js/issues/13179), [#13695](https://github.com/angular/angular.js/issues/13695)) + - correctly access `minErr` + ([0c1b54f0](https://github.com/angular/angular.js/commit/0c1b54f04cf5bd7c1fe42ac49b4fbfdf35c60979)) + - ensure animate runner is the same with and without animations + ([937942f5](https://github.com/angular/angular.js/commit/937942f5ada6de1bdacdf0ba465f6f118c270119), + [#13205](https://github.com/angular/angular.js/issues/13205), [#13347](https://github.com/angular/angular.js/issues/13347)) + - remove animation end event listeners on close + ([d9157849](https://github.com/angular/angular.js/commit/d9157849df224a3a8d2e0bf03099d137f51499f6), + [#13672](https://github.com/angular/angular.js/issues/13672)) + - consider options.delay value for closing timeout + ([592bf516](https://github.com/angular/angular.js/commit/592bf516e50b9729e446d9aa01f4d9ebdd72d187), + [#13355](https://github.com/angular/angular.js/issues/13355), [#13363](https://github.com/angular/angular.js/issues/13363)) +- **$controller:** allow identifiers containing `$` + ([2563ff7b](https://github.com/angular/angular.js/commit/2563ff7ba92d84af978e7e4131253190d4d00c20), + [#13736](https://github.com/angular/angular.js/issues/13736)) +- **$http:** throw if url passed is not a string + ([c5bf9dae](https://github.com/angular/angular.js/commit/c5bf9daef6dfdb3e4a2942c21155a9f67d92e237), + [#12925](https://github.com/angular/angular.js/issues/12925), [#13444](https://github.com/angular/angular.js/issues/13444)) +- **$parse:** handle interceptors with `undefined` expressions + ([7bb2414b](https://github.com/angular/angular.js/commit/7bb2414bf6461aa45a983fd322ae875f81814cc4)) +- **$resource:** don't allow using promises as `timeout` and log a warning + ([47486524](https://github.com/angular/angular.js/commit/474865242c89ba3e8143f0cd52f8c292979ea730)) +- **formatNumber:** cope with large and small number corner cases + ([9c49eb13](https://github.com/angular/angular.js/commit/9c49eb131a6100d58c965d01fb08bcd319032229), + [#13394](https://github.com/angular/angular.js/issues/13394), [#8674](https://github.com/angular/angular.js/issues/8674), [#12709](https://github.com/angular/angular.js/issues/12709), [#8705](https://github.com/angular/angular.js/issues/8705), [#12707](https://github.com/angular/angular.js/issues/12707), [#10246](https://github.com/angular/angular.js/issues/10246), [#10252](https://github.com/angular/angular.js/issues/10252)) +- **input:** + - fix URL validation being too strict + ([6610ae81](https://github.com/angular/angular.js/commit/6610ae816f78ee8fc1080b93a55bf19e4ce48d3e), + [#13528](https://github.com/angular/angular.js/issues/13528), [#13544](https://github.com/angular/angular.js/issues/13544)) + - add missing chars to URL validation regex + ([2995b54a](https://github.com/angular/angular.js/commit/2995b54afdb9a3a2a81b0076a6ac0a9001041163), + [#13379](https://github.com/angular/angular.js/issues/13379), [#13460](https://github.com/angular/angular.js/issues/13460)) +- **isArrayLike:** recognize empty instances of an Array subclass + ([323f9ab7](https://github.com/angular/angular.js/commit/323f9ab73696f223c245ddefd62a769fe102615e), + [#13560](https://github.com/angular/angular.js/issues/13560), [#13708](https://github.com/angular/angular.js/issues/13708)) +- **ngInclude:** do not compile template if original scope is destroyed + ([9590bcf0](https://github.com/angular/angular.js/commit/9590bcf0620cd507a7795c55f9a6f4a48bfedbc1)) +- **ngOptions:** + - don't skip `optgroup` elements with `value === ''` + ([85e392f3](https://github.com/angular/angular.js/commit/85e392f3543ef5285c7e90e843af0ab522cb0531), + [#13487](https://github.com/angular/angular.js/issues/13487), [#13489](https://github.com/angular/angular.js/issues/13489)) + - don't `$dirty` multiple select after compilation + ([f163c905](https://github.com/angular/angular.js/commit/f163c90555774426ccb14752d089fc707cb4029c), + [#13211](https://github.com/angular/angular.js/issues/13211), [#13326](https://github.com/angular/angular.js/issues/13326)) +- **select:** re-define `ngModelCtrl.$render` in the `select` directive's postLink function + ([529b2507](https://github.com/angular/angular.js/commit/529b2507bdb4fcc22dfa0f7ab462c79fc78d1413), + [#13583](https://github.com/angular/angular.js/issues/13583), [#13583](https://github.com/angular/angular.js/issues/13583), [#13663](https://github.com/angular/angular.js/issues/13663)) + +## Minor Features + +- **ngLocale:** add support for standalone months + ([54c4041e](https://github.com/angular/angular.js/commit/54c4041ebc0cc4df70cf6996f43a6aaaf56d46bd), + [#3744](https://github.com/angular/angular.js/issues/3744), [#10247](https://github.com/angular/angular.js/issues/10247), [#12642](https://github.com/angular/angular.js/issues/12642), [#12844](https://github.com/angular/angular.js/issues/12844)) +- **ngMock:** add support for `$animate.closeAndFlush()` + ([512c0811](https://github.com/angular/angular.js/commit/512c08118786a419fabbd063fa17d224aba125cf)) + + +## Performance Improvements + +- **ngAnimate:** speed up `areAnimationsAllowed` check + ([2d3303dd](https://github.com/angular/angular.js/commit/2d3303ddda6330c4f45b381b6b17346f6cfe2d97)) + + +## Breaking Changes + +While we do not deem the following to be a real breaking change we are highlighting it here in the +changelog to ensure that it does not surprise anyone. + +- **$resource:** due to [47486524](https://github.com/angular/angular.js/commit/474865242c89ba3e8143f0cd52f8c292979ea730), + +**Possible breaking change** for users who updated their code to provide a `timeout` +promise for a `$resource` request in version 1.4.8. + +Up to v1.4.7 (included), using a promise as a timeout in `$resource`, would silently +fail (i.e. have no effect). + +In v1.4.8, using a promise as timeout would have the (buggy) behaviour described +in https://github.com/angular/angular.js/pull/12657#issuecomment-152108887 +(i.e. it will work as expected for the first time you resolve the promise and will +cancel all subsequent requests after that - one has to re-create the resource +class. This is feature was not documented.) + +With this change, using a promise as timeout in 1.4.9 onwsards is not allowed. +It will log a warning and ignore the timeout value. + +If you need support for cancellable `$resource` actions, you should upgrade to +version 1.5 or higher. + + + +# 1.4.8 ice-manipulation (2015-11-19) + + +## Bug Fixes + +- **$animate:** ensure leave animation calls `close` callback + ([6bd6dbff](https://github.com/angular/angular.js/commit/6bd6dbff4961a601c03e9465442788781d329ba6), + [#12278](https://github.com/angular/angular.js/issues/12278), [#12096](https://github.com/angular/angular.js/issues/12096), [#13054](https://github.com/angular/angular.js/issues/13054)) +- **$cacheFactory:** check key exists before decreasing cache size count + ([2a5a52a7](https://github.com/angular/angular.js/commit/2a5a52a76ccf60c6e8c5d881e90e11a2666a6d3c), + [#12321](https://github.com/angular/angular.js/issues/12321), [#12329](https://github.com/angular/angular.js/issues/12329)) +- **$compile:** + - bind all directive controllers correctly when using `bindToController` + ([5d8861fb](https://github.com/angular/angular.js/commit/5d8861fb2f203e8a688b6044cbd1140cd79fd049), + [#11343](https://github.com/angular/angular.js/issues/11343), [#11345](https://github.com/angular/angular.js/issues/11345)) + - evaluate against the correct scope with bindToController on new scope + ([b9f7c453](https://github.com/angular/angular.js/commit/b9f7c453e00d6938106f414952f74d5e5fdcb993), + [#13021](https://github.com/angular/angular.js/issues/13021), [#13025](https://github.com/angular/angular.js/issues/13025)) + - fix scoping of transclusion directives inside replace directive + ([74da0340](https://github.com/angular/angular.js/commit/74da03407782d679951cd8f693860cea214f2580), + [#12975](https://github.com/angular/angular.js/issues/12975), [#12936](https://github.com/angular/angular.js/issues/12936), [#13244](https://github.com/angular/angular.js/issues/13244)) +- **$http:** apply `transformResponse` even when `data` is empty + ([c6909464](https://github.com/angular/angular.js/commit/c690946469e09cfe6b774e63dbe14ace92ce6cb7), + [#12976](https://github.com/angular/angular.js/issues/12976), [#12979](https://github.com/angular/angular.js/issues/12979)) +- **$location:** ensure `$locationChangeSuccess` fires even if URL ends with `#` + ([6f8ddb6d](https://github.com/angular/angular.js/commit/6f8ddb6d4329441e8d4a856978413aa9b9bd918f), + [#12175](https://github.com/angular/angular.js/issues/12175), [#13251](https://github.com/angular/angular.js/issues/13251)) +- **$parse:** evaluate once simple expressions only once + ([e4036824](https://github.com/angular/angular.js/commit/e403682444fa08af4f3491badf2f3a10d7595699), + [#12983](https://github.com/angular/angular.js/issues/12983), [#13002](https://github.com/angular/angular.js/issues/13002)) +- **$resource:** allow XHR request to be cancelled via a timeout promise + ([7170f9d9](https://github.com/angular/angular.js/commit/7170f9d9ca765c578f8d3eb4699860a9330a0a11), + [#12657](https://github.com/angular/angular.js/issues/12657), [#12675](https://github.com/angular/angular.js/issues/12675), [#10890](https://github.com/angular/angular.js/issues/10890), [#9332](https://github.com/angular/angular.js/issues/9332)) +- **$rootScope:** prevent IE9 memory leak when destroying scopes + ([87b0055c](https://github.com/angular/angular.js/commit/87b0055c80f40589c5bcf3765e59e872bcfae119), + [#10706](https://github.com/angular/angular.js/issues/10706), [#11786](https://github.com/angular/angular.js/issues/11786)) +- **Angular.js:** fix `isArrayLike` for unusual cases + ([70edec94](https://github.com/angular/angular.js/commit/70edec947c7b189694ae66b129568182e3369cab), + [#10186](https://github.com/angular/angular.js/issues/10186), [#8000](https://github.com/angular/angular.js/issues/8000), [#4855](https://github.com/angular/angular.js/issues/4855), [#4751](https://github.com/angular/angular.js/issues/4751), [#10272](https://github.com/angular/angular.js/issues/10272)) +- **isArrayLike:** handle jQuery objects of length 0 + ([d3da55c4](https://github.com/angular/angular.js/commit/d3da55c40f1e1ddceced5da51e364888ff9d82ff)) +- **jqLite:** + - deregister special `mouseenter` / `mouseleave` events correctly + ([22f66025](https://github.com/angular/angular.js/commit/22f66025db262417ebb78c1ce1f4d7058dca3fd3), + [#12795](https://github.com/angular/angular.js/issues/12795), [#12799](https://github.com/angular/angular.js/issues/12799)) + - ensure mouseenter works with svg elements on IE + ([c1f34e8e](https://github.com/angular/angular.js/commit/c1f34e8eeb5105767f6cbf4727b8c5664be2a261), + [#10259](https://github.com/angular/angular.js/issues/10259), [#10276](https://github.com/angular/angular.js/issues/10276)) +- **limitTo:** start at 0 if `begin` is negative and exceeds input length + ([4fc40bc9](https://github.com/angular/angular.js/commit/4fc40bc9320a1d5902e648b70fa79c7cf7e794c7), + [#12775](https://github.com/angular/angular.js/issues/12775), [#12781](https://github.com/angular/angular.js/issues/12781)) +- **merge:** + - ensure that jqlite->jqlite and DOM->DOM + ([2f8db1bf](https://github.com/angular/angular.js/commit/2f8db1bf01173b546a2868fc7b8b188c2383fbff)) + - clone elements instead of treating them like simple objects + ([838cf4be](https://github.com/angular/angular.js/commit/838cf4be3c671903796dbb69d95c0e5ac1516a06), + [#12286](https://github.com/angular/angular.js/issues/12286)) +- **ngAria:** don't add tabindex to radio and checkbox inputs + ([59f1f4e1](https://github.com/angular/angular.js/commit/59f1f4e19a02e6e6f4c41c15b0e9f3372d85cecc), + [#12492](https://github.com/angular/angular.js/issues/12492), [#13095](https://github.com/angular/angular.js/issues/13095)) +- **ngInput:** change URL_REGEXP to better match RFC3987 + ([cb51116d](https://github.com/angular/angular.js/commit/cb51116dbd225ccfdbc9a565a66a170e65d26331), + [#11341](https://github.com/angular/angular.js/issues/11341), [#11381](https://github.com/angular/angular.js/issues/11381)) +- **ngMock:** reset cache before every test + ([91b7cd9b](https://github.com/angular/angular.js/commit/91b7cd9b74d72a48d844c5c3e0e9dee03405e0ca), + [#13013](https://github.com/angular/angular.js/issues/13013)) +- **ngOptions:** + - skip comments and empty options when looking for options + ([0f58334b](https://github.com/angular/angular.js/commit/0f58334b7b9a9d3d6ff34e9754961b6f67731fae), + [#12190](https://github.com/angular/angular.js/issues/12190), [#13029](https://github.com/angular/angular.js/issues/13029), [#13033](https://github.com/angular/angular.js/issues/13033)) + - override select option registration to allow compilation of empty option + ([7b2ecf42](https://github.com/angular/angular.js/commit/7b2ecf42c697eb8d51a0f2d73b324bd900139e05), + [#11685](https://github.com/angular/angular.js/issues/11685), [#12972](https://github.com/angular/angular.js/issues/12972), [#12968](https://github.com/angular/angular.js/issues/12968), [#13012](https://github.com/angular/angular.js/issues/13012)) + + +## Performance Improvements + +- **$compile:** use static jquery data method to avoid creating new instances + ([55ad192e](https://github.com/angular/angular.js/commit/55ad192e4ab79295ab15ecaaf8f6b9e7932a0336)) +- **copy:** + - avoid regex in `isTypedArray` + ([19fab4a1](https://github.com/angular/angular.js/commit/19fab4a1d79d2445795273f1622344353cf4d104)) + - only validate/clear if the user specifies a destination + ([d1293540](https://github.com/angular/angular.js/commit/d1293540e13573eb9ea5f90730bb9c9710c345db), + [#12068](https://github.com/angular/angular.js/issues/12068)) +- **merge:** remove unnecessary wrapping of jqLite element + ([ce6a96b0](https://github.com/angular/angular.js/commit/ce6a96b0d76dd2e5ab2247ca3059d284575bc6f0), + [#13236](https://github.com/angular/angular.js/issues/13236)) + + +## Breaking Changes + + + +# 1.4.7 dark-luminescence (2015-09-29) + + +## Bug Fixes + +- **$compile:** use createMap() for $$observe listeners when initialized from attr interpolation + ([5a98e806](https://github.com/angular/angular.js/commit/5a98e806ef3c59916bb4668268125610b11effe8), + [#10446](https://github.com/angular/angular.js/issues/10446)) +- **$parse:** + - block assigning to fields of a constructor + ([a7f3761e](https://github.com/angular/angular.js/commit/a7f3761eda5309f76b73c6fb1d3173a270112899), + [#12860](https://github.com/angular/angular.js/issues/12860)) + - do not convert to string computed properties multiple times + ([698af191](https://github.com/angular/angular.js/commit/698af191ded2465ca4e0f97959b75fede5a531ab)) +- **filters:** ensure `formatNumber` observes i18n decimal separators + ([4994acd2](https://github.com/angular/angular.js/commit/4994acd26e582eec8a92b139bfc09ca79a9b8835), + [#10342](https://github.com/angular/angular.js/issues/10342), [#12850](https://github.com/angular/angular.js/issues/12850)) +- **jqLite:** properly handle dash-delimited node names in `jqLiteBuildFragment` + ([cdd1227a](https://github.com/angular/angular.js/commit/cdd1227a308edd34d31b67f338083b6e0c4c0db9), + [#10617](https://github.com/angular/angular.js/issues/10617), [#12759](https://github.com/angular/angular.js/issues/12759)) +- **ngAnimate:** + - ensure anchoring uses body as a container when needed + ([9d3704ca](https://github.com/angular/angular.js/commit/9d3704ca467081f16b71b011eb50c53d5cdb2f34), + [#12872](https://github.com/angular/angular.js/issues/12872)) + - callback detection should only use RAF when necessary + ([fa8c399f](https://github.com/angular/angular.js/commit/fa8c399fadc30b78710868fe59d2930fdc17c7a5)) +- **ngMessages:** prevent race condition with ngAnimate + ([7295c60f](https://github.com/angular/angular.js/commit/7295c60ffb9f2e4f32043c538ace740b187f565a), + [#12856](https://github.com/angular/angular.js/issues/12856), [#12903](https://github.com/angular/angular.js/issues/12903)) +- **ngOptions:** + - prevent frozen select ui in IE + ([dbc69851](https://github.com/angular/angular.js/commit/dbc698517ff620b3a6279f65d4a9b6e3c15087b9), + [#11314](https://github.com/angular/angular.js/issues/11314), [#11795](https://github.com/angular/angular.js/issues/11795)) + + +## Features + +- **$animateCss:** add support for temporary styles via `cleanupStyles` + ([e52d731b](https://github.com/angular/angular.js/commit/e52d731bfd1fbb6c616125fbde2fb365722254b7), + [#12930](https://github.com/angular/angular.js/issues/12930)) +- **$http:** add `$xhrFactory` service to enable creation of custom xhr objects + ([7a413df5](https://github.com/angular/angular.js/commit/7a413df5e47e04e20a1c93d35922050bbcbfb492), + [#2318](https://github.com/angular/angular.js/issues/2318), [#9319](https://github.com/angular/angular.js/issues/9319), [#12159](https://github.com/angular/angular.js/issues/12159)) + + +## Breaking Changes + + # 1.4.6 multiplicative-elevation (2015-09-17) @@ -170,8 +528,6 @@ the built-in pattern validator: ``` - - # 1.4.5 permanent-internship (2015-08-28) @@ -463,6 +819,43 @@ describe('$q.when', function() { }); ``` +- **form:** Due to [94533e57](https://github.com/angular/angular.js/commit/94533e570673e6b2eb92073955541fa289aabe02), + the `name` attribute of `form` elements can now only contain characters that can be evaluated as part + of an Angular expression. This is because Angular uses the value of `name` as an assignable expression + to set the form on the `$scope`. For example, `name="myForm"` assigns the form to `$scope.myForm` and + `name="myObj.myForm"` assigns it to `$scope.myObj.myForm`. + + Previously, it was possible to also use names such `name="my:name"`, because Angular used a special setter + function for the form name. Now the general, more robust `$parse` setter is used. + + The easiest way to migrate your code is therefore to remove all special characters from the `name` attribute. + + If you need to keep the special characters, you can use the following directive, which will replace + the `name` with a value that can be evaluated as an expression in the compile function, and then + re-set the original name in the postLink function. This ensures that (1), the form is published on + the scope, and (2), the form has the original name, which might be important if you are doing server-side + form submission. + + ```js + angular.module('myApp').directive('form', function() { + return { + restrict: 'E', + priority: 1000, + compile: function(element, attrs) { + var unsupportedCharacter = ':'; // change accordingly + var originalName = attrs.name; + if (attrs.name && attrs.name.indexOf(unsupportedCharacter) > 0) { + attrs.$set('name', 'this["' + originalName + '"]'); + } + + return postLinkFunction(scope, element) { + // Don't trigger $observers + element.setAttribute('name', originalName); + } + } + }; + }); + ``` # 1.4.3 foam-acceleration (2015-07-15) @@ -1340,7 +1733,6 @@ mechanism. - **ngMessages:** due to [c9a4421f](https://github.com/angular/angular.js/commit/c9a4421fc3c97448527eadef1f42eb2f487ec2e0), - The `ngMessagesInclude` attribute is now its own directive and that must be placed as a **child** element within the element with the ngMessages directive. (Keep in mind that the former behaviour of the @@ -1363,6 +1755,26 @@ end of the container containing the ngMessages directive). ``` +- **ngMessages:** due to [c9a4421f](https://github.com/angular/angular.js/commit/c9a4421fc3c97448527eadef1f42eb2f487ec2e0), + +it is no longer possible to use interpolation inside the `ngMessages` attribute expression. This technique +is generally not recommended, and can easily break when a directive implementation changes. In cases +where a simple expression is not possible, you can delegate accessing the object to a function: + +```html +
...
+``` +would become +```html +
...
+``` +where `ctrl.getMessages()` +```javascript +ctrl.getMessages = function($index) { + return ctrl.form['field_' + $index].$error; +} +``` + - **$http:** due to [5da1256](https://github.com/angular/angular.js/commit/5da1256fc2812d5b28fb0af0de81256054856369), `transformRequest` functions can no longer modify request headers. @@ -1818,6 +2230,12 @@ But in practice this is not what people want and so this change iterates over pr in the order they are returned by Object.keys(obj), which is almost always the order in which the properties were defined. +- **ngOptions:** due to [7fda214c](https://github.com/angular/angular.js/commit/7fda214c4f65a6a06b25cf5d5aff013a364e9cef), + +setting the ngOptions attribute expression after the element is compiled, will no longer trigger the ngOptions behavior. +This worked previously because the ngOptions logic was part of the select directive, while +it is now implemented in the ngOptions directive itself. + - **select:** due to [7fda214c](https://github.com/angular/angular.js/commit/7fda214c4f65a6a06b25cf5d5aff013a364e9cef), the `select` directive will now use strict comparison of the `ngModel` scope value against `option` @@ -2420,7 +2838,36 @@ We also added a documentation page focused on security, which contains some of t [#9578](https://github.com/angular/angular.js/issues/9578), [#9751](https://github.com/angular/angular.js/issues/9751)) +## Breaking Changes +- **$observe:** Due to [531a8de7](https://github.com/angular/angular.js/commit/531a8de72c439d8ddd064874bf364c00cedabb11), +observers no longer register on undefined attributes. For example, if you were using `$observe` on +an absent optional attribute to set a default value, the following would not work anymore: + +```html + +``` + +```js +// link function for directive myDir +link: function(scope, element, attr) { + attr.$observe('myAttr', function(newVal) { + scope.myValue = newVal ? newVal : 'myDefaultValue'; + }) +} +``` + +Instead, check if the attribute is set before registering the observer: + +```js +link: function(scope, element, attr) { + if (attr.myAttr) { + // register the observer + } else { + // set the default + } +} +``` # 1.3.0 superluminal-nudge (2014-10-13) diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 4a93f7090811..57c71041c7c0 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -71,7 +71,7 @@ chances of your issue being dealt with quickly: * **Angular Version(s)** - is it a regression? * **Browsers and Operating System** - is this a problem with all browsers or only IE8? * **Reproduce the Error** - provide a live example (using [Plunker][plunker] or - [JSFiddle][jsfiddle]) or a unambiguous set of steps. + [JSFiddle][jsfiddle]) or an unambiguous set of steps. * **Related Issues** - has a similar issue been reported before? * **Suggest a Fix** - if you can't fix the bug yourself, perhaps you can point to what might be causing the problem (line of code or commit) @@ -123,13 +123,19 @@ Before you submit your pull request consider the following guidelines: * If we suggest changes then: * Make the required updates. * Re-run the Angular test suite to ensure tests are still passing. - * Rebase your branch and force push to your GitHub repository (this will update your Pull Request): + * Commit your changes to your branch (e.g. `my-fix-branch`). + * Push the changes to your GitHub repository (this will update your Pull Request). + +If the PR gets too outdated we may ask you to rebase and force push to update the PR: ```shell git rebase master -i git push origin my-fix-branch -f ``` +*WARNING. Squashing or reverting commits and forced push thereafter may remove GitHub comments +on code that were previously made by you and others in your commits.* + That's it! Thank you for your contribution! #### After your pull request is merged diff --git a/Gruntfile.js b/Gruntfile.js index 53aaf59a9ad6..86139b9d03e9 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -162,7 +162,7 @@ module.exports = function(grunt) { '!src/angular.bind.js' // we ignore this file since contains an early return statement ], options: { - config: ".jscsrc" + config: '.jscsrc' } }, @@ -231,9 +231,9 @@ module.exports = function(grunt) { dest: 'build/angular-aria.js', src: util.wrap(files['angularModules']['ngAria'], 'module') }, - "promises-aplus-adapter": { + 'promises-aplus-adapter': { dest:'tmp/promises-aplus-adapter++.js', - src:['src/ng/q.js','lib/promises-aplus/promises-aplus-test-adapter.js'] + src:['src/ng/q.js', 'lib/promises-aplus/promises-aplus-test-adapter.js'] } }, @@ -253,7 +253,7 @@ module.exports = function(grunt) { }, - "ddescribe-iit": { + 'ddescribe-iit': { files: [ 'src/**/*.js', 'test/**/*.js', @@ -274,7 +274,7 @@ module.exports = function(grunt) { } }, - "merge-conflict": { + 'merge-conflict': { files: [ 'src/**/*', 'test/**/*', @@ -304,11 +304,11 @@ module.exports = function(grunt) { }, shell: { - "npm-install": { - command: path.normalize('scripts/npm/install-dependencies.sh') + 'npm-install': { + command: 'node scripts/npm/check-node-modules.js' }, - "promises-aplus-tests": { + 'promises-aplus-tests': { options: { stdout: false, stderr: true, @@ -339,8 +339,10 @@ module.exports = function(grunt) { grunt.task.run('shell:npm-install'); } + + //alias tasks - grunt.registerTask('test', 'Run unit, docs and e2e tests with Karma', ['jshint', 'jscs', 'package','test:unit','test:promises-aplus', 'tests:docs', 'test:protractor']); + grunt.registerTask('test', 'Run unit, docs and e2e tests with Karma', ['jshint', 'jscs', 'package', 'test:unit', 'test:promises-aplus', 'tests:docs', 'test:protractor']); grunt.registerTask('test:jqlite', 'Run the unit tests with Karma' , ['tests:jqlite']); grunt.registerTask('test:jquery', 'Run the jQuery unit tests with Karma', ['tests:jquery']); grunt.registerTask('test:modules', 'Run the Karma module tests with Karma', ['build', 'tests:modules']); @@ -350,11 +352,11 @@ module.exports = function(grunt) { grunt.registerTask('test:travis-protractor', 'Run the end to end tests with Protractor for Travis CI builds', ['connect:testserver', 'protractor:travis']); grunt.registerTask('test:ci-protractor', 'Run the end to end tests with Protractor for Jenkins CI builds', ['webdriver', 'connect:testserver', 'protractor:jenkins']); grunt.registerTask('test:e2e', 'Alias for test:protractor', ['test:protractor']); - grunt.registerTask('test:promises-aplus',['build:promises-aplus-adapter','shell:promises-aplus-tests']); + grunt.registerTask('test:promises-aplus',['build:promises-aplus-adapter', 'shell:promises-aplus-tests']); - grunt.registerTask('minify', ['bower','clean', 'build', 'minall']); + grunt.registerTask('minify', ['bower', 'clean', 'build', 'minall']); grunt.registerTask('webserver', ['connect:devserver']); - grunt.registerTask('package', ['bower','clean', 'buildall', 'minall', 'collect-errors', 'docs', 'copy', 'write', 'compress']); + grunt.registerTask('package', ['bower', 'validate-angular-files', 'clean', 'buildall', 'minall', 'collect-errors', 'docs', 'copy', 'write', 'compress']); grunt.registerTask('ci-checks', ['ddescribe-iit', 'merge-conflict', 'jshint', 'jscs']); grunt.registerTask('default', ['package']); }; diff --git a/README.md b/README.md index aac26f18c667..e13ed1145290 100644 --- a/README.md +++ b/README.md @@ -18,10 +18,10 @@ piece of cake. Best of all?? It makes development fun! * Developer Guide: http://docs.angularjs.org/guide * Contribution guidelines: [CONTRIBUTING.md](https://github.com/angular/angular.js/blob/master/CONTRIBUTING.md) * Dashboard: http://dashboard.angularjs.org - + Building AngularJS --------- -[Once you have your environment setup](http://docs.angularjs.org/misc/contribute) just run: +[Once you have your environment set up](http://docs.angularjs.org/misc/contribute) just run: grunt package @@ -43,3 +43,26 @@ To learn more about the grunt tasks, run `grunt --help` and also read our [![Analytics](https://ga-beacon.appspot.com/UA-8594346-11/angular.js/README.md?pixel)](https://github.com/igrigorik/ga-beacon) +What to Use AngularJS for and When to Use it +--------- +AngularJS is the next generation framework where each component is designed to work with every other component in an interconnected way like a well-oiled machine. AngularJS is JavaScript MVC made easy and done right. (Well it is not really MVC, read on, to understand what this means.) + +#### MVC, no, MV* done the right way! +MVC, short for Model-View-Controller, is a design pattern, i.e. how the code should be organized and how the different parts of an application separated for proper readability and debugging. Model is the data and the database. View is the user interface and what the user sees. Controller is the main link between Model and View. These are the three pillars of major programming frameworks present on the market today. On the other hand AngularJS works on MV*, short for Model-View-_Whatever_. The _Whatever_ is AngularJS's way of telling that you may create any kind of linking between the Model and the View here. + +Unlike other frameworks in any programming language, where MVC, the three separate components, each one has to be written and then connected by the programmer, AngularJS helps the programmer by asking him/her to just create these and everything else will be taken care of by AngularJS. + +#### Interconnection with HTML at the root level +AngularJS uses HTML to define the user's interface. AngularJS also enables the programmer to write new HTML tags (AngularJS Directives) and increase the readability and understandability of the HTML code. Directives are AngularJS’s way of bringing additional functionality to HTML. Directives achieve this by enabling us to invent our own HTML elements. This also helps in making the code DRY (Don't Repeat Yourself), which means once created, a new directive can be used anywhere within the application. + +#### Data Handling made simple +Data and Data Models in AngularJS are plain JavaScript objects and one can add and change properties directly on it and loop over objects and arrays at will. + +#### Two-way Data Binding +One of AngularJS's strongest features. Two-way Data Binding means that if something changes in the Model, the change gets reflected in the View instantaneously, and the same happens the other way around. This is also referred to as Reactive Programming, i.e. suppose `a = b + c` is being programmed and after this, if the value of `b` and/or `c` is changed then the value of `a` will be automatically updated to reflect the change. AngularJS uses its "scopes" as a glue between the Model and View and makes these updates in one available for the other. + +#### Less Written Code and Easily Maintable Code +Everything in AngularJS is created to enable the programmer ends up writing less code that is easily maintainable and readable by any other new person on the team. Believe it or not, one can write a complete working two-way data binded application in less than 10 lines of code. Try and see for yourself! + +#### Testing Ready +AngularJS has Dependency Injection, i.e. it takes care of providing all the necessary dependencies to its controllers whenever required. This helps in making the AngularJS code ready for unit testing by making use of mock dependencies created and injected. This makes AngularJS more modular and easily testable thus in turn helping a team create more robust applications. diff --git a/angularFiles.js b/angularFiles.js index 9d18fd831b6c..5cdc5bf5c74f 100755 --- a/angularFiles.js +++ b/angularFiles.js @@ -14,6 +14,7 @@ var angularFiles = { 'src/ng/anchorScroll.js', 'src/ng/animate.js', + 'src/ng/animateRunner.js', 'src/ng/animateCss.js', 'src/ng/browser.js', 'src/ng/cacheFactory.js', @@ -33,6 +34,7 @@ var angularFiles = { 'src/ng/q.js', 'src/ng/raf.js', 'src/ng/rootScope.js', + 'src/ng/rootElement.js', 'src/ng/sanitizeUri.js', 'src/ng/sce.js', 'src/ng/sniffer.js', @@ -84,7 +86,7 @@ var angularFiles = { ], 'angularLoader': [ - 'stringify.js', + 'src/stringify.js', 'src/minErr.js', 'src/loader.js' ], @@ -92,7 +94,6 @@ var angularFiles = { 'angularModules': { 'ngAnimate': [ 'src/ngAnimate/shared.js', - 'src/ngAnimate/body.js', 'src/ngAnimate/rafScheduler.js', 'src/ngAnimate/animateChildrenDirective.js', 'src/ngAnimate/animateCss.js', @@ -100,7 +101,6 @@ var angularFiles = { 'src/ngAnimate/animateJs.js', 'src/ngAnimate/animateJsDriver.js', 'src/ngAnimate/animateQueue.js', - 'src/ngAnimate/animateRunner.js', 'src/ngAnimate/animation.js', 'src/ngAnimate/module.js' ], diff --git a/bower.json b/bower.json index e1f774b6f6a6..ec7cfbbadcce 100644 --- a/bower.json +++ b/bower.json @@ -1,5 +1,6 @@ { "name": "AngularJS", + "license": "MIT", "devDependencies": { "jquery": "2.1.1", "closure-compiler": "https://dl.google.com/closure-compiler/compiler-20140814.zip", diff --git a/docs/app/assets/css/docs.css b/docs/app/assets/css/docs.css index 0bd14d669af8..89a15c49285a 100644 --- a/docs/app/assets/css/docs.css +++ b/docs/app/assets/css/docs.css @@ -124,7 +124,7 @@ h1,h2,h3,h4,h5,h6 { font-size:1.2em; padding:0; margin:0; - border-bottom:1px soild #aaa; + border-bottom:1px solid #aaa; margin-bottom:5px; } @@ -589,6 +589,12 @@ ul.events > li { 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; diff --git a/docs/app/assets/js/angular-bootstrap/bootstrap.js b/docs/app/assets/js/angular-bootstrap/bootstrap.js deleted file mode 100644 index 58aedde697e6..000000000000 --- a/docs/app/assets/js/angular-bootstrap/bootstrap.js +++ /dev/null @@ -1,442 +0,0 @@ -'use strict'; - -var directive = {}; - -directive.runnableExample = ['$templateCache', '$document', function($templateCache, $document) { - var exampleClassNameSelector = '.runnable-example-file'; - var doc = $document[0]; - var tpl = - ''; - - return { - restrict: 'C', - scope : true, - controller : ['$scope', function($scope) { - $scope.setTab = function(index) { - var tab = $scope.tabs[index]; - $scope.activeTabIndex = index; - $scope.$broadcast('tabChange', index, tab); - }; - }], - compile : function(element) { - element.html(tpl + element.html()); - return function(scope, element) { - var node = element[0]; - var examples = node.querySelectorAll(exampleClassNameSelector); - var tabs = [], now = Date.now(); - angular.forEach(examples, function(child, index) { - tabs.push(child.getAttribute('name')); - }); - - if(tabs.length > 0) { - scope.tabs = tabs; - scope.$on('tabChange', function(e, index, title) { - angular.forEach(examples, function(child) { - child.style.display = 'none'; - }); - var selected = examples[index]; - selected.style.display = 'block'; - }); - scope.setTab(0); - } - } - } - }; -}]; - -directive.dropdownToggle = - ['$document', '$location', '$window', - function ($document, $location, $window) { - var openElement = null, close; - return { - restrict: 'C', - link: function(scope, element, attrs) { - scope.$watch(function dropdownTogglePathWatch(){return $location.path();}, function dropdownTogglePathWatchAction() { - close && close(); - }); - - element.parent().on('click', function(event) { - close && close(); - }); - - element.on('click', function(event) { - event.preventDefault(); - event.stopPropagation(); - - var iWasOpen = false; - - if (openElement) { - iWasOpen = openElement === element; - close(); - } - - if (!iWasOpen){ - element.parent().addClass('open'); - openElement = element; - - close = function (event) { - event && event.preventDefault(); - event && event.stopPropagation(); - $document.off('click', close); - element.parent().removeClass('open'); - close = null; - openElement = null; - } - - $document.on('click', close); - } - }); - } - }; - }]; - -directive.syntax = function() { - return { - restrict: 'A', - link: function(scope, element, attrs) { - function makeLink(type, text, link, icon) { - return '' + - ' ' + text + - ''; - }; - - var html = ''; - var types = { - 'github' : { - text : 'View on Github', - key : 'syntaxGithub', - icon : 'icon-github' - }, - 'plunkr' : { - text : 'View on Plunkr', - key : 'syntaxPlunkr', - icon : 'icon-arrow-down' - }, - 'jsfiddle' : { - text : 'View on JSFiddle', - key : 'syntaxFiddle', - icon : 'icon-cloud' - } - }; - for(var type in types) { - var data = types[type]; - var link = attrs[data.key]; - if(link) { - html += makeLink(type, data.text, link, data.icon); - } - }; - - var nav = document.createElement('nav'); - nav.className = 'syntax-links'; - nav.innerHTML = html; - - var node = element[0]; - var par = node.parentNode; - par.insertBefore(nav, node); - } - } -} - -directive.tabbable = function() { - return { - restrict: 'C', - compile: function(element) { - var navTabs = angular.element(''), - tabContent = angular.element('
'); - - tabContent.append(element.contents()); - element.append(navTabs).append(tabContent); - }, - controller: ['$scope', '$element', function($scope, $element) { - var navTabs = $element.contents().eq(0), - ngModel = $element.controller('ngModel') || {}, - tabs = [], - selectedTab; - - ngModel.$render = function() { - var $viewValue = this.$viewValue; - - if (selectedTab ? (selectedTab.value != $viewValue) : $viewValue) { - if(selectedTab) { - selectedTab.paneElement.removeClass('active'); - selectedTab.tabElement.removeClass('active'); - selectedTab = null; - } - if($viewValue) { - for(var i = 0, ii = tabs.length; i < ii; i++) { - if ($viewValue == tabs[i].value) { - selectedTab = tabs[i]; - break; - } - } - if (selectedTab) { - selectedTab.paneElement.addClass('active'); - selectedTab.tabElement.addClass('active'); - } - } - - } - }; - - this.addPane = function(element, attr) { - var li = angular.element('
  • '), - a = li.find('a'), - tab = { - paneElement: element, - paneAttrs: attr, - tabElement: li - }; - - tabs.push(tab); - - attr.$observe('value', update)(); - attr.$observe('title', function(){ update(); a.text(tab.title); })(); - - function update() { - tab.title = attr.title; - tab.value = attr.value || attr.title; - if (!ngModel.$setViewValue && (!ngModel.$viewValue || tab == selectedTab)) { - // we are not part of angular - ngModel.$viewValue = tab.value; - } - ngModel.$render(); - } - - navTabs.append(li); - li.on('click', function(event) { - event.preventDefault(); - event.stopPropagation(); - if (ngModel.$setViewValue) { - $scope.$apply(function() { - ngModel.$setViewValue(tab.value); - ngModel.$render(); - }); - } else { - // we are not part of angular - ngModel.$viewValue = tab.value; - ngModel.$render(); - } - }); - - return function() { - tab.tabElement.remove(); - for(var i = 0, ii = tabs.length; i < ii; i++ ) { - if (tab == tabs[i]) { - tabs.splice(i, 1); - } - } - }; - } - }] - }; -}; - -directive.table = function() { - return { - restrict: 'E', - link: function(scope, element, attrs) { - if (!attrs['class']) { - element.addClass('table table-bordered table-striped code-table'); - } - } - }; -}; - -var popoverElement = function() { - var object = { - init : function() { - this.element = angular.element( - '
    ' + - '
    ' + - '
    ' + - '
    ' + - '
    ' + - '
    ' + - '
    ' - ); - this.node = this.element[0]; - this.element.css({ - 'display':'block', - 'position':'absolute' - }); - angular.element(document.body).append(this.element); - - var inner = this.element.children()[1]; - this.titleElement = angular.element(inner.childNodes[0].firstChild); - this.contentElement = angular.element(inner.childNodes[1]); - - //stop the click on the tooltip - this.element.on('click', function(event) { - event.preventDefault(); - event.stopPropagation(); - }); - - var self = this; - angular.element(document.body).on('click',function(event) { - if(self.visible()) self.hide(); - }); - }, - - show : function(x,y) { - this.element.addClass('visible'); - this.position(x || 0, y || 0); - }, - - hide : function() { - this.element.removeClass('visible'); - this.position(-9999,-9999); - }, - - visible : function() { - return this.position().y >= 0; - }, - - isSituatedAt : function(element) { - return this.besideElement ? element[0] == this.besideElement[0] : false; - }, - - title : function(value) { - return this.titleElement.html(value); - }, - - content : function(value) { - if(value && value.length > 0) { - value = marked(value); - } - return this.contentElement.html(value); - }, - - positionArrow : function(position) { - this.node.className = 'popover ' + position; - }, - - positionAway : function() { - this.besideElement = null; - this.hide(); - }, - - positionBeside : function(element) { - this.besideElement = element; - - var elm = element[0]; - var x = elm.offsetLeft; - var y = elm.offsetTop; - x -= 30; - y -= this.node.offsetHeight + 10; - this.show(x,y); - }, - - position : function(x,y) { - if(x != null && y != null) { - this.element.css('left',x + 'px'); - this.element.css('top', y + 'px'); - } - else { - return { - x : this.node.offsetLeft, - y : this.node.offsetTop - }; - } - } - }; - - object.init(); - object.hide(); - - return object; -}; - -directive.popover = ['popoverElement', function(popover) { - return { - restrict: 'A', - priority : 500, - link: function(scope, element, attrs) { - element.on('click',function(event) { - event.preventDefault(); - event.stopPropagation(); - if(popover.isSituatedAt(element) && popover.visible()) { - popover.title(''); - popover.content(''); - popover.positionAway(); - } - else { - popover.title(attrs.title); - popover.content(attrs.content); - popover.positionBeside(element); - } - }); - } - } -}]; - -directive.tabPane = function() { - return { - require: '^tabbable', - restrict: 'C', - link: function(scope, element, attrs, tabsCtrl) { - element.on('$remove', tabsCtrl.addPane(element, attrs)); - } - }; -}; - -directive.foldout = ['$http', '$animate','$window', function($http, $animate, $window) { - return { - restrict: 'A', - priority : 500, - link: function(scope, element, attrs) { - var container, loading, url = attrs.url; - if(/\/build\//.test($window.location.href)) { - url = '/build/docs' + url; - } - element.on('click',function() { - scope.$apply(function() { - if(!container) { - if(loading) return; - - loading = true; - var par = element.parent(); - container = angular.element('
    loading...
    '); - $animate.enter(container, null, par); - - $http.get(url, { cache : true }).success(function(html) { - loading = false; - - html = '
    ' + - '
    ' + - html + - '
    '; - container.html(html); - - //avoid showing the element if the user has already closed it - if(container.css('display') == 'block') { - container.css('display','none'); - $animate.addClass(container, 'ng-hide'); - } - }); - } - else { - container.hasClass('ng-hide') ? $animate.removeClass(container, 'ng-hide') : $animate.addClass(container, 'ng-hide'); - } - }); - }); - } - } -}]; - -angular.module('bootstrap', []) - .directive(directive) - .factory('popoverElement', popoverElement) - .run(function() { - marked.setOptions({ - gfm: true, - tables: true - }); - }); diff --git a/docs/app/assets/js/angular-bootstrap/dropdown-toggle.js b/docs/app/assets/js/angular-bootstrap/dropdown-toggle.js index a5cdac6a080f..7be6002ac546 100644 --- a/docs/app/assets/js/angular-bootstrap/dropdown-toggle.js +++ b/docs/app/assets/js/angular-bootstrap/dropdown-toggle.js @@ -54,7 +54,9 @@ angular.module('ui.bootstrap.dropdown', []) } }; - var closeDropdown = function() { + var closeDropdown = function(evt) { + if (evt && evt.which === 3) return; + openScope.$apply(function() { openScope.isOpen = false; }); diff --git a/docs/app/src/app.js b/docs/app/src/app.js index 76c8fef81029..157985f05705 100644 --- a/docs/app/src/app.js +++ b/docs/app/src/app.js @@ -13,7 +13,6 @@ angular.module('docsApp', [ 'search', 'tutorials', 'versions', - 'bootstrap', 'ui.bootstrap.dropdown' ]) diff --git a/docs/app/src/directives.js b/docs/app/src/directives.js index 4c3acf781000..c7c14eda1e3f 100644 --- a/docs/app/src/directives.js +++ b/docs/app/src/directives.js @@ -34,4 +34,15 @@ angular.module('directives', []) return function(scope, element) { $anchorScroll.yOffset = element; }; -}]); +}]) + +.directive('table', function() { + return { + restrict: 'E', + link: function(scope, element, attrs) { + if (!attrs['class']) { + element.addClass('table table-bordered table-striped code-table'); + } + } + }; +}); diff --git a/docs/app/src/errors.js b/docs/app/src/errors.js index bd7f6bbeef83..79f508ec3447 100644 --- a/docs/app/src/errors.js +++ b/docs/app/src/errors.js @@ -13,10 +13,10 @@ angular.module('errors', ['ngSanitize']) }; return function (text, target) { - var targetHtml = target ? ' target="' + target + '"' : ''; - if (!text) return text; + var targetHtml = target ? ' target="' + target + '"' : ''; + return $sanitize(text.replace(LINKY_URL_REGEXP, function (url) { if (STACK_TRACE_REGEXP.test(url)) { return url; @@ -34,6 +34,10 @@ angular.module('errors', ['ngSanitize']) .directive('errorDisplay', ['$location', 'errorLinkFilter', function ($location, errorLinkFilter) { + var encodeAngleBrackets = function (text) { + return text.replace(//g, '>'); + }; + var interpolate = function (formatString) { var formatArgs = arguments; return formatString.replace(/\{\d+\}/g, function (match) { @@ -51,12 +55,15 @@ angular.module('errors', ['ngSanitize']) link: function (scope, element, attrs) { var search = $location.search(), formatArgs = [attrs.errorDisplay], + formattedText, i; for (i = 0; angular.isDefined(search['p'+i]); i++) { formatArgs.push(search['p'+i]); } - element.html(errorLinkFilter(interpolate.apply(null, formatArgs), '_blank')); + + formattedText = encodeAngleBrackets(interpolate.apply(null, formatArgs)); + element.html(errorLinkFilter(formattedText, '_blank')); } }; }]); diff --git a/docs/app/src/examples.js b/docs/app/src/examples.js index 5db8bf4e88f7..1b71586e4b80 100644 --- a/docs/app/src/examples.js +++ b/docs/app/src/examples.js @@ -1,5 +1,55 @@ angular.module('examples', []) +.directive('runnableExample', ['$templateCache', '$document', function($templateCache, $document) { + var exampleClassNameSelector = '.runnable-example-file'; + var doc = $document[0]; + var tpl = + ''; + + return { + restrict: 'C', + scope : true, + controller : ['$scope', function($scope) { + $scope.setTab = function(index) { + var tab = $scope.tabs[index]; + $scope.activeTabIndex = index; + $scope.$broadcast('tabChange', index, tab); + }; + }], + compile : function(element) { + element.html(tpl + element.html()); + return function(scope, element) { + var node = element[0]; + var examples = node.querySelectorAll(exampleClassNameSelector); + var tabs = [], now = Date.now(); + angular.forEach(examples, function(child, index) { + tabs.push(child.getAttribute('name')); + }); + + if(tabs.length > 0) { + scope.tabs = tabs; + scope.$on('tabChange', function(e, index, title) { + angular.forEach(examples, function(child) { + child.style.display = 'none'; + }); + var selected = examples[index]; + selected.style.display = 'block'; + }); + scope.setTab(0); + } + }; + } + }; +}]) + .factory('formPostData', ['$document', function($document) { return function(url, newWindow, fields) { /** diff --git a/docs/app/test/.jshintrc b/docs/app/test/.jshintrc new file mode 100644 index 000000000000..f9d367561198 --- /dev/null +++ b/docs/app/test/.jshintrc @@ -0,0 +1,25 @@ +{ + "extends": "../../../.jshintrc-base", + "browser": true, + "globals": { + // AngularJS + "angular": false, + + // ngMocks + "module": false, + "inject": true, + + // Jasmine + "jasmine": false, + "describe": false, + "ddescribe": false, + "xdescribe": false, + "it": false, + "iit": false, + "xit": false, + "beforeEach": false, + "afterEach": false, + "spyOn": false, + "expect": false + } +} diff --git a/docs/app/test/errorsSpec.js b/docs/app/test/errorsSpec.js new file mode 100644 index 000000000000..f392057d356b --- /dev/null +++ b/docs/app/test/errorsSpec.js @@ -0,0 +1,166 @@ +'use strict'; + +describe('errors', function() { + // Mock `ngSanitize` module + angular. + module('ngSanitize', []). + value('$sanitize', jasmine.createSpy('$sanitize').andCallFake(angular.identity)); + + beforeEach(module('errors')); + + + describe('errorDisplay', function() { + var $sanitize; + var errorLinkFilter; + + beforeEach(inject(function(_$sanitize_, _errorLinkFilter_) { + $sanitize = _$sanitize_; + errorLinkFilter = _errorLinkFilter_; + })); + + + it('should return empty input unchanged', function() { + var inputs = [undefined, null, false, 0, '']; + var remaining = inputs.length; + + inputs.forEach(function(falsyValue) { + expect(errorLinkFilter(falsyValue)).toBe(falsyValue); + remaining--; + }); + + expect(remaining).toBe(0); + }); + + + it('should recognize URLs and convert them to ``', function() { + var urls = [ + ['ftp://foo/bar?baz#qux'], + ['http://foo/bar?baz#qux'], + ['https://foo/bar?baz#qux'], + ['mailto:foo_bar@baz.qux', null, 'foo_bar@baz.qux'], + ['foo_bar@baz.qux', 'mailto:foo_bar@baz.qux', 'foo_bar@baz.qux'] + ]; + var remaining = urls.length; + + urls.forEach(function(values) { + var actualUrl = values[0]; + var expectedUrl = values[1] || actualUrl; + var expectedText = values[2] || expectedUrl; + var anchor = '' + expectedText + ''; + + var input = 'start ' + actualUrl + ' end'; + var output = 'start ' + anchor + ' end'; + + expect(errorLinkFilter(input)).toBe(output); + remaining--; + }); + + expect(remaining).toBe(0); + }); + + + it('should not recognize stack-traces as URLs', function() { + var urls = [ + 'ftp://foo/bar?baz#qux:4:2', + 'http://foo/bar?baz#qux:4:2', + 'https://foo/bar?baz#qux:4:2', + 'mailto:foo_bar@baz.qux:4:2', + 'foo_bar@baz.qux:4:2' + ]; + var remaining = urls.length; + + urls.forEach(function(url) { + var input = 'start ' + url + ' end'; + + expect(errorLinkFilter(input)).toBe(input); + remaining--; + }); + + expect(remaining).toBe(0); + }); + + + it('should should set `[target]` if specified', function() { + var url = 'https://foo/bar?baz#qux'; + var target = '_blank'; + var outputWithoutTarget = '' + url + ''; + var outputWithTarget = '' + url + ''; + + expect(errorLinkFilter(url)).toBe(outputWithoutTarget); + expect(errorLinkFilter(url, target)).toBe(outputWithTarget); + }); + + + it('should truncate the contents of the generated `` to 60 characters', function() { + var looongUrl = 'https://foooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo'; + var truncatedUrl = 'https://foooooooooooooooooooooooooooooooooooooooooooooooo...'; + var output = '' + truncatedUrl + ''; + + expect(looongUrl.length).toBeGreaterThan(60); + expect(truncatedUrl.length).toBe(60); + expect(errorLinkFilter(looongUrl)).toBe(output); + }); + + + it('should pass the final string through `$sanitize`', function() { + $sanitize.reset(); + + var input = 'start https://foo/bar?baz#qux end'; + var output = errorLinkFilter(input); + + expect($sanitize.callCount).toBe(1); + expect($sanitize).toHaveBeenCalledWith(output); + }); + }); + + + describe('errorDisplay', function() { + var $compile; + var $location; + var $rootScope; + var errorLinkFilter; + + beforeEach(module(function($provide) { + $provide.decorator('errorLinkFilter', function() { + errorLinkFilter = jasmine.createSpy('errorLinkFilter'); + errorLinkFilter.andCallFake(angular.identity); + + return errorLinkFilter; + }); + })); + beforeEach(inject(function(_$compile_, _$location_, _$rootScope_) { + $compile = _$compile_; + $location = _$location_; + $rootScope = _$rootScope_; + })); + + + it('should set the element\s HTML', function() { + var elem = $compile('foo')($rootScope); + expect(elem.html()).toBe('bar'); + }); + + + it('should interpolate the contents against `$location.search()`', function() { + spyOn($location, 'search').andReturn({p0: 'foo', p1: 'bar'}); + + var elem = $compile('')($rootScope); + expect(elem.html()).toBe('foo = foo, bar = bar'); + }); + + + it('should pass the interpolated text through `errorLinkFilter`', function() { + $location.search = jasmine.createSpy('search').andReturn({p0: 'foo'}); + + var elem = $compile('')($rootScope); + expect(errorLinkFilter.callCount).toBe(1); + expect(errorLinkFilter).toHaveBeenCalledWith('foo = foo', '_blank'); + }); + + + it('should encode `<` and `>`', function() { + var elem = $compile('')($rootScope); + expect(elem.text()).toBe(''); + }); + }); +}); diff --git a/docs/config/index.js b/docs/config/index.js index 3bba1f92a5e5..9f6114f6b61d 100644 --- a/docs/config/index.js +++ b/docs/config/index.js @@ -170,4 +170,8 @@ module.exports = new Package('angularjs', [ jqueryDeployment, productionDeployment ]; +}) + +.config(function(generateKeywordsProcessor) { + generateKeywordsProcessor.docTypesToIgnore = ['componentGroup']; }); diff --git a/docs/config/processors/keywords.js b/docs/config/processors/keywords.js index 4994064aed85..62f01a9da163 100644 --- a/docs/config/processors/keywords.js +++ b/docs/config/processors/keywords.js @@ -16,9 +16,11 @@ module.exports = function generateKeywordsProcessor(log, readFilesProcessor) { ignoreWordsFile: undefined, areasToSearch: ['api', 'guide', 'misc', 'error', 'tutorial'], propertiesToIgnore: [], + docTypesToIgnore: [], $validate: { ignoreWordsFile: { }, areasToSearch: { presence: true }, + docTypesToIgnore: { }, propertiesToIgnore: { } }, $runAfter: ['memberDocsProcessor'], @@ -28,6 +30,7 @@ module.exports = function generateKeywordsProcessor(log, readFilesProcessor) { // Keywords to ignore var wordsToIgnore = []; var propertiesToIgnore; + var docTypesToIgnore; var areasToSearch; // Keywords start with "ng:" or one of $, _ or a letter @@ -47,6 +50,8 @@ module.exports = function generateKeywordsProcessor(log, readFilesProcessor) { areasToSearch = _.indexBy(this.areasToSearch); propertiesToIgnore = _.indexBy(this.propertiesToIgnore); log.debug('Properties to ignore', propertiesToIgnore); + docTypesToIgnore = _.indexBy(this.docTypesToIgnore); + log.debug('Doc types to ignore', docTypesToIgnore); var ignoreWordsMap = _.indexBy(wordsToIgnore); @@ -78,34 +83,36 @@ module.exports = function generateKeywordsProcessor(log, readFilesProcessor) { // We are only interested in docs that live in the right area docs = _.filter(docs, function(doc) { return areasToSearch[doc.area]; }); + docs = _.filter(docs, function(doc) { return !docTypesToIgnore[doc.docType]; }); _.forEach(docs, function(doc) { - var words = []; - var keywordMap = _.clone(ignoreWordsMap); - var members = []; - var membersMap = {}; - // Search each top level property of the document for search terms - _.forEach(doc, function(value, key) { + var words = []; + var keywordMap = _.clone(ignoreWordsMap); + var members = []; + var membersMap = {}; - if ( _.isString(value) && !propertiesToIgnore[key] ) { - extractWords(value, words, keywordMap); - } + // Search each top level property of the document for search terms + _.forEach(doc, function(value, key) { - if ( key === 'methods' || key === 'properties' || key === 'events' ) { - _.forEach(value, function(member) { - extractWords(member.name, members, membersMap); - }); - } - }); + if ( _.isString(value) && !propertiesToIgnore[key] ) { + extractWords(value, words, keywordMap); + } + + if ( key === 'methods' || key === 'properties' || key === 'events' ) { + _.forEach(value, function(member) { + extractWords(member.name, members, membersMap); + }); + } + }); - doc.searchTerms = { - titleWords: extractTitleWords(doc.name), - keywords: _.sortBy(words).join(' '), - members: _.sortBy(members).join(' ') - }; + doc.searchTerms = { + titleWords: extractTitleWords(doc.name), + keywords: _.sortBy(words).join(' '), + members: _.sortBy(members).join(' ') + }; }); diff --git a/docs/config/services/deployments/debug.js b/docs/config/services/deployments/debug.js index d97711e882ca..9fe969b60ad2 100644 --- a/docs/config/services/deployments/debug.js +++ b/docs/config/services/deployments/debug.js @@ -18,7 +18,6 @@ module.exports = function debugDeployment(getVersion) { '../angular-touch.js', '../angular-animate.js', 'components/marked-' + getVersion('marked', 'node_modules', 'package.json') + '/lib/marked.js', - 'js/angular-bootstrap/bootstrap.js', 'js/angular-bootstrap/dropdown-toggle.js', 'components/lunr.js-' + getVersion('lunr.js') + '/lunr.js', 'components/google-code-prettify-' + getVersion('google-code-prettify') + '/src/prettify.js', diff --git a/docs/config/services/deployments/default.js b/docs/config/services/deployments/default.js index 3765fdf405c2..31ab6041e45e 100644 --- a/docs/config/services/deployments/default.js +++ b/docs/config/services/deployments/default.js @@ -18,7 +18,6 @@ module.exports = function defaultDeployment(getVersion) { '../angular-touch.min.js', '../angular-animate.min.js', 'components/marked-' + getVersion('marked', 'node_modules', 'package.json') + '/lib/marked.js', - 'js/angular-bootstrap/bootstrap.min.js', 'js/angular-bootstrap/dropdown-toggle.min.js', 'components/lunr.js-' + getVersion('lunr.js') + '/lunr.min.js', 'components/google-code-prettify-' + getVersion('google-code-prettify') + '/src/prettify.js', diff --git a/docs/config/services/deployments/jquery.js b/docs/config/services/deployments/jquery.js index a54473061762..13099ef23f93 100644 --- a/docs/config/services/deployments/jquery.js +++ b/docs/config/services/deployments/jquery.js @@ -22,7 +22,6 @@ module.exports = function jqueryDeployment(getVersion) { '../angular-touch.min.js', '../angular-animate.min.js', 'components/marked-' + getVersion('marked', 'node_modules', 'package.json') + '/lib/marked.js', - 'js/angular-bootstrap/bootstrap.min.js', 'js/angular-bootstrap/dropdown-toggle.min.js', 'components/lunr.js-' + getVersion('lunr.js') + '/lunr.min.js', 'components/google-code-prettify-' + getVersion('google-code-prettify') + '/src/prettify.js', diff --git a/docs/config/services/deployments/production.js b/docs/config/services/deployments/production.js index 237c53e9460a..aa740ed5e8c9 100644 --- a/docs/config/services/deployments/production.js +++ b/docs/config/services/deployments/production.js @@ -21,7 +21,6 @@ module.exports = function productionDeployment(getVersion) { cdnUrl + '/angular-touch.min.js', cdnUrl + '/angular-animate.min.js', 'components/marked-' + getVersion('marked', 'node_modules', 'package.json') + '/lib/marked.js', - 'js/angular-bootstrap/bootstrap.min.js', 'js/angular-bootstrap/dropdown-toggle.min.js', 'components/lunr.js-' + getVersion('lunr.js') + '/lunr.min.js', 'components/google-code-prettify-' + getVersion('google-code-prettify') + '/src/prettify.js', diff --git a/docs/config/templates/runnableExample.template.html b/docs/config/templates/runnableExample.template.html index 0380c5ed8b5c..c01f1480b7d7 100644 --- a/docs/config/templates/runnableExample.template.html +++ b/docs/config/templates/runnableExample.template.html @@ -1,4 +1,4 @@ -{# Be aware that we need these extra new lines here or marked will not realise that the
    +{# Be aware that we need these extra new lines here or marked will not realize that the
    is HTML and wrap each line in a

    - thus breaking the HTML #}

    @@ -24,5 +24,5 @@
    -{# Be aware that we need these extra new lines here or marked will not realise that the
    +{# Be aware that we need these extra new lines here or marked will not realize that the
    above is HTML and wrap each line in a

    - thus breaking the HTML #} diff --git a/docs/content/error/$compile/ctreq.ngdoc b/docs/content/error/$compile/ctreq.ngdoc index d222a1bdc65f..8aedd10a38c5 100644 --- a/docs/content/error/$compile/ctreq.ngdoc +++ b/docs/content/error/$compile/ctreq.ngdoc @@ -8,7 +8,7 @@ but the required directive controller is not present on the current DOM element To resolve this error ensure that there is no typo in the required controller name and that the required directive controller is present on the current element. -If the required controller is expected to be on a ancestor element, make sure that you prefix the controller name in the `require` definition with `^`. +If the required controller is expected to be on an ancestor element, make sure that you prefix the controller name in the `require` definition with `^`. If the required controller is optionally requested, use `?` or `^?` to specify that. diff --git a/docs/content/error/$compile/iscp.ngdoc b/docs/content/error/$compile/iscp.ngdoc index 8facbcc4e0e3..95af99bc8928 100644 --- a/docs/content/error/$compile/iscp.ngdoc +++ b/docs/content/error/$compile/iscp.ngdoc @@ -12,9 +12,11 @@ myModule.directive('directiveName', function factory() { scope: { 'attrName': '@', // OK 'attrName2': '=localName', // OK - 'attrName3': 'name', // ERROR: missing mode @&= - 'attrName4': ' = name', // ERROR: extra spaces - 'attrName5': 'name=', // ERROR: must be prefixed with @&= + 'attrName3': '&?localName', // OK + 'attrName4': ' = name', // OK + 'attrName5': 'name', // ERROR: missing mode @&= + 'attrName6': 'name=', // ERROR: must be prefixed with @&= + 'attrName7': '=name?', // ERROR: ? must come directly after the mode } ... } diff --git a/docs/content/error/$injector/modulerr.ngdoc b/docs/content/error/$injector/modulerr.ngdoc index e957b64b1a01..4746bf66209f 100644 --- a/docs/content/error/$injector/modulerr.ngdoc +++ b/docs/content/error/$injector/modulerr.ngdoc @@ -6,6 +6,9 @@ This error occurs when a module fails to load due to some exception. The error message above should provide additional context. +A common reason why the module fails to load is that you've forgotten to +include the file with the defined module or that the file couldn't be loaded. + ### Using `ngRoute` In AngularJS `1.2.0` and later, `ngRoute` has been moved to its own module. @@ -24,4 +27,4 @@ angular.module('ng').filter('tel', function (){}); Instead create your own module and add it as a dependency to your application's top-level module. See [#9692](https://github.com/angular/angular.js/issues/9692) and -[#7709](https://github.com/angular/angular.js/issues/7709) for more information \ No newline at end of file +[#7709](https://github.com/angular/angular.js/issues/7709) for more information diff --git a/docs/content/error/$injector/unpr.ngdoc b/docs/content/error/$injector/unpr.ngdoc index cbf687b4d77e..c057baf743c1 100644 --- a/docs/content/error/$injector/unpr.ngdoc +++ b/docs/content/error/$injector/unpr.ngdoc @@ -81,3 +81,6 @@ angular.module('myModule', []) // a scope object cannot be injected into a service. }]); ``` + +If you encounter this error only with minified code, consider using `ngStrictDi` (see +{@link ng.directive:ngApp ngApp}) to provoke the error with the non-minified source. diff --git a/docs/content/error/$location/nobase.ngdoc b/docs/content/error/$location/nobase.ngdoc index 69500d43a767..ad90064cdba0 100644 --- a/docs/content/error/$location/nobase.ngdoc +++ b/docs/content/error/$location/nobase.ngdoc @@ -1,6 +1,6 @@ @ngdoc error @name $location:nobase -@fullName $location in HTML5 mode requires a tag to be present! +@fullName $location in HTML5 mode requires a `` tag to be present! @description If you configure {@link ng.$location `$location`} to use @@ -15,7 +15,7 @@ $locationProvider.html5Mode({ }); ``` -Note that removing the requirement for a tag will have adverse side effects when resolving +Note that removing the requirement for a `` tag will have adverse side effects when resolving relative paths with `$location` in IE9. The base URL is then used to resolve all relative URLs throughout the application regardless of the diff --git a/docs/content/error/$rootScope/inprog.ngdoc b/docs/content/error/$rootScope/inprog.ngdoc index fb5d09620a11..10da4c211516 100644 --- a/docs/content/error/$rootScope/inprog.ngdoc +++ b/docs/content/error/$rootScope/inprog.ngdoc @@ -100,7 +100,7 @@ To resolve this type of issue, either fix the api to be always synchronous or as your callback handler to always run asynchronously by using the `$timeout` service. ``` -function MyController($scope, thirdPartyComponent) { +function MyController($scope, $timeout, thirdPartyComponent) { thirdPartyComponent.getData(function(someData) { $timeout(function() { $scope.someData = someData; @@ -161,7 +161,7 @@ In this second scenario, we are already inside a `$digest` when the ngFocus dire call to `$apply()`, causing this error to be thrown. It is possible to workaround this problem by moving the call to set the focus outside of the digest, -by using `$timeout(fn, 0, false)`, where the `false` value tells Angular not to wrap this `fn` in a +by using `$timeout(fn, 0, false)`, where the `false` value tells Angular not to wrap this `fn` in an `$apply` block: ``` @@ -200,7 +200,7 @@ the top of the call stack. Once you have identified this call you work your way up the stack to see what the problem is. * If the second call was made in your application code then you should look at why this code has been -called from within a `$apply`/`$digest`. It may be a simple oversight or maybe it fits with the +called from within an `$apply`/`$digest`. It may be a simple oversight or maybe it fits with the sync/async scenario described earlier. * If the second call was made inside an Angular directive then it is likely that it matches the second diff --git a/docs/content/error/ngModel/nopromise.ngdoc b/docs/content/error/ngModel/nopromise.ngdoc new file mode 100644 index 000000000000..e20cc4e980a5 --- /dev/null +++ b/docs/content/error/ngModel/nopromise.ngdoc @@ -0,0 +1,28 @@ +@ngdoc error +@name ngModel:nopromise +@fullName No promise +@description + +The return value of an async validator, must always be a promise. If you want to return a +non-promise value, you can convert it to a promise using {@link ng.$q#resolve `$q.resolve()`} or +{@link ng.$q#reject `$q.reject()`}. + +Example: + +``` +.directive('asyncValidator', function($q) { + return { + require: 'ngModel', + link: function(scope, elem, attrs, ngModel) { + ngModel.$asyncValidators.myAsyncValidation = function(modelValue, viewValue) { + if (/* I don't need to hit the backend API */) { + return $q.resolve(); // to mark as valid or + // return $q.reject(); // to mark as invalid + } else { + // ...send a request to the backend and return a promise + } + }; + } + }; +}) +``` diff --git a/docs/content/guide/$location.ngdoc b/docs/content/guide/$location.ngdoc index 237c49a1b484..fd753879c6ff 100644 --- a/docs/content/guide/$location.ngdoc +++ b/docs/content/guide/$location.ngdoc @@ -330,8 +330,8 @@ reload to the original link. ### Relative links Be sure to check all relative links, images, scripts etc. Angular requires you to specify the url -base in the head of your main html file (``) unless `html5Mode.requireBase` is -set to `false` in the html5Mode definition object passed to `$locationProvider.html5Mode()`. With +base in the head of your main html file (``) unless `html5Mode.requireBase` +is set to `false` in the html5Mode definition object passed to `$locationProvider.html5Mode()`. With that, relative urls will always be resolved to this base url, even if the initial url of the document was different. @@ -339,6 +339,7 @@ There is one exception: Links that only contain a hash fragment (e.g. `` (i.e. the application exists in the "folder" +called `/base`). The URL `/base` is actually outside the application (it refers to the `base` file found +in the root `/` folder). + +If you wish to be able to navigate to the application via a URL such as `/base` then you should ensure that +you server is setup to redirect such requests to `/base/`. + +See https://github.com/angular/angular.js/issues/14018 for more information. + ### Sending links among different browsers Because of rewriting capability in HTML5 mode, your users will be able to open regular url links in @@ -356,15 +371,15 @@ legacy browsers and hashbang links in modern browser: ### Example -Here you can see two `$location` instances, both in **Html5 mode**, but on different browsers, so -that you can see the differences. These `$location` services are connected to a fake browsers. Each -input represents the address bar of the browser. +Here you can see two `$location` instances that show the difference between **Html5 mode** and **Html5 Fallback mode**. +Note that to simulate different levels of browser support, the `$location` instances are connected to +a fakeBrowser service, which you don't have to set up in actual projects. -Note that when you type hashbang url into first browser (or vice versa) it doesn't rewrite / +Note that when you type hashbang url into the first browser (or vice versa) it doesn't rewrite / redirect to regular / hashbang url, as this conversion happens only during parsing the initial URL = on page reload. -In these examples we use `` +In these examples we use ``. The inputs represent the address bar of the browser. #### Browser in HTML5 mode @@ -389,6 +404,7 @@ In these examples we use `` 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 }) @@ -538,6 +554,7 @@ In these examples we use `` 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 }) @@ -769,8 +786,8 @@ then uses the information it obtains to compose hashbang URLs (such as - Navigation outside the app - Use lower level API + Navigation outside the app + Use lower level API @@ -784,8 +801,8 @@ then uses the information it obtains to compose hashbang URLs (such as - Read access - Change to + Read access + Change to diff --git a/docs/content/guide/accessibility.ngdoc b/docs/content/guide/accessibility.ngdoc index 76f6fcfc498a..21556f515fd1 100644 --- a/docs/content/guide/accessibility.ngdoc +++ b/docs/content/guide/accessibility.ngdoc @@ -41,7 +41,7 @@ Currently, ngAria interfaces with the following directives:

    ngModel

    -Much of ngAria's heavy lifting happens in the {@link ngModel ngModel} +Much of ngAria's heavy lifting happens in the {@link ng.directive:ngModel ngModel} directive. For elements using ngModel, special attention is paid by ngAria if that element also has a role or type of `checkbox`, `radio`, `range` or `textbox`. @@ -134,14 +134,14 @@ attributes (if they have not been explicitly specified by the developer): ngAria will also add `tabIndex`, ensuring custom elements with these roles will be reachable from the keyboard. It is still up to **you** as a developer to **ensure custom controls will be -accessible**. As a rule, any time you create a widget involving user interaction, be sure to test +accessible**. As a rule, any time you create a widget involving user interaction, be sure to test it with your keyboard and at least one mobile and desktop screen reader.

    ngDisabled

    The `disabled` attribute is only valid for certain elements such as `button`, `input` and `textarea`. To properly disable custom element directives such as `` or ``, -using ngAria with [ngDisabled](https://docs.angularjs.org/api/ng/directive/ngDisabled) will also +using ngAria with {@link ng.directive:ngDisabled ngDisabled} will also add `aria-disabled`. This tells assistive technologies when a non-native input is disabled, helping custom controls to be more accessible. @@ -162,7 +162,7 @@ Becomes:

    ngShow

    ->The [ngShow](https://docs.angularjs.org/api/ng/directive/ngShow) directive shows or hides the +>The {@link ng.directive:ngShow ngShow} directive shows or hides the given HTML element based on the expression provided to the `ngShow` attribute. The element is shown or hidden by removing or adding the `.ng-hide` CSS class onto the element. @@ -199,7 +199,7 @@ Becomes:

    ngHide

    ->The [ngHide](https://docs.angularjs.org/api/ng/directive/ngHide) directive shows or hides the +>The {@link ng.directive:ngHide ngHide} directive shows or hides the given HTML element based on the expression provided to the `ngHide` attribute. The element is shown or hidden by removing or adding the `.ng-hide` CSS class onto the element. @@ -208,7 +208,7 @@ The default CSS for `ngHide`, the inverse method to `ngShow`, makes ngAria redun `display: none`. See explanation for {@link guide/accessibility#ngshow ngShow} when overriding the default CSS.

    ngClick and ngDblclick

    -If `ng-click` or `ng-dblclick` is encountered, ngAria will add `tabindex="0"` to any element not in +If `ng-click` or `ng-dblclick` is encountered, ngAria will add `tabindex="0"` to any element not in a node blacklist: * Button @@ -218,14 +218,14 @@ a node blacklist: * Select * Details/Summary -To fix widespread accessibility problems with `ng-click` on `div` elements, ngAria will +To fix widespread accessibility problems with `ng-click` on `div` elements, ngAria will dynamically bind a keypress event by default as long as the element isn't in the node blacklist. -You can turn this functionality on or off with the `bindKeypress` configuration option. +You can turn this functionality on or off with the `bindKeypress` configuration option. ngAria will also add the `button` role to communicate to users of assistive technologies. This can be disabled with the `bindRoleForClick` configuration option. -For `ng-dblclick`, you must still manually add `ng-keypress` and a role to non-interactive elements +For `ng-dblclick`, you must still manually add `ng-keypress` and a role to non-interactive elements such as `div` or `taco-button` to enable keyboard access.

    Example

    diff --git a/docs/content/guide/animations.ngdoc b/docs/content/guide/animations.ngdoc index 14b62f1f9d1b..0898789dd917 100644 --- a/docs/content/guide/animations.ngdoc +++ b/docs/content/guide/animations.ngdoc @@ -274,6 +274,37 @@ myModule.directive('my-directive', ['$animate', function($animate) { }]); ``` +## Preventing flicker before an animation starts + +When nesting elements with structural animations such as `ngIf` into elements that have class-based +animations such as `ngClass`, it sometimes happens that before the actual animation starts, there is a brief flicker or flash of content +where the animated element is briefly visible. + +To prevent this, you can apply styles to the `ng-[event]-prepare` class, which is added as soon as an animation is initialized, +but removed before the actual animation starts (after waiting for a $digest). This class is only added for *structural* +animations (`enter`, `move`, and `leave`). + +Here's an example where you might see flickering: + +```html +
    +
    +
    +
    +
    +``` + +It is possible that during the `enter` event, 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; +} + +/* Other animation styles ... */ +``` + ## More about animations For a full breakdown of each method available on `$animate`, see the {@link ng.$animate API documentation}. diff --git a/docs/content/guide/concepts.ngdoc b/docs/content/guide/concepts.ngdoc index 8c2a4a0cfc80..d72495a3ac33 100644 --- a/docs/content/guide/concepts.ngdoc +++ b/docs/content/guide/concepts.ngdoc @@ -76,7 +76,7 @@ stores/updates the value of the input field into/from a variable. The second kind of new markup are the double curly braces `{{ expression | filter }}`: When the compiler encounters this markup, it will replace it with the evaluated value of the markup. An
    {@link expression expression} in a template is a JavaScript-like code snippet that allows -to read and write variables. Note that those variables are not global variables. +Angular to read and write variables. Note that those variables are not global variables. Just like variables in a JavaScript function live in a scope, Angular provides a {@link scope scope} for the variables accessible to expressions. The values that are stored in variables on the scope are referred to as the *model* @@ -348,8 +348,7 @@ The following example shows how this is done with Angular: return { currencies: currencies, - convert: convert, - refresh: refresh + convert: convert }; }]); diff --git a/docs/content/guide/decorators.ngdoc b/docs/content/guide/decorators.ngdoc new file mode 100644 index 000000000000..5422f6d32dc5 --- /dev/null +++ b/docs/content/guide/decorators.ngdoc @@ -0,0 +1,486 @@ +@ngdoc overview +@name Decorators +@sortOrder 345 +@description + +# Decorators in AngularJS + +
    + **NOTE:** This guide is targeted towards developers who are already familiar with AngularJS basics. + If you're just getting started, we recommend the {@link tutorial/ tutorial} first. +
    + +## What are decorators? + +Decorators are a design pattern that is used to separate modification or *decoration* of a class without modifying the +original source code. In Angular, decorators are functions that allow a service, directive or filter to be modified +prior to its usage. + +## How to use decorators + +There are two ways to register decorators + +- `$provide.decorator`, and +- `module.decorator` + +Each provide access to a `$delegate`, which is the instantiated service/directive/filter, prior to being passed to the +service that required it. + +### $provide.decorator + +The {@link api/auto/service/$provide#decorator decorator function} allows access to a $delegate of the service once it +has been instantiated. For example: + +```js +angular.module('myApp', []) + +.config([ '$provide', function($provide) { + + $provide.decorator('$log', [ + '$delegate', + function $logDecorator($delegate) { + + var originalWarn = $delegate.warn; + $delegate.warn = function decoratedWarn(msg) { + msg = 'Decorated Warn: ' + msg; + originalWarn.apply($delegate, arguments); + }; + + return $delegate; + } + ]); +}]); +``` + +After the `$log` service has been instantiated the decorator is fired. The decorator function has a `$delegate` object +injected to provide access to the service that matches the selector in the decorator. This `$delegate` will be the +service you are decorating. The return value of the function *provided to the decorator* will take place of the service, +directive, or filter being decorated. + +
    + +The `$delegate` may be either modified or completely replaced. Given a service `myService` with a method `someFn`, the +following could all be viable solutions: + + +#### Completely Replace the $delegate +```js +angular.module('myApp', []) + +.config([ '$provide', function($provide) { + + $provide.decorator('myService', [ + '$delegate', + function myServiceDecorator($delegate) { + + var myDecoratedService = { + // new service object to replace myService + }; + return myDecoratedService; + } + ]); +}]); +``` + +#### Patch the $delegate +```js +angular.module('myApp', []) + +.config([ '$provide', function($provide) { + + $provide.decorator('myService', [ + '$delegate', + function myServiceDecorator($delegate) { + + var someFn = $delegate.someFn; + + function aNewFn() { + // new service function + someFn.apply($delegate, arguments); + } + + $delegate.someFn = aNewFn; + return $delegate; + } + ]); +}]); +``` + +#### Augment the $delegate +```js +angular.module('myApp', []) + +.config([ '$provide', function($provide) { + + $provide.decorator('myService', [ + '$delegate', + function myServiceDecorator($delegate) { + + function helperFn() { + // an additional fn to add to the service + } + + $delegate.aHelpfulAddition = helperFn; + return $delegate; + } + ]); +}]); +``` + +
    + Note that whatever is returned by the decorator function will replace that which is being decorated. For example, a + missing return statement will wipe out the entire object being decorated. +
    + +
    + +Decorators have different rules for different services. This is because services are registered in different ways. +Services are selected by name, however filters and directives are selected by appending `"Filter"` or `"Directive"` to +the end of the name. The `$delegate` provided is dictated by the type of service. + +| Service Type | Selector | $delegate | +|--------------|-------------------------------|-----------------------------------------------------------------------| +| Service | `serviceName` | The `object` or `function` returned by the service | +| Directive | `directiveName + 'Directive'` | An `Array.`{@link guide/decorators#drtvArray 1} | +| Filter | `filterName + 'Filter'` | The `function` returned by the filter | + +1. Multiple directives may be registered to the same selector/name + +
    + **NOTE:** Developers should take care in how and why they are modifying the `$delegate` for the service. Not only + should expectations for the consumer be kept, but some functionality (such as directive registration) does not take + place after decoration, but during creation/registration of the original service. This means, for example, that + an action such as pushing a directive object to a directive `$delegate` will likely result in unexpected behavior. + + Furthermore, great care should be taken when decorating core services, directives, or filters as this may unexpectedly + or adversely affect the functionality of the framework. +
    + +### module.decorator + +This {@link api/ng/type/angular.Module#decorator function} is the same as the `$provide.decorator` function except it is +exposed through the module API. This allows you to separate your decorator patterns from your module config blocks. The +main caveat here is that you will need to take note the order in which you create your decorators. + +Unlike in the module config block (which allows configuration of services prior to their creation), the service must be +registered prior to the decorator (see {@link guide/providers#provider-recipe Provider Recipe}). For example, the +following would not work because you are attempting to decorate outside of the configuration phase and the service +hasn't been created yet: + +```js +// will cause an error since 'someService' hasn't been registered +angular.module('myApp').decorator('someService', ...); + +angular.module('myApp').factory('someService', ...); +``` + +## Example Applications + +The following sections provide examples each of a service decorator, a directive decorator, and a filter decorator. + +### Service Decorator Example + +This example shows how we can replace the $log service with our own to display log messages. + + + + angular.module('myServiceDecorator', []). + + controller('Ctrl', [ + '$scope', + '$log', + '$timeout', + function($scope, $log, $timeout) { + var types = ['error', 'warn', 'log', 'info' ,'debug'], i; + + for (i = 0; i < types.length; i++) { + $log[types[i]](types[i] + ': message ' + (i + 1)); + } + + $timeout(function() { + $log.info('info: message logged in timeout'); + }); + } + ]). + + directive('myLog', [ + '$log', + function($log) { + return { + restrict: 'E', + template: '
    • {{l.message}}
    ', + scope: {}, + compile: function() { + return function(scope) { + scope.myLog = $log.stack; + }; + } + }; + } + ]). + + config([ + '$provide', + function($provide) { + + $provide.decorator('$log', [ + '$delegate', + function logDecorator($delegate) { + + var myLog = { + warn: function(msg) { + log(msg, 'warn'); + }, + error: function(msg) { + log(msg, 'error'); + }, + info: function(msg) { + log(msg, 'info'); + }, + debug: function(msg) { + log(msg, 'debug'); + }, + log: function(msg) { + log(msg, 'log'); + }, + stack: [] + }; + + function log(msg, type) { + myLog.stack.push({ type: type, message: msg.toString() }); + if (console && console[type]) console[type](msg); + } + + return myLog; + + } + ]); + + } + ]); +
    + + +
    +

    Logs

    + +
    +
    + + + li.warn { color: yellow; } + li.error { color: red; } + li.info { color: blue } + li.log { color: black } + li.debug { color: green } + + + + it('should display log messages in dom', function() { + element.all(by.repeater('l in myLog')).count().then(function(count) { + expect(count).toEqual(6); + }); + }); + +
    + +### Directive Decorator Example + +Failed interpolated expressions in `ng-href` attributes can easily go unnoticed. We can decorate `ngHref` to warn us of +those conditions. + + + + angular.module('urlDecorator', []). + + controller('Ctrl', ['$scope', function ($scope) { + $scope.id = 3; + $scope.warnCount = 0; // for testing + }]). + + config(['$provide', function($provide) { + + // matchExpressions looks for interpolation markup in the directive attribute, extracts the expressions + // from that markup (if they exist) and returns an array of those expressions + function matchExpressions(str) { + var exps = str.match(/{{([^}]+)}}/g); + + // if there isn't any, get out of here + if (exps === null) return; + + exps = exps.map(function(exp) { + var prop = exp.match(/[^{}]+/); + return prop === null ? null : prop[0]; + }); + + return exps; + } + + // remember: directives must be selected by appending 'Directive' to the directive selector + $provide.decorator('ngHrefDirective', [ + '$delegate', + '$log', + '$parse', + function($delegate, $log, $parse) { + + // store the original link fn + var originalLinkFn = $delegate[0].link; + + // replace the compile fn + $delegate[0].compile = function(tElem, tAttr) { + + // store the original exp in the directive attribute for our warning message + var originalExp = tAttr.ngHref; + + // get the interpolated expressions + var exps = matchExpressions(originalExp); + + // create and store the getters using $parse + var getters = exps.map(function(el) { + if (el) return $parse(el); + }); + + return function newLinkFn(scope, elem, attr) { + // fire the originalLinkFn + originalLinkFn.apply($delegate[0], arguments); + + // observe the directive attr and check the expressions + attr.$observe('ngHref', function(val) { + + // if we have getters and getters is an array... + if (getters && angular.isArray(getters)) { + + // loop through the getters and process them + angular.forEach(getters, function(g, idx) { + + // if val is truthy, then the warning won't log + var val = angular.isFunction(g) ? g(scope) : true; + if (!val) { + $log.warn('NgHref Warning: "' + exps[idx] + '" in the expression "' + originalExp + + '" is falsy!'); + + scope.warnCount++; // for testing + } + + }); + + } + + }); + + }; + + }; + + // get rid of the old link function since we return a link function in compile + delete $delegate[0].link; + + // return the $delegate + return $delegate; + + } + + ]); + + }]); + + + +
    + View Product {{ id }} + - id == 3, so no warning
    + View Product {{ id + 5 }} + - id + 5 == 8, so no warning
    + View Product {{ someOtherId }} + - someOtherId == undefined, so warn
    + View Product {{ someOtherId + 5 }} + - someOtherId + 5 == 5, so no warning
    +
    Warn Count: {{ warnCount }}
    +
    +
    + + + it('should warn when an expression in the interpolated value is falsy', function() { + var id3 = element(by.id('id3')); + var id8 = element(by.id('id8')); + var someOther = element(by.id('someOtherId')); + var someOther5 = element(by.id('someOtherId5')); + + expect(id3.getText()).toEqual('View Product 3'); + expect(id3.getAttribute('href')).toContain('/products/3/view'); + + expect(id8.getText()).toEqual('View Product 8'); + expect(id8.getAttribute('href')).toContain('/products/8/view'); + + expect(someOther.getText()).toEqual('View Product'); + expect(someOther.getAttribute('href')).toContain('/products//view'); + + expect(someOther5.getText()).toEqual('View Product 5'); + expect(someOther5.getAttribute('href')).toContain('/products/5/view'); + + expect(element(by.binding('warnCount')).getText()).toEqual('Warn Count: 1'); + }); + +
    + +### Filter Decorator Example + +Let's say we have created an app that uses the default format for many of our `Date` filters. Suddenly requirements have +changed (that never happens) and we need all of our default dates to be `'shortDate'` instead of `'mediumDate'`. + + + + angular.module('filterDecorator', []). + + controller('Ctrl', ['$scope', function ($scope) { + $scope.genesis = new Date(2010, 0, 5); + $scope.ngConf = new Date(2016, 4, 4); + }]). + + config(['$provide', function($provide) { + + $provide.decorator('dateFilter', [ + '$delegate', + function dateDecorator($delegate) { + + // store the original filter + var originalFilter = $delegate; + + // return our filter + return shortDateDefault; + + // shortDateDefault sets the format to shortDate if it is falsy + function shortDateDefault(date, format, timezone) { + if (!format) format = 'shortDate'; + + // return the result of the original filter + return originalFilter(date, format, timezone); + } + + } + + ]); + + }]); + + + +
    +
    Initial Commit default to short date: {{ genesis | date }}
    +
    ng-conf 2016 default short date: {{ ngConf | date }}
    +
    ng-conf 2016 with full date format: {{ ngConf | date:'fullDate' }}
    +
    +
    + + + it('should default date filter to short date format', function() { + expect(element(by.id('genesis')).getText()) + .toMatch(/Initial Commit default to short date: \d{1,2}\/\d{1,2}\/\d{2}/); + }); + + it('should still allow dates to be formatted', function() { + expect(element(by.id('ngConf')).getText()) + .toMatch(/ng-conf 2016 with full date format\: [A-Za-z]+, [A-Za-z]+ \d{1,2}, \d{4}/); + }); + +
    diff --git a/docs/content/guide/directive.ngdoc b/docs/content/guide/directive.ngdoc index 581a0f7e5b2e..85d8c4165acf 100644 --- a/docs/content/guide/directive.ngdoc +++ b/docs/content/guide/directive.ngdoc @@ -43,8 +43,7 @@ mirrors the process of compiling source code in Before we can write a directive, we need to know how Angular's {@link guide/compiler HTML compiler} determines when to use a given directive. -Similar to the terminology used when an [element **matches** a selector] -(https://developer.mozilla.org/en-US/docs/Web/API/Element.matches), we say an element **matches** a +Similar to the terminology used when an [element **matches** a selector](https://developer.mozilla.org/en-US/docs/Web/API/Element.matches), we say an element **matches** a directive when the directive is part of its declaration. In the following example, we say that the `` element **matches** the `ngModel` directive @@ -100,8 +99,13 @@ For example, the following forms are all equivalent and match the {@link ngBind} it('should show off bindings', function() { - expect(element(by.css('div[ng-controller="Controller"] span[ng-bind]')).getText()) - .toBe('Max Karl Ernst Ludwig Planck (April 23, 1858 – October 4, 1947)'); + 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)'); + }); }); @@ -142,63 +146,6 @@ directives when possible.
    - -### Text and attribute bindings - -During the compilation process the {@link ng.$compile compiler} matches text and attributes -using the {@link ng.$interpolate $interpolate} service to see if they contain embedded -expressions. These expressions are registered as {@link ng.$rootScope.Scope#$watch watches} -and will update as part of normal {@link ng.$rootScope.Scope#$digest digest} cycle. An -example of interpolation is shown below: - -```html -Hello {{username}}! -``` - - -### `ngAttr` attribute bindings - -Web browsers are sometimes picky about what values they consider valid for attributes. - -For example, considering this template: - -```html - - - -``` - -We would expect Angular to be able to bind to this, but when we check the console we see -something like `Error: Invalid value for attribute cx="{{cx}}"`. Because of the SVG DOM API's -restrictions, you cannot simply write `cx="{{cx}}"`. - -With `ng-attr-cx` you can work around this problem. - -If an attribute with a binding is prefixed with the `ngAttr` prefix (denormalized as `ng-attr-`) -then during the binding it will be applied to the corresponding unprefixed attribute. This allows -you to bind to attributes that would otherwise be eagerly processed by browsers -(e.g. an SVG element's `circle[cx]` attributes). When using `ngAttr`, the `allOrNothing` flag of -{@link ng.$interpolate $interpolate} is used, so if any expression in the interpolated string -results in `undefined`, the attribute is removed and not added to the element. - -For example, we could fix the example above by instead writing: - -```html - - - -``` - -If one wants to modify a camelcased attribute (SVG elements have valid camelcased attributes), such as `viewBox` on the `svg` element, one can use underscores to denote that the attribute to bind to is naturally camelcased. - -For example, to bind to `viewBox`, we can write: - -```html - - -``` - - ## Creating Directives First let's talk about the {@link ng.$compileProvider#directive API for registering directives}. Much like @@ -356,6 +303,7 @@ The `restrict` option is typically set to: * `'A'` - only matches attribute name * `'E'` - only matches element name * `'C'` - only matches class name +* `'M'` - only matches comment These restrictions can all be combined as needed: @@ -459,7 +407,7 @@ This is clearly not a great solution. What we want to be able to do is separate the scope inside a directive from the scope outside, and then map the outer scope to a directive's inner scope. We can do this by creating what -we call an **isolate scope**. To do this, we can use a directive's `scope` option: +we call an **isolate scope**. To do this, we can use a {@link $compile#-scope- directive's `scope`} option: @@ -588,14 +536,24 @@ want to reuse throughout your app. In this example we will build a directive that displays the current time. Once a second, it updates the DOM to reflect the current time. -Directives that want to modify the DOM typically use the `link` option. -`link` takes a function with the following signature, `function link(scope, element, attrs) { ... }` -where: +Directives that want to modify the DOM typically use the `link` option to register DOM listeners +as well as update the DOM. It is executed after the template has been cloned and is where +directive logic will be put. + + `link` takes a function with the following signature, +`function link(scope, element, attrs, controller, transcludeFn) { ... }`, where: * `scope` is an Angular scope object. * `element` is the jqLite-wrapped element that this directive matches. * `attrs` is a hash object with key-value pairs of normalized attribute names and their corresponding attribute values. +* `controller` is the directive's required controller instance(s) or its own controller (if any). + The exact value depends on the directive's require property. +* `transcludeFn` is a transclude linking function pre-bound to the correct transclusion scope. + +
    +For more details on the `link` option refer to the {@link ng.$compile#-link- `$compile` API} page. +
    In our `link` function, we want to update the displayed time once a second, or whenever a user changes the time formatting string that our directive binds to. We will use the `$interval` service @@ -689,6 +647,7 @@ To do this, we need to use the `transclude` option. return { restrict: 'E', transclude: true, + scope: {}, templateUrl: 'my-dialog.html' }; }); @@ -699,8 +658,7 @@ To do this, we need to use the `transclude` option.
    -
    -
    +
    @@ -722,7 +680,7 @@ that redefines `name` as `Jeff`. What do you think the `{{name}}` binding will r transclude: true, scope: {}, templateUrl: 'my-dialog.html', - link: function (scope, element) { + link: function (scope) { scope.name = 'Jeff'; } }; @@ -734,8 +692,7 @@ that redefines `name` as `Jeff`. What do you think the `{{name}}` binding will r
    -
    -
    +
    @@ -746,7 +703,7 @@ The `transclude` option changes the way scopes are nested. It makes it so that t transcluded directive have whatever scope is outside the directive, rather than whatever scope is on the inside. In doing so, it gives the contents access to the outside scope. -Note that if the directive did not create its own scope, then `scope` in `scope.name = 'Jeff';` would +Note that if the directive did not create its own scope, then `scope` in `scope.name = 'Jeff'` would reference the outside scope and we would see `Jeff` in the output. This behavior makes sense for a directive that wraps some content, because otherwise you'd have to @@ -819,9 +776,9 @@ function. Often it's desirable to pass data from the isolate scope via an expression to the parent scope, this can be done by passing a map of local variable names and values into the expression -wrapper fn. For example, the hideDialog function takes a message to display when the dialog is hidden. -This is specified in the directive by calling `close({message: 'closing for now'})`. Then the local -variable `message` will be available within the `on-close` expression. +wrapper function. For example, the `hideDialog` function takes a message to display when the dialog +is hidden. This is specified in the directive by calling `close({message: 'closing for now'})`. +Then the local variable `message` will be available within the `on-close` expression.
    **Best Practice:** use `&attr` in the `scope` option when you want your directive @@ -880,7 +837,7 @@ element? }]); - Drag ME + Drag Me @@ -903,7 +860,7 @@ to which tab is active. restrict: 'E', transclude: true, scope: {}, - controller: function($scope) { + controller: ['$scope', function($scope) { var panes = $scope.panes = []; $scope.select = function(pane) { @@ -919,13 +876,13 @@ to which tab is active. } panes.push(pane); }; - }, + }], templateUrl: 'my-tabs.html' }; }) .directive('myPane', function() { return { - require: '^myTabs', + require: '^^myTabs', restrict: 'E', transclude: true, scope: { @@ -941,11 +898,9 @@ to which tab is active. -

    Hello

    Lorem ipsum dolor sit amet

    -

    World

    Mauris elementum elementum enim at suscipit.

    counter: {{i || 0}}

    @@ -962,22 +917,25 @@ to which tab is active.
    -
    +
    +

    {{title}}

    +
    -The `myPane` directive has a `require` option with value `^myTabs`. When a directive uses this -option, `$compile` will throw an error unless the specified controller is found. The `^` prefix -means that this directive searches for the controller on its parents (without the `^` prefix, the -directive would look for the controller on just its own element). +The `myPane` directive has a `require` option with value `^^myTabs`. When a directive uses this +option, `$compile` will throw an error unless the specified controller is found. The `^^` prefix +means that this directive searches for the controller on its parents. (A `^` prefix would make the +directive look for the controller on its own element or its parents; without any prefix, the +directive would look on its own element only.) So where does this `myTabs` controller come from? Directives can specify controllers using the unsurprisingly named `controller` option. As you can see, the `myTabs` directive uses this option. Just like `ngController`, this option attaches a controller to the template of the directive. -If it is necessary to reference the controller or any functions bound to the controller's scope in -the template, you can use the option `controllerAs` to specify the name of the controller as an alias. +If it is necessary to reference the controller or any functions bound to the controller from the +template, you can use the option `controllerAs` to specify the name of the controller as an alias. The directive needs to define a scope for this configuration to be used. This is particularly useful in the case when the directive is used as a component. @@ -992,7 +950,7 @@ The corresponding parameter being sent to the `link` function will also be an ar angular.module('docsTabsExample', []) .directive('myPane', function() { return { - require: ['^myTabs', '^ngModel'], + require: ['^^myTabs', 'ngModel'], restrict: 'E', transclude: true, scope: { @@ -1028,4 +986,3 @@ available in the {@link guide/compiler compiler guide}. The {@link ng.$compile `$compile` API} page has a comprehensive list of directive options for reference. - diff --git a/docs/content/guide/expression.ngdoc b/docs/content/guide/expression.ngdoc index b5d0d4c7a9f5..8e62fcf036fd 100644 --- a/docs/content/guide/expression.ngdoc +++ b/docs/content/guide/expression.ngdoc @@ -5,8 +5,9 @@ # Angular Expressions -Angular expressions are JavaScript-like code snippets that are usually placed in bindings such as -`{{ expression }}`. +Angular expressions are JavaScript-like code snippets that are mainly placed in +interpolation bindings such as `{{ textBinding }}`, +but also used directly in directive attributes such as `ng-click="functionExpression()"`. For example, these are valid expressions in Angular: @@ -35,7 +36,9 @@ Angular expressions are like JavaScript expressions with the following differenc * **No RegExp Creation With Literal Notation:** You cannot create regular expressions in an Angular expression. - * **No Comma And Void Operators:** You cannot use `,` or `void` in an Angular expression. + * **No Object Creation With New Operator:** You cannot use `new` operator in an Angular expression. + + * **No Comma And Void Operators:** You cannot use `,` or `void` operators in an Angular expression. * **Filters:** You can use {@link guide/filter filters} within expressions to format data before displaying it. @@ -280,7 +283,7 @@ result is a non-undefined value (see value stabilization algorithm below). -### Why this feature +### Reasons for using one-time binding The main purpose of one-time binding expression is to provide a way to create a binding that gets deregistered and frees up resources once the binding is stabilized. diff --git a/docs/content/guide/filter.ngdoc b/docs/content/guide/filter.ngdoc index 6830cfdea459..bb1c49408abe 100644 --- a/docs/content/guide/filter.ngdoc +++ b/docs/content/guide/filter.ngdoc @@ -3,6 +3,8 @@ @sortOrder 280 @description +# Filters + A filter formats the value of an expression for display to the user. They can be used in view templates, controllers or services and it is easy to define your own filter. @@ -32,10 +34,13 @@ E.g. the markup `{{ 1234 | number:2 }}` formats the number 1234 with 2 decimal p ## Using filters in controllers, services, and directives -You can also use filters in controllers, services, and directives. For this, inject a dependency -with the name `Filter` to your controller/service/directive. E.g. using the dependency -`numberFilter` will inject the number filter. The injected argument is a function that takes the -value to format as first argument and filter parameters starting with the second argument. +You can also use filters in controllers, services, and directives. + +
    +For this, inject a dependency with the name `Filter` into your controller/service/directive. +E.g. a filter called `number` is injected by using the dependency `numberFilter`. The injected argument +is a function that takes the value to format as first argument, and filter parameters starting with the second argument. +
    The example below uses the filter called {@link ng.filter:filter `filter`}. This filter reduces arrays into sub arrays based on @@ -108,6 +113,7 @@ text upper-case. No filter: {{greeting}}
    Reverse: {{greeting|reverse}}
    Reverse + uppercase: {{greeting|reverse:true}}
    + Reverse, filtered in controller: {{filteredGreeting}}
    @@ -127,8 +133,9 @@ text upper-case. return out; }; }) - .controller('MyController', ['$scope', function($scope) { + .controller('MyController', ['$scope', 'reverseFilter', function($scope, reverseFilter) { $scope.greeting = 'hello'; + $scope.filteredGreeting = reverseFilter($scope.greeting); }]); diff --git a/docs/content/guide/forms.ngdoc b/docs/content/guide/forms.ngdoc index 336f26d7f07d..6e45b5737d58 100644 --- a/docs/content/guide/forms.ngdoc +++ b/docs/content/guide/forms.ngdoc @@ -383,7 +383,7 @@ In the following example we create two directives: return { require: 'ngModel', link: function(scope, elm, attrs, ctrl) { - var usernames = ['Jim', 'John', 'Jill', 'Jackie']; + var usernames = ['Jim', 'John', 'Jill', 'Jackie']; ctrl.$asyncValidators.username = function(modelValue, viewValue) { @@ -440,8 +440,7 @@ Note that you can alternatively use `ng-pattern` to further restrict the validat var EMAIL_REGEXP = /^[a-z0-9!#$%&'*+/=?^_`{|}~.-]+@example\.com$/i; return { - require: 'ngModel', - restrict: '', + 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) { diff --git a/docs/content/guide/index.ngdoc b/docs/content/guide/index.ngdoc index 1dacc7db86af..9701773eb898 100644 --- a/docs/content/guide/index.ngdoc +++ b/docs/content/guide/index.ngdoc @@ -91,7 +91,7 @@ This is a short list of libraries with specific support and documentation for wo ### Server-Specific * **Django:** [Tutorial](http://blog.mourafiq.com/post/55034504632/end-to-end-web-app-with-django-rest-framework), [Integrating AngularJS with Django](http://django-angular.readthedocs.org/en/latest/integration.html), [Getting Started with Django Rest Framework and AngularJS](http://blog.kevinastone.com/getting-started-with-django-rest-framework-and-angularjs.html) -* **FireBase:** [AngularFire](http://angularfire.com/), [Realtime Apps with AngularJS and FireBase (video)](http://www.youtube.com/watch?v=C7ZI7z7qnHU) +* **FireBase:** [AngularFire](http://angularfire.com/), [Firebase Foundations for AngularJS](http://blog.watchandcode.com/firebase-foundations/), [Realtime Apps with AngularJS and FireBase (video)](http://www.youtube.com/watch?v=C7ZI7z7qnHU) * **Google Cloud Platform: **[with Cloud Endpoints](https://cloud.google.com/developers/articles/angularjs-cloud-endpoints-recipe-for-building-modern-web-applications/), [with Go](https://github.com/GoogleCloudPlatform/appengine-angular-gotodos) * **Hood.ie:** [60 Minutes to Awesome](http://www.roberthorvick.com/2013/06/30/todomvc-angularjs-hood-ie-60-minutes-to-awesome/) * **MEAN Stack: **[Blog post](http://blog.mongodb.org/post/49262866911/the-mean-stack-mongodb-expressjs-angularjs-and), [Setup](http://thecodebarbarian.wordpress.com/2013/07/22/introduction-to-the-mean-stack-part-one-setting-up-your-tools/), [GDL Video](https://developers.google.com/live/shows/913996610) @@ -101,7 +101,7 @@ This is a short list of libraries with specific support and documentation for wo ## Learning Resources -###Books +### Books * [AngularJS: Up and Running](http://www.amazon.com/AngularJS-Running-Enhanced-Productivity-Structured/dp/1491901942) by Brad Green and Shyam Seshadri * [Mastering Web App Development](http://www.amazon.com/Mastering-Web-Application-Development-AngularJS/dp/1782161821) by Pawel Kozlowski and Pete Bacon Darwin * [AngularJS Directives](http://www.amazon.com/AngularJS-Directives-Alex-Vanston/dp/1783280336) by Alex Vanston @@ -111,10 +111,12 @@ This is a short list of libraries with specific support and documentation for wo * [AngularJS : Novice to Ninja](http://www.amazon.in/AngularJS-Novice-Ninja-Sandeep-Panda/dp/0992279453) by Sandeep Panda * [AngularJS UI Development](http://www.amazon.com/AngularJS-UI-Development-Amit-Ghart-ebook/dp/B00OXVAK7A) by Amit Gharat and Matthias Nehlsen * [Responsive Web Design with AngularJS](http://www.amazon.com/Responsive-Design-AngularJS-Sandeep-Kumar/dp/178439842X) by Sandeep Kumar Patel +* [Professional AngularJS](http://www.amazon.com/Professional-AngularJS-Valeri-Karpov/dp/1118832078/) -###Videos: +### Videos: * [egghead.io](http://egghead.io/) * [Angular on YouTube](http://youtube.com/angularjs) +* [Firebase Foundations for AngularJS](http://blog.watchandcode.com/firebase-foundations/) ### Courses * **Free online:** @@ -122,6 +124,7 @@ This is a short list of libraries with specific support and documentation for wo [CodeAcademy](http://www.codecademy.com/courses/javascript-advanced-en-2hJ3J/0/1), [CodeSchool](https://www.codeschool.com/courses/shaping-up-with-angular-js) * **Paid online:** + [The Angular Course (115 videos that show you how to build a full app)](http://watchandcode.com/courses/angular-course/), [Pluralsite (3 courses)](http://www.pluralsight.com/training/Courses/Find?highlight=true&searchTerm=angularjs), [Tuts+](https://tutsplus.com/course/easier-js-apps-with-angular/), [lynda.com](http://www.lynda.com/AngularJS-tutorials/Up-Running-AngularJS/133318-2.html), diff --git a/docs/content/guide/interpolation.ngdoc b/docs/content/guide/interpolation.ngdoc new file mode 100644 index 000000000000..6cb52f71273f --- /dev/null +++ b/docs/content/guide/interpolation.ngdoc @@ -0,0 +1,146 @@ +@ngdoc overview +@name Interpolation +@sortOrder 275 +@description + +# Interpolation and data-binding + +Interpolation markup with embedded {@link guide/expression expressions} is used by Angular to +provide data-binding to text nodes and attribute values. + +An example of interpolation is shown below: + +```html +Hello {{username}}! +``` + +### How text and attribute bindings work + +During the compilation process the {@link ng.$compile compiler} uses the {@link ng.$interpolate $interpolate} +service to see if text nodes and element attributes contain interpolation markup with embedded expressions. + +If that is the case, the compiler adds an interpolateDirective to the node and +registers {@link ng.$rootScope.Scope#$watch watches} on the computed interpolation function, +which will update the corresponding text nodes or attribute values as part of the +normal {@link ng.$rootScope.Scope#$digest digest} cycle. + +Note that the interpolateDirective has a priority of 100 and sets up the watch in the preLink function. + +### Binding to boolean attributes + +Attributes such as `disabled` are called `boolean` attributes, because their presence means `true` and +their absence means `false`. We cannot use normal attribute bindings with them, because the HTML +specification does not require browsers to preserve the values of boolean attributes. This means that +if we put an Angular interpolation expression into such an attribute then the binding information +would be lost, because the browser ignores the attribute value. + +In the following example, the interpolation information would be ignored and the browser would simply +interpret the attribute as present, meaning that the button would always be disabled. + + ```html + Disabled: + +``` + +For this reason, Angular provides special `ng`-prefixed directives for the following boolean attributes: +{@link ngDisabled `disabled`}, {@link ngRequired `required`}, {@link ngSelected `selected`}, +{@link ngChecked `checked`}, {@link ngReadonly `readOnly`} , and {@link ngOpen `open`}. + +These directives take an expression inside the attribute, and set the corresponding boolean attribute +to true when the expression evaluates to truthy. + + ```html + Disabled: + +``` + +### `ngAttr` for binding to arbitrary attributes + +Web browsers are sometimes picky about what values they consider valid for attributes. + +For example, considering this template: + +```html + + + +``` + +We would expect Angular to be able to bind to this, but when we check the console we see +something like `Error: Invalid value for attribute cx="{{cx}}"`. Because of the SVG DOM API's +restrictions, you cannot simply write `cx="{{cx}}"`. + +With `ng-attr-cx` you can work around this problem. + +If an attribute with a binding is prefixed with the `ngAttr` prefix (denormalized as `ng-attr-`) +then during the binding it will be applied to the corresponding unprefixed attribute. This allows +you to bind to attributes that would otherwise be eagerly processed by browsers +(e.g. an SVG element's `circle[cx]` attributes). When using `ngAttr`, the `allOrNothing` flag of +{@link ng.$interpolate $interpolate} is used, so if any expression in the interpolated string +results in `undefined`, the attribute is removed and not added to the element. + +For example, we could fix the example above by instead writing: + +```html + + + +``` + +If one wants to modify a camelcased attribute (SVG elements have valid camelcased attributes), +such as `viewBox` on the `svg` element, one can use underscores to denote that the attribute to bind +to is naturally camelcased. + +For example, to bind to `viewBox`, we can write: + +```html + + +``` + +Other attributes may also not work as expected when they contain interpolation markup, and +can be used with `ngAttr` instead. The following is a list of known problematic attributes: + +- **size** in ` diff --git a/src/ng/directive/ngModel.js b/src/ng/directive/ngModel.js index 034c386a5527..a200f75a32f8 100644 --- a/src/ng/directive/ngModel.js +++ b/src/ng/directive/ngModel.js @@ -22,7 +22,9 @@ var ngModelMinErr = minErr('ngModel'); * @ngdoc type * @name ngModel.NgModelController * - * @property {string} $viewValue Actual string value in the view. + * @property {*} $viewValue The actual value from the control's view. For `input` elements, this is a + * String. See {@link ngModel.NgModelController#$setViewValue} for information about when the $viewValue + * is set. * @property {*} $modelValue The value in the model that the control is bound to. * @property {Array.} $parsers Array of functions to execute, as a pipeline, whenever the control reads value from the DOM. The functions are called in array order, each passing @@ -433,11 +435,14 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$ * which may be caused by a pending debounced event or because the input is waiting for a some * future event. * - * If you have an input that uses `ng-model-options` to set up debounced events or events such - * as blur you can have a situation where there is a period when the `$viewValue` - * is out of synch with the ngModel's `$modelValue`. + * If you have an input that uses `ng-model-options` to set up debounced updates or updates that + * depend on special events such as blur, you can have a situation where there is a period when + * the `$viewValue` is out of sync with the ngModel's `$modelValue`. * - * In this case, you can run into difficulties if you try to update the ngModel's `$modelValue` + * In this case, you can use `$rollbackViewValue()` to manually cancel the debounced / future update + * and reset the input to the last committed view value. + * + * It is also possible that you run into difficulties if you try to update the ngModel's `$modelValue` * programmatically before these debounced/future events have resolved/occurred, because Angular's * dirty checking mechanism is not able to tell whether the model has actually changed or not. * @@ -450,39 +455,63 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$ * angular.module('cancel-update-example', []) * * .controller('CancelUpdateController', ['$scope', function($scope) { - * $scope.resetWithCancel = function(e) { - * if (e.keyCode == 27) { - * $scope.myForm.myInput1.$rollbackViewValue(); - * $scope.myValue = ''; - * } - * }; - * $scope.resetWithoutCancel = function(e) { + * $scope.model = {}; + * + * $scope.setEmpty = function(e, value, rollback) { * if (e.keyCode == 27) { - * $scope.myValue = ''; + * e.preventDefault(); + * if (rollback) { + * $scope.myForm[value].$rollbackViewValue(); + * } + * $scope.model[value] = ''; * } * }; * }]); * * *
    - *

    Try typing something in each input. See that the model only updates when you - * blur off the input. - *

    - *

    Now see what happens if you start typing then press the Escape key

    + *

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

    With $rollbackViewValue()

    - *
    - * myValue: "{{ myValue }}" + *
    + *

    Without $rollbackViewValue():

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

    Without $rollbackViewValue()

    - *
    - * myValue: "{{ myValue }}" + *
    + *

    With $rollbackViewValue():

    + * + * value2: "{{ model.value2 }}" + *
    *
    *
    *
    + + div { + display: table-cell; + } + div:nth-child(1) { + padding-right: 30px; + } + + * */ this.$rollbackViewValue = function() { @@ -596,7 +625,7 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$ forEach(ctrl.$asyncValidators, function(validator, name) { var promise = validator(modelValue, viewValue); if (!isPromiseLike(promise)) { - throw ngModelMinErr("$asyncValidators", + throw ngModelMinErr('nopromise', "Expected asynchronous validator to return a promise but got '{0}' instead.", promise); } setValidity(name, undefined); @@ -892,6 +921,22 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$ * - {@link ng.directive:select select} * - {@link ng.directive:textarea textarea} * + * # Complex Models (objects or collections) + * + * By default, `ngModel` watches the model by reference, not value. This is important to know when + * binding inputs to models that are objects (e.g. `Date`) or collections (e.g. arrays). If only properties of the + * object or collection change, `ngModel` will not be notified and so the input will not be re-rendered. + * + * The model must be assigned an entirely new object or collection before a re-rendering will occur. + * + * Some directives have options that will cause them to use a custom `$watchCollection` on the model expression + * - for example, `ngOptions` will do so when a `track by` clause is included in the comprehension expression or + * if the select is given the `multiple` attribute. + * + * The `$watchCollection()` method only does a shallow comparison, meaning that changing properties deeper than the + * first level of the object (or only changing the properties of an item in the collection if it's an array) will still + * not trigger a re-rendering of the model. + * * # CSS classes * The following CSS classes are added and removed on the associated input/select/textarea element * depending on the validity of the model. @@ -1139,12 +1184,13 @@ var DEFAULT_REGEXP = /(\s+|^)default(\s+|$)/;
    user.name = 
    +
    user.data = 
    angular.module('optionsExample', []) .controller('ExampleController', ['$scope', function($scope) { - $scope.user = { name: 'say', data: '' }; + $scope.user = { name: 'John', data: '' }; $scope.cancel = function(e) { if (e.keyCode == 27) { @@ -1159,20 +1205,20 @@ var DEFAULT_REGEXP = /(\s+|^)default(\s+|$)/; var other = element(by.model('user.data')); it('should allow custom events', function() { - input.sendKeys(' hello'); + input.sendKeys(' Doe'); input.click(); - expect(model.getText()).toEqual('say'); + expect(model.getText()).toEqual('John'); other.click(); - expect(model.getText()).toEqual('say hello'); + expect(model.getText()).toEqual('John Doe'); }); it('should $rollbackViewValue when model changes', function() { - input.sendKeys(' hello'); - expect(input.getAttribute('value')).toEqual('say hello'); + input.sendKeys(' Doe'); + expect(input.getAttribute('value')).toEqual('John Doe'); input.sendKeys(protractor.Key.ESCAPE); - expect(input.getAttribute('value')).toEqual('say'); + expect(input.getAttribute('value')).toEqual('John'); other.click(); - expect(model.getText()).toEqual('say'); + expect(model.getText()).toEqual('John'); }); @@ -1198,7 +1244,7 @@ var DEFAULT_REGEXP = /(\s+|^)default(\s+|$)/; angular.module('optionsExample', []) .controller('ExampleController', ['$scope', function($scope) { - $scope.user = { name: 'say' }; + $scope.user = { name: 'Igor' }; }]); diff --git a/src/ng/directive/ngOptions.js b/src/ng/directive/ngOptions.js index e07aa137f2a6..d950198fcbe9 100644 --- a/src/ng/directive/ngOptions.js +++ b/src/ng/directive/ngOptions.js @@ -33,19 +33,27 @@ var ngOptionsMinErr = minErr('ngOptions'); * * ## Complex Models (objects or collections) * - * **Note:** By default, `ngModel` watches the model by reference, not value. This is important when - * binding any input directive to a model that is an object or a collection. + * By default, `ngModel` watches the model by reference, not value. This is important to know when + * binding the select to a model that is an object or a collection. * - * Since this is a common situation for `ngOptions` the directive additionally watches the model using - * `$watchCollection` when the select has the `multiple` attribute or when there is a `track by` clause in - * the options expression. This allows ngOptions to trigger a re-rendering of the options even if the actual - * object/collection has not changed identity but only a property on the object or an item in the collection - * changes. + * One issue occurs if you want to preselect an option. For example, if you set + * the model to an object that is equal to an object in your collection, `ngOptions` won't be able to set the selection, + * because the objects are not identical. So by default, you should always reference the item in your collection + * for preselections, e.g.: `$scope.selected = $scope.collection[3]`. * - * Note that `$watchCollection` does a shallow comparison of the properties of the object (or the items in the collection - * if the model is an array). This means that changing a property deeper inside the object/collection that the - * first level will not trigger a re-rendering. + * Another solution is to use a `track by` clause, because then `ngOptions` will track the identity + * of the item not by reference, but by the result of the `track by` expression. For example, if your + * collection items have an id property, you would `track by item.id`. + * + * A different issue with objects or collections is that ngModel won't detect if an object property or + * a collection item changes. For that reason, `ngOptions` additionally watches the model using + * `$watchCollection`, when the expression contains a `track by` clause or the the select has the `multiple` attribute. + * This allows ngOptions to trigger a re-rendering of the options even if the actual object/collection + * has not changed identity, but only a property on the object or an item in the collection changes. * + * Note that `$watchCollection` does a shallow comparison of the properties of the object (or the items in the collection + * if the model is an array). This means that changing a property deeper than the first level inside the + * object/collection will not trigger a re-rendering. * * ## `select` **`as`** * @@ -58,17 +66,13 @@ var ngOptionsMinErr = minErr('ngOptions'); * ### `select` **`as`** and **`track by`** * *
    - * Do not use `select` **`as`** and **`track by`** in the same expression. They are not designed to work together. + * Be careful when using `select` **`as`** and **`track by`** in the same expression. *
    * - * Consider the following example: - * - * ```html - * - * ``` + * Given this array of items on the $scope: * * ```js - * $scope.values = [{ + * $scope.items = [{ * id: 1, * label: 'aLabel', * subItem: { name: 'aSubItem' } @@ -77,20 +81,33 @@ var ngOptionsMinErr = minErr('ngOptions'); * label: 'bLabel', * subItem: { name: 'bSubItem' } * }]; + * ``` + * + * This will work: * - * $scope.selected = { name: 'aSubItem' }; + * ```html + * + * ``` + * ```js + * $scope.selected = $scope.items[0]; * ``` * - * With the purpose of preserving the selection, the **`track by`** expression is always applied to the element - * of the data source (to `item` in this example). To calculate whether an element is selected, we do the - * following: + * but this will not work: + * + * ```html + * + * ``` + * ```js + * $scope.selected = $scope.items[0].subItem; + * ``` * - * 1. Apply **`track by`** to the elements in the array. In the example: `[1, 2]` - * 2. Apply **`track by`** to the already selected value in `ngModel`. - * In the example: this is not possible as **`track by`** refers to `item.id`, but the selected - * value from `ngModel` is `{name: 'aSubItem'}`, so the **`track by`** expression is applied to - * a wrong object, the selected element can't be found, `` appears as having no selected value. * * * @param {string} ngModel Assignable angular expression to data-bind to. @@ -392,11 +409,8 @@ var ngOptionsDirective = ['$compile', '$parse', function($compile, $parse) { var optionTemplate = document.createElement('option'), optGroupTemplate = document.createElement('optgroup'); - return { - restrict: 'A', - terminal: true, - require: ['select', '?ngModel'], - link: function(scope, selectElement, attr, ctrls) { + + function ngOptionsPostLink(scope, selectElement, attr, ctrls) { // if ngModel is not defined, we don't need to do anything var ngModelCtrl = ctrls[1]; @@ -451,7 +465,6 @@ var ngOptionsDirective = ['$compile', '$parse', function($compile, $parse) { unknownOption.remove(); }; - // Update the controller methods for multiple selectable options if (!multiple) { @@ -459,14 +472,20 @@ var ngOptionsDirective = ['$compile', '$parse', function($compile, $parse) { var option = options.getOptionFromViewValue(value); if (option && !option.disabled) { + // Don't update the option when it is already selected. + // For example, the browser will select the first option by default. In that case, + // most properties are set automatically - except the `selected` attribute, which we + // set always + if (selectElement[0].value !== option.selectValue) { removeUnknownOption(); removeEmptyOption(); selectElement[0].value = option.selectValue; option.element.selected = true; - option.element.setAttribute('selected', 'selected'); } + + option.element.setAttribute('selected', 'selected'); } else { if (value === null || providedEmptyOption) { removeUnknownOption(); @@ -579,11 +598,16 @@ var ngOptionsDirective = ['$compile', '$parse', function($compile, $parse) { function updateOptionElement(option, element) { option.element = element; element.disabled = option.disabled; - if (option.value !== element.value) element.value = option.selectValue; + // NOTE: The label must be set before the value, otherwise IE10/11/EDGE create unresponsive + // selects in certain circumstances when multiple selects are next to each other and display + // the option list in listbox style, i.e. the select is [multiple], or specifies a [size]. + // See https://github.com/angular/angular.js/issues/11314 for more info. + // This is unfortunately untestable with unit / e2e tests if (option.label !== element.label) { element.label = option.label; element.textContent = option.label; } + if (option.value !== element.value) element.value = option.selectValue; } function addOrReuseElement(parent, current, type, templateElement) { @@ -621,10 +645,15 @@ var ngOptionsDirective = ['$compile', '$parse', function($compile, $parse) { var emptyOption_ = emptyOption && emptyOption[0]; var unknownOption_ = unknownOption && unknownOption[0]; + // We cannot rely on the extracted empty option being the same as the compiled empty option, + // because the compiled empty option might have been replaced by a comment because + // it had an "element" transclusion directive on it (such as ngIf) if (emptyOption_ || unknownOption_) { while (current && (current === emptyOption_ || - current === unknownOption_)) { + current === unknownOption_ || + current.nodeType === NODE_TYPE_COMMENT || + (nodeName_(current) === 'option' && current.value === ''))) { current = current.nextSibling; } } @@ -714,14 +743,28 @@ var ngOptionsDirective = ['$compile', '$parse', function($compile, $parse) { // Check to see if the value has changed due to the update to the options if (!ngModelCtrl.$isEmpty(previousValue)) { var nextValue = selectCtrl.readValue(); - if (ngOptions.trackBy ? !equals(previousValue, nextValue) : previousValue !== nextValue) { + var isNotPrimitive = ngOptions.trackBy || multiple; + if (isNotPrimitive ? !equals(previousValue, nextValue) : previousValue !== nextValue) { ngModelCtrl.$setViewValue(nextValue); ngModelCtrl.$render(); } } } + } + return { + restrict: 'A', + terminal: true, + require: ['select', '?ngModel'], + link: { + pre: function ngOptionsPreLink(scope, selectElement, attr, ctrls) { + // Deactivate the SelectController.register method to prevent + // option directives from accidentally registering themselves + // (and unwanted $destroy handlers etc.) + ctrls[0].registerOption = noop; + }, + post: ngOptionsPostLink } }; }]; diff --git a/src/ng/directive/ngRepeat.js b/src/ng/directive/ngRepeat.js index 86da541df81b..833262543c86 100644 --- a/src/ng/directive/ngRepeat.js +++ b/src/ng/directive/ngRepeat.js @@ -43,7 +43,7 @@ * Version 1.4 removed the alphabetic sorting. We now rely on the order returned by the browser * when running `for key in myObj`. It seems that browsers generally follow the strategy of providing * keys in the order in which they were defined, although there are exceptions when keys are deleted - * and reinstated. See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/delete#Cross-browser_issues + * and reinstated. See the [MDN page on `delete` for more info](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/delete#Cross-browser_notes). * * If this is not desired, the recommended workaround is to convert your object into an array * that is sorted into the order that you prefer before providing it to `ngRepeat`. You could @@ -53,15 +53,21 @@ * * # Tracking and Duplicates * - * When the contents of the collection change, `ngRepeat` makes the corresponding changes to the DOM: + * `ngRepeat` uses {@link $rootScope.Scope#$watchCollection $watchCollection} to detect changes in + * the collection. When a change happens, ngRepeat then makes the corresponding changes to the DOM: * * * When an item is added, a new instance of the template is added to the DOM. * * When an item is removed, its template instance is removed from the DOM. * * When items are reordered, their respective templates are reordered in the DOM. * - * By default, `ngRepeat` does not allow duplicate items in arrays. This is because when - * there are duplicates, it is not possible to maintain a one-to-one mapping between collection - * items and DOM elements. + * To minimize creation of DOM elements, `ngRepeat` uses a function + * to "keep track" of all items in the collection and their corresponding DOM elements. + * For example, if an item is added to the collection, ngRepeat will know that all other items + * already have DOM elements, and will not re-render them. + * + * The default tracking function (which tracks items by their identity) does not allow + * duplicate items in arrays. This is because when there are duplicates, it is not possible + * to maintain a one-to-one mapping between collection items and DOM elements. * * If you do need to repeat duplicate items, you can substitute the default tracking behavior * with your own using the `track by` expression. @@ -74,7 +80,7 @@ * * ``` * - * You may use arbitrary expressions in `track by`, including references to custom functions + * You may also use arbitrary expressions in `track by`, including references to custom functions * on the scope: * ```html *
    @@ -82,10 +88,14 @@ *
    * ``` * - * If you are working with objects that have an identifier property, you can track + *
    + * If you are working with objects that have an identifier property, you should track * by the identifier instead of the whole object. Should you reload your data later, `ngRepeat` * will not have to rebuild the DOM elements for items it has already rendered, even if the - * JavaScript objects in the collection have been substituted for new ones: + * JavaScript objects in the collection have been substituted for new ones. For large collections, + * this signifincantly improves rendering performance. If you don't have a unique identifier, + * `track by $index` can also provide a performance boost. + *
    * ```html *
    * {{model.name}} @@ -498,7 +508,7 @@ var ngRepeatDirective = ['$parse', '$animate', function($parse, $animate) { if (getBlockStart(block) != nextNode) { // existing item which got moved - $animate.move(getBlockNodes(block.clone), null, jqLite(previousNode)); + $animate.move(getBlockNodes(block.clone), null, previousNode); } previousNode = getBlockEnd(block); updateScope(block.scope, index, valueIdentifier, value, keyIdentifier, key, collectionLength); @@ -510,8 +520,7 @@ var ngRepeatDirective = ['$parse', '$animate', function($parse, $animate) { var endNode = ngRepeatEndComment.cloneNode(false); clone[clone.length++] = endNode; - // TODO(perf): support naked previousNode in `enter` to avoid creation of jqLite wrapper? - $animate.enter(clone, null, jqLite(previousNode)); + $animate.enter(clone, null, previousNode); previousNode = endNode; // Note: We only need the first/last node of the cloned nodes. // However, we need to keep the reference to the jqlite wrapper as it might be changed later diff --git a/src/ng/directive/select.js b/src/ng/directive/select.js index df2bcee32437..a35f886fb757 100644 --- a/src/ng/directive/select.js +++ b/src/ng/directive/select.js @@ -2,6 +2,15 @@ var noopNgModelController = { $setViewValue: noop, $render: noop }; +function chromeHack(optionElement) { + // Workaround for https://code.google.com/p/chromium/issues/detail?id=381459 + // Adding an