Compare commits
420 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
cdda60a430 | ||
![]() |
7570356bfa | ||
![]() |
c4a97d822e | ||
![]() |
fc4dfc5eb1 | ||
![]() |
25d95dae6d | ||
![]() |
1c4fcce2a1 | ||
![]() |
6e73faaed7 | ||
![]() |
41c9910613 | ||
![]() |
aaddef213d | ||
![]() |
02f3aee1db | ||
![]() |
c27ba96093 | ||
![]() |
c5a474cb54 | ||
![]() |
d5264f5645 | ||
![]() |
0cd4b87021 | ||
![]() |
b1e7775a8a | ||
![]() |
87f5feff11 | ||
![]() |
c3ddc3d6b1 | ||
![]() |
cec39a7d16 | ||
![]() |
c6c8e15813 | ||
![]() |
752fd14fe5 | ||
![]() |
776067cd43 | ||
![]() |
e87a46be21 | ||
![]() |
89a7ff3ada | ||
![]() |
3d6e50dc02 | ||
![]() |
264950bbf2 | ||
![]() |
84c5be0b5b | ||
![]() |
eda8f2f8b9 | ||
![]() |
cc52945d00 | ||
![]() |
07f184a69d | ||
![]() |
a123ef58b1 | ||
![]() |
024126dde4 | ||
![]() |
4275c34818 | ||
![]() |
c4e6f585c5 | ||
![]() |
7467fd36b9 | ||
![]() |
aca01985fd | ||
![]() |
eb5e14e6e0 | ||
![]() |
b8af10902f | ||
![]() |
f411c9e5b9 | ||
![]() |
7f455e6eec | ||
![]() |
e36caafa52 | ||
![]() |
5e89d98876 | ||
![]() |
200dbd4860 | ||
![]() |
c90952884a | ||
![]() |
7c2d8fc672 | ||
![]() |
a50a688aaf | ||
![]() |
6ec7297e43 | ||
![]() |
f264cd1cb8 | ||
![]() |
fc17bddcde | ||
![]() |
0765626761 | ||
![]() |
f146b34042 | ||
![]() |
f899d6ea44 | ||
![]() |
c18d7a1469 | ||
![]() |
2c7ff82f31 | ||
![]() |
3f8f3a2daa | ||
![]() |
f5eeb1a714 | ||
![]() |
2af3d9c040 | ||
![]() |
4664acce50 | ||
![]() |
3797861dfe | ||
![]() |
d6d3984524 | ||
![]() |
c17f5c10cc | ||
![]() |
9cb318f5a2 | ||
![]() |
b026dd8b52 | ||
![]() |
0ebc316311 | ||
![]() |
dc412c5f02 | ||
![]() |
e80278cf02 | ||
![]() |
307699ac89 | ||
![]() |
4df0b7e9de | ||
![]() |
371831f9cb | ||
![]() |
8e305e7099 | ||
![]() |
481df830ad | ||
![]() |
14b4718cc2 | ||
![]() |
7b6e73cb98 | ||
![]() |
f2ca4634e2 | ||
![]() |
d30cf2f9d6 | ||
![]() |
e9cb6fbe87 | ||
![]() |
99960a98d2 | ||
![]() |
9ac3383d01 | ||
![]() |
06ac75724f | ||
![]() |
a1d691ecc8 | ||
![]() |
6e329721be | ||
![]() |
739bf5c325 | ||
![]() |
ee340b7c6c | ||
![]() |
17ddab98fb | ||
![]() |
4f65f473e4 | ||
![]() |
527a04d21e | ||
![]() |
f2ee468d76 | ||
![]() |
0320096538 | ||
![]() |
7abb48adfe | ||
![]() |
b40d3c0817 | ||
![]() |
0e5617152a | ||
![]() |
dba402344f | ||
![]() |
13d176302b | ||
![]() |
e3b801053a | ||
![]() |
6626739798 | ||
![]() |
396033da80 | ||
![]() |
02ee9d2938 | ||
![]() |
a4c7f183d7 | ||
![]() |
d690eec88f | ||
![]() |
10e4dfae27 | ||
![]() |
f8d948b46b | ||
![]() |
9cf78d5701 | ||
![]() |
45471dbbd6 | ||
![]() |
387e8386d1 | ||
![]() |
eec6e4be7a | ||
![]() |
b711f25892 | ||
![]() |
788f0453f7 | ||
![]() |
45b1775a53 | ||
![]() |
53e4ff72d2 | ||
![]() |
7813a7d129 | ||
![]() |
c0ced6dc2d | ||
![]() |
8157ee87b0 | ||
![]() |
ab051aba27 | ||
![]() |
5a61ef0c49 | ||
![]() |
c451dbda9f | ||
![]() |
0d38288078 | ||
![]() |
2e9b953e9d | ||
![]() |
a94383f168 | ||
![]() |
75c40ddd61 | ||
![]() |
86a75a0670 | ||
![]() |
c776825fdd | ||
![]() |
1cc9383d91 | ||
![]() |
8ed1e53e99 | ||
![]() |
7833c88ac4 | ||
![]() |
12f177399f | ||
![]() |
5be32366be | ||
![]() |
5b7d2eeabf | ||
![]() |
6cd10a1b10 | ||
![]() |
822652aa0d | ||
![]() |
cf47ace493 | ||
![]() |
0595f11950 | ||
![]() |
35df312ea4 | ||
![]() |
489eb8519e | ||
![]() |
b76a2dc2cb | ||
![]() |
f2f5f7fc6e | ||
![]() |
8ee23ba67b | ||
![]() |
ecb422b360 | ||
![]() |
60389d5441 | ||
![]() |
b186db70db | ||
![]() |
324b6f1b1a | ||
![]() |
cdba1d37a4 | ||
![]() |
dc42c97ee4 | ||
![]() |
bc00e8d312 | ||
![]() |
720b71d01f | ||
![]() |
1132b07c53 | ||
![]() |
9230194794 | ||
![]() |
d724896f04 | ||
![]() |
29866dfb91 | ||
![]() |
a249622159 | ||
![]() |
9f2393fb80 | ||
![]() |
d5f8040d0a | ||
![]() |
e0b8ea136b | ||
![]() |
879b2273c1 | ||
![]() |
f24972b1b1 | ||
![]() |
d2886b3bb4 | ||
![]() |
f296fea112 | ||
![]() |
2605fc46e7 | ||
![]() |
9d54b3a14b | ||
![]() |
d09a6283ed | ||
![]() |
1c168c3a44 | ||
![]() |
0f74479c47 | ||
![]() |
790bb949f6 | ||
![]() |
2adcad6dd2 | ||
![]() |
242ef1ace1 | ||
![]() |
842b6a1247 | ||
![]() |
98335529eb | ||
![]() |
ca7ee794bf | ||
![]() |
f9f2ba6faf | ||
![]() |
aea1d211d4 | ||
![]() |
57a518a36d | ||
![]() |
29b83189b0 | ||
![]() |
1d3df7885d | ||
![]() |
fd06ffa2af | ||
![]() |
36a1622dd1 | ||
![]() |
7a91b23cb5 | ||
![]() |
4b90b6a226 | ||
![]() |
b13daa4cdf | ||
![]() |
0c6f026828 | ||
![]() |
a2520bd267 | ||
![]() |
b928a209a4 | ||
![]() |
89e16ed6a5 | ||
![]() |
1a1f99af37 | ||
![]() |
df2cd37ed2 | ||
![]() |
12a71bc6bc | ||
![]() |
7d270c235a | ||
![]() |
b0b7248504 | ||
![]() |
78460c1848 | ||
![]() |
75b119eafc | ||
![]() |
64b0ae93f7 | ||
![]() |
7c0b25f5a6 | ||
![]() |
07b5df3a19 | ||
![]() |
e7023726f4 | ||
![]() |
a9ccd9254c | ||
![]() |
335f3271d2 | ||
![]() |
7f93f7ef47 | ||
![]() |
cf46a87fcd | ||
![]() |
ad6680f602 | ||
![]() |
5e287f67af | ||
![]() |
ecfe6e0609 | ||
![]() |
df9790dd11 | ||
![]() |
67cfc4c9bc | ||
![]() |
a68e623c80 | ||
![]() |
9e3915ba48 | ||
![]() |
ba2de61748 | ||
![]() |
a9a4edebe2 | ||
![]() |
64f2ffa166 | ||
![]() |
13020b9cc2 | ||
![]() |
96b96fba0f | ||
![]() |
2cbe53a9ba | ||
![]() |
48755114e5 | ||
![]() |
a5d5f67be7 | ||
![]() |
dfb58c44a2 | ||
![]() |
69948ce919 | ||
![]() |
3190ccf3b2 | ||
![]() |
a8ea8173aa | ||
![]() |
e13a49d1f0 | ||
![]() |
2f0b8f675a | ||
![]() |
c2aed033ba | ||
![]() |
0f8a780b0d | ||
![]() |
c5bc2e77c8 | ||
![]() |
079310dc7c | ||
![]() |
0d2cdf6165 | ||
![]() |
436dde271f | ||
![]() |
96891a076f | ||
![]() |
9ce0067bdf | ||
![]() |
345940bbc1 | ||
![]() |
c49507b289 | ||
![]() |
c730142508 | ||
![]() |
27aa00b15f | ||
![]() |
36a00a255b | ||
![]() |
0e3249c89b | ||
![]() |
920019ab70 | ||
![]() |
82c8b44db7 | ||
![]() |
bb3a307d5a | ||
![]() |
dcb0ddaf5e | ||
![]() |
4c1edd52c5 | ||
![]() |
d512e27979 | ||
![]() |
0619d82e0b | ||
![]() |
a4038d5b94 | ||
![]() |
e3d5e1fab7 | ||
![]() |
035036308a | ||
![]() |
0d29259d9b | ||
![]() |
26b0f3dc96 | ||
![]() |
5c9306b0fe | ||
![]() |
3befb0e4b9 | ||
![]() |
97bb88f10b | ||
![]() |
6c7467a58b | ||
![]() |
c579a85c12 | ||
![]() |
400fdd08fd | ||
![]() |
c1fe6c9c81 | ||
![]() |
c58a0bea91 | ||
![]() |
88a934b93c | ||
![]() |
cde5cced69 | ||
![]() |
472bedd3ea | ||
![]() |
d8a06d03bd | ||
![]() |
32020f9fb3 | ||
![]() |
d574b14934 | ||
![]() |
00c5d89e7d | ||
![]() |
d2c8aefe64 | ||
![]() |
ba796bbdd3 | ||
![]() |
a0bb2ba7b7 | ||
![]() |
f9fa3b5b6c | ||
![]() |
f89d438116 | ||
![]() |
1abe791d46 | ||
![]() |
1502ae78b6 | ||
![]() |
bad6e719de | ||
![]() |
8c7129f3d2 | ||
![]() |
dbff6f71e1 | ||
![]() |
fcd934ccf6 | ||
![]() |
45a8f340d9 | ||
![]() |
5856513405 | ||
![]() |
01fa3ee5c3 | ||
![]() |
b8d69ffdf3 | ||
![]() |
7ef60ec2a9 | ||
![]() |
83f905c0e1 | ||
![]() |
72ba3a3918 | ||
![]() |
df10597da4 | ||
![]() |
5db2e794a9 | ||
![]() |
486aa06747 | ||
![]() |
01ce1b32df | ||
![]() |
c78b0b2c51 | ||
![]() |
9ade1c3ea3 | ||
![]() |
315a4cfcd4 | ||
![]() |
11c04027ab | ||
![]() |
eabe3b4c39 | ||
![]() |
d471454675 | ||
![]() |
bf57776b59 | ||
![]() |
a32579ae5b | ||
![]() |
780601d27a | ||
![]() |
c909e731d7 | ||
![]() |
9b8eb42354 | ||
![]() |
0757174e8e | ||
![]() |
3a43cdefe8 | ||
![]() |
38c48beddd | ||
![]() |
ad5749fb04 | ||
![]() |
f6a838e9ee | ||
![]() |
a6d1f4aaf1 | ||
![]() |
eca8d11ee2 | ||
![]() |
a195b7dbe4 | ||
![]() |
083d7ec902 | ||
![]() |
9d2d0cae6d | ||
![]() |
c2f4a9bf71 | ||
![]() |
231095fe8a | ||
![]() |
28a532483a | ||
![]() |
83853a215d | ||
![]() |
8248307a99 | ||
![]() |
67bd88b19a | ||
![]() |
9f698b4de0 | ||
![]() |
742f3d6787 | ||
![]() |
323651bd38 | ||
![]() |
9d397eb5a1 | ||
![]() |
6114cd2bd4 | ||
![]() |
d494f7bd5e | ||
![]() |
ec6a7ab721 | ||
![]() |
ad6d2b4619 | ||
![]() |
c093390010 | ||
![]() |
acd69f2be2 | ||
![]() |
5d2f341653 | ||
![]() |
420d1c35f5 | ||
![]() |
08647267bb | ||
![]() |
215d50d2f6 | ||
![]() |
bf2cb6fa48 | ||
![]() |
e97a2d4123 | ||
![]() |
585e3f6adc | ||
![]() |
7f77ce1a48 | ||
![]() |
a1616ce181 | ||
![]() |
1c22dff714 | ||
![]() |
8d1d6e8f70 | ||
![]() |
e7f4aba5a3 | ||
![]() |
fdbe9f5d9f | ||
![]() |
8bead6bfdd | ||
![]() |
52dda73dbb | ||
![]() |
31b3888a2f | ||
![]() |
6f938470c2 | ||
![]() |
776c4afc03 | ||
![]() |
536dd647c6 | ||
![]() |
51d581ab27 | ||
![]() |
75294e7dad | ||
![]() |
04bada7a9d | ||
![]() |
2349143477 | ||
![]() |
e9bff5fe9f | ||
![]() |
411cb0cb92 | ||
![]() |
53e1fb3554 | ||
![]() |
2cb3b66640 | ||
![]() |
5af3144330 | ||
![]() |
e4043cbb3a | ||
![]() |
fff424a35f | ||
![]() |
b5d1c8b05a | ||
![]() |
d713e33cc4 | ||
![]() |
3d327d25f0 | ||
![]() |
077283bf0f | ||
![]() |
9ec25ea036 | ||
![]() |
878cfe669c | ||
![]() |
5f0be3cb2e | ||
![]() |
9e28e14c08 | ||
![]() |
954d002884 | ||
![]() |
0a48591e53 | ||
![]() |
d37c723951 | ||
![]() |
9078ca557e | ||
![]() |
2be1ef6ba0 | ||
![]() |
47c02efccb | ||
![]() |
d7ecfb432a | ||
![]() |
59abf4a33f | ||
![]() |
d6e715e726 | ||
![]() |
fcfcd1037c | ||
![]() |
f3ccd29e7b | ||
![]() |
5c0bdae809 | ||
![]() |
838902556b | ||
![]() |
c6872c02d8 | ||
![]() |
819982ea20 | ||
![]() |
f9daa136c3 | ||
![]() |
6a0d2ed6c8 | ||
![]() |
2c1f35e794 | ||
![]() |
5345e8da45 | ||
![]() |
e35269dd87 | ||
![]() |
60a03b7ef7 | ||
![]() |
305b5a3887 | ||
![]() |
bc549361d3 | ||
![]() |
084b627f2e | ||
![]() |
6755d00601 | ||
![]() |
cba1da3e44 | ||
![]() |
7be8bb1489 | ||
![]() |
c7c0c1f626 | ||
![]() |
3aa4629f92 | ||
![]() |
2d86dbb090 | ||
![]() |
91767ff0f9 | ||
![]() |
078b004ecc | ||
![]() |
930d204d83 | ||
![]() |
8d82cdfc77 | ||
![]() |
cb6996b5c3 | ||
![]() |
a4f7740332 | ||
![]() |
ba0faa2f77 | ||
![]() |
3e68029522 | ||
![]() |
b4e26b5828 | ||
![]() |
15cf7fcac2 | ||
![]() |
24ff0eb13b | ||
![]() |
cf86f72eb7 | ||
![]() |
61486f14f1 | ||
![]() |
d16a7f3ecc | ||
![]() |
82761ec50e | ||
![]() |
235bfa77a9 | ||
![]() |
299ae1bb1c | ||
![]() |
80f7522dab | ||
![]() |
028921e369 | ||
![]() |
a4e11bb524 | ||
![]() |
a4131752d2 | ||
![]() |
060dcfbba1 | ||
![]() |
4be7008f80 | ||
![]() |
4a0d05515e | ||
![]() |
83ab99c746 | ||
![]() |
270da1f69f | ||
![]() |
6b0e46e36c | ||
![]() |
3642707145 | ||
![]() |
0ea76edfd8 | ||
![]() |
d493a83b2b | ||
![]() |
f1721d5cef | ||
![]() |
5b3fd6aa82 | ||
![]() |
6f829180f7 | ||
![]() |
27b95ba64a | ||
![]() |
ef405b1e90 | ||
![]() |
441073bad5 |
@ -1,3 +1,3 @@
|
|||||||
2.1.1
|
3.2.0
|
||||||
# [NB: this comment has to be after the first line, see https://github.com/bazelbuild/bazelisk/issues/117]
|
# [NB: this comment has to be after the first line, see https://github.com/bazelbuild/bazelisk/issues/117]
|
||||||
# When updating the Bazel version you also need to update the RBE toolchains version in package.bzl
|
# When updating the Bazel version you also need to update the RBE toolchains version in package.bzl
|
||||||
|
@ -19,4 +19,12 @@ build --local_ram_resources=14336
|
|||||||
|
|
||||||
# All build executed remotely should be done using our RBE configuration.
|
# All build executed remotely should be done using our RBE configuration.
|
||||||
build:remote --google_default_credentials
|
build:remote --google_default_credentials
|
||||||
|
|
||||||
|
# Upload to GCP's Build Status viewer to allow for us to have better viewing of execution/build
|
||||||
|
# logs. This is only done on CI as the BES (GCP's Build Status viewer) API requires credentials
|
||||||
|
# from service accounts, rather than end user accounts.
|
||||||
|
build:remote --bes_backend=buildeventservice.googleapis.com
|
||||||
|
build:remote --bes_timeout=30s
|
||||||
|
build:remote --bes_results_url="https://source.cloud.google.com/results/invocations/"
|
||||||
|
|
||||||
build --config=remote
|
build --config=remote
|
||||||
|
@ -32,7 +32,7 @@ var_4_win: &cache_key_win_fallback v7-angular-win-node-12-{{ checksum ".bazelver
|
|||||||
|
|
||||||
# Cache key for the `components-repo-unit-tests` job. **Note** when updating the SHA in the
|
# Cache key for the `components-repo-unit-tests` job. **Note** when updating the SHA in the
|
||||||
# cache keys also update the SHA for the "COMPONENTS_REPO_COMMIT" environment variable.
|
# cache keys also update the SHA for the "COMPONENTS_REPO_COMMIT" environment variable.
|
||||||
var_5: &components_repo_unit_tests_cache_key v7-angular-components-189d98e8b01b33974328255f085de04251d61567
|
var_5: &components_repo_unit_tests_cache_key v7-angular-components-f428c00465dfcf8a020237f22532480eedbd2cb6
|
||||||
var_6: &components_repo_unit_tests_cache_key_fallback v7-angular-components-
|
var_6: &components_repo_unit_tests_cache_key_fallback v7-angular-components-
|
||||||
|
|
||||||
# Workspace initially persisted by the `setup` job, and then enhanced by `build-npm-packages` and
|
# Workspace initially persisted by the `setup` job, and then enhanced by `build-npm-packages` and
|
||||||
@ -67,9 +67,6 @@ var_10: &only_on_master
|
|||||||
# **NOTE 1**: Pin to exact images using an ID (SHA). See https://circleci.com/docs/2.0/circleci-images/#using-a-docker-image-id-to-pin-an-image-to-a-fixed-version.
|
# **NOTE 1**: Pin to exact images using an ID (SHA). See https://circleci.com/docs/2.0/circleci-images/#using-a-docker-image-id-to-pin-an-image-to-a-fixed-version.
|
||||||
# (Using the tag in not necessary when pinning by ID, but include it anyway for documentation purposes.)
|
# (Using the tag in not necessary when pinning by ID, but include it anyway for documentation purposes.)
|
||||||
# **NOTE 2**: If you change the version of the docker images, also change the `cache_key` suffix.
|
# **NOTE 2**: If you change the version of the docker images, also change the `cache_key` suffix.
|
||||||
# **NOTE 3**: If you change the version of the `*-browsers` docker image, make sure the
|
|
||||||
# `--versions.chrome` arg in `integration/bazel-schematics/test.sh` specifies a
|
|
||||||
# ChromeDriver version that is compatible with the Chrome version in the image.
|
|
||||||
executors:
|
executors:
|
||||||
default-executor:
|
default-executor:
|
||||||
parameters:
|
parameters:
|
||||||
@ -120,7 +117,7 @@ commands:
|
|||||||
sudo apt-get update
|
sudo apt-get update
|
||||||
# Install GTK+ graphical user interface (libgtk-3-0), advanced linux sound architecture (libasound2)
|
# Install GTK+ graphical user interface (libgtk-3-0), advanced linux sound architecture (libasound2)
|
||||||
# and network security service libraries (libnss3) & X11 Screen Saver extension library (libssx1)
|
# and network security service libraries (libnss3) & X11 Screen Saver extension library (libssx1)
|
||||||
# which are dependendies of chrome & needed for karma & protractor headless chrome tests.
|
# which are dependencies of chrome & needed for karma & protractor headless chrome tests.
|
||||||
# This is a very small install which takes around 7s in comparing to using the full
|
# This is a very small install which takes around 7s in comparing to using the full
|
||||||
# circleci/node:x.x.x-browsers image.
|
# circleci/node:x.x.x-browsers image.
|
||||||
sudo apt-get -y install libgtk-3-0 libasound2 libnss3 libxss1
|
sudo apt-get -y install libgtk-3-0 libasound2 libnss3 libxss1
|
||||||
@ -163,7 +160,7 @@ commands:
|
|||||||
description: Sets up a domain that resolves to the local host.
|
description: Sets up a domain that resolves to the local host.
|
||||||
steps:
|
steps:
|
||||||
- run:
|
- run:
|
||||||
name: Preparing environment for running tests on Saucelabs.
|
name: Preparing environment for running tests on Sauce Labs.
|
||||||
command: |
|
command: |
|
||||||
# For SauceLabs jobs, we set up a domain which resolves to the machine which launched
|
# For SauceLabs jobs, we set up a domain which resolves to the machine which launched
|
||||||
# the tunnel. We do this because devices are sometimes not able to properly resolve
|
# the tunnel. We do this because devices are sometimes not able to properly resolve
|
||||||
@ -175,13 +172,13 @@ commands:
|
|||||||
setSecretVar SAUCE_ACCESS_KEY $(echo $SAUCE_ACCESS_KEY | rev)
|
setSecretVar SAUCE_ACCESS_KEY $(echo $SAUCE_ACCESS_KEY | rev)
|
||||||
- run:
|
- run:
|
||||||
# Sets up a local domain in the machine's host file that resolves to the local
|
# Sets up a local domain in the machine's host file that resolves to the local
|
||||||
# host. This domain is helpful in Saucelabs tests where devices are not able to
|
# host. This domain is helpful in Sauce Labs tests where devices are not able to
|
||||||
# properly resolve `localhost` or `127.0.0.1` through the sauce-connect tunnel.
|
# properly resolve `localhost` or `127.0.0.1` through the sauce-connect tunnel.
|
||||||
name: Setting up alias domain for local host.
|
name: Setting up alias domain for local host.
|
||||||
command: echo "127.0.0.1 $SAUCE_LOCALHOST_ALIAS_DOMAIN" | sudo tee -a /etc/hosts
|
command: echo "127.0.0.1 $SAUCE_LOCALHOST_ALIAS_DOMAIN" | sudo tee -a /etc/hosts
|
||||||
|
|
||||||
# Normally this would be an individual job instead of a command.
|
# Normally this would be an individual job instead of a command.
|
||||||
# But startup and setup time for each invidual windows job are high enough to discourage
|
# But startup and setup time for each individual windows job are high enough to discourage
|
||||||
# many small jobs, so instead we use a command for setup unless the gain becomes significant.
|
# many small jobs, so instead we use a command for setup unless the gain becomes significant.
|
||||||
setup_win:
|
setup_win:
|
||||||
description: Setup windows node environment
|
description: Setup windows node environment
|
||||||
@ -599,8 +596,8 @@ jobs:
|
|||||||
- run:
|
- run:
|
||||||
name: Decrypt github credentials
|
name: Decrypt github credentials
|
||||||
# We need ensure that the same default digest is used for encoding and decoding with
|
# We need ensure that the same default digest is used for encoding and decoding with
|
||||||
# openssl. Openssl versions might have different default digests which can cause
|
# OpenSSL. OpenSSL versions might have different default digests which can cause
|
||||||
# decryption failures based on the installed openssl version. https://stackoverflow.com/a/39641378/4317734
|
# decryption failures based on the installed OpenSSL version. https://stackoverflow.com/a/39641378/4317734
|
||||||
command: 'openssl aes-256-cbc -d -in .circleci/github_token -md md5 -k "${KEY}" -out ~/.git_credentials'
|
command: 'openssl aes-256-cbc -d -in .circleci/github_token -md md5 -k "${KEY}" -out ~/.git_credentials'
|
||||||
- run: ./scripts/ci/publish-build-artifacts.sh
|
- run: ./scripts/ci/publish-build-artifacts.sh
|
||||||
|
|
||||||
|
@ -74,7 +74,7 @@ setPublicVar COMPONENTS_REPO_TMP_DIR "/tmp/angular-components-repo"
|
|||||||
setPublicVar COMPONENTS_REPO_URL "https://github.com/angular/components.git"
|
setPublicVar COMPONENTS_REPO_URL "https://github.com/angular/components.git"
|
||||||
setPublicVar COMPONENTS_REPO_BRANCH "master"
|
setPublicVar COMPONENTS_REPO_BRANCH "master"
|
||||||
# **NOTE**: When updating the commit SHA, also update the cache key in the CircleCI `config.yml`.
|
# **NOTE**: When updating the commit SHA, also update the cache key in the CircleCI `config.yml`.
|
||||||
setPublicVar COMPONENTS_REPO_COMMIT "189d98e8b01b33974328255f085de04251d61567"
|
setPublicVar COMPONENTS_REPO_COMMIT "f428c00465dfcf8a020237f22532480eedbd2cb6"
|
||||||
|
|
||||||
|
|
||||||
####################################################################################################
|
####################################################################################################
|
||||||
|
@ -60,14 +60,15 @@ if (require.resolve === module) {
|
|||||||
|
|
||||||
// Helpers
|
// Helpers
|
||||||
function _main(args) {
|
function _main(args) {
|
||||||
triggerWebhook(...args).
|
triggerWebhook(...args)
|
||||||
then(({statusCode, responseText}) => (200 <= statusCode && statusCode < 400) ?
|
.then(
|
||||||
console.log(`Status: ${statusCode}\n${responseText}`) :
|
({statusCode, responseText}) => (200 <= statusCode && statusCode < 400) ?
|
||||||
Promise.reject(new Error(`Request failed (status: ${statusCode}): ${responseText}`))).
|
console.log(`Status: ${statusCode}\n${responseText}`) :
|
||||||
catch(err => {
|
Promise.reject(new Error(`Request failed (status: ${statusCode}): ${responseText}`)))
|
||||||
console.error(err);
|
.catch(err => {
|
||||||
process.exit(1);
|
console.error(err);
|
||||||
});
|
process.exit(1);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function postJson(url, data) {
|
function postJson(url, data) {
|
||||||
@ -77,15 +78,12 @@ function postJson(url, data) {
|
|||||||
const statusCode = res.statusCode || -1;
|
const statusCode = res.statusCode || -1;
|
||||||
let responseText = '';
|
let responseText = '';
|
||||||
|
|
||||||
res.
|
res.on('error', reject)
|
||||||
on('error', reject).
|
.on('data', d => responseText += d)
|
||||||
on('data', d => responseText += d).
|
.on('end', () => resolve({statusCode, responseText}));
|
||||||
on('end', () => resolve({statusCode, responseText}));
|
|
||||||
};
|
};
|
||||||
|
|
||||||
request(url, opts, onResponse).
|
request(url, opts, onResponse).on('error', reject).end(JSON.stringify(data));
|
||||||
on('error', reject).
|
|
||||||
end(JSON.stringify(data));
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
4
.github/ISSUE_TEMPLATE/1-bug-report.md
vendored
4
.github/ISSUE_TEMPLATE/1-bug-report.md
vendored
@ -32,13 +32,13 @@ Existing issues often contain information about workarounds, resolution, or prog
|
|||||||
|
|
||||||
## 🔬 Minimal Reproduction
|
## 🔬 Minimal Reproduction
|
||||||
<!--
|
<!--
|
||||||
Please create and share minimal reproduction of the issue starting with this template: https://stackblitz.com/fork/angular-issue-repro2
|
Please create and share minimal reproduction of the issue starting with this template: https://stackblitz.com/fork/angular-ivy
|
||||||
-->
|
-->
|
||||||
<!-- ✍️--> https://stackblitz.com/...
|
<!-- ✍️--> https://stackblitz.com/...
|
||||||
|
|
||||||
<!--
|
<!--
|
||||||
If StackBlitz is not suitable for reproduction of your issue, please create a minimal GitHub repository with the reproduction of the issue.
|
If StackBlitz is not suitable for reproduction of your issue, please create a minimal GitHub repository with the reproduction of the issue.
|
||||||
A good way to make a minimal reproduction is to create a new app via `ng new repro-app` and add the minimum possible code to show the problem.
|
A good way to make a minimal reproduction is to create a new app via `ng new repro-app` and add the minimum possible code to show the problem.
|
||||||
Share the link to the repo below along with step-by-step instructions to reproduce the problem, as well as expected and actual behavior.
|
Share the link to the repo below along with step-by-step instructions to reproduce the problem, as well as expected and actual behavior.
|
||||||
|
|
||||||
Issues that don't have enough info and can't be reproduced will be closed.
|
Issues that don't have enough info and can't be reproduced will be closed.
|
||||||
|
6
.github/angular-robot.yml
vendored
6
.github/angular-robot.yml
vendored
@ -154,6 +154,12 @@ triage:
|
|||||||
-
|
-
|
||||||
- "type: RFC / Discussion / question"
|
- "type: RFC / Discussion / question"
|
||||||
- "comp: *"
|
- "comp: *"
|
||||||
|
-
|
||||||
|
- "type: confusing"
|
||||||
|
- "comp: *"
|
||||||
|
-
|
||||||
|
- "type: use-case"
|
||||||
|
- "comp: *"
|
||||||
|
|
||||||
# options for the triage PR plugin
|
# options for the triage PR plugin
|
||||||
triagePR:
|
triagePR:
|
||||||
|
3
.gitignore
vendored
3
.gitignore
vendored
@ -42,3 +42,6 @@ yarn-error.log
|
|||||||
|
|
||||||
.notes.md
|
.notes.md
|
||||||
baseline.json
|
baseline.json
|
||||||
|
|
||||||
|
# Ignore .history for the xyz.local-history VSCode extension
|
||||||
|
.history
|
||||||
|
145
.gitmessage
Normal file
145
.gitmessage
Normal file
@ -0,0 +1,145 @@
|
|||||||
|
<type>(<scope>): <summary>
|
||||||
|
|
||||||
|
<Describe the motivation behind this change - explain WHY you are making this change. Wrap all lines
|
||||||
|
at 100 characters.>
|
||||||
|
|
||||||
|
Fixes #<issue number>
|
||||||
|
|
||||||
|
# ────────────────────────────────────────── 100 chars ────────────────────────────────────────────┤
|
||||||
|
|
||||||
|
|
||||||
|
# Example Commit Messages
|
||||||
|
# =======================
|
||||||
|
|
||||||
|
|
||||||
|
# ─── Example: Simple refactor ────────────────────────────────────────────────────────────────────┤
|
||||||
|
# refactor(core): rename refreshDynamicEmbeddedViews to refreshEmbeddedViews
|
||||||
|
#
|
||||||
|
# Improve code readability. The original name no longer matches how the function is used.
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────────────────────────┤
|
||||||
|
|
||||||
|
|
||||||
|
# ─── Example: Simple docs change ─────────────────────────────────────────────────────────────────┤
|
||||||
|
# docs: clarify the service limitation in providers.md guide
|
||||||
|
#
|
||||||
|
# Fixes #36332
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────────────────────────┤
|
||||||
|
|
||||||
|
|
||||||
|
# ─── Example: A bug fix ──────────────────────────────────────────────────────────────────────────┤
|
||||||
|
# fix(ngcc): ensure lockfile is removed when `analyzeFn` fails
|
||||||
|
#
|
||||||
|
# Previously an error thrown in the `analyzeFn` would cause the ngcc process to exit immediately
|
||||||
|
# without removing the lockfile, and potentially before the unlocker process had been successfully
|
||||||
|
# spawned resulting in the lockfile being orphaned and left behind.
|
||||||
|
#
|
||||||
|
# Now we catch these errors and remove the lockfile as needed.
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────────────────────────┤
|
||||||
|
|
||||||
|
|
||||||
|
# ─── Example: Breaking change ────────────────────────────────────────────────────────────────────┤
|
||||||
|
# feat(bazel): simplify ng_package by dropping esm5 and fesm5
|
||||||
|
#
|
||||||
|
# esm5 and fesm5 distributions are no longer needed and have been deprecated in the past.
|
||||||
|
#
|
||||||
|
# https://v9.angular.io/guide/deprecations#esm5-and-fesm5-code-formats-in-angular-npm-packages
|
||||||
|
#
|
||||||
|
# This commit modifies ng_package to no longer distribute these two formats in npm packages built by
|
||||||
|
# ng_package (e.g. @angular/core).
|
||||||
|
#
|
||||||
|
# This commit intentionally doesn't fully clean up the ng_package rule to remove all traces of esm5
|
||||||
|
# and fems5 build artifacts as that is a bigger cleanup and currently we are narrowing down the
|
||||||
|
# scope of this change to the MVP needed for v10, which in this case is 'do not put esm5 and fesm5'
|
||||||
|
# into the npm packages.
|
||||||
|
#
|
||||||
|
# More cleanup to follow: https://angular-team.atlassian.net/browse/FW-2143
|
||||||
|
#
|
||||||
|
# BREAKING CHANGE: esm5 and fesm5 format is no longer distributed in Angular's npm packages e.g.
|
||||||
|
# @angular/core
|
||||||
|
#
|
||||||
|
# Angular CLI will automatically downlevel the code to es5 if differential loading is enabled in the
|
||||||
|
# Angular project, so no action is required from Angular CLI users.
|
||||||
|
#
|
||||||
|
# If you are not using Angular CLI to build your application or library, and you need to be able to
|
||||||
|
# build es5 artifacts, then you will need to downlevel the distributed Angular code to es5 on your
|
||||||
|
# own.
|
||||||
|
#
|
||||||
|
#
|
||||||
|
# Fixes #1234
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────────────────────────┤
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Angular Commit Message Format
|
||||||
|
# =============================
|
||||||
|
#
|
||||||
|
# The full specification of the Angular Commit Message Format can be found at
|
||||||
|
# https://github.com/angular/angular/blob/master/CONTRIBUTING.md#commit
|
||||||
|
#
|
||||||
|
# The following is an excerpt of the specification with the most commonly needed info.
|
||||||
|
#
|
||||||
|
# Each commit message consists of a *header*, a *body*, and a *footer*.
|
||||||
|
#
|
||||||
|
# <header>
|
||||||
|
# <BLANK LINE>
|
||||||
|
# <body>
|
||||||
|
# <BLANK LINE>
|
||||||
|
# <footer>
|
||||||
|
#
|
||||||
|
# The header is mandatory.
|
||||||
|
#
|
||||||
|
# The body is mandatory for all commits except for those of scope "docs". When the body is required
|
||||||
|
# it must be at least 20 characters long.
|
||||||
|
#
|
||||||
|
# The footer is optional.
|
||||||
|
#
|
||||||
|
# Any line of the commit message cannot be longer than 100 characters.
|
||||||
|
#
|
||||||
|
#
|
||||||
|
# Commit Message Header
|
||||||
|
# ---------------------
|
||||||
|
#
|
||||||
|
# <type>(<scope>): <short summary>
|
||||||
|
# │ │ │
|
||||||
|
# │ │ └─⫸ Summary in present tense. Not capitalized. No period at the end.
|
||||||
|
# │ │
|
||||||
|
# │ └─⫸ Commit Scope: animations|bazel|benchpress|common|compiler|compiler-cli|core|
|
||||||
|
# │ elements|forms|http|language-service|localize|platform-browser|
|
||||||
|
# │ platform-browser-dynamic|platform-server|platform-webworker|
|
||||||
|
# │ platform-webworker-dynamic|router|service-worker|upgrade|zone.js|
|
||||||
|
# │ packaging|changelog|dev-infra|docs-infra|migrations|ngcc|ve
|
||||||
|
# │ https://github.com/angular/angular/blob/master/CONTRIBUTING.md#scope
|
||||||
|
# │
|
||||||
|
# └─⫸ Commit Type: build|ci|docs|feat|fix|perf|refactor|style|test
|
||||||
|
# https://github.com/angular/angular/blob/master/CONTRIBUTING.md#type
|
||||||
|
#
|
||||||
|
#
|
||||||
|
# Commit Message Body
|
||||||
|
# ---------------------
|
||||||
|
#
|
||||||
|
# Just as in the summary, use the imperative, present tense: "fix" not "fixed" nor "fixes".
|
||||||
|
#
|
||||||
|
# Explain the motivation for the change in the commit message body. This commit message should
|
||||||
|
# explain WHY you are making the change. You can include a comparison of the previous behavior with
|
||||||
|
# the new behavior in order to illustrate the impact of the change.
|
||||||
|
#
|
||||||
|
#
|
||||||
|
# Commit Message Footer
|
||||||
|
# ---------------------
|
||||||
|
#
|
||||||
|
# The footer can contain information about breaking changes and is also the place to reference
|
||||||
|
# GitHub issues, Jira tickets, and other PRs that this commit closes or is related to.
|
||||||
|
#
|
||||||
|
# ```
|
||||||
|
# BREAKING CHANGE: <breaking change summary>
|
||||||
|
# <BLANK LINE>
|
||||||
|
# <breaking change description + migration instructions>
|
||||||
|
# <BLANK LINE>
|
||||||
|
# <BLANK LINE>
|
||||||
|
# Fixes #<issue number>
|
||||||
|
# ```
|
||||||
|
#
|
||||||
|
# Breaking Change section should start with the phrase "BREAKING CHANGE: " followed by a summary of
|
||||||
|
# the breaking change, a blank line, and a detailed description of the breaking change that also
|
||||||
|
# includes migration instructions.
|
||||||
|
#
|
@ -1,147 +0,0 @@
|
|||||||
import {exec} from 'shelljs';
|
|
||||||
|
|
||||||
import {MergeConfig} from './dev-infra/pr/merge/config';
|
|
||||||
|
|
||||||
// The configuration for `ng-dev commit-message` commands.
|
|
||||||
const commitMessage = {
|
|
||||||
'maxLength': 120,
|
|
||||||
'minBodyLength': 100,
|
|
||||||
'types': [
|
|
||||||
'build',
|
|
||||||
'ci',
|
|
||||||
'docs',
|
|
||||||
'feat',
|
|
||||||
'fix',
|
|
||||||
'perf',
|
|
||||||
'refactor',
|
|
||||||
'release',
|
|
||||||
'style',
|
|
||||||
'test',
|
|
||||||
],
|
|
||||||
'scopes': [
|
|
||||||
'animations',
|
|
||||||
'bazel',
|
|
||||||
'benchpress',
|
|
||||||
'changelog',
|
|
||||||
'common',
|
|
||||||
'compiler',
|
|
||||||
'compiler-cli',
|
|
||||||
'core',
|
|
||||||
'dev-infra',
|
|
||||||
'docs-infra',
|
|
||||||
'elements',
|
|
||||||
'forms',
|
|
||||||
'http',
|
|
||||||
'language-service',
|
|
||||||
'localize',
|
|
||||||
'ngcc',
|
|
||||||
'packaging',
|
|
||||||
'platform-browser',
|
|
||||||
'platform-browser-dynamic',
|
|
||||||
'platform-server',
|
|
||||||
'platform-webworker',
|
|
||||||
'platform-webworker-dynamic',
|
|
||||||
'router',
|
|
||||||
'service-worker',
|
|
||||||
'upgrade',
|
|
||||||
've',
|
|
||||||
'zone.js',
|
|
||||||
]
|
|
||||||
};
|
|
||||||
|
|
||||||
// The configuration for `ng-dev format` commands.
|
|
||||||
const format = {
|
|
||||||
'clang-format': {
|
|
||||||
'matchers': [
|
|
||||||
'dev-infra/**/*.{js,ts}',
|
|
||||||
'packages/**/*.{js,ts}',
|
|
||||||
'!packages/zone.js',
|
|
||||||
'!packages/common/locales/**/*.{js,ts}',
|
|
||||||
'!packages/common/src/i18n/available_locales.ts',
|
|
||||||
'!packages/common/src/i18n/currencies.ts',
|
|
||||||
'!packages/common/src/i18n/locale_en.ts',
|
|
||||||
'modules/benchmarks/**/*.{js,ts}',
|
|
||||||
'modules/playground/**/*.{js,ts}',
|
|
||||||
'tools/**/*.{js,ts}',
|
|
||||||
'!tools/gulp-tasks/cldr/extract.js',
|
|
||||||
'!tools/public_api_guard/**/*.d.ts',
|
|
||||||
'!tools/ts-api-guardian/test/fixtures/**',
|
|
||||||
'*.{js,ts}',
|
|
||||||
'!**/node_modules/**',
|
|
||||||
'!**/dist/**',
|
|
||||||
'!**/built/**',
|
|
||||||
'!shims_for_IE.js',
|
|
||||||
]
|
|
||||||
},
|
|
||||||
'buildifier': true
|
|
||||||
};
|
|
||||||
|
|
||||||
/** Github metadata information for `ng-dev` commands. */
|
|
||||||
const github = {
|
|
||||||
owner: 'angular',
|
|
||||||
name: 'angular',
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the name of the current patch branch. The patch branch is determined by
|
|
||||||
* looking for upstream branches that follow the format of `{major}.{minor}.x`.
|
|
||||||
*/
|
|
||||||
const getPatchBranchName = (): string => {
|
|
||||||
const branches =
|
|
||||||
exec(
|
|
||||||
`git ls-remote --heads https://github.com/${github.owner}/${github.name}.git`,
|
|
||||||
{silent: true})
|
|
||||||
.trim()
|
|
||||||
.split('\n');
|
|
||||||
|
|
||||||
for (let i = branches.length - 1; i >= 0; i--) {
|
|
||||||
const branchName = branches[i];
|
|
||||||
const matches = branchName.match(/refs\/heads\/([0-9]+\.[0-9]+\.x)/);
|
|
||||||
if (matches !== null) {
|
|
||||||
return matches[1];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
throw Error('Could not determine patch branch name.');
|
|
||||||
};
|
|
||||||
|
|
||||||
// Configuration for the `ng-dev pr merge` command. The command can be used
|
|
||||||
// for merging upstream pull requests into branches based on a PR target label.
|
|
||||||
const merge = () => {
|
|
||||||
const patchBranch = getPatchBranchName();
|
|
||||||
const config: MergeConfig = {
|
|
||||||
githubApiMerge: false,
|
|
||||||
claSignedLabel: 'cla: yes',
|
|
||||||
mergeReadyLabel: /^PR action: merge(-assistance)?/,
|
|
||||||
commitMessageFixupLabel: 'commit message fixup',
|
|
||||||
labels: [
|
|
||||||
{
|
|
||||||
pattern: 'PR target: master-only',
|
|
||||||
branches: ['master'],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
pattern: 'PR target: patch-only',
|
|
||||||
branches: [patchBranch],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
pattern: 'PR target: master & patch',
|
|
||||||
branches: ['master', patchBranch],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
requiredBaseCommits: {
|
|
||||||
// PRs that target either `master` or the patch branch, need to be rebased
|
|
||||||
// on top of the latest commit message validation fix.
|
|
||||||
'master': '4341743b4a6d7e23c6f944aa9e34166b701369a1',
|
|
||||||
[patchBranch]: '2a53f471592f424538802907aca1f60f1177a86d'
|
|
||||||
},
|
|
||||||
};
|
|
||||||
return config;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Export function to build ng-dev configuration object.
|
|
||||||
module.exports = {
|
|
||||||
commitMessage,
|
|
||||||
format,
|
|
||||||
github,
|
|
||||||
merge,
|
|
||||||
};
|
|
52
.ng-dev/commit-message.ts
Normal file
52
.ng-dev/commit-message.ts
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
import {CommitMessageConfig} from '../dev-infra/commit-message/config';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The configuration for `ng-dev commit-message` commands.
|
||||||
|
*/
|
||||||
|
export const commitMessage: CommitMessageConfig = {
|
||||||
|
maxLineLength: 120,
|
||||||
|
minBodyLength: 20,
|
||||||
|
minBodyLengthTypeExcludes: ['docs'],
|
||||||
|
types: [
|
||||||
|
'build',
|
||||||
|
'ci',
|
||||||
|
'docs',
|
||||||
|
'feat',
|
||||||
|
'fix',
|
||||||
|
'perf',
|
||||||
|
'refactor',
|
||||||
|
'release',
|
||||||
|
'style',
|
||||||
|
'test',
|
||||||
|
],
|
||||||
|
scopes: [
|
||||||
|
'animations',
|
||||||
|
'bazel',
|
||||||
|
'benchpress',
|
||||||
|
'changelog',
|
||||||
|
'common',
|
||||||
|
'compiler',
|
||||||
|
'compiler-cli',
|
||||||
|
'core',
|
||||||
|
'dev-infra',
|
||||||
|
'docs-infra',
|
||||||
|
'elements',
|
||||||
|
'forms',
|
||||||
|
'http',
|
||||||
|
'language-service',
|
||||||
|
'localize',
|
||||||
|
'migrations',
|
||||||
|
'ngcc',
|
||||||
|
'packaging',
|
||||||
|
'platform-browser',
|
||||||
|
'platform-browser-dynamic',
|
||||||
|
'platform-server',
|
||||||
|
'platform-webworker',
|
||||||
|
'platform-webworker-dynamic',
|
||||||
|
'router',
|
||||||
|
'service-worker',
|
||||||
|
'upgrade',
|
||||||
|
've',
|
||||||
|
'zone.js',
|
||||||
|
]
|
||||||
|
};
|
11
.ng-dev/config.ts
Normal file
11
.ng-dev/config.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import {commitMessage} from './commit-message';
|
||||||
|
import {format} from './format';
|
||||||
|
import {github} from './github';
|
||||||
|
import {merge} from './merge';
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
commitMessage,
|
||||||
|
format,
|
||||||
|
github,
|
||||||
|
merge,
|
||||||
|
};
|
22
.ng-dev/format.ts
Normal file
22
.ng-dev/format.ts
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
import {FormatConfig} from '../dev-infra/format/config';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configuration for the `ng-dev format` command.
|
||||||
|
*/
|
||||||
|
export const format: FormatConfig = {
|
||||||
|
'clang-format': {
|
||||||
|
'matchers': [
|
||||||
|
'**/*.{js,ts}',
|
||||||
|
// TODO: burn down format failures and remove aio and integration exceptions.
|
||||||
|
'!aio/**',
|
||||||
|
'!integration/**',
|
||||||
|
// Both third_party and .yarn are directories containing copied code which should
|
||||||
|
// not be modified.
|
||||||
|
'!third_party/**',
|
||||||
|
'!.yarn/**',
|
||||||
|
// Do not format d.ts files as they are generated
|
||||||
|
'!**/*.d.ts',
|
||||||
|
]
|
||||||
|
},
|
||||||
|
'buildifier': true
|
||||||
|
};
|
15
.ng-dev/gitconfig
Normal file
15
.ng-dev/gitconfig
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
# The file is inert unless it's explicitly included into the local git config via:
|
||||||
|
#
|
||||||
|
# ```
|
||||||
|
# git config --add include.path '../.ng-dev/gitconfig'
|
||||||
|
# ```
|
||||||
|
#
|
||||||
|
# Calling that command will append the following into `.git/config` of the current git workspace
|
||||||
|
# (i.e. $GIT_DIR, typically `angular/.git/config`):
|
||||||
|
#
|
||||||
|
# ```
|
||||||
|
# [include]
|
||||||
|
# path = ../.ng-dev/gitconfig
|
||||||
|
# ```
|
||||||
|
[commit]
|
||||||
|
template = .gitmessage
|
11
.ng-dev/github.ts
Normal file
11
.ng-dev/github.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import {GithubConfig} from '../dev-infra/utils/config';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Github configuration for the `ng-dev` command. This repository is used as
|
||||||
|
* remote for the merge script and other utilities like `ng-dev pr rebase`.
|
||||||
|
*/
|
||||||
|
|
||||||
|
export const github: GithubConfig = {
|
||||||
|
owner: 'angular',
|
||||||
|
name: 'angular'
|
||||||
|
};
|
38
.ng-dev/merge.ts
Normal file
38
.ng-dev/merge.ts
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
import {MergeConfig} from '../dev-infra/pr/merge/config';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configuration for the merge tool in `ng-dev`. This sets up the labels which
|
||||||
|
* are respected by the merge script (e.g. the target labels).
|
||||||
|
*/
|
||||||
|
export const merge = (): MergeConfig => {
|
||||||
|
// TODO: resume dynamically determining patch branch
|
||||||
|
const patch = '10.0.x';
|
||||||
|
return {
|
||||||
|
githubApiMerge: false,
|
||||||
|
claSignedLabel: 'cla: yes',
|
||||||
|
mergeReadyLabel: /^PR action: merge(-assistance)?/,
|
||||||
|
caretakerNoteLabel: 'PR action: merge-assistance',
|
||||||
|
commitMessageFixupLabel: 'commit message fixup',
|
||||||
|
labels: [
|
||||||
|
{
|
||||||
|
pattern: 'PR target: master-only',
|
||||||
|
branches: ['master'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
pattern: 'PR target: patch-only',
|
||||||
|
branches: [patch],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
pattern: 'PR target: master & patch',
|
||||||
|
branches: ['master', patch],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
requiredBaseCommits: {
|
||||||
|
// PRs that target either `master` or the patch branch, need to be rebased
|
||||||
|
// on top of the latest commit message validation fix.
|
||||||
|
// These SHAs are the commits that update the required license text in the header.
|
||||||
|
'master': '5aeb9a4124922d8ac08eb73b8f322905a32b0b3a',
|
||||||
|
[patch]: '27b95ba64a5d99757f4042073fd1860e20e3ed24'
|
||||||
|
},
|
||||||
|
};
|
||||||
|
};
|
317
.pullapprove.yml
317
.pullapprove.yml
@ -34,41 +34,8 @@
|
|||||||
####################################################################################
|
####################################################################################
|
||||||
# GitHub usernames
|
# GitHub usernames
|
||||||
####################################################################################
|
####################################################################################
|
||||||
# aikidave - Dave Shevitz
|
# See reviewer list under `required-minimum-review` group. Team member names and
|
||||||
# alan-agius4 - Alan Agius
|
# usernames are managed there.
|
||||||
# alxhub - Alex Rickabaugh
|
|
||||||
# AndrewKushnir - Andrew Kushnir
|
|
||||||
# andrewseguin - Andrew Seguin
|
|
||||||
# atscott - Andrew Scott
|
|
||||||
# ayazhafiz - Ayaz Hafiz
|
|
||||||
# clydin - Charles Lyding
|
|
||||||
# crisbeto - Kristiyan Kostadinov
|
|
||||||
# dennispbrown - Denny Brown
|
|
||||||
# devversion - Paul Gschwendtner
|
|
||||||
# dgp1130 - Doug Parker
|
|
||||||
# filipesilva - Filipe Silva
|
|
||||||
# gkalpak - Georgios Kalpakas
|
|
||||||
# gregmagolan - Greg Magolan
|
|
||||||
# IgorMinar - Igor Minar
|
|
||||||
# jbogarthyde - Judy Bogart
|
|
||||||
# jelbourn - Jeremy Elbourn
|
|
||||||
# JiaLiPassion - Jia Li
|
|
||||||
# JoostK - Joost Koehoorn
|
|
||||||
# josephperrott - Joey Perrott
|
|
||||||
# juleskremer - Jules Kremer
|
|
||||||
# kapunahelewong - Kapunahele Wong
|
|
||||||
# kara - Kara Erickson
|
|
||||||
# kyliau - Keen Yee Liau
|
|
||||||
# manughub - Manu Murthy
|
|
||||||
# matsko - Matias Niemela
|
|
||||||
# mgechev - Minko Gechev
|
|
||||||
# mhevery - Miško Hevery
|
|
||||||
# michaelprentice - Michael Prentice
|
|
||||||
# mmalerba - Miles Malerba
|
|
||||||
# petebacondarwin - Pete Bacon Darwin
|
|
||||||
# pkozlowski-opensource - Pawel Kozlowski
|
|
||||||
# robwormald - Rob Wormald
|
|
||||||
# StephenFluin - Stephen Fluin
|
|
||||||
|
|
||||||
|
|
||||||
####################################################################################
|
####################################################################################
|
||||||
@ -80,8 +47,8 @@
|
|||||||
# Used for approving minor changes, large-scale refactorings, and in emergency situations.
|
# Used for approving minor changes, large-scale refactorings, and in emergency situations.
|
||||||
#
|
#
|
||||||
# IgorMinar
|
# IgorMinar
|
||||||
|
# jelbourn
|
||||||
# josephperrott
|
# josephperrott
|
||||||
# kara
|
|
||||||
# mhevery
|
# mhevery
|
||||||
#
|
#
|
||||||
# =========================================================
|
# =========================================================
|
||||||
@ -100,8 +67,35 @@ version: 3
|
|||||||
# Meta field that goes unused by PullApprove to allow for defining aliases to be
|
# Meta field that goes unused by PullApprove to allow for defining aliases to be
|
||||||
# used throughout the config.
|
# used throughout the config.
|
||||||
meta:
|
meta:
|
||||||
1: &can-be-global-approved "\"global-approvers\" not in groups.approved"
|
# The following groups have no file based conditions and will be initially `active` on all PRs
|
||||||
2: &can-be-global-docs-approved "\"global-docs-approvers\" not in groups.approved"
|
# - `global-approvers`
|
||||||
|
# - `global-docs-approvers`
|
||||||
|
# - `required-minimum-review`
|
||||||
|
#
|
||||||
|
# By checking the number of active/pending/rejected groups when these are excluded, we can determine
|
||||||
|
# if any other groups are matched.
|
||||||
|
#
|
||||||
|
# Note: Because all inactive groups start as pending, we are only checking pending and rejected active groups.
|
||||||
|
#
|
||||||
|
# Also note that the ordering of groups matters in this file. The only groups visible to the current
|
||||||
|
# one are those that appear above it.
|
||||||
|
no-groups-above-this-pending: &no-groups-above-this-pending
|
||||||
|
len(groups.active.pending.exclude("required-minimum-review").exclude("global-approvers").exclude("global-docs-approvers")) == 0
|
||||||
|
no-groups-above-this-rejected: &no-groups-above-this-rejected
|
||||||
|
len(groups.active.rejected.exclude("required-minimum-review").exclude("global-approvers").exclude("global-docs-approvers")) == 0
|
||||||
|
no-groups-above-this-active: &no-groups-above-this-active
|
||||||
|
len(groups.active.exclude("required-minimum-review").exclude("global-approvers").exclude("global-docs-approvers")) == 0
|
||||||
|
|
||||||
|
can-be-global-approved: &can-be-global-approved "\"global-approvers\" not in groups.approved"
|
||||||
|
can-be-global-docs-approved: &can-be-global-docs-approved "\"global-docs-approvers\" not in groups.approved"
|
||||||
|
defaults: &defaults
|
||||||
|
reviews:
|
||||||
|
# Authors provide their approval implicitly, this approval allows for a reviewer
|
||||||
|
# from a group not to need a review specifically for an area of the repository
|
||||||
|
# they own. This is coupled with the `required-minimum-review` group which requires
|
||||||
|
# that all PRs are reviewed by at least one team member who is not the author of
|
||||||
|
# the PR.
|
||||||
|
author_value: 1
|
||||||
|
|
||||||
# turn on 'draft' support
|
# turn on 'draft' support
|
||||||
# https://docs.pullapprove.com/config/github-api-version/
|
# https://docs.pullapprove.com/config/github-api-version/
|
||||||
@ -121,6 +115,54 @@ pullapprove_conditions:
|
|||||||
|
|
||||||
|
|
||||||
groups:
|
groups:
|
||||||
|
# =========================================================
|
||||||
|
# Require review on all PRs
|
||||||
|
#
|
||||||
|
# All PRs require at least one review. This rule will not
|
||||||
|
# request any reviewers, however will require that at least
|
||||||
|
# one review is provided before the group is satisfied.
|
||||||
|
# =========================================================
|
||||||
|
required-minimum-review:
|
||||||
|
reviews:
|
||||||
|
request: 0 # Do not request any reviews from the reviewer group
|
||||||
|
required: 1 # Require that all PRs have approval from at least one of the users in the group
|
||||||
|
author_value: 0 # The author of the PR cannot provide an approval for themself
|
||||||
|
reviewers:
|
||||||
|
users:
|
||||||
|
- aikidave # Dave Shevitz
|
||||||
|
- alan-agius4 # Alan Agius
|
||||||
|
- alxhub # Alex Rickabaugh
|
||||||
|
- AndrewKushnir # Andrew Kushnir
|
||||||
|
- andrewseguin # Andrew Seguin
|
||||||
|
- atscott # Andrew Scott
|
||||||
|
- ayazhafiz # Ayaz Hafiz
|
||||||
|
- clydin # Charles Lyding
|
||||||
|
- crisbeto # Kristiyan Kostadinov
|
||||||
|
- dennispbrown # Denny Brown
|
||||||
|
- devversion # Paul Gschwendtner
|
||||||
|
- dgp1130 # Doug Parker
|
||||||
|
- filipesilva # Filipe Silva
|
||||||
|
- gkalpak # Georgios Kalpakas
|
||||||
|
- gregmagolan # Greg Magolan
|
||||||
|
- IgorMinar # Igor Minar
|
||||||
|
- jbogarthyde # Judy Bogart
|
||||||
|
- jelbourn # Jeremy Elbourn
|
||||||
|
- JiaLiPassion # Jia Li
|
||||||
|
- JoostK # Joost Koehoorn
|
||||||
|
- josephperrott # Joey Perrott
|
||||||
|
- juleskremer # Jules Kremer
|
||||||
|
- kapunahelewong # Kapunahele Wong
|
||||||
|
- kara # Kara Erickson
|
||||||
|
- kyliau # Keen Yee Liau
|
||||||
|
- manughub # Manu Murthy
|
||||||
|
- mgechev # Minko Gechev
|
||||||
|
- mhevery # Miško Hevery
|
||||||
|
- michaelprentice # Michael Prentice
|
||||||
|
- mmalerba # Miles Malerba
|
||||||
|
- petebacondarwin # Pete Bacon Darwin
|
||||||
|
- pkozlowski-opensource # Pawel Kozlowski
|
||||||
|
- StephenFluin # Stephen Fluin
|
||||||
|
|
||||||
# =========================================================
|
# =========================================================
|
||||||
# Global Approvers
|
# Global Approvers
|
||||||
#
|
#
|
||||||
@ -161,6 +203,7 @@ groups:
|
|||||||
# Framework: Animations
|
# Framework: Animations
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-animations:
|
fw-animations:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -178,13 +221,16 @@ groups:
|
|||||||
])
|
])
|
||||||
reviewers:
|
reviewers:
|
||||||
users:
|
users:
|
||||||
- matsko
|
- crisbeto
|
||||||
|
- IgorMinar
|
||||||
|
- jelbourn
|
||||||
|
|
||||||
|
|
||||||
# =========================================================
|
# =========================================================
|
||||||
# Framework: Compiler
|
# Framework: Compiler
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-compiler:
|
fw-compiler:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -203,13 +249,13 @@ groups:
|
|||||||
- alxhub
|
- alxhub
|
||||||
- AndrewKushnir
|
- AndrewKushnir
|
||||||
- JoostK
|
- JoostK
|
||||||
- kara
|
|
||||||
|
|
||||||
|
|
||||||
# =========================================================
|
# =========================================================
|
||||||
# Framework: Compiler / ngcc
|
# Framework: Compiler / ngcc
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-ngcc:
|
fw-ngcc:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -226,6 +272,7 @@ groups:
|
|||||||
# Framework: Migrations
|
# Framework: Migrations
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-migrations:
|
fw-migrations:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -235,13 +282,13 @@ groups:
|
|||||||
- alxhub
|
- alxhub
|
||||||
- crisbeto
|
- crisbeto
|
||||||
- devversion
|
- devversion
|
||||||
- kara
|
|
||||||
|
|
||||||
|
|
||||||
# =========================================================
|
# =========================================================
|
||||||
# Framework: Core
|
# Framework: Core
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-core:
|
fw-core:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -315,26 +362,38 @@ groups:
|
|||||||
'aio/content/guide/ngmodule-vs-jsmodule.md',
|
'aio/content/guide/ngmodule-vs-jsmodule.md',
|
||||||
'aio/content/guide/module-types.md',
|
'aio/content/guide/module-types.md',
|
||||||
'aio/content/guide/template-syntax.md',
|
'aio/content/guide/template-syntax.md',
|
||||||
|
'aio/content/guide/built-in-template-functions.md',
|
||||||
'aio/content/examples/built-in-template-functions/**',
|
'aio/content/examples/built-in-template-functions/**',
|
||||||
|
'aio/content/guide/event-binding.md',
|
||||||
'aio/content/examples/event-binding/**',
|
'aio/content/examples/event-binding/**',
|
||||||
|
'aio/content/guide/interpolation.md',
|
||||||
'aio/content/examples/interpolation/**',
|
'aio/content/examples/interpolation/**',
|
||||||
'aio/content/examples/template-syntax/**',
|
'aio/content/examples/template-syntax/**',
|
||||||
'aio/content/images/guide/template-syntax/**',
|
'aio/content/images/guide/template-syntax/**',
|
||||||
|
'aio/content/guide/binding-syntax.md',
|
||||||
'aio/content/examples/binding-syntax/**',
|
'aio/content/examples/binding-syntax/**',
|
||||||
|
'aio/content/guide/property-binding.md',
|
||||||
'aio/content/examples/property-binding/**',
|
'aio/content/examples/property-binding/**',
|
||||||
|
'aio/content/guide/attribute-binding.md',
|
||||||
'aio/content/examples/attribute-binding/**',
|
'aio/content/examples/attribute-binding/**',
|
||||||
|
'aio/content/guide/two-way-binding.md',
|
||||||
'aio/content/examples/two-way-binding/**',
|
'aio/content/examples/two-way-binding/**',
|
||||||
|
'aio/content/guide/built-in-directives.md',
|
||||||
'aio/content/examples/built-in-directives/**',
|
'aio/content/examples/built-in-directives/**',
|
||||||
'aio/content/images/guide/built-in-directives/**',
|
'aio/content/images/guide/built-in-directives/**',
|
||||||
|
'aio/content/guide/template-reference-variables.md',
|
||||||
'aio/content/examples/template-reference-variables/**',
|
'aio/content/examples/template-reference-variables/**',
|
||||||
|
'aio/content/guide/inputs-outputs.md',
|
||||||
'aio/content/examples/inputs-outputs/**',
|
'aio/content/examples/inputs-outputs/**',
|
||||||
'aio/content/images/guide/inputs-outputs/**',
|
'aio/content/images/guide/inputs-outputs/**',
|
||||||
|
'aio/content/guide/template-expression-operators.md',
|
||||||
'aio/content/examples/template-expression-operators/**',
|
'aio/content/examples/template-expression-operators/**',
|
||||||
'aio/content/guide/pipes.md',
|
'aio/content/guide/pipes.md',
|
||||||
'aio/content/examples/pipes/**',
|
'aio/content/examples/pipes/**',
|
||||||
'aio/content/images/guide/pipes/**',
|
'aio/content/images/guide/pipes/**',
|
||||||
'aio/content/guide/providers.md',
|
'aio/content/guide/providers.md',
|
||||||
'aio/content/examples/providers/**',
|
'aio/content/examples/providers/**',
|
||||||
|
'aio/content/images/guide/providers/**',
|
||||||
'aio/content/guide/singleton-services.md',
|
'aio/content/guide/singleton-services.md',
|
||||||
'aio/content/guide/set-document-title.md',
|
'aio/content/guide/set-document-title.md',
|
||||||
'aio/content/examples/set-document-title/**',
|
'aio/content/examples/set-document-title/**',
|
||||||
@ -342,7 +401,9 @@ groups:
|
|||||||
'aio/content/guide/sharing-ngmodules.md',
|
'aio/content/guide/sharing-ngmodules.md',
|
||||||
'aio/content/guide/structural-directives.md',
|
'aio/content/guide/structural-directives.md',
|
||||||
'aio/content/examples/structural-directives/**',
|
'aio/content/examples/structural-directives/**',
|
||||||
|
'aio/content/guide/svg-in-templates.md',
|
||||||
'aio/content/images/guide/structural-directives/**',
|
'aio/content/images/guide/structural-directives/**',
|
||||||
|
'aio/content/guide/template-statements.md',
|
||||||
'aio/content/guide/user-input.md',
|
'aio/content/guide/user-input.md',
|
||||||
'aio/content/examples/user-input/**',
|
'aio/content/examples/user-input/**',
|
||||||
'aio/content/images/guide/user-input/**'
|
'aio/content/images/guide/user-input/**'
|
||||||
@ -352,7 +413,7 @@ groups:
|
|||||||
- alxhub
|
- alxhub
|
||||||
- AndrewKushnir
|
- AndrewKushnir
|
||||||
- atscott
|
- atscott
|
||||||
- kara
|
- ~kara # do not request reviews from Kara, but allow her to approve PRs
|
||||||
- mhevery
|
- mhevery
|
||||||
- pkozlowski-opensource
|
- pkozlowski-opensource
|
||||||
|
|
||||||
@ -361,6 +422,7 @@ groups:
|
|||||||
# Framework: Http
|
# Framework: Http
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-http:
|
fw-http:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -382,6 +444,7 @@ groups:
|
|||||||
# Framework: Elements
|
# Framework: Elements
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-elements:
|
fw-elements:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -402,6 +465,7 @@ groups:
|
|||||||
# Framework: Forms
|
# Framework: Forms
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-forms:
|
fw-forms:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -434,6 +498,7 @@ groups:
|
|||||||
# Framework: i18n
|
# Framework: i18n
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-i18n:
|
fw-i18n:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -467,6 +532,7 @@ groups:
|
|||||||
# Framework: Platform Server
|
# Framework: Platform Server
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-platform-server:
|
fw-platform-server:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -486,6 +552,7 @@ groups:
|
|||||||
# Framework: Router
|
# Framework: Router
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-router:
|
fw-router:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -495,6 +562,7 @@ groups:
|
|||||||
'packages/examples/router/**',
|
'packages/examples/router/**',
|
||||||
'aio/content/guide/router.md',
|
'aio/content/guide/router.md',
|
||||||
'aio/content/guide/router-tutorial.md',
|
'aio/content/guide/router-tutorial.md',
|
||||||
|
'aio/content/guide/router-tutorial-toh.md',
|
||||||
'aio/content/examples/router-tutorial/**',
|
'aio/content/examples/router-tutorial/**',
|
||||||
'aio/content/examples/router/**',
|
'aio/content/examples/router/**',
|
||||||
'aio/content/images/guide/router/**'
|
'aio/content/images/guide/router/**'
|
||||||
@ -508,6 +576,7 @@ groups:
|
|||||||
# Framework: Service Worker
|
# Framework: Service Worker
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-service-worker:
|
fw-service-worker:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -535,6 +604,7 @@ groups:
|
|||||||
# Framework: Upgrade
|
# Framework: Upgrade
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-upgrade:
|
fw-upgrade:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -565,6 +635,7 @@ groups:
|
|||||||
# Framework: Testing
|
# Framework: Testing
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-testing:
|
fw-testing:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -572,6 +643,14 @@ groups:
|
|||||||
contains_any_globs(files.exclude('packages/compiler-cli/**'), [
|
contains_any_globs(files.exclude('packages/compiler-cli/**'), [
|
||||||
'**/testing/**',
|
'**/testing/**',
|
||||||
'aio/content/guide/testing.md',
|
'aio/content/guide/testing.md',
|
||||||
|
'aio/content/guide/test-debugging.md',
|
||||||
|
'aio/content/guide/testing-attribute-directives.md',
|
||||||
|
'aio/content/guide/testing-code-coverage.md',
|
||||||
|
'aio/content/guide/testing-components-basics.md',
|
||||||
|
'aio/content/guide/testing-components-scenarios.md',
|
||||||
|
'aio/content/guide/testing-pipes.md',
|
||||||
|
'aio/content/guide/testing-services.md',
|
||||||
|
'aio/content/guide/testing-utility-apis.md',
|
||||||
'aio/content/examples/testing/**',
|
'aio/content/examples/testing/**',
|
||||||
'aio/content/images/guide/testing/**'
|
'aio/content/images/guide/testing/**'
|
||||||
])
|
])
|
||||||
@ -579,7 +658,6 @@ groups:
|
|||||||
users:
|
users:
|
||||||
- AndrewKushnir
|
- AndrewKushnir
|
||||||
- IgorMinar
|
- IgorMinar
|
||||||
- kara
|
|
||||||
- pkozlowski-opensource
|
- pkozlowski-opensource
|
||||||
|
|
||||||
|
|
||||||
@ -587,6 +665,7 @@ groups:
|
|||||||
# Framework: Benchmarks
|
# Framework: Benchmarks
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-benchmarks:
|
fw-benchmarks:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- >
|
- >
|
||||||
@ -596,7 +675,6 @@ groups:
|
|||||||
reviewers:
|
reviewers:
|
||||||
users:
|
users:
|
||||||
- IgorMinar
|
- IgorMinar
|
||||||
- kara
|
|
||||||
- pkozlowski-opensource
|
- pkozlowski-opensource
|
||||||
|
|
||||||
|
|
||||||
@ -604,6 +682,7 @@ groups:
|
|||||||
# Framework: Playground
|
# Framework: Playground
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-playground:
|
fw-playground:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- >
|
- >
|
||||||
@ -613,13 +692,15 @@ groups:
|
|||||||
reviewers:
|
reviewers:
|
||||||
users:
|
users:
|
||||||
- IgorMinar
|
- IgorMinar
|
||||||
- kara
|
- jelbourn
|
||||||
|
- pkozlowski-opensource
|
||||||
|
|
||||||
|
|
||||||
# =========================================================
|
# =========================================================
|
||||||
# Framework: Security
|
# Framework: Security
|
||||||
# =========================================================
|
# =========================================================
|
||||||
fw-security:
|
fw-security:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -637,18 +718,25 @@ groups:
|
|||||||
users:
|
users:
|
||||||
- IgorMinar
|
- IgorMinar
|
||||||
- mhevery
|
- mhevery
|
||||||
|
- jelbourn
|
||||||
|
- pkozlowski-opensource
|
||||||
|
reviews:
|
||||||
|
request: -1 # request reviews from everyone
|
||||||
|
required: 2 # require at least 2 approvals
|
||||||
|
reviewed_for: required
|
||||||
|
|
||||||
|
|
||||||
# =========================================================
|
# =========================================================
|
||||||
# Bazel
|
# Bazel
|
||||||
# =========================================================
|
# =========================================================
|
||||||
bazel:
|
bazel:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
- >
|
- >
|
||||||
contains_any_globs(files, [
|
contains_any_globs(files, [
|
||||||
'packages/bazel/**',
|
'packages/bazel/**',
|
||||||
'aio/content/guide/bazel.md'
|
|
||||||
])
|
])
|
||||||
reviewers:
|
reviewers:
|
||||||
users:
|
users:
|
||||||
@ -661,6 +749,7 @@ groups:
|
|||||||
# Language Service
|
# Language Service
|
||||||
# =========================================================
|
# =========================================================
|
||||||
language-service:
|
language-service:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -680,6 +769,7 @@ groups:
|
|||||||
# zone.js
|
# zone.js
|
||||||
# =========================================================
|
# =========================================================
|
||||||
zone-js:
|
zone-js:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -698,6 +788,7 @@ groups:
|
|||||||
# Benchpress
|
# Benchpress
|
||||||
# =========================================================
|
# =========================================================
|
||||||
benchpress:
|
benchpress:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -708,12 +799,14 @@ groups:
|
|||||||
reviewers:
|
reviewers:
|
||||||
users:
|
users:
|
||||||
- alxhub
|
- alxhub
|
||||||
|
- josephperrott
|
||||||
|
|
||||||
|
|
||||||
# =========================================================
|
# =========================================================
|
||||||
# Integration Tests
|
# Integration Tests
|
||||||
# =========================================================
|
# =========================================================
|
||||||
integration-tests:
|
integration-tests:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- >
|
- >
|
||||||
@ -724,7 +817,6 @@ groups:
|
|||||||
users:
|
users:
|
||||||
- IgorMinar
|
- IgorMinar
|
||||||
- josephperrott
|
- josephperrott
|
||||||
- kara
|
|
||||||
- mhevery
|
- mhevery
|
||||||
|
|
||||||
|
|
||||||
@ -732,6 +824,7 @@ groups:
|
|||||||
# Docs: Gettings Started & Tutorial
|
# Docs: Gettings Started & Tutorial
|
||||||
# =========================================================
|
# =========================================================
|
||||||
docs-getting-started-and-tutorial:
|
docs-getting-started-and-tutorial:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -764,6 +857,7 @@ groups:
|
|||||||
# Docs: Marketing
|
# Docs: Marketing
|
||||||
# =========================================================
|
# =========================================================
|
||||||
docs-marketing:
|
docs-marketing:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -786,6 +880,7 @@ groups:
|
|||||||
# Docs: Observables
|
# Docs: Observables
|
||||||
# =========================================================
|
# =========================================================
|
||||||
docs-observables:
|
docs-observables:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -811,6 +906,7 @@ groups:
|
|||||||
# Docs: Packaging, Tooling, Releasing
|
# Docs: Packaging, Tooling, Releasing
|
||||||
# =========================================================
|
# =========================================================
|
||||||
docs-packaging-and-releasing:
|
docs-packaging-and-releasing:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -830,20 +926,47 @@ groups:
|
|||||||
'aio/content/guide/migration-localize.md',
|
'aio/content/guide/migration-localize.md',
|
||||||
'aio/content/guide/migration-module-with-providers.md',
|
'aio/content/guide/migration-module-with-providers.md',
|
||||||
'aio/content/guide/static-query-migration.md',
|
'aio/content/guide/static-query-migration.md',
|
||||||
'aio/content/guide/updating-to-version-9.md',
|
'aio/content/guide/updating-to-version-10.md',
|
||||||
'aio/content/guide/ivy-compatibility.md',
|
'aio/content/guide/ivy-compatibility.md',
|
||||||
'aio/content/guide/ivy-compatibility-examples.md'
|
'aio/content/guide/ivy-compatibility-examples.md'
|
||||||
])
|
])
|
||||||
reviewers:
|
reviewers:
|
||||||
users:
|
users:
|
||||||
- IgorMinar
|
- IgorMinar
|
||||||
- kara
|
- jelbourn
|
||||||
|
|
||||||
|
|
||||||
|
# =========================================================
|
||||||
|
# Tooling: Compiler API shared with Angular CLI
|
||||||
|
#
|
||||||
|
# Changing this API might break Angular CLI, so we require
|
||||||
|
# the CLI team to approve changes here.
|
||||||
|
# =========================================================
|
||||||
|
tooling-cli-shared-api:
|
||||||
|
conditions:
|
||||||
|
- *can-be-global-approved
|
||||||
|
- *can-be-global-docs-approved
|
||||||
|
- >
|
||||||
|
contains_any_globs(files, [
|
||||||
|
'packages/compiler-cli/src/tooling.ts'
|
||||||
|
])
|
||||||
|
reviewers:
|
||||||
|
users:
|
||||||
|
- alan-agius4
|
||||||
|
- clydin
|
||||||
|
- kyliau
|
||||||
|
- IgorMinar
|
||||||
|
reviews:
|
||||||
|
request: -1 # request reviews from everyone
|
||||||
|
required: 2 # require at least 2 approvals
|
||||||
|
reviewed_for: required
|
||||||
|
|
||||||
|
|
||||||
# =========================================================
|
# =========================================================
|
||||||
# Docs: CLI
|
# Docs: CLI
|
||||||
# =========================================================
|
# =========================================================
|
||||||
docs-cli:
|
docs-cli:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -860,8 +983,12 @@ groups:
|
|||||||
'aio/content/images/guide/deployment/**',
|
'aio/content/images/guide/deployment/**',
|
||||||
'aio/content/guide/file-structure.md',
|
'aio/content/guide/file-structure.md',
|
||||||
'aio/content/guide/ivy.md',
|
'aio/content/guide/ivy.md',
|
||||||
|
'aio/content/guide/strict-mode.md',
|
||||||
'aio/content/guide/web-worker.md',
|
'aio/content/guide/web-worker.md',
|
||||||
'aio/content/guide/workspace-config.md',
|
'aio/content/guide/workspace-config.md',
|
||||||
|
'aio/content/guide/migration-solution-style-tsconfig.md',
|
||||||
|
'aio/content/guide/migration-update-module-and-target-compiler-options.md',
|
||||||
|
'aio/content/guide/migration-update-libraries-tslib.md',
|
||||||
])
|
])
|
||||||
reviewers:
|
reviewers:
|
||||||
users:
|
users:
|
||||||
@ -874,6 +1001,7 @@ groups:
|
|||||||
# Docs: CLI Libraries
|
# Docs: CLI Libraries
|
||||||
# =========================================================
|
# =========================================================
|
||||||
docs-libraries:
|
docs-libraries:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -894,6 +1022,7 @@ groups:
|
|||||||
# Docs: Schematics
|
# Docs: Schematics
|
||||||
# =========================================================
|
# =========================================================
|
||||||
docs-schematics:
|
docs-schematics:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -916,6 +1045,7 @@ groups:
|
|||||||
# Docs-infra
|
# Docs-infra
|
||||||
# =========================================================
|
# =========================================================
|
||||||
docs-infra:
|
docs-infra:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- *can-be-global-docs-approved
|
- *can-be-global-docs-approved
|
||||||
@ -945,14 +1075,16 @@ groups:
|
|||||||
# Dev-infra
|
# Dev-infra
|
||||||
# =========================================================
|
# =========================================================
|
||||||
dev-infra:
|
dev-infra:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- >
|
- >
|
||||||
contains_any_globs(files.exclude("CHANGELOG.md"), [
|
contains_any_globs(files.exclude("CHANGELOG.md").exclude("packages/compiler-cli/**/BUILD.bazel"), [
|
||||||
'*',
|
'*',
|
||||||
'.circleci/**',
|
'.circleci/**',
|
||||||
'.devcontainer/**',
|
'.devcontainer/**',
|
||||||
'.github/**',
|
'.github/**',
|
||||||
|
'.ng-dev/**',
|
||||||
'.vscode/**',
|
'.vscode/**',
|
||||||
'.yarn/**',
|
'.yarn/**',
|
||||||
'dev-infra/**',
|
'dev-infra/**',
|
||||||
@ -968,8 +1100,6 @@ groups:
|
|||||||
'docs/TOOLS.md',
|
'docs/TOOLS.md',
|
||||||
'docs/TRIAGE_AND_LABELS.md',
|
'docs/TRIAGE_AND_LABELS.md',
|
||||||
'goldens/*',
|
'goldens/*',
|
||||||
'modules/e2e_util/e2e_util.ts',
|
|
||||||
'modules/e2e_util/perf_util.ts',
|
|
||||||
'modules/*',
|
'modules/*',
|
||||||
'packages/*',
|
'packages/*',
|
||||||
'packages/examples/test-utils/**',
|
'packages/examples/test-utils/**',
|
||||||
@ -977,15 +1107,10 @@ groups:
|
|||||||
'packages/examples/*',
|
'packages/examples/*',
|
||||||
'scripts/**',
|
'scripts/**',
|
||||||
'third_party/**',
|
'third_party/**',
|
||||||
'tools/brotli-cli/**',
|
|
||||||
'tools/browsers/**',
|
|
||||||
'tools/build/**',
|
'tools/build/**',
|
||||||
'tools/circular_dependency_test/**',
|
'tools/circular_dependency_test/**',
|
||||||
'tools/contributing-stats/**',
|
'tools/contributing-stats/**',
|
||||||
'tools/components/**',
|
|
||||||
'tools/gulp-tasks/**',
|
'tools/gulp-tasks/**',
|
||||||
'tools/ng_rollup_bundle/**',
|
|
||||||
'tools/ngcontainer/**',
|
|
||||||
'tools/npm/**',
|
'tools/npm/**',
|
||||||
'tools/npm_integration_test/**',
|
'tools/npm_integration_test/**',
|
||||||
'tools/rxjs/**',
|
'tools/rxjs/**',
|
||||||
@ -1015,7 +1140,10 @@ groups:
|
|||||||
# Public API
|
# Public API
|
||||||
# =========================================================
|
# =========================================================
|
||||||
public-api:
|
public-api:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
|
- *no-groups-above-this-pending
|
||||||
|
- *no-groups-above-this-rejected
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- >
|
- >
|
||||||
contains_any_globs(files, [
|
contains_any_globs(files, [
|
||||||
@ -1029,15 +1157,27 @@ groups:
|
|||||||
])
|
])
|
||||||
reviewers:
|
reviewers:
|
||||||
users:
|
users:
|
||||||
|
- AndrewKushnir
|
||||||
- IgorMinar
|
- IgorMinar
|
||||||
- kara
|
- alxhub
|
||||||
|
- atscott
|
||||||
|
- jelbourn
|
||||||
|
- petebacondarwin
|
||||||
|
- pkozlowski-opensource
|
||||||
|
reviews:
|
||||||
|
request: 4 # Request reviews from four people
|
||||||
|
required: 3 # Require that three people approve
|
||||||
|
reviewed_for: required
|
||||||
|
|
||||||
|
|
||||||
# ================================================
|
# ================================================
|
||||||
# Size tracking
|
# Size tracking
|
||||||
# ================================================
|
# ================================================
|
||||||
size-tracking:
|
size-tracking:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
|
- *no-groups-above-this-pending
|
||||||
|
- *no-groups-above-this-rejected
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- >
|
- >
|
||||||
contains_any_globs(files, [
|
contains_any_globs(files, [
|
||||||
@ -1045,15 +1185,27 @@ groups:
|
|||||||
])
|
])
|
||||||
reviewers:
|
reviewers:
|
||||||
users:
|
users:
|
||||||
|
- AndrewKushnir
|
||||||
- IgorMinar
|
- IgorMinar
|
||||||
- kara
|
- alxhub
|
||||||
|
- atscott
|
||||||
|
- jelbourn
|
||||||
|
- petebacondarwin
|
||||||
|
- pkozlowski-opensource
|
||||||
|
reviews:
|
||||||
|
request: 4 # Request reviews from four people
|
||||||
|
required: 2 # Require that two people approve
|
||||||
|
reviewed_for: required
|
||||||
|
|
||||||
|
|
||||||
# ================================================
|
# ================================================
|
||||||
# Circular dependencies
|
# Circular dependencies
|
||||||
# ================================================
|
# ================================================
|
||||||
circular-dependencies:
|
circular-dependencies:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
|
- *no-groups-above-this-pending
|
||||||
|
- *no-groups-above-this-rejected
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- >
|
- >
|
||||||
contains_any_globs(files, [
|
contains_any_globs(files, [
|
||||||
@ -1061,9 +1213,13 @@ groups:
|
|||||||
])
|
])
|
||||||
reviewers:
|
reviewers:
|
||||||
users:
|
users:
|
||||||
|
- AndrewKushnir
|
||||||
- IgorMinar
|
- IgorMinar
|
||||||
- josephperrott
|
- alxhub
|
||||||
- kara
|
- atscott
|
||||||
|
- jelbourn
|
||||||
|
- petebacondarwin
|
||||||
|
- pkozlowski-opensource
|
||||||
|
|
||||||
|
|
||||||
####################################################################################
|
####################################################################################
|
||||||
@ -1074,6 +1230,7 @@ groups:
|
|||||||
# Code Ownership
|
# Code Ownership
|
||||||
# =========================================================
|
# =========================================================
|
||||||
code-ownership:
|
code-ownership:
|
||||||
|
<<: *defaults
|
||||||
conditions:
|
conditions:
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
- >
|
- >
|
||||||
@ -1082,19 +1239,43 @@ groups:
|
|||||||
])
|
])
|
||||||
reviewers:
|
reviewers:
|
||||||
users:
|
users:
|
||||||
|
- AndrewKushnir
|
||||||
- IgorMinar
|
- IgorMinar
|
||||||
|
- alxhub
|
||||||
|
- atscott
|
||||||
|
- jelbourn
|
||||||
|
- josephperrott
|
||||||
|
- mhevery
|
||||||
|
|
||||||
|
|
||||||
# ====================================================
|
# ====================================================
|
||||||
# Catch all for if no groups match the code change
|
# Catch all for if no groups match the code change
|
||||||
# ====================================================
|
# ====================================================
|
||||||
fallback:
|
fallback:
|
||||||
|
<<: *defaults
|
||||||
|
# A group is considered to be `active` for a PR if at least one of group's
|
||||||
|
# conditions matches the PR.
|
||||||
|
#
|
||||||
|
# The PullApprove CI check should fail if a PR has no `active` groups, as
|
||||||
|
# this indicates the PR is modifying a file that has no owner.
|
||||||
|
#
|
||||||
|
# This is enforced through the pullapprove verification check done
|
||||||
|
# as part of the CircleCI lint job. Failures in this lint job should be
|
||||||
|
# fixed as part of the PR. This can be done by updating the
|
||||||
|
# `.pullapprove.yml` file cover the unmatched path.
|
||||||
|
# The pullapprove verification script is part of the ng-dev tool and can be
|
||||||
|
# run locally with the command: `yarn -s ng-dev pullapprove verify`
|
||||||
|
#
|
||||||
|
# For cases in which the verification check fails to ensure coverage, this
|
||||||
|
# group will be active. The expectation is that this should be remedied
|
||||||
|
# before merging the PR as described above. In an emergency situation
|
||||||
|
# `global-approvers` can still approve PRs that match this `fallback` rule,
|
||||||
|
# but that should be an exception and not an expectation.
|
||||||
conditions:
|
conditions:
|
||||||
|
- *no-groups-above-this-active
|
||||||
|
# When any of the `global-*` groups is approved, they cause other groups to deactivate.
|
||||||
|
# In those cases, the condition above would evaluate to `true` while in reality, only a global
|
||||||
|
# approval has been provided. To ensure we don't activate the fallback group in such cases,
|
||||||
|
# ensure that no explicit global approval has been provided.
|
||||||
- *can-be-global-approved
|
- *can-be-global-approved
|
||||||
# Groups which are found to have matching conditions are `active`
|
- *can-be-global-docs-approved
|
||||||
# according to PullApprove. If no groups are matched and considered
|
|
||||||
# active, we still want to have a review occur.
|
|
||||||
- len(groups.active) == 0
|
|
||||||
reviewers:
|
|
||||||
users:
|
|
||||||
- IgorMinar
|
|
||||||
|
3
.vscode/recommended-settings.json
vendored
3
.vscode/recommended-settings.json
vendored
@ -26,6 +26,7 @@
|
|||||||
"**/bazel-out": true,
|
"**/bazel-out": true,
|
||||||
"**/dist": true,
|
"**/dist": true,
|
||||||
"**/aio/src/generated": true,
|
"**/aio/src/generated": true,
|
||||||
|
".history": true,
|
||||||
},
|
},
|
||||||
"git.ignoreLimitWarning": true,
|
"git.ignoreLimitWarning": true,
|
||||||
}
|
}
|
@ -2,7 +2,6 @@ package(default_visibility = ["//visibility:public"])
|
|||||||
|
|
||||||
exports_files([
|
exports_files([
|
||||||
"LICENSE",
|
"LICENSE",
|
||||||
"protractor-perf.conf.js",
|
|
||||||
"karma-js.conf.js",
|
"karma-js.conf.js",
|
||||||
"browser-providers.conf.js",
|
"browser-providers.conf.js",
|
||||||
"scripts/ci/track-payload-size.sh",
|
"scripts/ci/track-payload-size.sh",
|
||||||
@ -25,7 +24,7 @@ filegroup(
|
|||||||
"//packages/zone.js/dist:zone-testing.js",
|
"//packages/zone.js/dist:zone-testing.js",
|
||||||
"//packages/zone.js/dist:task-tracking.js",
|
"//packages/zone.js/dist:task-tracking.js",
|
||||||
"//:test-events.js",
|
"//:test-events.js",
|
||||||
"//:shims_for_IE.js",
|
"//:third_party/shims_for_IE.js",
|
||||||
# Including systemjs because it defines `__eval`, which produces correct stack traces.
|
# Including systemjs because it defines `__eval`, which produces correct stack traces.
|
||||||
"@npm//:node_modules/systemjs/dist/system.src.js",
|
"@npm//:node_modules/systemjs/dist/system.src.js",
|
||||||
"@npm//:node_modules/reflect-metadata/Reflect.js",
|
"@npm//:node_modules/reflect-metadata/Reflect.js",
|
||||||
|
839
CHANGELOG.md
839
CHANGELOG.md
File diff suppressed because it is too large
Load Diff
297
CONTRIBUTING.md
297
CONTRIBUTING.md
@ -1,7 +1,7 @@
|
|||||||
# Contributing to Angular
|
# Contributing to Angular
|
||||||
|
|
||||||
We would love for you to contribute to Angular and help make it even better than it is
|
We would love for you to contribute to Angular and help make it even better than it is today!
|
||||||
today! As a contributor, here are the guidelines we would like you to follow:
|
As a contributor, here are the guidelines we would like you to follow:
|
||||||
|
|
||||||
- [Code of Conduct](#coc)
|
- [Code of Conduct](#coc)
|
||||||
- [Question or Problem?](#question)
|
- [Question or Problem?](#question)
|
||||||
@ -12,50 +12,63 @@ today! As a contributor, here are the guidelines we would like you to follow:
|
|||||||
- [Commit Message Guidelines](#commit)
|
- [Commit Message Guidelines](#commit)
|
||||||
- [Signing the CLA](#cla)
|
- [Signing the CLA](#cla)
|
||||||
|
|
||||||
|
|
||||||
## <a name="coc"></a> Code of Conduct
|
## <a name="coc"></a> Code of Conduct
|
||||||
Help us keep Angular open and inclusive. Please read and follow our [Code of Conduct][coc].
|
|
||||||
|
Help us keep Angular open and inclusive.
|
||||||
|
Please read and follow our [Code of Conduct][coc].
|
||||||
|
|
||||||
|
|
||||||
## <a name="question"></a> Got a Question or Problem?
|
## <a name="question"></a> Got a Question or Problem?
|
||||||
|
|
||||||
Do not open issues for general support questions as we want to keep GitHub issues for bug reports and feature requests. You've got much better chances of getting your question answered on [Stack Overflow](https://stackoverflow.com/questions/tagged/angular) where the questions should be tagged with tag `angular`.
|
Do not open issues for general support questions as we want to keep GitHub issues for bug reports and feature requests.
|
||||||
|
Instead, we recommend using [Stack Overflow](https://stackoverflow.com/questions/tagged/angular) to ask support-related questions. When creating a new question on Stack Overflow, make sure to add the `angular` tag.
|
||||||
|
|
||||||
Stack Overflow is a much better place to ask questions since:
|
Stack Overflow is a much better place to ask questions since:
|
||||||
|
|
||||||
- there are thousands of people willing to help on Stack Overflow
|
- there are thousands of people willing to help on Stack Overflow
|
||||||
- questions and answers stay available for public viewing so your question / answer might help someone else
|
- questions and answers stay available for public viewing so your question/answer might help someone else
|
||||||
- Stack Overflow's voting system assures that the best answers are prominently visible.
|
- Stack Overflow's voting system assures that the best answers are prominently visible.
|
||||||
|
|
||||||
To save your and our time, we will systematically close all issues that are requests for general support and redirect people to Stack Overflow.
|
To save your and our time, we will systematically close all issues that are requests for general support and redirect people to Stack Overflow.
|
||||||
|
|
||||||
If you would like to chat about the question in real-time, you can reach out via [our gitter channel][gitter].
|
If you would like to chat about the question in real-time, you can reach out via [our gitter channel][gitter].
|
||||||
|
|
||||||
|
|
||||||
## <a name="issue"></a> Found a Bug?
|
## <a name="issue"></a> Found a Bug?
|
||||||
If you find a bug in the source code, you can help us by
|
|
||||||
[submitting an issue](#submit-issue) to our [GitHub Repository][github]. Even better, you can
|
If you find a bug in the source code, you can help us by [submitting an issue](#submit-issue) to our [GitHub Repository][github].
|
||||||
[submit a Pull Request](#submit-pr) with a fix.
|
Even better, you can [submit a Pull Request](#submit-pr) with a fix.
|
||||||
|
|
||||||
|
|
||||||
## <a name="feature"></a> Missing a Feature?
|
## <a name="feature"></a> Missing a Feature?
|
||||||
You can *request* a new feature by [submitting an issue](#submit-issue) to our GitHub
|
You can *request* a new feature by [submitting an issue](#submit-issue) to our GitHub Repository.
|
||||||
Repository. If you would like to *implement* a new feature, please submit an issue with
|
If you would like to *implement* a new feature, please consider the size of the change in order to determine the right steps to proceed:
|
||||||
a proposal for your work first, to be sure that we can use it.
|
|
||||||
Please consider what kind of change it is:
|
* For a **Major Feature**, first open an issue and outline your proposal so that it can be discussed.
|
||||||
|
This process allows us to better coordinate our efforts, prevent duplication of work, and help you to craft the change so that it is successfully accepted into the project.
|
||||||
|
|
||||||
|
**Note**: Adding a new topic to the documentation, or significantly re-writing a topic, counts as a major feature.
|
||||||
|
|
||||||
* For a **Major Feature**, first open an issue and outline your proposal so that it can be
|
|
||||||
discussed. This will also allow us to better coordinate our efforts, prevent duplication of work,
|
|
||||||
and help you to craft the change so that it is successfully accepted into the project.
|
|
||||||
* **Small Features** can be crafted and directly [submitted as a Pull Request](#submit-pr).
|
* **Small Features** can be crafted and directly [submitted as a Pull Request](#submit-pr).
|
||||||
|
|
||||||
|
|
||||||
## <a name="submit"></a> Submission Guidelines
|
## <a name="submit"></a> Submission Guidelines
|
||||||
|
|
||||||
|
|
||||||
### <a name="submit-issue"></a> Submitting an Issue
|
### <a name="submit-issue"></a> Submitting an Issue
|
||||||
|
|
||||||
Before you submit an issue, please search the issue tracker, maybe an issue for your problem already exists and the discussion might inform you of workarounds readily available.
|
Before you submit an issue, please search the issue tracker, maybe an issue for your problem already exists and the discussion might inform you of workarounds readily available.
|
||||||
|
|
||||||
We want to fix all the issues as soon as possible, but before fixing a bug we need to reproduce and confirm it. In order to reproduce bugs, we will systematically ask you to provide a minimal reproduction. Having a minimal reproducible scenario gives us a wealth of important information without going back & forth to you with additional questions.
|
We want to fix all the issues as soon as possible, but before fixing a bug we need to reproduce and confirm it.
|
||||||
|
In order to reproduce bugs, we require that you provide a minimal reproduction.
|
||||||
|
Having a minimal reproducible scenario gives us a wealth of important information without going back and forth to you with additional questions.
|
||||||
|
|
||||||
A minimal reproduction allows us to quickly confirm a bug (or point out a coding problem) as well as confirm that we are fixing the right problem.
|
A minimal reproduction allows us to quickly confirm a bug (or point out a coding problem) as well as confirm that we are fixing the right problem.
|
||||||
|
|
||||||
We will be insisting on a minimal reproduction scenario in order to save maintainers time and ultimately be able to fix more bugs. Interestingly, from our experience, users often find coding problems themselves while preparing a minimal reproduction. We understand that sometimes it might be hard to extract essential bits of code from a larger codebase but we really need to isolate the problem before we can fix it.
|
We require a minimal reproduction to save maintainers' time and ultimately be able to fix more bugs.
|
||||||
|
Often, developers find coding problems themselves while preparing a minimal reproduction.
|
||||||
|
We understand that sometimes it might be hard to extract essential bits of code from a larger codebase but we really need to isolate the problem before we can fix it.
|
||||||
|
|
||||||
Unfortunately, we are not able to investigate / fix bugs without a minimal reproduction, so if we don't hear back from you, we are going to close an issue that doesn't have enough info to be reproduced.
|
Unfortunately, we are not able to investigate / fix bugs without a minimal reproduction, so if we don't hear back from you, we are going to close an issue that doesn't have enough info to be reproduced.
|
||||||
|
|
||||||
@ -63,57 +76,66 @@ You can file new issues by selecting from our [new issue templates](https://gith
|
|||||||
|
|
||||||
|
|
||||||
### <a name="submit-pr"></a> Submitting a Pull Request (PR)
|
### <a name="submit-pr"></a> Submitting a Pull Request (PR)
|
||||||
|
|
||||||
Before you submit your Pull Request (PR) consider the following guidelines:
|
Before you submit your Pull Request (PR) consider the following guidelines:
|
||||||
|
|
||||||
1. Search [GitHub](https://github.com/angular/angular/pulls) for an open or closed PR
|
1. Search [GitHub](https://github.com/angular/angular/pulls) for an open or closed PR that relates to your submission.
|
||||||
that relates to your submission. You don't want to duplicate effort.
|
You don't want to duplicate existing efforts.
|
||||||
1. Be sure that an issue describes the problem you're fixing, or documents the design for the feature you'd like to add.
|
|
||||||
Discussing the design up front helps to ensure that we're ready to accept your work.
|
2. Be sure that an issue describes the problem you're fixing, or documents the design for the feature you'd like to add.
|
||||||
1. Please sign our [Contributor License Agreement (CLA)](#cla) before sending PRs.
|
Discussing the design upfront helps to ensure that we're ready to accept your work.
|
||||||
We cannot accept code without this. Make sure you sign with the primary email address of the Git identity that has been granted access to the Angular repository.
|
|
||||||
1. Fork the angular/angular repo.
|
3. Please sign our [Contributor License Agreement (CLA)](#cla) before sending PRs.
|
||||||
1. Make your changes in a new git branch:
|
We cannot accept code without a signed CLA.
|
||||||
|
Make sure you author all contributed Git commits with email address associated with your CLA signature.
|
||||||
|
|
||||||
|
4. Fork the angular/angular repo.
|
||||||
|
|
||||||
|
5. Make your changes in a new git branch:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
git checkout -b my-fix-branch master
|
git checkout -b my-fix-branch master
|
||||||
```
|
```
|
||||||
|
|
||||||
1. Create your patch, **including appropriate test cases**.
|
6. Create your patch, **including appropriate test cases**.
|
||||||
1. Follow our [Coding Rules](#rules).
|
|
||||||
1. Run the full Angular test suite, as described in the [developer documentation][dev-doc],
|
7. Follow our [Coding Rules](#rules).
|
||||||
and ensure that all tests pass.
|
|
||||||
1. Commit your changes using a descriptive commit message that follows our
|
8. Run the full Angular test suite, as described in the [developer documentation][dev-doc], and ensure that all tests pass.
|
||||||
[commit message conventions](#commit). Adherence to these conventions
|
|
||||||
is necessary because release notes are automatically generated from these messages.
|
9. Commit your changes using a descriptive commit message that follows our [commit message conventions](#commit).
|
||||||
|
Adherence to these conventions is necessary because release notes are automatically generated from these messages.
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
git commit -a
|
git commit -a
|
||||||
```
|
```
|
||||||
Note: the optional commit `-a` command line option will automatically "add" and "rm" edited files.
|
Note: the optional commit `-a` command line option will automatically "add" and "rm" edited files.
|
||||||
|
|
||||||
1. Push your branch to GitHub:
|
10. Push your branch to GitHub:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
git push origin my-fix-branch
|
git push origin my-fix-branch
|
||||||
```
|
```
|
||||||
|
|
||||||
1. In GitHub, send a pull request to `angular:master`.
|
11. In GitHub, send a pull request to `angular:master`.
|
||||||
* If we suggest changes then:
|
|
||||||
* Make the required updates.
|
|
||||||
* Re-run the Angular test suites to ensure tests are still passing.
|
|
||||||
* Rebase your branch and force push to your GitHub repository (this will update your Pull Request):
|
|
||||||
|
|
||||||
```shell
|
If we ask for changes via code reviews then:
|
||||||
git rebase master -i
|
|
||||||
git push -f
|
* Make the required updates.
|
||||||
```
|
* Re-run the Angular test suites to ensure tests are still passing.
|
||||||
|
* Rebase your branch and force push to your GitHub repository (this will update your Pull Request):
|
||||||
|
|
||||||
|
```shell
|
||||||
|
git rebase master -i
|
||||||
|
git push -f
|
||||||
|
```
|
||||||
|
|
||||||
That's it! Thank you for your contribution!
|
That's it! Thank you for your contribution!
|
||||||
|
|
||||||
|
|
||||||
#### After your pull request is merged
|
#### After your pull request is merged
|
||||||
|
|
||||||
After your pull request is merged, you can safely delete your branch and pull the changes
|
After your pull request is merged, you can safely delete your branch and pull the changes from the main (upstream) repository:
|
||||||
from the main (upstream) repository:
|
|
||||||
|
|
||||||
* Delete the remote branch on GitHub either through the GitHub web UI or your local shell as follows:
|
* Delete the remote branch on GitHub either through the GitHub web UI or your local shell as follows:
|
||||||
|
|
||||||
@ -139,55 +161,66 @@ from the main (upstream) repository:
|
|||||||
git pull --ff upstream master
|
git pull --ff upstream master
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
## <a name="rules"></a> Coding Rules
|
## <a name="rules"></a> Coding Rules
|
||||||
To ensure consistency throughout the source code, keep these rules in mind as you are working:
|
To ensure consistency throughout the source code, keep these rules in mind as you are working:
|
||||||
|
|
||||||
* All features or bug fixes **must be tested** by one or more specs (unit-tests).
|
* All features or bug fixes **must be tested** by one or more specs (unit-tests).
|
||||||
* All public API methods **must be documented**. (Details TBC).
|
* All public API methods **must be documented**.
|
||||||
* We follow [Google's JavaScript Style Guide][js-style-guide], but wrap all code at
|
* We follow [Google's JavaScript Style Guide][js-style-guide], but wrap all code at **100 characters**.
|
||||||
**100 characters**. An automated formatter is available, see
|
|
||||||
[DEVELOPER.md](docs/DEVELOPER.md#clang-format).
|
|
||||||
|
|
||||||
## <a name="commit"></a> Commit Message Guidelines
|
An automated formatter is available, see [DEVELOPER.md](docs/DEVELOPER.md#clang-format).
|
||||||
|
|
||||||
We have very precise rules over how our git commit messages can be formatted. This leads to **more
|
|
||||||
readable messages** that are easy to follow when looking through the **project history**. But also,
|
|
||||||
we use the git commit messages to **generate the Angular change log**.
|
|
||||||
|
|
||||||
### Commit Message Format
|
## <a name="commit"></a> Commit Message Format
|
||||||
Each commit message consists of a **header**, a **body** and a **footer**. The header has a special
|
|
||||||
format that includes a **type**, a **scope** and a **subject**:
|
*This specification is inspired and supersedes the [AngularJS commit message format][commit-message-format].*
|
||||||
|
|
||||||
|
We have very precise rules over how our Git commit messages must be formatted.
|
||||||
|
This format leads to **easier to read commit history**.
|
||||||
|
|
||||||
|
Each commit message consists of a **header**, a **body**, and a **footer**.
|
||||||
|
|
||||||
|
|
||||||
```
|
```
|
||||||
<type>(<scope>): <subject>
|
<header>
|
||||||
<BLANK LINE>
|
<BLANK LINE>
|
||||||
<body>
|
<body>
|
||||||
<BLANK LINE>
|
<BLANK LINE>
|
||||||
<footer>
|
<footer>
|
||||||
```
|
```
|
||||||
|
|
||||||
The **header** is mandatory and the **scope** of the header is optional.
|
The `header` is mandatory and must conform to the [Commit Message Header](#commit-header) format.
|
||||||
|
|
||||||
Any line of the commit message cannot be longer than 100 characters! This allows the message to be easier
|
The `body` is mandatory for all commits except for those of scope "docs".
|
||||||
to read on GitHub as well as in various git tools.
|
When the body is required it must be at least 20 characters long.
|
||||||
|
|
||||||
The footer should contain a [closing reference to an issue](https://help.github.com/articles/closing-issues-via-commit-messages/) if any.
|
The `footer` is optional.
|
||||||
|
|
||||||
Samples: (even more [samples](https://github.com/angular/angular/commits/master))
|
Any line of the commit message cannot be longer than 100 characters.
|
||||||
|
|
||||||
|
|
||||||
|
#### <a href="commit-header"></a>Commit Message Header
|
||||||
|
|
||||||
```
|
```
|
||||||
docs(changelog): update changelog to beta.5
|
<type>(<scope>): <short summary>
|
||||||
```
|
│ │ │
|
||||||
```
|
│ │ └─⫸ Summary in present tense. Not capitalized. No period at the end.
|
||||||
fix(release): need to depend on latest rxjs and zone.js
|
│ │
|
||||||
|
│ └─⫸ Commit Scope: animations|bazel|benchpress|common|compiler|compiler-cli|core|
|
||||||
The version in our package.json gets copied to the one we publish, and users need the latest of these.
|
│ elements|forms|http|language-service|localize|platform-browser|
|
||||||
|
│ platform-browser-dynamic|platform-server|platform-webworker|
|
||||||
|
│ platform-webworker-dynamic|router|service-worker|upgrade|zone.js|
|
||||||
|
│ packaging|changelog|dev-infra|docs-infra|migrations|ngcc|ve
|
||||||
|
│
|
||||||
|
└─⫸ Commit Type: build|ci|docs|feat|fix|perf|refactor|style|test
|
||||||
```
|
```
|
||||||
|
|
||||||
### Revert
|
The `<type>` and `<summary>` fields are mandatory, the `(<scope>)` field is optional.
|
||||||
If the commit reverts a previous commit, it should begin with `revert: `, followed by the header of the reverted commit. In the body it should say: `This reverts commit <hash>.`, where the hash is the SHA of the commit being reverted.
|
|
||||||
|
|
||||||
|
##### Type
|
||||||
|
|
||||||
### Type
|
|
||||||
Must be one of the following:
|
Must be one of the following:
|
||||||
|
|
||||||
* **build**: Changes that affect the build system or external dependencies (example scopes: gulp, broccoli, npm)
|
* **build**: Changes that affect the build system or external dependencies (example scopes: gulp, broccoli, npm)
|
||||||
@ -200,66 +233,95 @@ Must be one of the following:
|
|||||||
* **style**: Changes that do not affect the meaning of the code (white-space, formatting, missing semi-colons, etc)
|
* **style**: Changes that do not affect the meaning of the code (white-space, formatting, missing semi-colons, etc)
|
||||||
* **test**: Adding missing tests or correcting existing tests
|
* **test**: Adding missing tests or correcting existing tests
|
||||||
|
|
||||||
### Scope
|
|
||||||
|
##### Scope
|
||||||
The scope should be the name of the npm package affected (as perceived by the person reading the changelog generated from commit messages).
|
The scope should be the name of the npm package affected (as perceived by the person reading the changelog generated from commit messages).
|
||||||
|
|
||||||
The following is the list of supported scopes:
|
The following is the list of supported scopes:
|
||||||
|
|
||||||
* **animations**
|
* `animations`
|
||||||
* **bazel**
|
* `bazel`
|
||||||
* **benchpress**
|
* `benchpress`
|
||||||
* **common**
|
* `common`
|
||||||
* **compiler**
|
* `compiler`
|
||||||
* **compiler-cli**
|
* `compiler-cli`
|
||||||
* **core**
|
* `core`
|
||||||
* **elements**
|
* `elements`
|
||||||
* **forms**
|
* `forms`
|
||||||
* **http**
|
* `http`
|
||||||
* **language-service**
|
* `language-service`
|
||||||
* **localize**
|
* `localize`
|
||||||
* **platform-browser**
|
* `platform-browser`
|
||||||
* **platform-browser-dynamic**
|
* `platform-browser-dynamic`
|
||||||
* **platform-server**
|
* `platform-server`
|
||||||
* **platform-webworker**
|
* `platform-webworker`
|
||||||
* **platform-webworker-dynamic**
|
* `platform-webworker-dynamic`
|
||||||
* **router**
|
* `router`
|
||||||
* **service-worker**
|
* `service-worker`
|
||||||
* **upgrade**
|
* `upgrade`
|
||||||
* **zone.js**
|
* `zone.js`
|
||||||
|
|
||||||
There are currently a few exceptions to the "use package name" rule:
|
There are currently a few exceptions to the "use package name" rule:
|
||||||
|
|
||||||
* **packaging**: used for changes that change the npm package layout in all of our packages, e.g.
|
* `packaging`: used for changes that change the npm package layout in all of our packages, e.g. public path changes, package.json changes done to all packages, d.ts file/format changes, changes to bundles, etc.
|
||||||
public path changes, package.json changes done to all packages, d.ts file/format changes, changes
|
|
||||||
to bundles, etc.
|
|
||||||
* **changelog**: used for updating the release notes in CHANGELOG.md
|
|
||||||
* **docs-infra**: used for docs-app (angular.io) related changes within the /aio directory of the
|
|
||||||
repo
|
|
||||||
* **dev-infra**: used for dev-infra related changes within the directories /scripts, /tools and /dev-infra
|
|
||||||
* **ngcc**: used for changes to the [Angular Compatibility Compiler](./packages/compiler-cli/ngcc/README.md)
|
|
||||||
* **ve**: used for changes specific to ViewEngine (legacy compiler/renderer).
|
|
||||||
* none/empty string: useful for `style`, `test` and `refactor` changes that are done across all
|
|
||||||
packages (e.g. `style: add missing semicolons`) and for docs changes that are not related to a
|
|
||||||
specific package (e.g. `docs: fix typo in tutorial`).
|
|
||||||
|
|
||||||
### Subject
|
* `changelog`: used for updating the release notes in CHANGELOG.md
|
||||||
The subject contains a succinct description of the change:
|
|
||||||
|
* `dev-infra`: used for dev-infra related changes within the directories /scripts, /tools and /dev-infra
|
||||||
|
|
||||||
|
* `docs-infra`: used for docs-app (angular.io) related changes within the /aio directory of the repo
|
||||||
|
|
||||||
|
* `migrations`: used for changes to the `ng update` migrations.
|
||||||
|
|
||||||
|
* `ngcc`: used for changes to the [Angular Compatibility Compiler](./packages/compiler-cli/ngcc/README.md)
|
||||||
|
|
||||||
|
* `ve`: used for changes specific to ViewEngine (legacy compiler/renderer).
|
||||||
|
|
||||||
|
* none/empty string: useful for `style`, `test` and `refactor` changes that are done across all packages (e.g. `style: add missing semicolons`) and for docs changes that are not related to a specific package (e.g. `docs: fix typo in tutorial`).
|
||||||
|
|
||||||
|
|
||||||
|
##### Summary
|
||||||
|
|
||||||
|
Use the summary field to provide a succinct description of the change:
|
||||||
|
|
||||||
* use the imperative, present tense: "change" not "changed" nor "changes"
|
* use the imperative, present tense: "change" not "changed" nor "changes"
|
||||||
* don't capitalize the first letter
|
* don't capitalize the first letter
|
||||||
* no dot (.) at the end
|
* no dot (.) at the end
|
||||||
|
|
||||||
### Body
|
|
||||||
Just as in the **subject**, use the imperative, present tense: "change" not "changed" nor "changes".
|
|
||||||
The body should include the motivation for the change and contrast this with previous behavior.
|
|
||||||
|
|
||||||
### Footer
|
#### Commit Message Body
|
||||||
The footer should contain any information about **Breaking Changes** and is also the place to
|
|
||||||
reference GitHub issues that this commit **Closes**.
|
|
||||||
|
|
||||||
**Breaking Changes** should start with the word `BREAKING CHANGE:` with a space or two newlines. The rest of the commit message is then used for this.
|
Just as in the summary, use the imperative, present tense: "fix" not "fixed" nor "fixes".
|
||||||
|
|
||||||
|
Explain the motivation for the change in the commit message body. This commit message should explain _why_ you are making the change.
|
||||||
|
You can include a comparison of the previous behavior with the new behavior in order to illustrate the impact of the change.
|
||||||
|
|
||||||
|
|
||||||
|
#### Commit Message Footer
|
||||||
|
|
||||||
|
The footer can contain information about breaking changes and is also the place to reference GitHub issues, Jira tickets, and other PRs that this commit closes or is related to.
|
||||||
|
|
||||||
|
```
|
||||||
|
BREAKING CHANGE: <breaking change summary>
|
||||||
|
<BLANK LINE>
|
||||||
|
<breaking change description + migration instructions>
|
||||||
|
<BLANK LINE>
|
||||||
|
<BLANK LINE>
|
||||||
|
Fixes #<issue number>
|
||||||
|
```
|
||||||
|
|
||||||
|
Breaking Change section should start with the phrase "BREAKING CHANGE: " followed by a summary of the breaking change, a blank line, and a detailed description of the breaking change that also includes migration instructions.
|
||||||
|
|
||||||
|
|
||||||
|
### Revert commits
|
||||||
|
|
||||||
|
If the commit reverts a previous commit, it should begin with `revert: `, followed by the header of the reverted commit.
|
||||||
|
|
||||||
|
The content of the commit message body should contain:
|
||||||
|
|
||||||
|
- information about the SHA of the commit being reverted in the following format: `This reverts commit <SHA>`,
|
||||||
|
- a clear description of the reason for reverting the commit message.
|
||||||
|
|
||||||
A detailed explanation can be found in this [document][commit-message-format].
|
|
||||||
|
|
||||||
## <a name="cla"></a> Signing the CLA
|
## <a name="cla"></a> Signing the CLA
|
||||||
|
|
||||||
@ -270,18 +332,17 @@ changes to be accepted, the CLA must be signed. It's a quick process, we promise
|
|||||||
* For corporations, we'll need you to
|
* For corporations, we'll need you to
|
||||||
[print, sign and one of scan+email, fax or mail the form][corporate-cla].
|
[print, sign and one of scan+email, fax or mail the form][corporate-cla].
|
||||||
|
|
||||||
<hr>
|
If you have more than one GitHub accounts, or multiple email addresses associated with a single GitHub account, you must sign the CLA using the primary email address of the GitHub account used to author Git commits and send pull requests.
|
||||||
|
|
||||||
If you have more than one Git identity, you must make sure that you sign the CLA using the primary email address associated with the ID that has been granted access to the Angular repository. Git identities can be associated with more than one email address, and only one is primary. Here are some links to help you sort out multiple Git identities and email addresses:
|
The following documents can help you sort out issues with GitHub accounts and multiple email addresses:
|
||||||
|
|
||||||
* https://help.github.com/articles/setting-your-commit-email-address-in-git/
|
* https://help.github.com/articles/setting-your-commit-email-address-in-git/
|
||||||
* https://stackoverflow.com/questions/37245303/what-does-usera-committed-with-userb-13-days-ago-on-github-mean
|
* https://stackoverflow.com/questions/37245303/what-does-usera-committed-with-userb-13-days-ago-on-github-mean
|
||||||
* https://help.github.com/articles/about-commit-email-addresses/
|
* https://help.github.com/articles/about-commit-email-addresses/
|
||||||
* https://help.github.com/articles/blocking-command-line-pushes-that-expose-your-personal-email-address/
|
* https://help.github.com/articles/blocking-command-line-pushes-that-expose-your-personal-email-address/
|
||||||
|
|
||||||
Note that if you have more than one Git identity, it is important to verify that you are logged in with the same ID with which you signed the CLA, before you commit changes. If not, your PR will fail the CLA check.
|
|
||||||
|
|
||||||
<hr>
|
|
||||||
|
|
||||||
[angular-group]: https://groups.google.com/forum/#!forum/angular
|
[angular-group]: https://groups.google.com/forum/#!forum/angular
|
||||||
[coc]: https://github.com/angular/code-of-conduct/blob/master/CODE_OF_CONDUCT.md
|
[coc]: https://github.com/angular/code-of-conduct/blob/master/CODE_OF_CONDUCT.md
|
||||||
|
15
WORKSPACE
15
WORKSPACE
@ -8,8 +8,8 @@ load("@bazel_tools//tools/build_defs/repo:http.bzl", "http_archive")
|
|||||||
# Fetch rules_nodejs so we can install our npm dependencies
|
# Fetch rules_nodejs so we can install our npm dependencies
|
||||||
http_archive(
|
http_archive(
|
||||||
name = "build_bazel_rules_nodejs",
|
name = "build_bazel_rules_nodejs",
|
||||||
sha256 = "f9e7b9f42ae202cc2d2ce6d698ccb49a9f7f7ea572a78fd451696d03ef2ee116",
|
sha256 = "84abf7ac4234a70924628baa9a73a5a5cbad944c4358cf9abdb4aab29c9a5b77",
|
||||||
urls = ["https://github.com/bazelbuild/rules_nodejs/releases/download/1.6.0/rules_nodejs-1.6.0.tar.gz"],
|
urls = ["https://github.com/bazelbuild/rules_nodejs/releases/download/1.7.0/rules_nodejs-1.7.0.tar.gz"],
|
||||||
)
|
)
|
||||||
|
|
||||||
# Check the rules_nodejs version and download npm dependencies
|
# Check the rules_nodejs version and download npm dependencies
|
||||||
@ -17,7 +17,7 @@ http_archive(
|
|||||||
# assert on that.
|
# assert on that.
|
||||||
load("@build_bazel_rules_nodejs//:index.bzl", "check_rules_nodejs_version", "node_repositories", "yarn_install")
|
load("@build_bazel_rules_nodejs//:index.bzl", "check_rules_nodejs_version", "node_repositories", "yarn_install")
|
||||||
|
|
||||||
check_rules_nodejs_version(minimum_version_string = "1.6.0")
|
check_rules_nodejs_version(minimum_version_string = "1.7.0")
|
||||||
|
|
||||||
# Setup the Node.js toolchain
|
# Setup the Node.js toolchain
|
||||||
node_repositories(
|
node_repositories(
|
||||||
@ -64,7 +64,7 @@ load("@io_bazel_rules_webtesting//web:repositories.bzl", "web_test_repositories"
|
|||||||
|
|
||||||
web_test_repositories()
|
web_test_repositories()
|
||||||
|
|
||||||
load("//tools/browsers:browser_repositories.bzl", "browser_repositories")
|
load("//dev-infra/browsers:browser_repositories.bzl", "browser_repositories")
|
||||||
|
|
||||||
browser_repositories()
|
browser_repositories()
|
||||||
|
|
||||||
@ -91,17 +91,18 @@ rbe_autoconfig(
|
|||||||
# Need to specify a base container digest in order to ensure that we can use the checked-in
|
# Need to specify a base container digest in order to ensure that we can use the checked-in
|
||||||
# platform configurations for the "ubuntu16_04" image. Otherwise the autoconfig rule would
|
# platform configurations for the "ubuntu16_04" image. Otherwise the autoconfig rule would
|
||||||
# need to pull the image and run it in order determine the toolchain configuration. See:
|
# need to pull the image and run it in order determine the toolchain configuration. See:
|
||||||
# https://github.com/bazelbuild/bazel-toolchains/blob/1.1.2/configs/ubuntu16_04_clang/versions.bzl
|
# https://github.com/bazelbuild/bazel-toolchains/blob/3.2.0/configs/ubuntu16_04_clang/versions.bzl
|
||||||
base_container_digest = "sha256:1ab40405810effefa0b2f45824d6d608634ccddbf06366760c341ef6fbead011",
|
base_container_digest = "sha256:5e750dd878df9fcf4e185c6f52b9826090f6e532b097f286913a428290622332",
|
||||||
# Note that if you change the `digest`, you might also need to update the
|
# Note that if you change the `digest`, you might also need to update the
|
||||||
# `base_container_digest` to make sure marketplace.gcr.io/google/rbe-ubuntu16-04-webtest:<digest>
|
# `base_container_digest` to make sure marketplace.gcr.io/google/rbe-ubuntu16-04-webtest:<digest>
|
||||||
# and marketplace.gcr.io/google/rbe-ubuntu16-04:<base_container_digest> have
|
# and marketplace.gcr.io/google/rbe-ubuntu16-04:<base_container_digest> have
|
||||||
# the same Clang and JDK installed. Clang is needed because of the dependency on
|
# the same Clang and JDK installed. Clang is needed because of the dependency on
|
||||||
# @com_google_protobuf. Java is needed for the Bazel's test executor Java tool.
|
# @com_google_protobuf. Java is needed for the Bazel's test executor Java tool.
|
||||||
digest = "sha256:0b8fa87db4b8e5366717a7164342a029d1348d2feea7ecc4b18c780bc2507059",
|
digest = "sha256:f743114235a43355bf8324e2ba0fa6a597236fe06f7bc99aaa9ac703631c306b",
|
||||||
env = clang_env(),
|
env = clang_env(),
|
||||||
registry = "marketplace.gcr.io",
|
registry = "marketplace.gcr.io",
|
||||||
# We can't use the default "ubuntu16_04" RBE image provided by the autoconfig because we need
|
# We can't use the default "ubuntu16_04" RBE image provided by the autoconfig because we need
|
||||||
# a specific Linux kernel that comes with "libx11" in order to run headless browser tests.
|
# a specific Linux kernel that comes with "libx11" in order to run headless browser tests.
|
||||||
repository = "google/rbe-ubuntu16-04-webtest",
|
repository = "google/rbe-ubuntu16-04-webtest",
|
||||||
|
use_checked_in_confs = "Force",
|
||||||
)
|
)
|
||||||
|
@ -58,7 +58,7 @@
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
"styles": [
|
"styles": [
|
||||||
"src/styles.scss"
|
"src/styles/main.scss"
|
||||||
],
|
],
|
||||||
"scripts": [],
|
"scripts": [],
|
||||||
"budgets": [
|
"budgets": [
|
||||||
@ -158,7 +158,7 @@
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
"styles": [
|
"styles": [
|
||||||
"src/styles.scss"
|
"src/styles/main.scss"
|
||||||
],
|
],
|
||||||
"scripts": []
|
"scripts": []
|
||||||
}
|
}
|
||||||
@ -193,4 +193,4 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"defaultProject": "site"
|
"defaultProject": "site"
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# CLI Overview and Command Reference
|
# CLI Overview and Command Reference
|
||||||
|
|
||||||
The Angular CLI is a command-line interface tool that you use to initialize, develop, scaffold, and maintain Angular applications. You can use the tool directly in a command shell, or indirectly through an interactive UI such as [Angular Console](https://angularconsole.com).
|
The Angular CLI is a command-line interface tool that you use to initialize, develop, scaffold, and maintain Angular applications directly from a command shell.
|
||||||
|
|
||||||
## Installing Angular CLI
|
## Installing Angular CLI
|
||||||
|
|
||||||
@ -109,9 +109,3 @@ Options that specify files can be given as absolute paths, or as paths relative
|
|||||||
The [ng generate](cli/generate) and [ng add](cli/add) commands take as an argument the artifact or library to be generated or added to the current project.
|
The [ng generate](cli/generate) and [ng add](cli/add) commands take as an argument the artifact or library to be generated or added to the current project.
|
||||||
In addition to any general options, each artifact or library defines its own options in a *schematic*.
|
In addition to any general options, each artifact or library defines its own options in a *schematic*.
|
||||||
Schematic options are supplied to the command in the same format as immediate command options.
|
Schematic options are supplied to the command in the same format as immediate command options.
|
||||||
|
|
||||||
|
|
||||||
### Building with Bazel
|
|
||||||
|
|
||||||
Optionally, you can configure the Angular CLI to use [Bazel](https://docs.bazel.build) as the build tool. For more information, see [Building with Bazel](guide/bazel).
|
|
||||||
|
|
||||||
|
1
aio/content/examples/.gitignore
vendored
1
aio/content/examples/.gitignore
vendored
@ -18,6 +18,7 @@
|
|||||||
**/src/karma.conf.js
|
**/src/karma.conf.js
|
||||||
**/.angular-cli.json
|
**/.angular-cli.json
|
||||||
**/.editorconfig
|
**/.editorconfig
|
||||||
|
**/.gitignore
|
||||||
**/angular.json
|
**/angular.json
|
||||||
**/tsconfig.json
|
**/tsconfig.json
|
||||||
**/bs-config.e2e.json
|
**/bs-config.e2e.json
|
||||||
|
@ -32,15 +32,15 @@ export const slideInAnimation =
|
|||||||
// #enddocregion style-view
|
// #enddocregion style-view
|
||||||
// #docregion query
|
// #docregion query
|
||||||
query(':enter', [
|
query(':enter', [
|
||||||
style({ left: '-100%'})
|
style({ left: '-100%' })
|
||||||
]),
|
]),
|
||||||
query(':leave', animateChild()),
|
query(':leave', animateChild()),
|
||||||
group([
|
group([
|
||||||
query(':leave', [
|
query(':leave', [
|
||||||
animate('300ms ease-out', style({ left: '100%'}))
|
animate('300ms ease-out', style({ left: '100%' }))
|
||||||
]),
|
]),
|
||||||
query(':enter', [
|
query(':enter', [
|
||||||
animate('300ms ease-out', style({ left: '0%'}))
|
animate('300ms ease-out', style({ left: '0%' }))
|
||||||
])
|
])
|
||||||
]),
|
]),
|
||||||
query(':enter', animateChild()),
|
query(':enter', animateChild()),
|
||||||
@ -56,15 +56,15 @@ export const slideInAnimation =
|
|||||||
})
|
})
|
||||||
]),
|
]),
|
||||||
query(':enter', [
|
query(':enter', [
|
||||||
style({ left: '-100%'})
|
style({ left: '-100%' })
|
||||||
]),
|
]),
|
||||||
query(':leave', animateChild()),
|
query(':leave', animateChild()),
|
||||||
group([
|
group([
|
||||||
query(':leave', [
|
query(':leave', [
|
||||||
animate('200ms ease-out', style({ left: '100%'}))
|
animate('200ms ease-out', style({ left: '100%' }))
|
||||||
]),
|
]),
|
||||||
query(':enter', [
|
query(':enter', [
|
||||||
animate('300ms ease-out', style({ left: '0%'}))
|
animate('300ms ease-out', style({ left: '0%' }))
|
||||||
])
|
])
|
||||||
]),
|
]),
|
||||||
query(':enter', animateChild()),
|
query(':enter', animateChild()),
|
||||||
|
@ -17,7 +17,7 @@ Toggle All Animations <input type="checkbox" [checked]="!animationsDisabled" (cl
|
|||||||
</nav>
|
</nav>
|
||||||
|
|
||||||
<!-- #docregion route-animations-outlet -->
|
<!-- #docregion route-animations-outlet -->
|
||||||
<div [@routeAnimations]="prepareRoute(outlet)" >
|
<div [@routeAnimations]="prepareRoute(outlet)">
|
||||||
<router-outlet #outlet="outlet"></router-outlet>
|
<router-outlet #outlet="outlet"></router-outlet>
|
||||||
</div>
|
</div>
|
||||||
<!-- #enddocregion route-animations-outlet -->
|
<!-- #enddocregion route-animations-outlet -->
|
||||||
|
@ -3,7 +3,7 @@ import { NgModule } from '@angular/core';
|
|||||||
import { BrowserModule } from '@angular/platform-browser';
|
import { BrowserModule } from '@angular/platform-browser';
|
||||||
import { FormsModule } from '@angular/forms';
|
import { FormsModule } from '@angular/forms';
|
||||||
|
|
||||||
import { AppComponent } from './app.component';
|
import { AppComponent } from './app.component';
|
||||||
import { HeroFormComponent } from './hero-form/hero-form.component';
|
import { HeroFormComponent } from './hero-form/hero-form.component';
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
|
@ -200,13 +200,4 @@
|
|||||||
(ngModelChange)="model.name = $event">
|
(ngModelChange)="model.name = $event">
|
||||||
TODO: remove this: {{model.name}}
|
TODO: remove this: {{model.name}}
|
||||||
<!-- #enddocregion ngModel-3-->
|
<!-- #enddocregion ngModel-3-->
|
||||||
<hr>
|
|
||||||
<!-- #docregion ngModelName-2 -->
|
|
||||||
<input type="text" class="form-control" id="name"
|
|
||||||
required
|
|
||||||
[(ngModel)]="model.name" name="name"
|
|
||||||
#spy>
|
|
||||||
<br>TODO: remove this: {{spy.className}}
|
|
||||||
<!-- #enddocregion ngModelName-2 -->
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
// #docregion , v1, final
|
// #docregion , v1, final
|
||||||
import { Component } from '@angular/core';
|
import { Component } from '@angular/core';
|
||||||
|
|
||||||
import { Hero } from '../hero';
|
import { Hero } from '../hero';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-hero-form',
|
selector: 'app-hero-form',
|
||||||
|
@ -76,15 +76,15 @@ export class ConfigService {
|
|||||||
console.error('An error occurred:', error.error.message);
|
console.error('An error occurred:', error.error.message);
|
||||||
} else {
|
} else {
|
||||||
// The backend returned an unsuccessful response code.
|
// The backend returned an unsuccessful response code.
|
||||||
// The response body may contain clues as to what went wrong,
|
// The response body may contain clues as to what went wrong.
|
||||||
console.error(
|
console.error(
|
||||||
`Backend returned code ${error.status}, ` +
|
`Backend returned code ${error.status}, ` +
|
||||||
`body was: ${error.error}`);
|
`body was: ${error.error}`);
|
||||||
}
|
}
|
||||||
// return an observable with a user-facing error message
|
// Return an observable with a user-facing error message.
|
||||||
return throwError(
|
return throwError(
|
||||||
'Something bad happened; please try again later.');
|
'Something bad happened; please try again later.');
|
||||||
};
|
}
|
||||||
// #enddocregion handleError
|
// #enddocregion handleError
|
||||||
|
|
||||||
makeIntentionalError() {
|
makeIntentionalError() {
|
||||||
|
@ -13,13 +13,13 @@ import { searchUrl } from '../package-search/package-search.service';
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* If request is cachable (e.g., package search) and
|
* If request is cacheable (e.g., package search) and
|
||||||
* response is in cache return the cached response as observable.
|
* response is in cache return the cached response as observable.
|
||||||
* If has 'x-refresh' header that is true,
|
* If has 'x-refresh' header that is true,
|
||||||
* then also re-run the package search, using response from next(),
|
* then also re-run the package search, using response from next(),
|
||||||
* returning an observable that emits the cached response first.
|
* returning an observable that emits the cached response first.
|
||||||
*
|
*
|
||||||
* If not in cache or not cachable,
|
* If not in cache or not cacheable,
|
||||||
* pass request through to next()
|
* pass request through to next()
|
||||||
*/
|
*/
|
||||||
// #docregion v1
|
// #docregion v1
|
||||||
@ -28,8 +28,8 @@ export class CachingInterceptor implements HttpInterceptor {
|
|||||||
constructor(private cache: RequestCache) {}
|
constructor(private cache: RequestCache) {}
|
||||||
|
|
||||||
intercept(req: HttpRequest<any>, next: HttpHandler) {
|
intercept(req: HttpRequest<any>, next: HttpHandler) {
|
||||||
// continue if not cachable.
|
// continue if not cacheable.
|
||||||
if (!isCachable(req)) { return next.handle(req); }
|
if (!isCacheable(req)) { return next.handle(req); }
|
||||||
|
|
||||||
const cachedResponse = this.cache.get(req);
|
const cachedResponse = this.cache.get(req);
|
||||||
// #enddocregion v1
|
// #enddocregion v1
|
||||||
@ -51,11 +51,11 @@ export class CachingInterceptor implements HttpInterceptor {
|
|||||||
// #enddocregion v1
|
// #enddocregion v1
|
||||||
|
|
||||||
|
|
||||||
/** Is this request cachable? */
|
/** Is this request cacheable? */
|
||||||
function isCachable(req: HttpRequest<any>) {
|
function isCacheable(req: HttpRequest<any>) {
|
||||||
// Only GET requests are cachable
|
// Only GET requests are cacheable
|
||||||
return req.method === 'GET' &&
|
return req.method === 'GET' &&
|
||||||
// Only npm package search is cachable in this app
|
// Only npm package search is cacheable in this app
|
||||||
-1 < req.url.indexOf(searchUrl);
|
-1 < req.url.indexOf(searchUrl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
12
aio/content/examples/i18n/stackblitz.json
Normal file
12
aio/content/examples/i18n/stackblitz.json
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
{
|
||||||
|
"description": "i18n",
|
||||||
|
"files":[
|
||||||
|
"!**/*.d.ts",
|
||||||
|
"!**/*.js",
|
||||||
|
"!**/*.[0-9].*",
|
||||||
|
"!doc-files/**/*",
|
||||||
|
"**/*.xlf"
|
||||||
|
],
|
||||||
|
"file": "src/app/app.component.ts",
|
||||||
|
"tags": ["Angular", "i18n", "internationalization"]
|
||||||
|
}
|
@ -1,9 +0,0 @@
|
|||||||
{
|
|
||||||
"files": [
|
|
||||||
"!dist/",
|
|
||||||
"!**/*.d.ts",
|
|
||||||
"!src/**/*.js",
|
|
||||||
"!doc-files/**/*",
|
|
||||||
"**/*.xlf"
|
|
||||||
]
|
|
||||||
}
|
|
@ -6,5 +6,5 @@ import { Component } from '@angular/core';
|
|||||||
templateUrl: './app.component.html'
|
templateUrl: './app.component.html'
|
||||||
})
|
})
|
||||||
export class AppComponent {
|
export class AppComponent {
|
||||||
birthday = new Date(1988, 3, 15); // April 15, 1988
|
birthday = new Date(1988, 3, 15); // April 15, 1988 -- since month parameter is zero-based
|
||||||
}
|
}
|
||||||
|
@ -8,5 +8,5 @@ import { Component } from '@angular/core';
|
|||||||
// #enddocregion hero-birthday-template
|
// #enddocregion hero-birthday-template
|
||||||
})
|
})
|
||||||
export class HeroBirthdayComponent {
|
export class HeroBirthdayComponent {
|
||||||
birthday = new Date(1988, 3, 15); // April 15, 1988
|
birthday = new Date(1988, 3, 15); // April 15, 1988 -- since month parameter is zero-based
|
||||||
}
|
}
|
||||||
|
@ -12,7 +12,7 @@ import { Component } from '@angular/core';
|
|||||||
})
|
})
|
||||||
// #docregion class
|
// #docregion class
|
||||||
export class HeroBirthday2Component {
|
export class HeroBirthday2Component {
|
||||||
birthday = new Date(1988, 3, 15); // April 15, 1988
|
birthday = new Date(1988, 3, 15); // April 15, 1988 -- since month parameter is zero-based
|
||||||
toggle = true; // start with true == shortDate
|
toggle = true; // start with true == shortDate
|
||||||
|
|
||||||
get format() { return this.toggle ? 'shortDate' : 'fullDate'; }
|
get format() { return this.toggle ? 'shortDate' : 'fullDate'; }
|
||||||
|
7
aio/content/examples/providers/src/app/user.service.2.ts
Normal file
7
aio/content/examples/providers/src/app/user.service.2.ts
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
import { Injectable } from '@angular/core';
|
||||||
|
|
||||||
|
@Injectable({
|
||||||
|
providedIn: 'any',
|
||||||
|
})
|
||||||
|
export class UserService {
|
||||||
|
}
|
@ -6,13 +6,11 @@ import { Routes, RouterModule } from '@angular/router'; // CLI imports router
|
|||||||
const routes: Routes = [
|
const routes: Routes = [
|
||||||
{ path: 'first-component', component: FirstComponent },
|
{ path: 'first-component', component: FirstComponent },
|
||||||
{ path: 'second-component', component: SecondComponent },
|
{ path: 'second-component', component: SecondComponent },
|
||||||
// #enddocregion routes
|
// #enddocregion routes, routes-with-wildcard
|
||||||
{ path: '', redirectTo: '/first-component', pathMatch: 'full' }, // redirect to `first-component`
|
{ path: '', redirectTo: '/first-component', pathMatch: 'full' }, // redirect to `first-component`
|
||||||
{ path: '**', component: FirstComponent },
|
// #docregion routes-with-wildcard
|
||||||
// #enddocregion redirect
|
|
||||||
{ path: '**', component: PageNotFoundComponent }, // Wildcard route for a 404 page
|
{ path: '**', component: PageNotFoundComponent }, // Wildcard route for a 404 page
|
||||||
// #docregion routes
|
// #docregion routes
|
||||||
// #docregion redirect
|
|
||||||
];
|
];
|
||||||
// #enddocregion routes, routes-with-wildcard, redirect
|
// #enddocregion routes, routes-with-wildcard, redirect
|
||||||
|
|
||||||
|
@ -16,5 +16,12 @@
|
|||||||
"@angular/core": "^7.2.0"
|
"@angular/core": "^7.2.0"
|
||||||
},
|
},
|
||||||
// #docregion collection
|
// #docregion collection
|
||||||
"schematics": "./schematics/collection.json"
|
"schematics": "./schematics/collection.json",
|
||||||
}
|
// #enddocregion collection
|
||||||
|
// #docregion ng-add
|
||||||
|
"ng-add": {
|
||||||
|
"save": "devDependencies"
|
||||||
|
}
|
||||||
|
// #enddocregion ng-add
|
||||||
|
// #docregion collection
|
||||||
|
}
|
||||||
|
@ -33,7 +33,7 @@ export class HeroesComponent implements OnInit {
|
|||||||
|
|
||||||
onSelect(hero: Hero): void {
|
onSelect(hero: Hero): void {
|
||||||
this.selectedHero = hero;
|
this.selectedHero = hero;
|
||||||
this.messageService.add(`HeroService: Selected hero id=${hero.id}`);
|
this.messageService.add(`HeroesComponent: Selected hero id=${hero.id}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
// #docregion getHeroes
|
// #docregion getHeroes
|
||||||
|
@ -20,7 +20,7 @@ work well for all users, including those who rely on assistive technologies.
|
|||||||
|
|
||||||
Building accessible web experience often involves setting [ARIA attributes](https://developers.google.com/web/fundamentals/accessibility/semantics-aria)
|
Building accessible web experience often involves setting [ARIA attributes](https://developers.google.com/web/fundamentals/accessibility/semantics-aria)
|
||||||
to provide semantic meaning where it might otherwise be missing.
|
to provide semantic meaning where it might otherwise be missing.
|
||||||
Use [attribute binding](guide/template-syntax#attribute-binding) template syntax to control the values of accessibility-related attributes.
|
Use [attribute binding](guide/attribute-binding) template syntax to control the values of accessibility-related attributes.
|
||||||
|
|
||||||
When binding to ARIA attributes in Angular, you must use the `attr.` prefix, as the ARIA
|
When binding to ARIA attributes in Angular, you must use the `attr.` prefix, as the ARIA
|
||||||
specification depends specifically on HTML attributes rather than properties of DOM elements.
|
specification depends specifically on HTML attributes rather than properties of DOM elements.
|
||||||
@ -44,7 +44,7 @@ NOTE:
|
|||||||
|
|
||||||
By convention, HTML attributes use lowercase names (`tabindex`), while properties use camelCase names (`tabIndex`).
|
By convention, HTML attributes use lowercase names (`tabindex`), while properties use camelCase names (`tabIndex`).
|
||||||
|
|
||||||
See the [Template Syntax](guide/template-syntax#html-attribute-vs-dom-property) guide for more background on the difference between attributes and properties.
|
See the [Binding syntax](guide/binding-syntax#html-attribute-vs-dom-property) guide for more background on the difference between attributes and properties.
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -74,8 +74,7 @@ The following table lists some of the key AngularJS template features with their
|
|||||||
The context of the binding is implied and is always the
|
The context of the binding is implied and is always the
|
||||||
associated component, so it needs no reference variable.
|
associated component, so it needs no reference variable.
|
||||||
|
|
||||||
For more information, see the [Interpolation](guide/template-syntax#interpolation)
|
For more information, see the [Interpolation](guide/interpolation) guide.
|
||||||
section of the [Template Syntax](guide/template-syntax) page.
|
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
</tr>
|
</tr>
|
||||||
@ -141,8 +140,8 @@ The following table lists some of the key AngularJS template features with their
|
|||||||
|
|
||||||
Angular has true template input variables that are explicitly defined using the `let` keyword.
|
Angular has true template input variables that are explicitly defined using the `let` keyword.
|
||||||
|
|
||||||
For more information, see the [ngFor micro-syntax](guide/template-syntax#microsyntax)
|
For more information, see the [ngFor micro-syntax](guide/built-in-directives#microsyntax)
|
||||||
section of the [Template Syntax](guide/template-syntax) page.
|
section of the [Built-in Directives](guide/built-in-directives) page.
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
</tr>
|
</tr>
|
||||||
@ -258,8 +257,7 @@ The following are some of the key AngularJS built-in directives and their equiva
|
|||||||
Angular also has **class binding**, which is a good way to add or remove a single class,
|
Angular also has **class binding**, which is a good way to add or remove a single class,
|
||||||
as shown in the third example.
|
as shown in the third example.
|
||||||
|
|
||||||
For more information see the [Attribute, class, and style bindings](guide/template-syntax#other-bindings)
|
For more information see [Attribute, class, and style bindings](guide/attribute-binding) page.
|
||||||
section of the [Template Syntax](guide/template-syntax) page.
|
|
||||||
|
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
@ -309,8 +307,7 @@ The following are some of the key AngularJS built-in directives and their equiva
|
|||||||
|
|
||||||
For a list of DOM events, see: https://developer.mozilla.org/en-US/docs/Web/Events.
|
For a list of DOM events, see: https://developer.mozilla.org/en-US/docs/Web/Events.
|
||||||
|
|
||||||
For more information, see the [Event binding](guide/template-syntax#event-binding)
|
For more information, see the [Event binding](guide/event-binding) page.
|
||||||
section of the [Template Syntax](guide/template-syntax) page.
|
|
||||||
|
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
@ -407,8 +404,7 @@ The following are some of the key AngularJS built-in directives and their equiva
|
|||||||
Angular uses property binding; there is no built-in *href* directive.
|
Angular uses property binding; there is no built-in *href* directive.
|
||||||
Place the element's `href` property in square brackets and set it to a quoted template expression.
|
Place the element's `href` property in square brackets and set it to a quoted template expression.
|
||||||
|
|
||||||
For more information see the [Property binding](guide/template-syntax#property-binding)
|
For more information see the [Property binding](guide/property-binding) page.
|
||||||
section of the [Template Syntax](guide/template-syntax) page.
|
|
||||||
|
|
||||||
In Angular, `href` is no longer used for routing. Routing uses `routerLink`, as shown in the following example.
|
In Angular, `href` is no longer used for routing. Routing uses `routerLink`, as shown in the following example.
|
||||||
|
|
||||||
@ -487,8 +483,8 @@ The following are some of the key AngularJS built-in directives and their equiva
|
|||||||
and event binding (from the view to the component), thereby providing two-way binding.
|
and event binding (from the view to the component), thereby providing two-way binding.
|
||||||
|
|
||||||
For more information on two-way binding with `ngModel`, see the [NgModel—Two-way binding to
|
For more information on two-way binding with `ngModel`, see the [NgModel—Two-way binding to
|
||||||
form elements with `[(ngModel)]`](../guide/template-syntax.html#ngModel)
|
form elements with `[(ngModel)]`](../guide/built-in-directives#ngModel)
|
||||||
section of the [Template Syntax](guide/template-syntax) page.
|
section of the [Built-in directives](guide/built-in-directives) page.
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
</tr>
|
</tr>
|
||||||
@ -570,8 +566,7 @@ The following are some of the key AngularJS built-in directives and their equiva
|
|||||||
|
|
||||||
In this example, the `<div>` element is hidden if the `favoriteHero` variable is not truthy.
|
In this example, the `<div>` element is hidden if the `favoriteHero` variable is not truthy.
|
||||||
|
|
||||||
For more information on property binding, see the [Property binding](guide/template-syntax#property-binding)
|
For more information on property binding, see the [Property binding](guide/property-binding) page.
|
||||||
section of the [Template Syntax](guide/template-syntax) page.
|
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
</tr>
|
</tr>
|
||||||
@ -604,8 +599,7 @@ The following are some of the key AngularJS built-in directives and their equiva
|
|||||||
Angular uses property binding; there is no built-in *src* directive.
|
Angular uses property binding; there is no built-in *src* directive.
|
||||||
Place the `src` property in square brackets and set it to a quoted template expression.
|
Place the `src` property in square brackets and set it to a quoted template expression.
|
||||||
|
|
||||||
For more information on property binding, see the [Property binding](guide/template-syntax#property-binding)
|
For more information on property binding, see the [Property binding](guide/property-binding) page.
|
||||||
section of the [Template Syntax](guide/template-syntax) page.
|
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
</tr>
|
</tr>
|
||||||
@ -644,11 +638,11 @@ The following are some of the key AngularJS built-in directives and their equiva
|
|||||||
|
|
||||||
Angular also has **style binding**, which is good way to set a single style. This is shown in the second example.
|
Angular also has **style binding**, which is good way to set a single style. This is shown in the second example.
|
||||||
|
|
||||||
For more information on style binding, see the [Style binding](guide/template-syntax#style-binding) section of the
|
For more information on style binding, see the [Style binding](guide/attribute-binding#style-binding) section of the
|
||||||
[Template Syntax](guide/template-syntax) page.
|
[Attribute binding](guide/attribute-binding) page.
|
||||||
|
|
||||||
For more information on the `ngStyle` directive, see [NgStyle](guide/template-syntax#ngStyle)
|
For more information on the `ngStyle` directive, see the [NgStyle](guide/built-in-directives#ngStyle)
|
||||||
section of the [Template Syntax](guide/template-syntax) page.
|
section of the [Built-in directives](guide/built-in-directives) page.
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
</tr>
|
</tr>
|
||||||
@ -704,8 +698,8 @@ The following are some of the key AngularJS built-in directives and their equiva
|
|||||||
|
|
||||||
The (*) before `ngSwitchCase` and `ngSwitchDefault` is required in this example.
|
The (*) before `ngSwitchCase` and `ngSwitchDefault` is required in this example.
|
||||||
|
|
||||||
For more information, see [The NgSwitch directives](guide/template-syntax#ngSwitch)
|
For more information, see [The NgSwitch directives](guide/built-in-directives#ngSwitch)
|
||||||
section of the [Template Syntax](guide/template-syntax) page.
|
section of the [Built-in directives](guide/built-in-directives) page.
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
</tr>
|
</tr>
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Angular compiler options
|
# Angular compiler options
|
||||||
|
|
||||||
When you use [AOT compilation](guide/aot-compiler), you can control how your application is compiled by specifying *template* compiler options in the `tsconfig.json` [TypeScript configuration file](guide/typescript-configuration).
|
When you use [AOT compilation](guide/aot-compiler), you can control how your application is compiled by specifying *template* compiler options in the [TypeScript configuration file](guide/typescript-configuration).
|
||||||
|
|
||||||
The template options object, `angularCompilerOptions`, is a sibling to the `compilerOptions` object that supplies standard options to the TypeScript compiler.
|
The template options object, `angularCompilerOptions`, is a sibling to the `compilerOptions` object that supplies standard options to the TypeScript compiler.
|
||||||
|
|
||||||
@ -21,11 +21,11 @@ The template options object, `angularCompilerOptions`, is a sibling to the `comp
|
|||||||
{@a tsconfig-extends}
|
{@a tsconfig-extends}
|
||||||
## Configuration inheritance with extends
|
## Configuration inheritance with extends
|
||||||
|
|
||||||
Like the TypeScript compiler, The Angular AOT compiler also supports `extends` in the `angularCompilerOptions` section of the TypeScript configuration file, `tsconfig.json`.
|
Like the TypeScript compiler, The Angular AOT compiler also supports `extends` in the `angularCompilerOptions` section of the TypeScript configuration file.
|
||||||
The `extends` property is at the top level, parallel to `compilerOptions` and `angularCompilerOptions`.
|
The `extends` property is at the top level, parallel to `compilerOptions` and `angularCompilerOptions`.
|
||||||
|
|
||||||
A TypeScript configuration can inherit settings from another file using the `extends` property.
|
A TypeScript configuration can inherit settings from another file using the `extends` property.
|
||||||
The configuration options from the base file are loaded first, then overridden by those in the inheriting `tsconfig` file.
|
The configuration options from the base file are loaded first, then overridden by those in the inheriting configuration file.
|
||||||
|
|
||||||
For example:
|
For example:
|
||||||
|
|
||||||
@ -186,7 +186,7 @@ For library projects generated with the CLI, the dev configuration default is `t
|
|||||||
|
|
||||||
When `true` (recommended), reports an error for a supplied parameter whose injection type cannot be determined. When `false` (currently the default), constructor parameters of classes marked with `@Injectable` whose type cannot be resolved produce a warning.
|
When `true` (recommended), reports an error for a supplied parameter whose injection type cannot be determined. When `false` (currently the default), constructor parameters of classes marked with `@Injectable` whose type cannot be resolved produce a warning.
|
||||||
|
|
||||||
When you use the CLI command `ng new`, it is set to `true` by default in the generated project's configuration.
|
When you use the CLI command `ng new --strict`, it is set to `true` in the generated project's configuration.
|
||||||
|
|
||||||
### `strictTemplates`
|
### `strictTemplates`
|
||||||
|
|
||||||
@ -194,6 +194,7 @@ When `true`, enables [strict template type checking](guide/template-typecheck#st
|
|||||||
|
|
||||||
Additional strictness flags allow you to enable and disable specific types of strict template type checking. See [troubleshooting template errors](guide/template-typecheck#troubleshooting-template-errors).
|
Additional strictness flags allow you to enable and disable specific types of strict template type checking. See [troubleshooting template errors](guide/template-typecheck#troubleshooting-template-errors).
|
||||||
|
|
||||||
|
When you use the CLI command `ng new --strict`, it is set to `true` in the generated project's configuration.
|
||||||
|
|
||||||
### `trace`
|
### `trace`
|
||||||
|
|
||||||
|
@ -105,7 +105,7 @@ For help in understanding and resolving these problems, see [AOT Metadata Errors
|
|||||||
|
|
||||||
### Configuring AOT compilation
|
### Configuring AOT compilation
|
||||||
|
|
||||||
You can provide options in the `tsconfig.json` [TypeScript configuration file](guide/typescript-configuration) that control the compilation process. See [Angular compiler options](guide/angular-compiler-options) for a complete list of available options.
|
You can provide options in the [TypeScript configuration file](guide/typescript-configuration) that controls the compilation process. See [Angular compiler options](guide/angular-compiler-options) for a complete list of available options.
|
||||||
|
|
||||||
## Phase 1: Code analysis
|
## Phase 1: Code analysis
|
||||||
|
|
||||||
@ -211,7 +211,7 @@ The compiler later reports the error if it needs that piece of metadata to gener
|
|||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
If you want `ngc` to report syntax errors immediately rather than produce a `.metadata.json` file with errors, set the `strictMetadataEmit` option in the TypeScript configuration file, `tsconfig.json`.
|
If you want `ngc` to report syntax errors immediately rather than produce a `.metadata.json` file with errors, set the `strictMetadataEmit` option in the TypeScript configuration file.
|
||||||
|
|
||||||
```
|
```
|
||||||
"angularCompilerOptions": {
|
"angularCompilerOptions": {
|
||||||
@ -548,7 +548,7 @@ It does not, however, rewrite the `.d.ts` file, so TypeScript doesn't recognize
|
|||||||
One of the Angular compiler's most helpful features is the ability to type-check expressions within templates, and catch any errors before they cause crashes at runtime.
|
One of the Angular compiler's most helpful features is the ability to type-check expressions within templates, and catch any errors before they cause crashes at runtime.
|
||||||
In the template type-checking phase, the Angular template compiler uses the TypeScript compiler to validate the binding expressions in templates.
|
In the template type-checking phase, the Angular template compiler uses the TypeScript compiler to validate the binding expressions in templates.
|
||||||
|
|
||||||
Enable this phase explicitly by adding the compiler option `"fullTemplateTypeCheck"` in the `"angularCompilerOptions"` of the project's `tsconfig.json`
|
Enable this phase explicitly by adding the compiler option `"fullTemplateTypeCheck"` in the `"angularCompilerOptions"` of the project's TypeScript configuration file
|
||||||
(see [Angular Compiler Options](guide/angular-compiler-options)).
|
(see [Angular Compiler Options](guide/angular-compiler-options)).
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
@ -623,7 +623,7 @@ For more information about input type narrowing, see [Input setter coercion](gui
|
|||||||
|
|
||||||
### Non-null type assertion operator
|
### Non-null type assertion operator
|
||||||
|
|
||||||
Use the [non-null type assertion operator](guide/template-syntax#non-null-assertion-operator) to suppress the `Object is possibly 'undefined'` error when it is inconvenient to use `*ngIf` or when some constraint in the component ensures that the expression is always non-null when the binding expression is interpolated.
|
Use the [non-null type assertion operator](guide/template-expression-operators#non-null-assertion-operator) to suppress the `Object is possibly 'undefined'` error when it is inconvenient to use `*ngIf` or when some constraint in the component ensures that the expression is always non-null when the binding expression is interpolated.
|
||||||
|
|
||||||
In the following example, the `person` and `address` properties are always set together, implying that `address` is always non-null if `person` is non-null.
|
In the following example, the `person` and `address` properties are always set together, implying that `address` is always non-null if `person` is non-null.
|
||||||
There is no convenient way to describe this constraint to TypeScript and the template compiler, but the error is suppressed in the example by using `address!.street`.
|
There is no convenient way to describe this constraint to TypeScript and the template compiler, but the error is suppressed in the example by using `address!.street`.
|
||||||
|
@ -430,7 +430,7 @@ Angular does something similar with the `DOCUMENT` token so you can inject the b
|
|||||||
|
|
||||||
```ts
|
```ts
|
||||||
import { Inject } from '@angular/core';
|
import { Inject } from '@angular/core';
|
||||||
import { DOCUMENT } from '@angular/platform-browser';
|
import { DOCUMENT } from '@angular/common';
|
||||||
|
|
||||||
@Component({ ... })
|
@Component({ ... })
|
||||||
export class MyComponent {
|
export class MyComponent {
|
||||||
|
@ -92,7 +92,7 @@ This example from the `HeroListComponent` template uses three of these forms.
|
|||||||
* The `{{hero.name}}` [*interpolation*](guide/displaying-data#interpolation)
|
* The `{{hero.name}}` [*interpolation*](guide/displaying-data#interpolation)
|
||||||
displays the component's `hero.name` property value within the `<li>` element.
|
displays the component's `hero.name` property value within the `<li>` element.
|
||||||
|
|
||||||
* The `[hero]` [*property binding*](guide/template-syntax#property-binding) passes the value of
|
* The `[hero]` [*property binding*](guide/property-binding) passes the value of
|
||||||
`selectedHero` from the parent `HeroListComponent` to the `hero` property of the child `HeroDetailComponent`.
|
`selectedHero` from the parent `HeroListComponent` to the `hero` property of the child `HeroDetailComponent`.
|
||||||
|
|
||||||
* The `(click)` [*event binding*](guide/user-input#binding-to-user-input-events) calls the component's `selectHero` method when the user clicks a hero's name.
|
* The `(click)` [*event binding*](guide/user-input#binding-to-user-input-events) calls the component's `selectHero` method when the user clicks a hero's name.
|
||||||
@ -126,7 +126,7 @@ Angular pipes let you declare display-value transformations in your template HTM
|
|||||||
|
|
||||||
Angular defines various pipes, such as the [date](https://angular.io/api/common/DatePipe) pipe and [currency](https://angular.io/api/common/CurrencyPipe) pipe; for a complete list, see the [Pipes API list](https://angular.io/api?type=pipe). You can also define new pipes.
|
Angular defines various pipes, such as the [date](https://angular.io/api/common/DatePipe) pipe and [currency](https://angular.io/api/common/CurrencyPipe) pipe; for a complete list, see the [Pipes API list](https://angular.io/api?type=pipe). You can also define new pipes.
|
||||||
|
|
||||||
To specify a value transformation in an HTML template, use the [pipe operator (|)](https://angular.io/guide/template-syntax#pipe).
|
To specify a value transformation in an HTML template, use the [pipe operator (|)](https://angular.io/guide/template-expression-operators#pipe).
|
||||||
|
|
||||||
`{{interpolated_value | pipe_name}}`
|
`{{interpolated_value | pipe_name}}`
|
||||||
|
|
||||||
@ -179,9 +179,9 @@ The `ngModel` directive, which implements two-way data binding, is an example of
|
|||||||
<code-example path="architecture/src/app/hero-detail.component.html" header="src/app/hero-detail.component.html (ngModel)" region="ngModel"></code-example>
|
<code-example path="architecture/src/app/hero-detail.component.html" header="src/app/hero-detail.component.html (ngModel)" region="ngModel"></code-example>
|
||||||
|
|
||||||
Angular has more pre-defined directives that either alter the layout structure
|
Angular has more pre-defined directives that either alter the layout structure
|
||||||
(for example, [ngSwitch](guide/template-syntax#ngSwitch))
|
(for example, [ngSwitch](guide/built-in-directives#ngSwitch))
|
||||||
or modify aspects of DOM elements and components
|
or modify aspects of DOM elements and components
|
||||||
(for example, [ngStyle](guide/template-syntax#ngStyle) and [ngClass](guide/template-syntax#ngClass)).
|
(for example, [ngStyle](guide/built-in-directives#ngStyle) and [ngClass](guide/built-in-directives#ngClass)).
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
@ -11,7 +11,7 @@ The basic building blocks are *NgModules*, which provide a compilation context f
|
|||||||
|
|
||||||
* Components use *services*, which provide specific functionality not directly related to views. Service providers can be *injected* into components as *dependencies*, making your code modular, reusable, and efficient.
|
* Components use *services*, which provide specific functionality not directly related to views. Service providers can be *injected* into components as *dependencies*, making your code modular, reusable, and efficient.
|
||||||
|
|
||||||
Both components and services are simply classes, with *decorators* that mark their type and provide metadata that tells Angular how to use them.
|
Modules, components and services are classes that use *decorators*. These decorators mark their type and provide metadata that tells Angular how to use them.
|
||||||
|
|
||||||
* The metadata for a component class associates it with a *template* that defines a view. A template combines ordinary HTML with Angular *directives* and *binding markup* that allow Angular to modify the HTML before rendering it for display.
|
* The metadata for a component class associates it with a *template* that defines a view. A template combines ordinary HTML with Angular *directives* and *binding markup* that allow Angular to modify the HTML before rendering it for display.
|
||||||
|
|
||||||
|
303
aio/content/guide/attribute-binding.md
Normal file
303
aio/content/guide/attribute-binding.md
Normal file
@ -0,0 +1,303 @@
|
|||||||
|
# Attribute, class, and style bindings
|
||||||
|
|
||||||
|
The template syntax provides specialized one-way bindings for scenarios less well-suited to property binding.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
See the <live-example></live-example> for a working example containing the code snippets in this guide.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
## Attribute binding
|
||||||
|
|
||||||
|
Set the value of an attribute directly with an **attribute binding**. This is the only exception to the rule that a binding sets a target property and the only binding that creates and sets an attribute.
|
||||||
|
|
||||||
|
Usually, setting an element property with a [property binding](guide/property-binding)
|
||||||
|
is preferable to setting the attribute with a string. However, sometimes
|
||||||
|
there is no element property to bind, so attribute binding is the solution.
|
||||||
|
|
||||||
|
Consider the [ARIA](https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA) and
|
||||||
|
[SVG](https://developer.mozilla.org/en-US/docs/Web/SVG). They are purely attributes, don't correspond to element properties, and don't set element properties. In these cases, there are no property targets to bind to.
|
||||||
|
|
||||||
|
Attribute binding syntax resembles property binding, but
|
||||||
|
instead of an element property between brackets, start with the prefix `attr`,
|
||||||
|
followed by a dot (`.`), and the name of the attribute.
|
||||||
|
You then set the attribute value, using an expression that resolves to a string,
|
||||||
|
or remove the attribute when the expression resolves to `null`.
|
||||||
|
|
||||||
|
One of the primary use cases for attribute binding
|
||||||
|
is to set ARIA attributes, as in this example:
|
||||||
|
|
||||||
|
<code-example path="attribute-binding/src/app/app.component.html" region="attrib-binding-aria" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
{@a colspan}
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
#### `colspan` and `colSpan`
|
||||||
|
|
||||||
|
Notice the difference between the `colspan` attribute and the `colSpan` property.
|
||||||
|
|
||||||
|
If you wrote something like this:
|
||||||
|
|
||||||
|
<code-example language="html">
|
||||||
|
<tr><td colspan="{{1 + 1}}">Three-Four</td></tr>
|
||||||
|
</code-example>
|
||||||
|
|
||||||
|
You'd get this error:
|
||||||
|
|
||||||
|
<code-example language="bash">
|
||||||
|
Template parse errors:
|
||||||
|
Can't bind to 'colspan' since it isn't a known native property
|
||||||
|
</code-example>
|
||||||
|
|
||||||
|
As the message says, the `<td>` element does not have a `colspan` property. This is true
|
||||||
|
because `colspan` is an attribute—`colSpan`, with a capital `S`, is the
|
||||||
|
corresponding property. Interpolation and property binding can set only *properties*, not attributes.
|
||||||
|
|
||||||
|
Instead, you'd use property binding and write it like this:
|
||||||
|
|
||||||
|
<code-example path="attribute-binding/src/app/app.component.html" region="colSpan" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
{@a class-binding}
|
||||||
|
|
||||||
|
## Class binding
|
||||||
|
|
||||||
|
Here's how to set the `class` attribute without a binding in plain HTML:
|
||||||
|
|
||||||
|
```html
|
||||||
|
<!-- standard class attribute setting -->
|
||||||
|
<div class="foo bar">Some text</div>
|
||||||
|
```
|
||||||
|
|
||||||
|
You can also add and remove CSS class names from an element's `class` attribute with a **class binding**.
|
||||||
|
|
||||||
|
To create a single class binding, start with the prefix `class` followed by a dot (`.`) and the name of the CSS class (for example, `[class.foo]="hasFoo"`).
|
||||||
|
Angular adds the class when the bound expression is truthy, and it removes the class when the expression is falsy (with the exception of `undefined`, see [styling delegation](#styling-delegation)).
|
||||||
|
|
||||||
|
To create a binding to multiple classes, use a generic `[class]` binding without the dot (for example, `[class]="classExpr"`).
|
||||||
|
The expression can be a space-delimited string of class names, or you can format it as an object with class names as the keys and truthy/falsy expressions as the values.
|
||||||
|
With object format, Angular will add a class only if its associated value is truthy.
|
||||||
|
|
||||||
|
It's important to note that with any object-like expression (`object`, `Array`, `Map`, `Set`, etc), the identity of the object must change for the class list to be updated.
|
||||||
|
Updating the property without changing object identity will have no effect.
|
||||||
|
|
||||||
|
If there are multiple bindings to the same class name, conflicts are resolved using [styling precedence](#styling-precedence).
|
||||||
|
|
||||||
|
<style>
|
||||||
|
td, th {vertical-align: top}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<table width="100%">
|
||||||
|
<col width="15%">
|
||||||
|
</col>
|
||||||
|
<col width="20%">
|
||||||
|
</col>
|
||||||
|
<col width="35%">
|
||||||
|
</col>
|
||||||
|
<col width="30%">
|
||||||
|
</col>
|
||||||
|
<tr>
|
||||||
|
<th>
|
||||||
|
Binding Type
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
Syntax
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
Input Type
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
Example Input Values
|
||||||
|
</th>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>Single class binding</td>
|
||||||
|
<td><code>[class.foo]="hasFoo"</code></td>
|
||||||
|
<td><code>boolean | undefined | null</code></td>
|
||||||
|
<td><code>true</code>, <code>false</code></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td rowspan=3>Multi-class binding</td>
|
||||||
|
<td rowspan=3><code>[class]="classExpr"</code></td>
|
||||||
|
<td><code>string</code></td>
|
||||||
|
<td><code>"my-class-1 my-class-2 my-class-3"</code></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>{[key: string]: boolean | undefined | null}</code></td>
|
||||||
|
<td><code>{foo: true, bar: false}</code></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>Array</code><<code>string</code>></td>
|
||||||
|
<td><code>['foo', 'bar']</code></td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
|
||||||
|
The [NgClass](guide/built-in-directives/#ngclass) directive can be used as an alternative to direct `[class]` bindings.
|
||||||
|
However, using the above class binding syntax without `NgClass` is preferred because due to improvements in class binding in Angular, `NgClass` no longer provides significant value, and might eventually be removed in the future.
|
||||||
|
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
## Style binding
|
||||||
|
|
||||||
|
Here's how to set the `style` attribute without a binding in plain HTML:
|
||||||
|
|
||||||
|
```html
|
||||||
|
<!-- standard style attribute setting -->
|
||||||
|
<div style="color: blue">Some text</div>
|
||||||
|
```
|
||||||
|
|
||||||
|
You can also set styles dynamically with a **style binding**.
|
||||||
|
|
||||||
|
To create a single style binding, start with the prefix `style` followed by a dot (`.`) and the name of the CSS style property (for example, `[style.width]="width"`).
|
||||||
|
The property will be set to the value of the bound expression, which is normally a string.
|
||||||
|
Optionally, you can add a unit extension like `em` or `%`, which requires a number type.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
Note that a _style property_ name can be written in either
|
||||||
|
[dash-case](guide/glossary#dash-case), as shown above, or
|
||||||
|
[camelCase](guide/glossary#camelcase), such as `fontSize`.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
If there are multiple styles you'd like to toggle, you can bind to the `[style]` property directly without the dot (for example, `[style]="styleExpr"`).
|
||||||
|
The expression attached to the `[style]` binding is most often a string list of styles like `"width: 100px; height: 100px;"`.
|
||||||
|
|
||||||
|
You can also format the expression as an object with style names as the keys and style values as the values, like `{width: '100px', height: '100px'}`.
|
||||||
|
It's important to note that with any object-like expression (`object`, `Array`, `Map`, `Set`, etc), the identity of the object must change for the class list to be updated.
|
||||||
|
Updating the property without changing object identity will have no effect.
|
||||||
|
|
||||||
|
If there are multiple bindings to the same style property, conflicts are resolved using [styling precedence rules](#styling-precedence).
|
||||||
|
|
||||||
|
<style>
|
||||||
|
td, th {vertical-align: top}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<table width="100%">
|
||||||
|
<col width="15%">
|
||||||
|
</col>
|
||||||
|
<col width="20%">
|
||||||
|
</col>
|
||||||
|
<col width="35%">
|
||||||
|
</col>
|
||||||
|
<col width="30%">
|
||||||
|
</col>
|
||||||
|
<tr>
|
||||||
|
<th>
|
||||||
|
Binding Type
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
Syntax
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
Input Type
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
Example Input Values
|
||||||
|
</th>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>Single style binding</td>
|
||||||
|
<td><code>[style.width]="width"</code></td>
|
||||||
|
<td><code>string | undefined | null</code></td>
|
||||||
|
<td><code>"100px"</code></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<tr>
|
||||||
|
<td>Single style binding with units</td>
|
||||||
|
<td><code>[style.width.px]="width"</code></td>
|
||||||
|
<td><code>number | undefined | null</code></td>
|
||||||
|
<td><code>100</code></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td rowspan=3>Multi-style binding</td>
|
||||||
|
<td rowspan=3><code>[style]="styleExpr"</code></td>
|
||||||
|
<td><code>string</code></td>
|
||||||
|
<td><code>"width: 100px; height: 100px"</code></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>{[key: string]: string | undefined | null}</code></td>
|
||||||
|
<td><code>{width: '100px', height: '100px'}</code></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>Array</code><<code>string</code>></td>
|
||||||
|
<td><code>['width', '100px']</code></td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
The [NgStyle](guide/built-in-directives/#ngstyle) directive can be used as an alternative to direct `[style]` bindings.
|
||||||
|
However, using the above style binding syntax without `NgStyle` is preferred because due to improvements in style binding in Angular, `NgStyle` no longer provides significant value, and might eventually be removed in the future.
|
||||||
|
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
{@a styling-precedence}
|
||||||
|
|
||||||
|
## Styling Precedence
|
||||||
|
|
||||||
|
A single HTML element can have its CSS class list and style values bound to multiple sources (for example, host bindings from multiple directives).
|
||||||
|
|
||||||
|
When there are multiple bindings to the same class name or style property, Angular uses a set of precedence rules to resolve conflicts and determine which classes or styles are ultimately applied to the element.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
<h4>Styling precedence (highest to lowest)</h4>
|
||||||
|
|
||||||
|
1. Template bindings
|
||||||
|
1. Property binding (for example, `<div [class.foo]="hasFoo">` or `<div [style.color]="color">`)
|
||||||
|
1. Map binding (for example, `<div [class]="classExpr">` or `<div [style]="styleExpr">`)
|
||||||
|
1. Static value (for example, `<div class="foo">` or `<div style="color: blue">`)
|
||||||
|
1. Directive host bindings
|
||||||
|
1. Property binding (for example, `host: {'[class.foo]': 'hasFoo'}` or `host: {'[style.color]': 'color'}`)
|
||||||
|
1. Map binding (for example, `host: {'[class]': 'classExpr'}` or `host: {'[style]': 'styleExpr'}`)
|
||||||
|
1. Static value (for example, `host: {'class': 'foo'}` or `host: {'style': 'color: blue'}`)
|
||||||
|
1. Component host bindings
|
||||||
|
1. Property binding (for example, `host: {'[class.foo]': 'hasFoo'}` or `host: {'[style.color]': 'color'}`)
|
||||||
|
1. Map binding (for example, `host: {'[class]': 'classExpr'}` or `host: {'[style]': 'styleExpr'}`)
|
||||||
|
1. Static value (for example, `host: {'class': 'foo'}` or `host: {'style': 'color: blue'}`)
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
The more specific a class or style binding is, the higher its precedence.
|
||||||
|
|
||||||
|
A binding to a specific class (for example, `[class.foo]`) will take precedence over a generic `[class]` binding, and a binding to a specific style (for example, `[style.bar]`) will take precedence over a generic `[style]` binding.
|
||||||
|
|
||||||
|
<code-example path="attribute-binding/src/app/app.component.html" region="basic-specificity" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Specificity rules also apply when it comes to bindings that originate from different sources.
|
||||||
|
It's possible for an element to have bindings in the template where it's declared, from host bindings on matched directives, and from host bindings on matched components.
|
||||||
|
|
||||||
|
Template bindings are the most specific because they apply to the element directly and exclusively, so they have the highest precedence.
|
||||||
|
|
||||||
|
Directive host bindings are considered less specific because directives can be used in multiple locations, so they have a lower precedence than template bindings.
|
||||||
|
|
||||||
|
Directives often augment component behavior, so host bindings from components have the lowest precedence.
|
||||||
|
|
||||||
|
<code-example path="attribute-binding/src/app/app.component.html" region="source-specificity" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
In addition, bindings take precedence over static attributes.
|
||||||
|
|
||||||
|
In the following case, `class` and `[class]` have similar specificity, but the `[class]` binding will take precedence because it is dynamic.
|
||||||
|
|
||||||
|
<code-example path="attribute-binding/src/app/app.component.html" region="dynamic-priority" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
{@a styling-delegation}
|
||||||
|
### Delegating to styles with lower precedence
|
||||||
|
|
||||||
|
It is possible for higher precedence styles to "delegate" to lower precedence styles using `undefined` values.
|
||||||
|
Whereas setting a style property to `null` ensures the style is removed, setting it to `undefined` will cause Angular to fall back to the next-highest precedence binding to that style.
|
||||||
|
|
||||||
|
For example, consider the following template:
|
||||||
|
|
||||||
|
<code-example path="attribute-binding/src/app/app.component.html" region="style-delegation" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Imagine that the `dirWithHostBinding` directive and the `comp-with-host-binding` component both have a `[style.width]` host binding.
|
||||||
|
In that case, if `dirWithHostBinding` sets its binding to `undefined`, the `width` property will fall back to the value of the `comp-with-host-binding` host binding.
|
||||||
|
However, if `dirWithHostBinding` sets its binding to `null`, the `width` property will be removed entirely.
|
@ -18,12 +18,12 @@ There are three kinds of directives in Angular:
|
|||||||
You saw a component for the first time in the [Getting Started](start "Getting Started with Angular") tutorial.
|
You saw a component for the first time in the [Getting Started](start "Getting Started with Angular") tutorial.
|
||||||
|
|
||||||
*Structural Directives* change the structure of the view.
|
*Structural Directives* change the structure of the view.
|
||||||
Two examples are [NgFor](guide/template-syntax#ngFor) and [NgIf](guide/template-syntax#ngIf).
|
Two examples are [NgFor](guide/built-in-directives#ngFor) and [NgIf](guide/built-in-directives#ngIf).
|
||||||
Learn about them in the [Structural Directives](guide/structural-directives) guide.
|
Learn about them in the [Structural Directives](guide/structural-directives) guide.
|
||||||
|
|
||||||
*Attribute directives* are used as attributes of elements.
|
*Attribute directives* are used as attributes of elements.
|
||||||
The built-in [NgStyle](guide/template-syntax#ngStyle) directive in the
|
The built-in [NgStyle](guide/built-in-directives#ngStyle) directive in the
|
||||||
[Template Syntax](guide/template-syntax) guide, for example,
|
[Built-in directives](guide/built-in-directives) guide, for example,
|
||||||
can change several element styles at the same time.
|
can change several element styles at the same time.
|
||||||
|
|
||||||
## Build a simple attribute directive
|
## Build a simple attribute directive
|
||||||
|
@ -1,122 +0,0 @@
|
|||||||
# Building with Bazel
|
|
||||||
|
|
||||||
This guide explains how to build and test Angular apps with Bazel.
|
|
||||||
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
|
||||||
|
|
||||||
This guide assumes you are already familiar with developing and building Angular applications using the [CLI](cli).
|
|
||||||
|
|
||||||
It describes features which are part of Angular Labs, and are not considered a stable, supported API.
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
## Using Bazel with the Angular CLI
|
|
||||||
|
|
||||||
The `@angular/bazel` package provides a builder that allows Angular CLI to use Bazel as the build tool.
|
|
||||||
|
|
||||||
To opt-in an existing application, run
|
|
||||||
|
|
||||||
```sh
|
|
||||||
ng add @angular/bazel
|
|
||||||
```
|
|
||||||
|
|
||||||
To use Bazel in a new application, first install `@angular/bazel` globally
|
|
||||||
|
|
||||||
```sh
|
|
||||||
npm install -g @angular/bazel
|
|
||||||
```
|
|
||||||
|
|
||||||
then create the new application with
|
|
||||||
|
|
||||||
```sh
|
|
||||||
ng new --collection=@angular/bazel
|
|
||||||
```
|
|
||||||
|
|
||||||
Now when you use Angular CLI build commands such as `ng build` and `ng serve`,
|
|
||||||
Bazel is used behind the scenes.
|
|
||||||
Outputs from Bazel appear in the `dist/bin` folder.
|
|
||||||
|
|
||||||
> The command-line output includes extra logging from Bazel.
|
|
||||||
> We plan to reduce this in the future.
|
|
||||||
|
|
||||||
### Removing Bazel
|
|
||||||
|
|
||||||
If you need to opt-out from using Bazel, you can restore the backup files:
|
|
||||||
|
|
||||||
- `/angular.json.bak` replaces `/angular.json`
|
|
||||||
|
|
||||||
## Advanced configuration
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
|
||||||
|
|
||||||
Editing the Bazel configuration may prevent you opting out of Bazel.
|
|
||||||
Custom behaviors driven by Bazel won't be available in other Builders.
|
|
||||||
|
|
||||||
This section assumes you are familiar with [Bazel](https://docs.bazel.build).
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
You can manually adjust the Bazel configuration to:
|
|
||||||
|
|
||||||
* customize the build steps
|
|
||||||
* parallellize the build for scale and incrementality
|
|
||||||
|
|
||||||
Create the initial Bazel configuration files by running the following command:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
ng build --leaveBazelFilesOnDisk
|
|
||||||
```
|
|
||||||
|
|
||||||
Now you'll find new files in the Angular workspace:
|
|
||||||
|
|
||||||
* `/WORKSPACE` tells Bazel how to download external dependencies.
|
|
||||||
* `/BUILD.bazel` and `/src/BUILD.bazel` tell Bazel about your source code.
|
|
||||||
|
|
||||||
You can find a full-featured example with custom Bazel configurations at https://github.com/bazelbuild/rules_nodejs/tree/master/examples/angular.
|
|
||||||
|
|
||||||
Documentation for using Bazel for frontend projects is linked from https://docs.bazel.build/versions/master/bazel-and-javascript.html.
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## Running Bazel directly
|
|
||||||
|
|
||||||
In some cases you'll want to bypass the Angular CLI builder, and run the Bazel CLI directly.
|
|
||||||
The Bazel tool is managed by the `@bazel/bazelisk` package (similar to how Node.js can be managed by `nvm`).
|
|
||||||
You can install it globally to get the `bazelisk` command in your path, or use `$(npm bin)/bazelisk` in place of bazelisk below.
|
|
||||||
|
|
||||||
The common commands in Bazel are:
|
|
||||||
|
|
||||||
* `bazelisk build [targets]`: Compile the default output artifacts of the given targets.
|
|
||||||
* `bazelisk test [targets]`: For whichever `*_test` targets are found in the patterns, run the tests.
|
|
||||||
* `bazelisk run [target]`: Compile the program represented by target, and then run it.
|
|
||||||
|
|
||||||
To repeat the command any time the inputs change (watch mode), replace `bazelisk` with `ibazel` in these commands.
|
|
||||||
|
|
||||||
The output locations are printed in the output.
|
|
||||||
|
|
||||||
Full documentation for the Bazel CLI is at https://docs.bazel.build/versions/master/command-line-reference.html.
|
|
||||||
|
|
||||||
|
|
||||||
## Querying the build graph
|
|
||||||
|
|
||||||
Because Bazel constructs a graph out of your targets, you can find lots of useful information.
|
|
||||||
|
|
||||||
Using the graphviz optional dependency, you'll have a program `dot`, which you can use with `bazel query`:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
$ bazel query --output=graph ... | dot -Tpng > graph.png
|
|
||||||
```
|
|
||||||
|
|
||||||
See https://docs.bazel.build/versions/master/query-how-to.html for more details on `bazel query`.
|
|
||||||
|
|
||||||
|
|
||||||
## Customizing `BUILD.bazel` files
|
|
||||||
|
|
||||||
"Rules" are like plugins for Bazel. Many rule sets are available. This guide documents the ones maintained by the Angular team at Google.
|
|
||||||
|
|
||||||
Rules are used in `BUILD.bazel` files, which are markers for the packages in your workspace. Each `BUILD.bazel` file declares a separate package to Bazel, though you can have more coarse-grained distributions so that the packages you publish (for example, to `npm`) can be made up of many Bazel packages.
|
|
||||||
|
|
||||||
In the `BUILD.bazel` file, each rule must first be imported, using the `load` statement. Then the rule is called with some attributes, and the result of calling the rule is that you've declared to Bazel how it can derive some outputs given some inputs and dependencies. Then later, when you run a `bazel` command line, Bazel loads all the rules you've declared to determine an absolute ordering of what needs to be run. Note that only the rules needed to produce the requested output will actually be executed.
|
|
||||||
|
|
||||||
A list of common rules for frontend development is documented in the README at https://github.com/bazelbuild/rules_nodejs/.
|
|
318
aio/content/guide/binding-syntax.md
Normal file
318
aio/content/guide/binding-syntax.md
Normal file
@ -0,0 +1,318 @@
|
|||||||
|
|
||||||
|
# Binding syntax: an overview
|
||||||
|
|
||||||
|
Data-binding is a mechanism for coordinating what users see, specifically
|
||||||
|
with application data values.
|
||||||
|
While you could push values to and pull values from HTML,
|
||||||
|
the application is easier to write, read, and maintain if you turn these tasks over to a binding framework.
|
||||||
|
You simply declare bindings between binding sources, target HTML elements, and let the framework do the rest.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
See the <live-example></live-example> for a working example containing the code snippets in this guide.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
Angular provides many kinds of data-binding. Binding types can be grouped into three categories distinguished by the direction of data flow:
|
||||||
|
|
||||||
|
* From the _source-to-view_
|
||||||
|
* From _view-to-source_
|
||||||
|
* Two-way sequence: _view-to-source-to-view_
|
||||||
|
|
||||||
|
<style>
|
||||||
|
td, th {vertical-align: top}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<table width="100%">
|
||||||
|
<col width="30%">
|
||||||
|
</col>
|
||||||
|
<col width="50%">
|
||||||
|
</col>
|
||||||
|
<col width="20%">
|
||||||
|
</col>
|
||||||
|
<tr>
|
||||||
|
<th>
|
||||||
|
Type
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
Syntax
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
Category
|
||||||
|
</th>
|
||||||
|
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
Interpolation<br>
|
||||||
|
Property<br>
|
||||||
|
Attribute<br>
|
||||||
|
Class<br>
|
||||||
|
Style
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
|
||||||
|
<code-example>
|
||||||
|
{{expression}}
|
||||||
|
[target]="expression"
|
||||||
|
bind-target="expression"
|
||||||
|
</code-example>
|
||||||
|
|
||||||
|
</td>
|
||||||
|
|
||||||
|
<td>
|
||||||
|
One-way<br>from data source<br>to view target
|
||||||
|
</td>
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
Event
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<code-example>
|
||||||
|
(target)="statement"
|
||||||
|
on-target="statement"
|
||||||
|
</code-example>
|
||||||
|
</td>
|
||||||
|
|
||||||
|
<td>
|
||||||
|
One-way<br>from view target<br>to data source
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
Two-way
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<code-example>
|
||||||
|
[(target)]="expression"
|
||||||
|
bindon-target="expression"
|
||||||
|
</code-example>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
Two-way
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
Binding types other than interpolation have a **target name** to the left of the equal sign, either surrounded by punctuation, `[]` or `()`,
|
||||||
|
or preceded by a prefix: `bind-`, `on-`, `bindon-`.
|
||||||
|
|
||||||
|
The *target* of a binding is the property or event inside the binding punctuation: `[]`, `()` or `[()]`.
|
||||||
|
|
||||||
|
Every public member of a **source** directive is automatically available for binding.
|
||||||
|
You don't have to do anything special to access a directive member in a template expression or statement.
|
||||||
|
|
||||||
|
|
||||||
|
### Data-binding and HTML
|
||||||
|
|
||||||
|
In the normal course of HTML development, you create a visual structure with HTML elements, and
|
||||||
|
you modify those elements by setting element attributes with string constants.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<div class="special">Plain old HTML</div>
|
||||||
|
<img src="images/item.png">
|
||||||
|
<button disabled>Save</button>
|
||||||
|
```
|
||||||
|
|
||||||
|
With data-binding, you can control things like the state of a button:
|
||||||
|
|
||||||
|
<code-example path="binding-syntax/src/app/app.component.html" region="disabled-button" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Notice that the binding is to the `disabled` property of the button's DOM element,
|
||||||
|
**not** the attribute. This applies to data-binding in general. Data-binding works with *properties* of DOM elements, components, and directives, not HTML *attributes*.
|
||||||
|
|
||||||
|
{@a html-attribute-vs-dom-property}
|
||||||
|
|
||||||
|
### HTML attribute vs. DOM property
|
||||||
|
|
||||||
|
The distinction between an HTML attribute and a DOM property is key to understanding
|
||||||
|
how Angular binding works. **Attributes are defined by HTML. Properties are accessed from DOM (Document Object Model) nodes.**
|
||||||
|
|
||||||
|
* A few HTML attributes have 1:1 mapping to properties; for example, `id`.
|
||||||
|
|
||||||
|
* Some HTML attributes don't have corresponding properties; for example, `aria-*`.
|
||||||
|
|
||||||
|
* Some DOM properties don't have corresponding attributes; for example, `textContent`.
|
||||||
|
|
||||||
|
It is important to remember that *HTML attribute* and the *DOM property* are different things, even when they have the same name.
|
||||||
|
In Angular, the only role of HTML attributes is to initialize element and directive state.
|
||||||
|
|
||||||
|
**Template binding works with *properties* and *events*, not *attributes*.**
|
||||||
|
|
||||||
|
When you write a data-binding, you're dealing exclusively with the *DOM properties* and *events* of the target object.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
This general rule can help you build a mental model of attributes and DOM properties:
|
||||||
|
**Attributes initialize DOM properties and then they are done.
|
||||||
|
Property values can change; attribute values can't.**
|
||||||
|
|
||||||
|
There is one exception to this rule.
|
||||||
|
Attributes can be changed by `setAttribute()`, which re-initializes corresponding DOM properties.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
For more information, see the [MDN Interfaces documentation](https://developer.mozilla.org/en-US/docs/Web/API#Interfaces) which has API docs for all the standard DOM elements and their properties.
|
||||||
|
Comparing the [`<td>` attributes](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/td) attributes to the [`<td>` properties](https://developer.mozilla.org/en-US/docs/Web/API/HTMLTableCellElement) provides a helpful example for differentiation.
|
||||||
|
In particular, you can navigate from the attributes page to the properties via "DOM interface" link, and navigate the inheritance hierarchy up to `HTMLTableCellElement`.
|
||||||
|
|
||||||
|
|
||||||
|
#### Example 1: an `<input>`
|
||||||
|
|
||||||
|
When the browser renders `<input type="text" value="Sarah">`, it creates a
|
||||||
|
corresponding DOM node with a `value` property initialized to "Sarah".
|
||||||
|
|
||||||
|
```html
|
||||||
|
<input type="text" value="Sarah">
|
||||||
|
```
|
||||||
|
|
||||||
|
When the user enters "Sally" into the `<input>`, the DOM element `value` *property* becomes "Sally".
|
||||||
|
However, if you look at the HTML attribute `value` using `input.getAttribute('value')`, you can see that the *attribute* remains unchanged—it returns "Sarah".
|
||||||
|
|
||||||
|
The HTML attribute `value` specifies the *initial* value; the DOM `value` property is the *current* value.
|
||||||
|
|
||||||
|
To see attributes versus DOM properties in a functioning app, see the <live-example name="binding-syntax"></live-example> especially for binding syntax.
|
||||||
|
|
||||||
|
#### Example 2: a disabled button
|
||||||
|
|
||||||
|
The `disabled` attribute is another example. A button's `disabled`
|
||||||
|
*property* is `false` by default so the button is enabled.
|
||||||
|
|
||||||
|
When you add the `disabled` *attribute*, its presence alone
|
||||||
|
initializes the button's `disabled` *property* to `true`
|
||||||
|
so the button is disabled.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<button disabled>Test Button</button>
|
||||||
|
```
|
||||||
|
|
||||||
|
Adding and removing the `disabled` *attribute* disables and enables the button.
|
||||||
|
However, the value of the *attribute* is irrelevant,
|
||||||
|
which is why you cannot enable a button by writing `<button disabled="false">Still Disabled</button>`.
|
||||||
|
|
||||||
|
To control the state of the button, set the `disabled` *property*,
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
Though you could technically set the `[attr.disabled]` attribute binding, the values are different in that the property binding requires to a boolean value, while its corresponding attribute binding relies on whether the value is `null` or not. Consider the following:
|
||||||
|
|
||||||
|
```html
|
||||||
|
<input [disabled]="condition ? true : false">
|
||||||
|
<input [attr.disabled]="condition ? 'disabled' : null">
|
||||||
|
```
|
||||||
|
|
||||||
|
Generally, use property binding over attribute binding as it is more intuitive (being a boolean value), has a shorter syntax, and is more performant.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
To see the `disabled` button example in a functioning app, see the <live-example name="binding-syntax"></live-example> especially for binding syntax. This example shows you how to toggle the disabled property from the component.
|
||||||
|
|
||||||
|
## Binding types and targets
|
||||||
|
|
||||||
|
The **target of a data-binding** is something in the DOM.
|
||||||
|
Depending on the binding type, the target can be a property (element, component, or directive),
|
||||||
|
an event (element, component, or directive), or sometimes an attribute name.
|
||||||
|
The following table summarizes the targets for the different binding types.
|
||||||
|
|
||||||
|
<style>
|
||||||
|
td, th {vertical-align: top}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<table width="100%">
|
||||||
|
<col width="10%">
|
||||||
|
</col>
|
||||||
|
<col width="15%">
|
||||||
|
</col>
|
||||||
|
<col width="75%">
|
||||||
|
</col>
|
||||||
|
<tr>
|
||||||
|
<th>
|
||||||
|
Type
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
Target
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
Examples
|
||||||
|
</th>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
Property
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
Element property<br>
|
||||||
|
Component property<br>
|
||||||
|
Directive property
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<code>src</code>, <code>hero</code>, and <code>ngClass</code> in the following:
|
||||||
|
<code-example path="template-syntax/src/app/app.component.html" region="property-binding-syntax-1"></code-example>
|
||||||
|
<!-- For more information, see [Property Binding](guide/property-binding). -->
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
Event
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
Element event<br>
|
||||||
|
Component event<br>
|
||||||
|
Directive event
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<code>click</code>, <code>deleteRequest</code>, and <code>myClick</code> in the following:
|
||||||
|
<code-example path="template-syntax/src/app/app.component.html" region="event-binding-syntax-1"></code-example>
|
||||||
|
<!-- KW--Why don't these links work in the table? -->
|
||||||
|
<!-- <div>For more information, see [Event Binding](guide/event-binding).</div> -->
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
Two-way
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
Event and property
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<code-example path="template-syntax/src/app/app.component.html" region="2-way-binding-syntax-1"></code-example>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
Attribute
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
Attribute
|
||||||
|
(the exception)
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<code-example path="template-syntax/src/app/app.component.html" region="attribute-binding-syntax-1"></code-example>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
Class
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<code>class</code> property
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<code-example path="template-syntax/src/app/app.component.html" region="class-binding-syntax-1"></code-example>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
Style
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<code>style</code> property
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<code-example path="template-syntax/src/app/app.component.html" region="style-binding-syntax-1"></code-example>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
|
@ -54,16 +54,17 @@ Angular supports most recent browsers. This includes the following specific vers
|
|||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
<div> 11, 10*, 9* ("compatibility view" mode not supported) </div>
|
<div> 11, 10*, 9* ("compatibility view" mode not supported) </div>
|
||||||
<div>*deprecated in v10, see the <a href="/guide/deprecations#ie-9-10">deprecations guide</a>.</div>
|
<div>*deprecated in v10, see the {@link guide/deprecations#ie-9-10-and-mobile deprecations guide}.</div>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>
|
<td>
|
||||||
IE Mobile
|
IE Mobile*
|
||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
11
|
11
|
||||||
|
<div>*deprecated in v10, see the {@link guide/deprecations#ie-9-10-and-mobile deprecations guide}.</div>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
|
@ -262,6 +262,33 @@ Each budget entry is a JSON object with the following properties:
|
|||||||
|
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
|
{@a commonjs }
|
||||||
|
## Configuring CommonJS dependencies
|
||||||
|
|
||||||
|
<div class="alert is-important">
|
||||||
|
|
||||||
|
It is recommended that you avoid depending on CommonJS modules in your Angular applications.
|
||||||
|
Depending on CommonJS modules can prevent bundlers and minifiers from optimizing your application, which results in larger bundle sizes.
|
||||||
|
Instead, it is recommended that you use [ECMAScript modules](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/import) in your entire application.
|
||||||
|
For more information, see [How CommonJS is making your bundles larger](https://web.dev/commonjs-larger-bundles/).
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
The Angular CLI outputs warnings if it detects that your browser application depends on CommonJS modules.
|
||||||
|
To disable these warnings, you can add the CommonJS module name to `allowedCommonJsDependencies` option in the `build` options located in `angular.json` file.
|
||||||
|
|
||||||
|
<code-example lang="json">
|
||||||
|
"build": {
|
||||||
|
"builder": "@angular-devkit/build-angular:browser",
|
||||||
|
"options": {
|
||||||
|
"allowedCommonJsDependencies": [
|
||||||
|
"lodash"
|
||||||
|
]
|
||||||
|
...
|
||||||
|
}
|
||||||
|
...
|
||||||
|
},
|
||||||
|
</code-example>
|
||||||
|
|
||||||
{@a browser-compat}
|
{@a browser-compat}
|
||||||
|
|
||||||
|
435
aio/content/guide/built-in-directives.md
Normal file
435
aio/content/guide/built-in-directives.md
Normal file
@ -0,0 +1,435 @@
|
|||||||
|
# Built-in directives
|
||||||
|
|
||||||
|
Angular offers two kinds of built-in directives: [_attribute_ directives](guide/attribute-directives) and [_structural_ directives](guide/structural-directives).
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
See the <live-example></live-example> for a working example containing the code snippets in this guide.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
For more detail, including how to build your own custom directives, see [Attribute Directives](guide/attribute-directives) and [Structural Directives](guide/structural-directives).
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
{@a attribute-directives}
|
||||||
|
|
||||||
|
## Built-in attribute directives
|
||||||
|
|
||||||
|
Attribute directives listen to and modify the behavior of
|
||||||
|
other HTML elements, attributes, properties, and components.
|
||||||
|
You usually apply them to elements as if they were HTML attributes, hence the name.
|
||||||
|
|
||||||
|
Many NgModules such as the [`RouterModule`](guide/router "Routing and Navigation")
|
||||||
|
and the [`FormsModule`](guide/forms "Forms") define their own attribute directives.
|
||||||
|
The most common attribute directives are as follows:
|
||||||
|
|
||||||
|
* [`NgClass`](guide/built-in-directives#ngClass)—adds and removes a set of CSS classes.
|
||||||
|
* [`NgStyle`](guide/built-in-directives#ngStyle)—adds and removes a set of HTML styles.
|
||||||
|
* [`NgModel`](guide/built-in-directives#ngModel)—adds two-way data binding to an HTML form element.
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
{@a ngClass}
|
||||||
|
|
||||||
|
## `NgClass`
|
||||||
|
|
||||||
|
Add or remove several CSS classes simultaneously with `ngClass`.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="special-div" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
To add or remove a *single* class, use [class binding](guide/attribute-binding#class-binding) rather than `NgClass`.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
Consider a `setCurrentClasses()` component method that sets a component property,
|
||||||
|
`currentClasses`, with an object that adds or removes three classes based on the
|
||||||
|
`true`/`false` state of three other component properties. Each key of the object is a CSS class name; its value is `true` if the class should be added,
|
||||||
|
`false` if it should be removed.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.ts" region="setClasses" header="src/app/app.component.ts"></code-example>
|
||||||
|
|
||||||
|
Adding an `ngClass` property binding to `currentClasses` sets the element's classes accordingly:
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgClass-1" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
Remember that in this situation you'd call `setCurrentClasses()`,
|
||||||
|
both initially and when the dependent properties change.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
{@a ngStyle}
|
||||||
|
|
||||||
|
## `NgStyle`
|
||||||
|
|
||||||
|
Use `NgStyle` to set many inline styles simultaneously and dynamically, based on the state of the component.
|
||||||
|
|
||||||
|
### Without `NgStyle`
|
||||||
|
|
||||||
|
For context, consider setting a *single* style value with [style binding](guide/attribute-binding#style-binding), without `NgStyle`.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="without-ng-style" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
However, to set *many* inline styles at the same time, use the `NgStyle` directive.
|
||||||
|
|
||||||
|
The following is a `setCurrentStyles()` method that sets a component
|
||||||
|
property, `currentStyles`, with an object that defines three styles,
|
||||||
|
based on the state of three other component properties:
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.ts" region="setStyles" header="src/app/app.component.ts"></code-example>
|
||||||
|
|
||||||
|
Adding an `ngStyle` property binding to `currentStyles` sets the element's styles accordingly:
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgStyle-2" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
Remember to call `setCurrentStyles()`, both initially and when the dependent properties change.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
{@a ngModel}
|
||||||
|
|
||||||
|
## `[(ngModel)]`: Two-way binding
|
||||||
|
|
||||||
|
The `NgModel` directive allows you to display a data property and
|
||||||
|
update that property when the user makes changes. Here's an example:
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" header="src/app/app.component.html (NgModel example)" region="NgModel-1"></code-example>
|
||||||
|
|
||||||
|
|
||||||
|
### Import `FormsModule` to use `ngModel`
|
||||||
|
|
||||||
|
Before using the `ngModel` directive in a two-way data binding,
|
||||||
|
you must import the `FormsModule` and add it to the NgModule's `imports` list.
|
||||||
|
Learn more about the `FormsModule` and `ngModel` in [Forms](guide/forms#ngModel).
|
||||||
|
|
||||||
|
Remember to import the `FormsModule` to make `[(ngModel)]` available as follows:
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.module.ts" header="src/app/app.module.ts (FormsModule import)" region="import-forms-module"></code-example>
|
||||||
|
|
||||||
|
|
||||||
|
You could achieve the same result with separate bindings to
|
||||||
|
the `<input>` element's `value` property and `input` event:
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="without-NgModel" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
To streamline the syntax, the `ngModel` directive hides the details behind its own `ngModel` input and `ngModelChange` output properties:
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgModelChange" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
The `ngModel` data property sets the element's value property and the `ngModelChange` event property
|
||||||
|
listens for changes to the element's value.
|
||||||
|
|
||||||
|
### `NgModel` and value accessors
|
||||||
|
|
||||||
|
The details are specific to each kind of element and therefore the `NgModel` directive only works for an element
|
||||||
|
supported by a [ControlValueAccessor](api/forms/ControlValueAccessor)
|
||||||
|
that adapts an element to this protocol.
|
||||||
|
Angular provides *value accessors* for all of the basic HTML form elements and the
|
||||||
|
[Forms](guide/forms) guide shows how to bind to them.
|
||||||
|
|
||||||
|
You can't apply `[(ngModel)]` to a non-form native element or a
|
||||||
|
third-party custom component until you write a suitable value accessor. For more information, see
|
||||||
|
the API documentation on [DefaultValueAccessor](https://angular.io/api/forms/DefaultValueAccessor).
|
||||||
|
|
||||||
|
You don't need a value accessor for an Angular component that
|
||||||
|
you write because you can name the value and event properties
|
||||||
|
to suit Angular's basic [two-way binding syntax](guide/two-way-binding)
|
||||||
|
and skip `NgModel` altogether.
|
||||||
|
The `sizer` in the
|
||||||
|
[Two-way Binding](guide/two-way-binding) section is an example of this technique.
|
||||||
|
|
||||||
|
Separate `ngModel` bindings are an improvement over binding to the
|
||||||
|
element's native properties, but you can streamline the binding with a
|
||||||
|
single declaration using the `[(ngModel)]` syntax:
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgModel-1" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
This `[(ngModel)]` syntax can only _set_ a data-bound property.
|
||||||
|
If you need to do something more, you can write the expanded form;
|
||||||
|
for example, the following changes the `<input>` value to uppercase:
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="uppercase" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Here are all variations in action, including the uppercase version:
|
||||||
|
|
||||||
|
<div class="lightbox">
|
||||||
|
<img src='generated/images/guide/built-in-directives/ng-model-anim.gif' alt="NgModel variations">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
{@a structural-directives}
|
||||||
|
|
||||||
|
## Built-in _structural_ directives
|
||||||
|
|
||||||
|
Structural directives are responsible for HTML layout.
|
||||||
|
They shape or reshape the DOM's structure, typically by adding, removing, and manipulating
|
||||||
|
the host elements to which they are attached.
|
||||||
|
|
||||||
|
This section is an introduction to the common built-in structural directives:
|
||||||
|
|
||||||
|
* [`NgIf`](guide/built-in-directives#ngIf)—conditionally creates or destroys subviews from the template.
|
||||||
|
* [`NgFor`](guide/built-in-directives#ngFor)—repeat a node for each item in a list.
|
||||||
|
* [`NgSwitch`](guide/built-in-directives#ngSwitch)—a set of directives that switch among alternative views.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
The deep details of structural directives are covered in the
|
||||||
|
[Structural Directives](guide/structural-directives) guide,
|
||||||
|
which explains the following:
|
||||||
|
|
||||||
|
* Why you
|
||||||
|
[prefix the directive name with an asterisk (\*)](guide/structural-directives#the-asterisk--prefix).
|
||||||
|
* Using [`<ng-container>`](guide/structural-directives#ngcontainer "<ng-container>")
|
||||||
|
to group elements when there is no suitable host element for the directive.
|
||||||
|
* How to write your own structural directive.
|
||||||
|
* That you can only apply [one structural directive](guide/structural-directives#one-per-element "one per host element") to an element.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
{@a ngIf}
|
||||||
|
|
||||||
|
## NgIf
|
||||||
|
|
||||||
|
You can add or remove an element from the DOM by applying an `NgIf` directive to
|
||||||
|
a host element.
|
||||||
|
Bind the directive to a condition expression like `isActive` in this example.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgIf-1" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
Don't forget the asterisk (`*`) in front of `ngIf`. For more information
|
||||||
|
on the asterisk, see the [asterisk (*) prefix](guide/structural-directives#the-asterisk--prefix) section of
|
||||||
|
[Structural Directives](guide/structural-directives).
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
When the `isActive` expression returns a truthy value, `NgIf` adds the
|
||||||
|
`ItemDetailComponent` to the DOM.
|
||||||
|
When the expression is falsy, `NgIf` removes the `ItemDetailComponent`
|
||||||
|
from the DOM, destroying that component and all of its sub-components.
|
||||||
|
|
||||||
|
|
||||||
|
### Show/hide vs. `NgIf`
|
||||||
|
|
||||||
|
Hiding an element is different from removing it with `NgIf`.
|
||||||
|
For comparison, the following example shows how to control
|
||||||
|
the visibility of an element with a
|
||||||
|
[class](guide/attribute-binding#class-binding) or [style](guide/attribute-binding#style-binding) binding.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgIf-3" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
When you hide an element, that element and all of its descendants remain in the DOM.
|
||||||
|
All components for those elements stay in memory and
|
||||||
|
Angular may continue to check for changes.
|
||||||
|
You could be holding onto considerable computing resources and degrading performance
|
||||||
|
unnecessarily.
|
||||||
|
|
||||||
|
`NgIf` works differently. When `NgIf` is `false`, Angular removes the element and its descendants from the DOM.
|
||||||
|
It destroys their components, freeing up resources, which
|
||||||
|
results in a better user experience.
|
||||||
|
|
||||||
|
If you are hiding large component trees, consider `NgIf` as a more
|
||||||
|
efficient alternative to showing/hiding.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
For more information on `NgIf` and `ngIfElse`, see the [API documentation about NgIf](api/common/NgIf).
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
### Guard against null
|
||||||
|
|
||||||
|
Another advantage of `ngIf` is that you can use it to guard against null. Show/hide
|
||||||
|
is best suited for very simple use cases, so when you need a guard, opt instead for `ngIf`. Angular will throw an error if a nested expression tries to access a property of `null`.
|
||||||
|
|
||||||
|
The following shows `NgIf` guarding two `<div>`s.
|
||||||
|
The `currentCustomer` name appears only when there is a `currentCustomer`.
|
||||||
|
The `nullCustomer` will not be displayed as long as it is `null`.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgIf-2" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgIf-2b" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
See also the
|
||||||
|
[safe navigation operator](guide/template-expression-operators#safe-navigation-operator "Safe navigation operator (?.)") below.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
{@a ngFor}
|
||||||
|
## `NgFor`
|
||||||
|
|
||||||
|
`NgFor` is a repeater directive—a way to present a list of items.
|
||||||
|
You define a block of HTML that defines how a single item should be displayed
|
||||||
|
and then you tell Angular to use that block as a template for rendering each item in the list.
|
||||||
|
The text assigned to `*ngFor` is the instruction that guides the repeater process.
|
||||||
|
|
||||||
|
The following example shows `NgFor` applied to a simple `<div>`. (Don't forget the asterisk (`*`) in front of `ngFor`.)
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgFor-1" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
Don't forget the asterisk (`*`) in front of `ngFor`. For more information
|
||||||
|
on the asterisk, see the [asterisk (*) prefix](guide/structural-directives#the-asterisk--prefix) section of
|
||||||
|
[Structural Directives](guide/structural-directives).
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
You can also apply an `NgFor` to a component element, as in the following example.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgFor-2" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
{@a microsyntax}
|
||||||
|
|
||||||
|
<div class="callout is-critical">
|
||||||
|
<header>*ngFor microsyntax</header>
|
||||||
|
|
||||||
|
The string assigned to `*ngFor` is not a [template expression](guide/interpolation). Rather,
|
||||||
|
it's a *microsyntax*—a little language of its own that Angular interprets.
|
||||||
|
The string `"let item of items"` means:
|
||||||
|
|
||||||
|
> *Take each item in the `items` array, store it in the local `item` looping variable, and
|
||||||
|
make it available to the templated HTML for each iteration.*
|
||||||
|
|
||||||
|
Angular translates this instruction into an `<ng-template>` around the host element,
|
||||||
|
then uses this template repeatedly to create a new set of elements and bindings for each `item`
|
||||||
|
in the list.
|
||||||
|
For more information about microsyntax, see the [Structural Directives](guide/structural-directives#microsyntax) guide.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
{@a template-input-variable}
|
||||||
|
|
||||||
|
{@a template-input-variables}
|
||||||
|
|
||||||
|
### Template input variables
|
||||||
|
|
||||||
|
The `let` keyword before `item` creates a template input variable called `item`.
|
||||||
|
The `ngFor` directive iterates over the `items` array returned by the parent component's `items` property
|
||||||
|
and sets `item` to the current item from the array during each iteration.
|
||||||
|
|
||||||
|
Reference `item` within the `ngFor` host element
|
||||||
|
as well as within its descendants to access the item's properties.
|
||||||
|
The following example references `item` first in an interpolation
|
||||||
|
and then passes in a binding to the `item` property of the `<app-item-detail>` component.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgFor-1-2" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
For more information about template input variables, see
|
||||||
|
[Structural Directives](guide/structural-directives#template-input-variable).
|
||||||
|
|
||||||
|
### `*ngFor` with `index`
|
||||||
|
|
||||||
|
The `index` property of the `NgFor` directive context
|
||||||
|
returns the zero-based index of the item in each iteration.
|
||||||
|
You can capture the `index` in a template input variable and use it in the template.
|
||||||
|
|
||||||
|
The next example captures the `index` in a variable named `i` and displays it with the item name.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgFor-3" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
`NgFor` is implemented by the `NgForOf` directive. Read more about the other `NgForOf` context values such as `last`, `even`,
|
||||||
|
and `odd` in the [NgForOf API reference](api/common/NgForOf).
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{@a trackBy}
|
||||||
|
### *ngFor with `trackBy`
|
||||||
|
|
||||||
|
If you use `NgFor` with large lists, a small change to one item, such as removing or adding an item, can trigger a cascade of DOM manipulations. For example, re-querying the server could reset a list with all new item objects, even when those items were previously displayed. In this case, Angular sees only a fresh list of new object references and has no choice but to replace the old DOM elements with all new DOM elements.
|
||||||
|
|
||||||
|
You can make this more efficient with `trackBy`.
|
||||||
|
Add a method to the component that returns the value `NgFor` should track.
|
||||||
|
In this case, that value is the hero's `id`. If the `id` has already been rendered,
|
||||||
|
Angular keeps track of it and doesn't re-query the server for the same `id`.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.ts" region="trackByItems" header="src/app/app.component.ts"></code-example>
|
||||||
|
|
||||||
|
In the microsyntax expression, set `trackBy` to the `trackByItems()` method.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="trackBy" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Here is an illustration of the `trackBy` effect.
|
||||||
|
"Reset items" creates new items with the same `item.id`s.
|
||||||
|
"Change ids" creates new items with new `item.id`s.
|
||||||
|
|
||||||
|
* With no `trackBy`, both buttons trigger complete DOM element replacement.
|
||||||
|
* With `trackBy`, only changing the `id` triggers element replacement.
|
||||||
|
|
||||||
|
<div class="lightbox">
|
||||||
|
<img src="generated/images/guide/built-in-directives/ngfor-trackby.gif" alt="Animation of trackBy">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
Built-in directives use only public APIs; that is,
|
||||||
|
they do not have special access to any private APIs that other directives can't access.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
{@a ngSwitch}
|
||||||
|
## The `NgSwitch` directives
|
||||||
|
|
||||||
|
NgSwitch is like the JavaScript `switch` statement.
|
||||||
|
It displays one element from among several possible elements, based on a switch condition.
|
||||||
|
Angular puts only the selected element into the DOM.
|
||||||
|
<!-- API Flagged -->
|
||||||
|
`NgSwitch` is actually a set of three, cooperating directives:
|
||||||
|
`NgSwitch`, `NgSwitchCase`, and `NgSwitchDefault` as in the following example.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgSwitch" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
<div class="lightbox">
|
||||||
|
<img src="generated/images/guide/built-in-directives/ngswitch.gif" alt="Animation of NgSwitch">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
`NgSwitch` is the controller directive. Bind it to an expression that returns
|
||||||
|
the *switch value*, such as `feature`. Though the `feature` value in this
|
||||||
|
example is a string, the switch value can be of any type.
|
||||||
|
|
||||||
|
**Bind to `[ngSwitch]`**. You'll get an error if you try to set `*ngSwitch` because
|
||||||
|
`NgSwitch` is an *attribute* directive, not a *structural* directive.
|
||||||
|
Rather than touching the DOM directly, it changes the behavior of its companion directives.
|
||||||
|
|
||||||
|
**Bind to `*ngSwitchCase` and `*ngSwitchDefault`**.
|
||||||
|
The `NgSwitchCase` and `NgSwitchDefault` directives are _structural_ directives
|
||||||
|
because they add or remove elements from the DOM.
|
||||||
|
|
||||||
|
* `NgSwitchCase` adds its element to the DOM when its bound value equals the switch value and removes
|
||||||
|
its bound value when it doesn't equal the switch value.
|
||||||
|
|
||||||
|
* `NgSwitchDefault` adds its element to the DOM when there is no selected `NgSwitchCase`.
|
||||||
|
|
||||||
|
The switch directives are particularly useful for adding and removing *component elements*.
|
||||||
|
This example switches among four `item` components defined in the `item-switch.components.ts` file.
|
||||||
|
Each component has an `item` [input property](guide/inputs-outputs#input "Input property")
|
||||||
|
which is bound to the `currentItem` of the parent component.
|
||||||
|
|
||||||
|
Switch directives work as well with native elements and web components too.
|
||||||
|
For example, you could replace the `<app-best-item>` switch case with the following.
|
||||||
|
|
||||||
|
<code-example path="built-in-directives/src/app/app.component.html" region="NgSwitch-div" header="src/app/app.component.html"></code-example>
|
@ -25,7 +25,7 @@ in which two or more components share information.
|
|||||||
## Pass data from parent to child with input binding
|
## Pass data from parent to child with input binding
|
||||||
|
|
||||||
`HeroChildComponent` has two ***input properties***,
|
`HeroChildComponent` has two ***input properties***,
|
||||||
typically adorned with [@Input decorations](guide/template-syntax#inputs-outputs).
|
typically adorned with [@Input() decorator](guide/inputs-outputs#input).
|
||||||
|
|
||||||
|
|
||||||
<code-example path="component-interaction/src/app/hero-child.component.ts" header="component-interaction/src/app/hero-child.component.ts">
|
<code-example path="component-interaction/src/app/hero-child.component.ts" header="component-interaction/src/app/hero-child.component.ts">
|
||||||
@ -180,7 +180,7 @@ The child component exposes an `EventEmitter` property with which it `emits` eve
|
|||||||
The parent binds to that event property and reacts to those events.
|
The parent binds to that event property and reacts to those events.
|
||||||
|
|
||||||
The child's `EventEmitter` property is an ***output property***,
|
The child's `EventEmitter` property is an ***output property***,
|
||||||
typically adorned with an [@Output decoration](guide/template-syntax#inputs-outputs)
|
typically adorned with an [@Output() decorator](guide/inputs-outputs#output)
|
||||||
as seen in this `VoterComponent`:
|
as seen in this `VoterComponent`:
|
||||||
|
|
||||||
|
|
||||||
|
@ -109,7 +109,9 @@ To learn more, see [Schematics Overview](guide/schematics) and [Schematics for
|
|||||||
|
|
||||||
## Publishing your library
|
## Publishing your library
|
||||||
|
|
||||||
Use the Angular CLI and the npm package manager to build and publish your library as an npm package. It is not recommended to publish Ivy libraries to NPM repositories. Before publishing a library to NPM, build it using the `--prod` flag which will use the older compiler and runtime known as View Engine instead of Ivy.
|
Use the Angular CLI and the npm package manager to build and publish your library as an npm package.
|
||||||
|
|
||||||
|
Before publishing a library to NPM, build it using the `--prod` flag which will use the older compiler and runtime known as View Engine instead of Ivy.
|
||||||
|
|
||||||
<code-example language="bash">
|
<code-example language="bash">
|
||||||
ng build my-lib --prod
|
ng build my-lib --prod
|
||||||
@ -119,6 +121,14 @@ npm publish
|
|||||||
|
|
||||||
If you've never published a package in npm before, you must create a user account. Read more in [Publishing npm Packages](https://docs.npmjs.com/getting-started/publishing-npm-packages).
|
If you've never published a package in npm before, you must create a user account. Read more in [Publishing npm Packages](https://docs.npmjs.com/getting-started/publishing-npm-packages).
|
||||||
|
|
||||||
|
<div class="alert is-important">
|
||||||
|
|
||||||
|
For now, it is not recommended to publish Ivy libraries to NPM because Ivy generated code is not backward compatible with View Engine, so apps using View Engine will not be able to consume them. Furthermore, the internal Ivy instructions are not yet stable, which can potentially break consumers using a different Angular version from the one used to build the library.
|
||||||
|
|
||||||
|
When a published library is used in an Ivy app, the Angular CLI will automatically convert it to Ivy using a tool known as the Angular compatibility compiler (`ngcc`). Thus, publishing your libraries using the View Engine compiler ensures that they can be transparently consumed by both View Engine and Ivy apps.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
{@a lib-assets}
|
{@a lib-assets}
|
||||||
|
|
||||||
## Managing assets in a library
|
## Managing assets in a library
|
||||||
|
@ -197,11 +197,11 @@ Like `EvenBetterLogger`, `HeroService` needs to know if the user is authorized
|
|||||||
That authorization can change during the course of a single application session,
|
That authorization can change during the course of a single application session,
|
||||||
as when you log in a different user.
|
as when you log in a different user.
|
||||||
|
|
||||||
Let's say you don't want to inject `UserService` directly into `HeroService`, because you don't want to complicate that service with security-sensitive information.
|
Imagine that you don't want to inject `UserService` directly into `HeroService`, because you don't want to complicate that service with security-sensitive information.
|
||||||
`HeroService` won't have direct access to the user information to decide
|
`HeroService` won't have direct access to the user information to decide
|
||||||
who is authorized and who isn't.
|
who is authorized and who isn't.
|
||||||
|
|
||||||
To resolve this, we give the `HeroService` constructor a boolean flag to control display of secret heroes.
|
To resolve this, give the `HeroService` constructor a boolean flag to control display of secret heroes.
|
||||||
|
|
||||||
<code-example path="dependency-injection/src/app/heroes/hero.service.ts" region="internals" header="src/app/heroes/hero.service.ts (excerpt)"></code-example>
|
<code-example path="dependency-injection/src/app/heroes/hero.service.ts" region="internals" header="src/app/heroes/hero.service.ts (excerpt)"></code-example>
|
||||||
|
|
||||||
@ -262,7 +262,7 @@ For example, when bootstrapping an application, you can register many initialize
|
|||||||
|
|
||||||
```
|
```
|
||||||
export const APP_TOKENS = [
|
export const APP_TOKENS = [
|
||||||
{ provide: PLATFORM_INITIALIZER, useFactory: platformInitialized, multi: true },
|
{ provide: PLATFORM_INITIALIZER, useFactory: platformInitialized, multi: true },
|
||||||
{ provide: APP_INITIALIZER, useFactory: delayBootstrapping, multi: true },
|
{ provide: APP_INITIALIZER, useFactory: delayBootstrapping, multi: true },
|
||||||
{ provide: APP_BOOTSTRAP_LISTENER, useFactory: appBootstrapped, multi: true },
|
{ provide: APP_BOOTSTRAP_LISTENER, useFactory: appBootstrapped, multi: true },
|
||||||
];
|
];
|
||||||
@ -284,6 +284,23 @@ Search for [Constants in API documentation](api?type=const) to find more built-i
|
|||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
Note that the reference to the array returned for a `multi` provider is shared between all the
|
||||||
|
places where the token is injected. We recommend avoiding mutations of the array (especially for
|
||||||
|
predefined tokens) as it may lead to unexpected behavior in other parts of the app that inject
|
||||||
|
the same token. You can prevent the value from being mutated by setting its type to `ReadonlyArray`.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
You can use `ReadonlyArray` to type your `multi` provider, so TypeScript triggers an error in case
|
||||||
|
of unwanted array mutations:
|
||||||
|
|
||||||
|
```
|
||||||
|
constructor(@Inject(MULTI_PROVIDER) multiProvider: ReadonlyArray<MultiProvider>) {
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
{@a tree-shakable-provider}
|
{@a tree-shakable-provider}
|
||||||
{@a tree-shakable-providers}
|
{@a tree-shakable-providers}
|
||||||
|
|
||||||
|
@ -320,7 +320,7 @@ You can dramatically reduce launch time by only loading the application modules
|
|||||||
absolutely must be present when the app starts.
|
absolutely must be present when the app starts.
|
||||||
|
|
||||||
Configure the Angular Router to defer loading of all other modules (and their associated code), either by
|
Configure the Angular Router to defer loading of all other modules (and their associated code), either by
|
||||||
[waiting until the app has launched](guide/router#preloading "Preloading")
|
[waiting until the app has launched](guide/router-tutorial-toh#preloading "Preloading")
|
||||||
or by [_lazy loading_](guide/router#lazy-loading "Lazy loading")
|
or by [_lazy loading_](guide/router#lazy-loading "Lazy loading")
|
||||||
them on demand.
|
them on demand.
|
||||||
|
|
||||||
@ -328,7 +328,7 @@ them on demand.
|
|||||||
|
|
||||||
<header>Don't eagerly import something from a lazy-loaded module</header>
|
<header>Don't eagerly import something from a lazy-loaded module</header>
|
||||||
|
|
||||||
If you mean to lazy-load a module, be careful not import it
|
If you mean to lazy-load a module, be careful not to import it
|
||||||
in a file that's eagerly loaded when the app starts (such as the root `AppModule`).
|
in a file that's eagerly loaded when the app starts (such as the root `AppModule`).
|
||||||
If you do that, the module will be loaded immediately.
|
If you do that, the module will be loaded immediately.
|
||||||
|
|
||||||
@ -469,7 +469,7 @@ The following configurations determine your requirements.
|
|||||||
|
|
||||||
* TypeScript configuration
|
* TypeScript configuration
|
||||||
|
|
||||||
In the TypeScript configuration file, `tsconfig.json`, the "target" option in the `compilerOptions` section determines the ECMAScript target version that the code is compiled to.
|
In the TypeScript configuration file, the "target" option in the `compilerOptions` section determines the ECMAScript target version that the code is compiled to.
|
||||||
Modern browsers support ES2015 natively, while ES5 is more commonly used to support legacy browsers.
|
Modern browsers support ES2015 natively, while ES5 is more commonly used to support legacy browsers.
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
@ -35,6 +35,7 @@ v9 - v12
|
|||||||
|
|
||||||
| Area | API or Feature | May be removed in |
|
| Area | API or Feature | May be removed in |
|
||||||
| ----------------------------- | --------------------------------------------------------------------------- | ----------------- |
|
| ----------------------------- | --------------------------------------------------------------------------- | ----------------- |
|
||||||
|
| `@angular/bazel` | [`Bazel builder and schematics`](#bazelbuilder) | v10 |
|
||||||
| `@angular/common` | [`ReflectiveInjector`](#reflectiveinjector) | <!--v8--> v11 |
|
| `@angular/common` | [`ReflectiveInjector`](#reflectiveinjector) | <!--v8--> v11 |
|
||||||
| `@angular/common` | [`CurrencyPipe` - `DEFAULT_CURRENCY_CODE`](api/common/CurrencyPipe#currency-code-deprecation) | <!--v9--> v11 |
|
| `@angular/common` | [`CurrencyPipe` - `DEFAULT_CURRENCY_CODE`](api/common/CurrencyPipe#currency-code-deprecation) | <!--v9--> v11 |
|
||||||
| `@angular/core` | [`CollectionChangeRecord`](#core) | <!--v7--> v11 |
|
| `@angular/core` | [`CollectionChangeRecord`](#core) | <!--v7--> v11 |
|
||||||
@ -59,7 +60,7 @@ v9 - v12
|
|||||||
| `@angular/core/testing` | [`TestBed.get`](#testing) | <!--v9--> v12 |
|
| `@angular/core/testing` | [`TestBed.get`](#testing) | <!--v9--> v12 |
|
||||||
| `@angular/router` | [`ActivatedRoute` params and `queryParams` properties](#activatedroute-props) | unspecified |
|
| `@angular/router` | [`ActivatedRoute` params and `queryParams` properties](#activatedroute-props) | unspecified |
|
||||||
| template syntax | [`/deep/`, `>>>`, and `::ng-deep`](#deep-component-style-selector) | <!--v7--> unspecified |
|
| template syntax | [`/deep/`, `>>>`, and `::ng-deep`](#deep-component-style-selector) | <!--v7--> unspecified |
|
||||||
| browser support | [`IE 9 and 10`](#ie-9-10) | <!--v10--> v11 |
|
| browser support | [`IE 9 and 10, IE mobile`](#ie-9-10-and-mobile) | <!--v10--> v11 |
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -160,7 +161,11 @@ Tip: In the [API reference section](api) of this doc site, deprecated APIs are i
|
|||||||
|
|
||||||
This section lists all of the currently-deprecated features, which includes template syntax, configuration options, and any other deprecations not listed in the [Deprecated APIs](#deprecated-apis) section above. It also includes deprecated API usage scenarios or API combinations, to augment the information above.
|
This section lists all of the currently-deprecated features, which includes template syntax, configuration options, and any other deprecations not listed in the [Deprecated APIs](#deprecated-apis) section above. It also includes deprecated API usage scenarios or API combinations, to augment the information above.
|
||||||
|
|
||||||
|
{@a bazelbuilder}
|
||||||
|
### Bazel builder and schematics
|
||||||
|
|
||||||
|
Bazel builder and schematics were introduced in Angular Labs to let users try out Bazel without having to manage Bazel version and BUILD files.
|
||||||
|
This feature has been deprecated. For more information, please refer to the [migration doc](https://github.com/angular/angular/blob/master/packages/bazel/src/schematics/README.md).
|
||||||
|
|
||||||
{@a wtf}
|
{@a wtf}
|
||||||
### Web Tracing Framework integration
|
### Web Tracing Framework integration
|
||||||
@ -459,17 +464,17 @@ export class MyModule {
|
|||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
{@a ie-9-10}
|
{@a ie-9-10-and-mobile}
|
||||||
### IE 9 and 10 support
|
### IE 9, 10, and IE mobile support
|
||||||
|
|
||||||
Support for IE 9 and 10 has been deprecated and will be removed in a future version.
|
Support for IE 9 and 10 has been deprecated, as well as support for IE Mobile. These will be dropped in a future version.
|
||||||
Supporting outdated browsers like these increases bundle size, code complexity, and test load, and also requires time and effort that could be spent on improvements to the framework.
|
Supporting outdated browsers like these increases bundle size, code complexity, and test load, and also requires time and effort that could be spent on improvements to the framework.
|
||||||
For example, fixing issues can be more difficult, as a straightforward fix for modern browsers could break old ones that have quirks due to not receiving updates from vendors.
|
For example, fixing issues can be more difficult, as a straightforward fix for modern browsers could break old ones that have quirks due to not receiving updates from vendors.
|
||||||
|
|
||||||
The final decision was made on three key points:
|
The final decision was made on three key points:
|
||||||
* __Vendor support__: Microsoft dropped support of IE 9 and 10 on 1/12/16, meaning they no longer provide security updates or technical support.
|
* __Vendor support__: Microsoft dropped support of IE 9 and 10 on 1/12/16, meaning they no longer provide security updates or technical support. Additionally, Microsoft dropped support for Windows 10 Mobile in December 2019.
|
||||||
* __Usage statistics__: We looked at usage trends for IE 9 and 10 from various sources and all indicated that usage percentages were extremely small (fractions of 1%).
|
* __Usage statistics__: We looked at usage trends for IE 9 and 10 (as well as IE Mobile) from various sources and all indicated that usage percentages were extremely small (fractions of 1%).
|
||||||
* __Feedback from partners__: We also reached out to some of our Angular customers and none expressed concern about dropping IE 9 and 10 support.
|
* __Feedback from partners__: We also reached out to some of our Angular customers and none expressed concern about dropping IE 9, 10, nor IE Mobile support.
|
||||||
|
|
||||||
|
|
||||||
{@a wrapped-value}
|
{@a wrapped-value}
|
||||||
@ -485,6 +490,56 @@ If you rely on the behavior that the same object instance should cause change de
|
|||||||
- Clone the resulting value so that it has a new identity.
|
- Clone the resulting value so that it has a new identity.
|
||||||
- Explicitly call [`ChangeDetectorRef.detectChanges()`](api/core/ChangeDetectorRef#detectchanges) to force the update.
|
- Explicitly call [`ChangeDetectorRef.detectChanges()`](api/core/ChangeDetectorRef#detectchanges) to force the update.
|
||||||
|
|
||||||
|
{@a deprecated-cli-flags}
|
||||||
|
## Deprecated CLI APIs and Options
|
||||||
|
|
||||||
|
This section contains a complete list all of the currently deprecated CLI flags.
|
||||||
|
|
||||||
|
### @angular-devkit/build-angular
|
||||||
|
|
||||||
|
| API/Option | May be removed in | Notes |
|
||||||
|
| ------------------------------- | ----------------- |-------------------------------------------------------------------------------- |
|
||||||
|
| `i18nFile` | <!--v9--> v11 | Specified in the project locale configuration in version 9 and later. |
|
||||||
|
| `i18nFormat` | <!--v9--> v11 | Format is now automatically detected. |
|
||||||
|
| `i18nLocale` | <!--v9--> v11 | New [localization option](/guide/i18n#localize-config) in version 9 and later. |
|
||||||
|
| `lazyModules` | <!--v9--> v11 | Used with deprecated SystemJsNgModuleLoader. |
|
||||||
|
| `rebaseRootRelativeCssUrls` | <!--v8--> v11 | Intended only to assist with specific migration issues. |
|
||||||
|
| `scripts[].lazy` | <!--v8--> v11 | Renamed to `scripts[].inject`. |
|
||||||
|
| `styles[].lazy` | <!--v8--> v11 | Renamed to `styles[].inject`. |
|
||||||
|
| `i18nFormat` | <!--v9--> v11 | Renamed to `format` to simplify the user experience. |
|
||||||
|
| `i18nLocale` | <!--v9--> v11 | Redundant with project’s source locale. |
|
||||||
|
| `scripts[].lazy` | <!--v8--> v11 | Renamed to `scripts[].inject`. |
|
||||||
|
| `styles[].lazy` | <!--v8--> v11 | Renamed to `styles[].inject`. |
|
||||||
|
| `i18nFile` | <!--v9--> v11 | Specified in the project locale configuration in version 9 and later. |
|
||||||
|
| `i18nFormat` | <!--v9--> v11 | Format is now automatically detected. |
|
||||||
|
| `i18nLocale` | <!--v9--> v11 | New [localization option](/guide/i18n#localize-config) in version 9 and later. |
|
||||||
|
| `lazyModules` | <!--v9--> v11 | Used with deprecated SystemJsNgModuleLoader. |
|
||||||
|
|
||||||
|
### @angular-devkit/core
|
||||||
|
|
||||||
|
| API/Option | May be removed in | Notes |
|
||||||
|
| ------------------------------- | ----------------- |-------------------------------------------------------------------------------- |
|
||||||
|
| `ModuleNotFoundException` | <!--v8--> v10 | Not used within projects. Used with Tooling API only. Not Yarn PnP compatible and not used in the Angular CLI. Use Node.js [require.resolve](https://nodejs.org/api/modules.html#modules_require_resolve_request_options).|
|
||||||
|
| `resolve` | <!--v8--> v10 | Not used within projects. Used with Tooling API only. Not Yarn PnP compatible and not used in the Angular CLI. Use Node.js [require.resolve](https://nodejs.org/api/modules.html#modules_require_resolve_request_options).|
|
||||||
|
| `setResolveHook` | <!--v8--> v10 | Not used within projects. Used with Tooling API only. Not Yarn PnP compatible and not used in the Angular CLI. Use Node.js [require.resolve](https://nodejs.org/api/modules.html#modules_require_resolve_request_options).|
|
||||||
|
| `ResolveOptions` | <!--v8--> v10 | Not used within projects. Used with Tooling API only. Not Yarn PnP compatible and not used in the Angular CLI. Use Node.js [require.resolve](https://nodejs.org/api/modules.html#modules_require_resolve_request_options).|
|
||||||
|
| `terminal` | <!--v8--> v10 | Unused implementation of terminal codes (color). |
|
||||||
|
| `isObservable` | <!--v8--> v10 | Not used within projects. Used with Tooling API only. Use `isObservable` function from the `rxjs` package.|
|
||||||
|
|
||||||
|
### @ngtools/webpack
|
||||||
|
|
||||||
|
| API/Option | May be removed in | Notes |
|
||||||
|
| ------------------------------- | ----------------- |-------------------------------------------------------------------------------- |
|
||||||
|
| `discoverLazyRoutes` | <!--v9--> TBD | Used with deprecated SystemJsNgModuleLoader. |
|
||||||
|
| `additionalLazyModules` | <!--v9--> TBD | Used with deprecated SystemJsNgModuleLoader. |
|
||||||
|
| `additionalLazyModuleResources` | <!--v9--> TBD | Used with deprecated SystemJsNgModuleLoader. |
|
||||||
|
|
||||||
|
### @schematics/angular
|
||||||
|
|
||||||
|
| API/Option | May be removed in | Notes |
|
||||||
|
| ------------------------------- | ----------------- |-------------------------------------------------------------------------------- |
|
||||||
|
| `entryComponent` | <!--v9--> TBD | No longer needed with Ivy. |
|
||||||
|
|
||||||
{@a removed}
|
{@a removed}
|
||||||
## Removed APIs
|
## Removed APIs
|
||||||
|
|
||||||
|
@ -153,14 +153,14 @@ It marks that `<li>` element (and its children) as the "repeater template":
|
|||||||
<div class="alert is-important">
|
<div class="alert is-important">
|
||||||
|
|
||||||
Don't forget the leading asterisk (\*) in `*ngFor`. It is an essential part of the syntax.
|
Don't forget the leading asterisk (\*) in `*ngFor`. It is an essential part of the syntax.
|
||||||
Read more about `ngFor` and `*` in the [ngFor section](guide/template-syntax#ngfor) of the [Template Syntax](guide/template-syntax) page.
|
Read more about `ngFor` and `*` in the [ngFor section](guide/built-in-directives#ngfor) of the [Built-in directives](guide/built-in-directives) page.
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
Notice the `hero` in the `ngFor` double-quoted instruction;
|
Notice the `hero` in the `ngFor` double-quoted instruction;
|
||||||
it is an example of a template input variable. Read
|
it is an example of a template input variable. Read
|
||||||
more about template input variables in the [microsyntax](guide/template-syntax#microsyntax) section of
|
more about template input variables in the [microsyntax](guide/built-in-directives#microsyntax) section of
|
||||||
the [Template Syntax](guide/template-syntax) page.
|
the [Built-in directives](guide/built-in-directives) page.
|
||||||
|
|
||||||
Angular duplicates the `<li>` for each item in the list, setting the `hero` variable
|
Angular duplicates the `<li>` for each item in the list, setting the `hero` variable
|
||||||
to the item (the hero) in the current iteration. Angular uses that variable as the
|
to the item (the hero) in the current iteration. Angular uses that variable as the
|
||||||
@ -255,7 +255,7 @@ To see it in action, add the following paragraph at the bottom of the template:
|
|||||||
<div class="alert is-important">
|
<div class="alert is-important">
|
||||||
|
|
||||||
Don't forget the leading asterisk (\*) in `*ngIf`. It is an essential part of the syntax.
|
Don't forget the leading asterisk (\*) in `*ngIf`. It is an essential part of the syntax.
|
||||||
Read more about `ngIf` and `*` in the [ngIf section](guide/template-syntax#ngIf) of the [Template Syntax](guide/template-syntax) page.
|
Read more about `ngIf` and `*` in the [ngIf section](guide/built-in-directives#ngIf) of the [Built-in directives](guide/built-in-directives) page.
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -266,7 +266,7 @@ When the component's list of heroes has more than three items, Angular adds the
|
|||||||
to the DOM and the message appears.
|
to the DOM and the message appears.
|
||||||
If there are three or fewer items, Angular omits the paragraph, so no message appears.
|
If there are three or fewer items, Angular omits the paragraph, so no message appears.
|
||||||
|
|
||||||
For more information, see [template expressions](guide/template-syntax#template-expressions).
|
For more information, see [template expression operators](guide/interpolation#template-expressions).
|
||||||
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
@ -931,7 +931,7 @@ If you do, be sure to set the `id` attribute - not the `name` attribute! The doc
|
|||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
## Alerts and Calllouts
|
## Alerts and Callouts
|
||||||
|
|
||||||
Alerts and callouts present warnings, extra detail or references to other pages. They can also be used to provide commentary that _enriches_ the reader's understanding of the content being presented.
|
Alerts and callouts present warnings, extra detail or references to other pages. They can also be used to provide commentary that _enriches_ the reader's understanding of the content being presented.
|
||||||
|
|
||||||
|
@ -131,22 +131,6 @@ The `createComponent()` method returns a reference to the loaded component.
|
|||||||
Use that reference to interact with the component by assigning to its properties or calling its methods.
|
Use that reference to interact with the component by assigning to its properties or calling its methods.
|
||||||
|
|
||||||
|
|
||||||
{@a selector-references}
|
|
||||||
|
|
||||||
|
|
||||||
#### Selector references
|
|
||||||
|
|
||||||
Generally, the Angular compiler generates a `ComponentFactory`
|
|
||||||
for any component referenced in a template. However, there are
|
|
||||||
no selector references in the templates for
|
|
||||||
dynamically loaded components since they load at runtime.
|
|
||||||
|
|
||||||
To ensure that the compiler still generates a factory,
|
|
||||||
add dynamically loaded components to the `NgModule`'s `entryComponents` array:
|
|
||||||
|
|
||||||
<code-example path="dynamic-component-loader/src/app/app.module.ts" region="entry-components" header="src/app/app.module.ts (entry components)"></code-example>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
{@a common-interface}
|
{@a common-interface}
|
||||||
|
|
||||||
|
@ -117,9 +117,9 @@ The recently-developed [custom elements](https://developer.mozilla.org/en-US/doc
|
|||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
In browsers that support Custom Elements natively, the specification requires developers use ES2015 classes to define Custom Elements - developers can opt-in to this by setting the `target: "es2015"` property in their project's `tsconfig.json`. As Custom Element and ES2015 support may not be available in all browsers, developers can instead choose to use a polyfill to support older browsers and ES5 code.
|
In browsers that support Custom Elements natively, the specification requires developers use ES2015 classes to define Custom Elements - developers can opt-in to this by setting the `target: "es2015"` property in their project's [TypeScript configuration file](/guide/typescript-configuration). As Custom Element and ES2015 support may not be available in all browsers, developers can instead choose to use a polyfill to support older browsers and ES5 code.
|
||||||
|
|
||||||
Use the [Angular CLI](cli) to automatically set up your project with the correct polyfill: `ng add @angular/elements --name=*your_project_name*`.
|
Use the [Angular CLI](cli) to automatically set up your project with the correct polyfill: `ng add @angular/elements --project=*your_project_name*`.
|
||||||
- For more information about polyfills, see [polyfill documentation](https://www.webcomponents.org/polyfills).
|
- For more information about polyfills, see [polyfill documentation](https://www.webcomponents.org/polyfills).
|
||||||
|
|
||||||
- For more information about Angular browser support, see [Browser Support](guide/browser-support).
|
- For more information about Angular browser support, see [Browser Support](guide/browser-support).
|
||||||
@ -186,7 +186,7 @@ aDialog.content = 123; // <-- ERROR: TypeScript knows this should be a string.
|
|||||||
aDialog.body = 'News'; // <-- ERROR: TypeScript knows there is no `body` property on `aDialog`.
|
aDialog.body = 'News'; // <-- ERROR: TypeScript knows there is no `body` property on `aDialog`.
|
||||||
```
|
```
|
||||||
|
|
||||||
This is a good way to quickly get TypeScript features, such as type checking and autocomplete support, for you custom element. But it can get cumbersome if you need it in several places, because you have to cast the return type on every occurrence.
|
This is a good way to quickly get TypeScript features, such as type checking and autocomplete support, for your custom element. But it can get cumbersome if you need it in several places, because you have to cast the return type on every occurrence.
|
||||||
|
|
||||||
An alternative way, that only requires defining each custom element's type once, is augmenting the `HTMLElementTagNameMap`, which TypeScript uses to infer the type of a returned element based on its tag name (for DOM methods such as `document.createElement()`, `document.querySelector()`, etc.):
|
An alternative way, that only requires defining each custom element's type once, is augmenting the `HTMLElementTagNameMap`, which TypeScript uses to infer the type of a returned element based on its tag name (for DOM methods such as `document.createElement()`, `document.querySelector()`, etc.):
|
||||||
|
|
||||||
|
108
aio/content/guide/event-binding.md
Normal file
108
aio/content/guide/event-binding.md
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
# Event binding `(event)`
|
||||||
|
|
||||||
|
Event binding allows you to listen for certain events such as
|
||||||
|
keystrokes, mouse movements, clicks, and touches.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
See the <live-example></live-example> for a working example containing the code snippets in this guide.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
Angular event binding syntax consists of a **target event** name
|
||||||
|
within parentheses on the left of an equal sign, and a quoted
|
||||||
|
template statement on the right.
|
||||||
|
The following event binding listens for the button's click events, calling
|
||||||
|
the component's `onSave()` method whenever a click occurs:
|
||||||
|
|
||||||
|
<div class="lightbox">
|
||||||
|
<img src='generated/images/guide/template-syntax/syntax-diagram.svg' alt="Syntax diagram">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
## Target event
|
||||||
|
|
||||||
|
As above, the target is the button's click event.
|
||||||
|
|
||||||
|
<code-example path="event-binding/src/app/app.component.html" region="event-binding-1" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Alternatively, use the `on-` prefix, known as the canonical form:
|
||||||
|
|
||||||
|
<code-example path="event-binding/src/app/app.component.html" region="event-binding-2" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Element events may be the more common targets, but Angular looks first to see if the name matches an event property
|
||||||
|
of a known directive, as it does in the following example:
|
||||||
|
|
||||||
|
<code-example path="event-binding/src/app/app.component.html" region="custom-directive" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
If the name fails to match an element event or an output property of a known directive,
|
||||||
|
Angular reports an “unknown directive” error.
|
||||||
|
|
||||||
|
## *$event* and event handling statements
|
||||||
|
|
||||||
|
In an event binding, Angular sets up an event handler for the target event.
|
||||||
|
|
||||||
|
When the event is raised, the handler executes the template statement.
|
||||||
|
The template statement typically involves a receiver, which performs an action
|
||||||
|
in response to the event, such as storing a value from the HTML control
|
||||||
|
into a model.
|
||||||
|
|
||||||
|
The binding conveys information about the event. This information can include data values such as an event object, string, or number named `$event`.
|
||||||
|
|
||||||
|
The target event determines the shape of the `$event` object.
|
||||||
|
If the target event is a native DOM element event, then `$event` is a
|
||||||
|
[DOM event object](https://developer.mozilla.org/en-US/docs/Web/Events),
|
||||||
|
with properties such as `target` and `target.value`.
|
||||||
|
|
||||||
|
Consider this example:
|
||||||
|
|
||||||
|
<code-example path="event-binding/src/app/app.component.html" region="event-binding-3" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
This code sets the `<input>` `value` property by binding to the `name` property.
|
||||||
|
To listen for changes to the value, the code binds to the `input`
|
||||||
|
event of the `<input>` element.
|
||||||
|
When the user makes changes, the `input` event is raised, and the binding executes
|
||||||
|
the statement within a context that includes the DOM event object, `$event`.
|
||||||
|
|
||||||
|
To update the `name` property, the changed text is retrieved by following the path `$event.target.value`.
|
||||||
|
|
||||||
|
If the event belongs to a directive—recall that components
|
||||||
|
are directives—`$event` has whatever shape the directive produces.
|
||||||
|
|
||||||
|
## Custom events with `EventEmitter`
|
||||||
|
|
||||||
|
Directives typically raise custom events with an Angular [EventEmitter](api/core/EventEmitter).
|
||||||
|
The directive creates an `EventEmitter` and exposes it as a property.
|
||||||
|
The directive calls `EventEmitter.emit(payload)` to fire an event, passing in a message payload, which can be anything.
|
||||||
|
Parent directives listen for the event by binding to this property and accessing the payload through the `$event` object.
|
||||||
|
|
||||||
|
Consider an `ItemDetailComponent` that presents item information and responds to user actions.
|
||||||
|
Although the `ItemDetailComponent` has a delete button, it doesn't know how to delete the hero. It can only raise an event reporting the user's delete request.
|
||||||
|
|
||||||
|
Here are the pertinent excerpts from that `ItemDetailComponent`:
|
||||||
|
|
||||||
|
<code-example path="event-binding/src/app/item-detail/item-detail.component.html" header="src/app/item-detail/item-detail.component.html (template)" region="line-through"></code-example>
|
||||||
|
|
||||||
|
<code-example path="event-binding/src/app/item-detail/item-detail.component.ts" header="src/app/item-detail/item-detail.component.ts (deleteRequest)" region="deleteRequest"></code-example>
|
||||||
|
|
||||||
|
The component defines a `deleteRequest` property that returns an `EventEmitter`.
|
||||||
|
When the user clicks *delete*, the component invokes the `delete()` method,
|
||||||
|
telling the `EventEmitter` to emit an `Item` object.
|
||||||
|
|
||||||
|
Now imagine a hosting parent component that binds to the `deleteRequest` event
|
||||||
|
of the `ItemDetailComponent`.
|
||||||
|
|
||||||
|
<code-example path="event-binding/src/app/app.component.html" header="src/app/app.component.html (event-binding-to-component)" region="event-binding-to-component"></code-example>
|
||||||
|
|
||||||
|
When the `deleteRequest` event fires, Angular calls the parent component's
|
||||||
|
`deleteItem()` method, passing the *item-to-delete* (emitted by `ItemDetail`)
|
||||||
|
in the `$event` variable.
|
||||||
|
|
||||||
|
## Template statements have side effects
|
||||||
|
|
||||||
|
Though [template expressions](guide/interpolation#template-expressions) shouldn't have [side effects](guide/property-binding#avoid-side-effects), template
|
||||||
|
statements usually do. The `deleteItem()` method does have
|
||||||
|
a side effect: it deletes an item.
|
||||||
|
|
||||||
|
Deleting an item updates the model, and depending on your code, triggers
|
||||||
|
other changes including queries and saving to a remote server.
|
||||||
|
These changes propagate through the system and ultimately display in this and other views.
|
@ -40,7 +40,8 @@ The top level of the workspace contains workspace-wide configuration files, conf
|
|||||||
| `package-lock.json` | Provides version information for all packages installed into `node_modules` by the npm client. See [npm documentation](https://docs.npmjs.com/files/package-lock.json) for details. If you use the yarn client, this file will be [yarn.lock](https://yarnpkg.com/lang/en/docs/yarn-lock/) instead. |
|
| `package-lock.json` | Provides version information for all packages installed into `node_modules` by the npm client. See [npm documentation](https://docs.npmjs.com/files/package-lock.json) for details. If you use the yarn client, this file will be [yarn.lock](https://yarnpkg.com/lang/en/docs/yarn-lock/) instead. |
|
||||||
| `src/` | Source files for the root-level application project. |
|
| `src/` | Source files for the root-level application project. |
|
||||||
| `node_modules/` | Provides [npm packages](guide/npm-packages) to the entire workspace. Workspace-wide `node_modules` dependencies are visible to all projects. |
|
| `node_modules/` | Provides [npm packages](guide/npm-packages) to the entire workspace. Workspace-wide `node_modules` dependencies are visible to all projects. |
|
||||||
| `tsconfig.json` | Default [TypeScript](https://www.typescriptlang.org/) configuration for projects in the workspace. |
|
| `tsconfig.json` | The `tsconfig.json` file is a ["Solution Style"](https://www.typescriptlang.org/docs/handbook/release-notes/typescript-3-9.html#support-for-solution-style-tsconfigjson-files) TypeScript configuration file. Code editors and TypeScript’s language server use this file to improve development experience. Compilers do not use this file. |
|
||||||
|
| `tsconfig.base.json` | The base [TypeScript](https://www.typescriptlang.org/) configuration for projects in the workspace. All other configuration files inherit from this base file. For more information, see the [Configuration inheritance with extends](https://www.typescriptlang.org/docs/handbook/tsconfig-json.html#configuration-inheritance-with-extends) section of the TypeScript documentation.|
|
||||||
| `tslint.json` | Default [TSLint](https://palantir.github.io/tslint/) configuration for projects in the workspace. |
|
| `tslint.json` | Default [TSLint](https://palantir.github.io/tslint/) configuration for projects in the workspace. |
|
||||||
|
|
||||||
|
|
||||||
@ -54,7 +55,7 @@ This initial root-level application is the *default app* for CLI commands (unles
|
|||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
Besides using the CLI on the command line, you can also use an interactive development environment like [Angular Console](https://angularconsole.com/), or manipulate files directly in the app's source folder and configuration files.
|
Besides using the CLI on the command line, you can also manipulate files directly in the app's source folder and configuration files.
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -77,6 +78,12 @@ Files at the top level of `src/` support testing and running your application. S
|
|||||||
| `styles.sass` | Lists CSS files that supply styles for a project. The extension reflects the style preprocessor you have configured for the project. |
|
| `styles.sass` | Lists CSS files that supply styles for a project. The extension reflects the style preprocessor you have configured for the project. |
|
||||||
| `test.ts` | The main entry point for your unit tests, with some Angular-specific configuration. You don't typically need to edit this file. |
|
| `test.ts` | The main entry point for your unit tests, with some Angular-specific configuration. You don't typically need to edit this file. |
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
If you create an application using Angular's strict mode, you will also have an additional `package.json` file in the `src/app` directory. For more information, see [Strict mode](/guide/strict-mode).
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
{@a app-src}
|
{@a app-src}
|
||||||
|
|
||||||
Inside the `src/` folder, the `app/` folder contains your project's logic and data.
|
Inside the `src/` folder, the `app/` folder contains your project's logic and data.
|
||||||
@ -89,13 +96,14 @@ Angular components, templates, and styles go here.
|
|||||||
| `app/app.component.css` | Defines the base CSS stylesheet for the root `AppComponent`. |
|
| `app/app.component.css` | Defines the base CSS stylesheet for the root `AppComponent`. |
|
||||||
| `app/app.component.spec.ts` | Defines a unit test for the root `AppComponent`. |
|
| `app/app.component.spec.ts` | Defines a unit test for the root `AppComponent`. |
|
||||||
| `app/app.module.ts` | Defines the root module, named `AppModule`, that tells Angular how to assemble the application. Initially declares only the `AppComponent`. As you add more components to the app, they must be declared here. |
|
| `app/app.module.ts` | Defines the root module, named `AppModule`, that tells Angular how to assemble the application. Initially declares only the `AppComponent`. As you add more components to the app, they must be declared here. |
|
||||||
|
| `app/package.json` | This file is generated only in applications created using `--strict` mode. This file is not used by package managers. It is used to tell the tools and bundlers whether the code under this directory is free of non-local [side-effects](guide/strict-mode#side-effect). |
|
||||||
|
|
||||||
### Application configuration files
|
### Application configuration files
|
||||||
|
|
||||||
The application-specific configuration files for the root application reside at the workspace root level.
|
The application-specific configuration files for the root application reside at the workspace root level.
|
||||||
For a multi-project workspace, project-specific configuration files are in the project root, under `projects/project-name/`.
|
For a multi-project workspace, project-specific configuration files are in the project root, under `projects/project-name/`.
|
||||||
|
|
||||||
Project-specific [TypeScript](https://www.typescriptlang.org/) configuration files inherit from the workspace-wide `tsconfig.json`, and project-specific [TSLint](https://palantir.github.io/tslint/) configuration files inherit from the workspace-wide `tslint.json`.
|
Project-specific [TypeScript](https://www.typescriptlang.org/) configuration files inherit from the workspace-wide `tsconfig.base.json`, and project-specific [TSLint](https://palantir.github.io/tslint/) configuration files inherit from the workspace-wide `tslint.json`.
|
||||||
|
|
||||||
| APPLICATION-SPECIFIC CONFIG FILES | PURPOSE |
|
| APPLICATION-SPECIFIC CONFIG FILES | PURPOSE |
|
||||||
| :--------------------- | :------------------------------------------|
|
| :--------------------- | :------------------------------------------|
|
||||||
|
@ -4,22 +4,27 @@ Handling user input with forms is the cornerstone of many common applications. A
|
|||||||
|
|
||||||
Angular provides two different approaches to handling user input through forms: reactive and template-driven. Both capture user input events from the view, validate the user input, create a form model and data model to update, and provide a way to track changes.
|
Angular provides two different approaches to handling user input through forms: reactive and template-driven. Both capture user input events from the view, validate the user input, create a form model and data model to update, and provide a way to track changes.
|
||||||
|
|
||||||
Reactive and template-driven forms process and manage form data differently. Each offers different advantages.
|
|
||||||
|
|
||||||
**In general:**
|
|
||||||
|
|
||||||
* **Reactive forms** are more robust: they're more scalable, reusable, and testable. If forms are a key part of your application, or you're already using reactive patterns for building your application, use reactive forms.
|
|
||||||
* **Template-driven forms** are useful for adding a simple form to an app, such as an email list signup form. They're easy to add to an app, but they don't scale as well as reactive forms. If you have very basic form requirements and logic that can be managed solely in the template, use template-driven forms.
|
|
||||||
|
|
||||||
This guide provides information to help you decide which type of form works best for your situation. It introduces the common building blocks used by both approaches. It also summarizes the key differences between the two approaches, and demonstrates those differences in the context of setup, data flow, and testing.
|
This guide provides information to help you decide which type of form works best for your situation. It introduces the common building blocks used by both approaches. It also summarizes the key differences between the two approaches, and demonstrates those differences in the context of setup, data flow, and testing.
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
## Prerequisites
|
||||||
|
|
||||||
**Note:** For complete information about each kind of form, see [Reactive Forms](guide/reactive-forms) and [Template-driven Forms](guide/forms).
|
This guide assumes that you have a basic understanding of the following.
|
||||||
|
|
||||||
</div>
|
* [TypeScript](https://www.typescriptlang.org/docs/home.html "The TypeScript language") and HTML5 programming.
|
||||||
|
|
||||||
## Key differences
|
* Angular app-design fundamentals, as described in [Angular Concepts](guide/architecture "Introduction to Angular concepts.").
|
||||||
|
|
||||||
|
* The basics of [Angular template syntax](guide/architecture-components#template-syntax "Template syntax intro").
|
||||||
|
|
||||||
|
## Choosing an approach
|
||||||
|
|
||||||
|
Reactive forms and template-driven forms process and manage form data differently. Each approach offers different advantages.
|
||||||
|
|
||||||
|
* **Reactive forms** provide direct, explicit access to the underlying forms object model. Compared to template-driven forms, they are more robust: they're more scalable, reusable, and testable. If forms are a key part of your application, or you're already using reactive patterns for building your application, use reactive forms.
|
||||||
|
|
||||||
|
* **Template-driven forms** rely on directives in the template to create and manipulate the underlying object model. They are useful for adding a simple form to an app, such as an email list signup form. They're easy to add to an app, but they don't scale as well as reactive forms. If you have very basic form requirements and logic that can be managed solely in the template, template-driven forms could be a good fit.
|
||||||
|
|
||||||
|
### Key differences
|
||||||
|
|
||||||
The table below summarizes the key differences between reactive and template-driven forms.
|
The table below summarizes the key differences between reactive and template-driven forms.
|
||||||
|
|
||||||
@ -30,17 +35,33 @@ The table below summarizes the key differences between reactive and template-dri
|
|||||||
|
|
||||||
||Reactive|Template-driven|
|
||Reactive|Template-driven|
|
||||||
|--- |--- |--- |
|
|--- |--- |--- |
|
||||||
|Setup (form model)|More explicit, created in component class|Less explicit, created by directives|
|
|[Setup of form model](#setup) | Explicit, created in component class | Implicit, created by directives |
|
||||||
|Data model|Structured|Unstructured|
|
|[Data model](#data-flow-in-forms) | Structured and immutable | Unstructured and mutable |
|
||||||
|Predictability|Synchronous|Asynchronous|
|
|Predictability | Synchronous | Asynchronous |
|
||||||
|Form validation|Functions|Directives|
|
|[Form validation](#validation) | Functions | Directives |
|
||||||
|Mutability|Immutable|Mutable|
|
|
||||||
|Scalability|Low-level API access|Abstraction on top of APIs|
|
|
||||||
|
|
||||||
## Common foundation
|
### Scalability
|
||||||
|
|
||||||
Both reactive and template-driven forms share underlying building blocks.
|
If forms are a central part of your application, scalability is very important. Being able to reuse form models across components is critical.
|
||||||
|
|
||||||
|
Reactive forms are more scalable than template-driven forms. They provide direct access to the underlying form API, and synchronous access to the form data model, making creating large-scale forms easier.
|
||||||
|
Reactive forms require less setup for testing, and testing does not require deep understanding of change detection to properly test form updates and validation.
|
||||||
|
|
||||||
|
Template-driven forms focus on simple scenarios and are not as reusable.
|
||||||
|
They abstract away the underlying form API, and provide only asynchronous access to the form data model.
|
||||||
|
The abstraction of template-driven forms also affects testing.
|
||||||
|
Tests are deeply reliant on manual change detection execution to run properly, and require more setup.
|
||||||
|
|
||||||
|
{@a setup}
|
||||||
|
|
||||||
|
## Setting up the form model
|
||||||
|
|
||||||
|
Both reactive and template-driven forms track value changes between the form input elements that users interact with and the form data in your component model.
|
||||||
|
The two approaches share underlying building blocks, but differ in how you create and manage the common form-control instances.
|
||||||
|
|
||||||
|
### Common form foundation classes
|
||||||
|
|
||||||
|
Both reactive and template-driven forms are built on the following base classes.
|
||||||
|
|
||||||
* `FormControl` tracks the value and validation status of an individual form control.
|
* `FormControl` tracks the value and validation status of an individual form control.
|
||||||
|
|
||||||
@ -50,59 +71,59 @@ Both reactive and template-driven forms share underlying building blocks.
|
|||||||
|
|
||||||
* `ControlValueAccessor` creates a bridge between Angular `FormControl` instances and native DOM elements.
|
* `ControlValueAccessor` creates a bridge between Angular `FormControl` instances and native DOM elements.
|
||||||
|
|
||||||
See the [Form model setup](#setup-the-form-model) section below for an introduction to how these control instances are created and managed with reactive and template-driven forms. Further details are provided in the [data flow section](#data-flow-in-forms) of this guide.
|
|
||||||
|
|
||||||
{@a setup-the-form-model}
|
{@a setup-the-form-model}
|
||||||
|
|
||||||
## Form model setup
|
|
||||||
|
|
||||||
Reactive and template-driven forms both use a form model to track value changes between Angular forms and form input elements. The examples below show how the form model is defined and created.
|
|
||||||
|
|
||||||
### Setup in reactive forms
|
### Setup in reactive forms
|
||||||
|
|
||||||
Here's a component with an input field for a single control implemented using reactive forms.
|
With reactive forms, you define the form model directly in the component class.
|
||||||
|
The `[formControl]` directive links the explicitly created `FormControl` instance to a specific form element in the view, using an internal value accessor.
|
||||||
|
|
||||||
|
The following component implements an input field for a single control, using reactive forms. In this example, the form model is the `FormControl` instance.
|
||||||
|
|
||||||
<code-example path="forms-overview/src/app/reactive/favorite-color/favorite-color.component.ts">
|
<code-example path="forms-overview/src/app/reactive/favorite-color/favorite-color.component.ts">
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
The source of truth provides the value and status of the form element at a given point in time. In reactive forms, the form model is the source of truth. In the example above, the form model is the `FormControl` instance.
|
Figure 1 shows how, in reactive forms, the form model is the source of truth; it provides the value and status of the form element at any given point in time, through the `[formControl]` directive on the input element.
|
||||||
|
|
||||||
|
**Figure 1.** *Direct access to forms model in a reactive form.*
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src="generated/images/guide/forms-overview/key-diff-reactive-forms.png" alt="Reactive forms key differences">
|
<img src="generated/images/guide/forms-overview/key-diff-reactive-forms.png" alt="Reactive forms key differences">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
With reactive forms, the form model is explicitly defined in the component class. The reactive form directive (in this case, `FormControlDirective`) then links the existing `FormControl` instance to a specific form element in the view using a value accessor (`ControlValueAccessor` instance).
|
|
||||||
|
|
||||||
### Setup in template-driven forms
|
### Setup in template-driven forms
|
||||||
|
|
||||||
Here's the same component with an input field for a single control implemented using template-driven forms.
|
In template-driven forms, the form model is implicit, rather than explicit. The directive `NgModel` creates and manages a `FormControl` instance for a given form element.
|
||||||
|
|
||||||
|
The following component implements the same input field for a single control, using template-driven forms.
|
||||||
|
|
||||||
<code-example path="forms-overview/src/app/template/favorite-color/favorite-color.component.ts">
|
<code-example path="forms-overview/src/app/template/favorite-color/favorite-color.component.ts">
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
In template-driven forms, the source of truth is the template.
|
In a template-driven form the source of truth is the template. You do not have direct programmatic access to the `FormControl` instance, as shown in Figure 2.
|
||||||
|
|
||||||
|
**Figure 2.** *Indirect access to forms model in a template-driven form.*
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src="generated/images/guide/forms-overview/key-diff-td-forms.png" alt="Template-driven forms key differences">
|
<img src="generated/images/guide/forms-overview/key-diff-td-forms.png" alt="Template-driven forms key differences">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
The abstraction of the form model promotes simplicity over structure. The template-driven form directive `NgModel` is responsible for creating and managing the `FormControl` instance for a given form element. It's less explicit, but you no longer have direct control over the form model.
|
|
||||||
|
|
||||||
{@a data-flow-in-forms}
|
{@a data-flow-in-forms}
|
||||||
|
|
||||||
## Data flow in forms
|
## Data flow in forms
|
||||||
|
|
||||||
When building forms in Angular, it's important to understand how the framework handles data flowing from the user or from programmatic changes. Reactive and template-driven forms follow two different strategies when handling form input. The data flow examples below begin with the favorite color input field example from above, and then show how changes to favorite color are handled in reactive forms compared to template-driven forms.
|
When an application contains a form, Angular must keep the view in sync with the component model and the component model in sync with the view.
|
||||||
|
As users change values and make selections through the view, the new values must be reflected in the data model.
|
||||||
|
Similarly, when the program logic changes values in the data model, those values must be reflected in the view.
|
||||||
|
|
||||||
|
Reactive and template-driven forms differ in how they handle data flowing from the user or from programmatic changes.
|
||||||
|
The following diagrams illustrate both kinds of data flow for each type of form, using the favorite-color input field defined above.
|
||||||
|
|
||||||
### Data flow in reactive forms
|
### Data flow in reactive forms
|
||||||
|
|
||||||
As described above, in reactive forms each form element in the view is directly linked to a form model (`FormControl` instance). Updates from the view to the model and from the model to the view are synchronous and aren't dependent on the UI rendered. The diagrams below use the same favorite color example to demonstrate how data flows when an input field's value is changed from the view and then from the model.
|
In reactive forms each form element in the view is directly linked to the form model (a `FormControl` instance). Updates from the view to the model and from the model to the view are synchronous and do not depend on how the UI is rendered.
|
||||||
|
|
||||||
<div class="lightbox">
|
The view-to-model diagram shows how data flows when an input field's value is changed from the view through the following steps.
|
||||||
<img src="generated/images/guide/forms-overview/dataflow-reactive-forms-vtm.png" alt="Reactive forms data flow - view to model" width="100%">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
The steps below outline the data flow from view to model.
|
|
||||||
|
|
||||||
1. The user types a value into the input element, in this case the favorite color *Blue*.
|
1. The user types a value into the input element, in this case the favorite color *Blue*.
|
||||||
1. The form input element emits an "input" event with the latest value.
|
1. The form input element emits an "input" event with the latest value.
|
||||||
@ -111,25 +132,25 @@ The steps below outline the data flow from view to model.
|
|||||||
1. Any subscribers to the `valueChanges` observable receive the new value.
|
1. Any subscribers to the `valueChanges` observable receive the new value.
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src="generated/images/guide/forms-overview/dataflow-reactive-forms-mtv.png" alt="Reactive forms data flow - model to view" width="100%">
|
<img src="generated/images/guide/forms-overview/dataflow-reactive-forms-vtm.png" alt="Reactive forms data flow - view to model">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
The steps below outline the data flow from model to view.
|
The model-to-view diagram shows how a programmatic change to the model is propagated to the view through the following steps.
|
||||||
|
|
||||||
1. The user calls the `favoriteColorControl.setValue()` method, which updates the `FormControl` value.
|
1. The user calls the `favoriteColorControl.setValue()` method, which updates the `FormControl` value.
|
||||||
1. The `FormControl` instance emits the new value through the `valueChanges` observable.
|
1. The `FormControl` instance emits the new value through the `valueChanges` observable.
|
||||||
1. Any subscribers to the `valueChanges` observable receive the new value.
|
1. Any subscribers to the `valueChanges` observable receive the new value.
|
||||||
1. The control value accessor on the form input element updates the element with the new value.
|
1. The control value accessor on the form input element updates the element with the new value.
|
||||||
|
|
||||||
### Data flow in template-driven forms
|
|
||||||
|
|
||||||
In template-driven forms, each form element is linked to a directive that manages the form model internally. The diagrams below use the same favorite color example to demonstrate how data flows when an input field's value is changed from the view and then from the model.
|
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src="generated/images/guide/forms-overview/dataflow-td-forms-vtm.png" alt="Template-driven forms data flow - view to model" width="100%">
|
<img src="generated/images/guide/forms-overview/dataflow-reactive-forms-mtv.png" alt="Reactive forms data flow - model to view">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
The steps below outline the data flow from view to model when the input value changes from *Red* to *Blue*.
|
### Data flow in template-driven forms
|
||||||
|
|
||||||
|
In template-driven forms, each form element is linked to a directive that manages the form model internally.
|
||||||
|
|
||||||
|
The view-to-model diagram shows how data flows when an input field's value is changed from the view through the following steps.
|
||||||
|
|
||||||
1. The user types *Blue* into the input element.
|
1. The user types *Blue* into the input element.
|
||||||
1. The input element emits an "input" event with the value *Blue*.
|
1. The input element emits an "input" event with the value *Blue*.
|
||||||
@ -141,10 +162,10 @@ The steps below outline the data flow from view to model when the input value ch
|
|||||||
is updated to the value emitted by the `ngModelChange` event (*Blue*).
|
is updated to the value emitted by the `ngModelChange` event (*Blue*).
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src="generated/images/guide/forms-overview/dataflow-td-forms-mtv.png" alt="Template-driven forms data flow - model to view" width="100%">
|
<img src="generated/images/guide/forms-overview/dataflow-td-forms-vtm.png" alt="Template-driven forms data flow - view to model" width="100%">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
The steps below outline the data flow from model to view when the `favoriteColor` changes from *Blue* to *Red*.
|
The model-to-view diagram shows how data flows from model to view when the `favoriteColor` changes from *Blue* to *Red*, through the following steps
|
||||||
|
|
||||||
1. The `favoriteColor` value is updated in the component.
|
1. The `favoriteColor` value is updated in the component.
|
||||||
1. Change detection begins.
|
1. Change detection begins.
|
||||||
@ -156,6 +177,30 @@ The steps below outline the data flow from model to view when the `favoriteColor
|
|||||||
1. Any subscribers to the `valueChanges` observable receive the new value.
|
1. Any subscribers to the `valueChanges` observable receive the new value.
|
||||||
1. The control value accessor updates the form input element in the view with the latest `favoriteColor` value.
|
1. The control value accessor updates the form input element in the view with the latest `favoriteColor` value.
|
||||||
|
|
||||||
|
<div class="lightbox">
|
||||||
|
<img src="generated/images/guide/forms-overview/dataflow-td-forms-mtv.png" alt="Template-driven forms data flow - model to view" width="100%">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
### Mutability of the data model
|
||||||
|
|
||||||
|
The change-tracking method plays a role in the efficiency of your application.
|
||||||
|
|
||||||
|
* **Reactive forms** keep the data model pure by providing it as an immutable data structure.
|
||||||
|
Each time a change is triggered on the data model, the `FormControl` instance returns a new data model rather than updating the existing data model.
|
||||||
|
This gives you the ability to track unique changes to the data model through the control's observable.
|
||||||
|
Change detection is more efficient because it only needs to update on unique changes.
|
||||||
|
Because data updates follow reactive patterns, you can integrate with observable operators to transform data.
|
||||||
|
|
||||||
|
* **Template-driven** forms rely on mutability with two-way data binding to update the data model in the component as changes are made in the template.
|
||||||
|
Because there are no unique changes to track on the data model when using two-way data binding, change detection is less efficient at determining when updates are required.
|
||||||
|
|
||||||
|
The difference is demonstrated in the previous examples that use the favorite-color input element.
|
||||||
|
|
||||||
|
* With reactive forms, the **`FormControl` instance** always returns a new value when the control's value is updated.
|
||||||
|
|
||||||
|
* With template-driven forms, the **favorite color property** is always modified to its new value.
|
||||||
|
|
||||||
|
{@a validation}
|
||||||
## Form validation
|
## Form validation
|
||||||
|
|
||||||
Validation is an integral part of managing any set of forms. Whether you're checking for required fields or querying an external API for an existing username, Angular provides a set of built-in validators as well as the ability to create custom validators.
|
Validation is an integral part of managing any set of forms. Whether you're checking for required fields or querying an external API for an existing username, Angular provides a set of built-in validators as well as the ability to create custom validators.
|
||||||
@ -167,36 +212,37 @@ For more information, see [Form Validation](guide/form-validation).
|
|||||||
|
|
||||||
## Testing
|
## Testing
|
||||||
|
|
||||||
Testing plays a large part in complex applications and a simpler testing strategy is useful when validating that your forms function correctly. Reactive forms and template-driven forms have different levels of reliance on rendering the UI to perform assertions based on form control and form field changes. The following examples demonstrate the process of testing forms with reactive and template-driven forms.
|
Testing plays a large part in complex applications. A simpler testing strategy is useful when validating that your forms function correctly.
|
||||||
|
Reactive forms and template-driven forms have different levels of reliance on rendering the UI to perform assertions based on form control and form field changes.
|
||||||
|
The following examples demonstrate the process of testing forms with reactive and template-driven forms.
|
||||||
|
|
||||||
### Testing reactive forms
|
### Testing reactive forms
|
||||||
|
|
||||||
Reactive forms provide a relatively easy testing strategy because they provide synchronous access to the form and data models, and they can be tested without rendering the UI. In these tests, status and data are queried and manipulated through the control without interacting with the change detection cycle.
|
Reactive forms provide a relatively easy testing strategy because they provide synchronous access to the form and data models, and they can be tested without rendering the UI.
|
||||||
|
In these tests, status and data are queried and manipulated through the control without interacting with the change detection cycle.
|
||||||
|
|
||||||
The following tests use the favorite color components mentioned earlier to verify the data flows from view to model and model to view for a reactive form.
|
The following tests use the favorite-color components from previous examples to verify the view-to-model and model-to-view data flows for a reactive form.
|
||||||
|
|
||||||
The following test verifies the data flow from view to model.
|
**Verifying view-to-model data flow**
|
||||||
|
|
||||||
<code-example path="forms-overview/src/app/reactive/favorite-color/favorite-color.component.spec.ts" region="view-to-model" header="Favorite color test - view to model">
|
The first example performs the following steps to verify the view-to-model data flow.
|
||||||
</code-example>
|
|
||||||
|
|
||||||
Here are the steps performed in the view to model test.
|
|
||||||
|
|
||||||
1. Query the view for the form input element, and create a custom "input" event for the test.
|
1. Query the view for the form input element, and create a custom "input" event for the test.
|
||||||
1. Set the new value for the input to *Red*, and dispatch the "input" event on the form input element.
|
1. Set the new value for the input to *Red*, and dispatch the "input" event on the form input element.
|
||||||
1. Assert that the component's `favoriteColorControl` value matches the value from the input.
|
1. Assert that the component's `favoriteColorControl` value matches the value from the input.
|
||||||
|
|
||||||
The following test verifies the data flow from model to view.
|
<code-example path="forms-overview/src/app/reactive/favorite-color/favorite-color.component.spec.ts" region="view-to-model" header="Favorite color test - view to model">
|
||||||
|
|
||||||
<code-example path="forms-overview/src/app/reactive/favorite-color/favorite-color.component.spec.ts" region="model-to-view" header="Favorite color test - model to view">
|
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
Here are the steps performed in the model to view test.
|
The next example performs the following steps to verify the model-to-view data flow.
|
||||||
|
|
||||||
1. Use the `favoriteColorControl`, a `FormControl` instance, to set the new value.
|
1. Use the `favoriteColorControl`, a `FormControl` instance, to set the new value.
|
||||||
1. Query the view for the form input element.
|
1. Query the view for the form input element.
|
||||||
1. Assert that the new value set on the control matches the value in the input.
|
1. Assert that the new value set on the control matches the value in the input.
|
||||||
|
|
||||||
|
<code-example path="forms-overview/src/app/reactive/favorite-color/favorite-color.component.spec.ts" region="model-to-view" header="Favorite color test - model to view">
|
||||||
|
</code-example>
|
||||||
|
|
||||||
### Testing template-driven forms
|
### Testing template-driven forms
|
||||||
|
|
||||||
Writing tests with template-driven forms requires a detailed knowledge of the change detection process and an understanding of how directives run on each cycle to ensure that elements are queried, tested, or changed at the correct time.
|
Writing tests with template-driven forms requires a detailed knowledge of the change detection process and an understanding of how directives run on each cycle to ensure that elements are queried, tested, or changed at the correct time.
|
||||||
@ -228,46 +274,17 @@ Here are the steps performed in the model to view test.
|
|||||||
1. Query the view for the form input element.
|
1. Query the view for the form input element.
|
||||||
1. Assert that the input value matches the value of the `favoriteColor` property in the component instance.
|
1. Assert that the input value matches the value of the `favoriteColor` property in the component instance.
|
||||||
|
|
||||||
## Mutability
|
|
||||||
|
|
||||||
The change tracking method plays a role in the efficiency of your application.
|
|
||||||
|
|
||||||
|
|
||||||
* **Reactive forms** keep the data model pure by providing it as an immutable data structure. Each time a change is triggered on the data model, the `FormControl` instance returns a new data model rather than updating the existing data model. This gives you the ability to track unique changes to the data model through the control's observable. This provides one way for change detection to be more efficient because it only needs to update on unique changes. It also follows reactive patterns that integrate with observable operators to transform data.
|
|
||||||
|
|
||||||
* **Template-driven** forms rely on mutability with two-way data binding to update the data model in the component as changes are made in the template. Because there are no unique changes to track on the data model when using two-way data binding, change detection is less efficient at determining when updates are required.
|
|
||||||
|
|
||||||
The difference is demonstrated in the examples above using the **favorite color** input element.
|
|
||||||
|
|
||||||
|
|
||||||
* With reactive forms, the **`FormControl` instance** always returns a new value when the control's value is updated.
|
|
||||||
|
|
||||||
* With template-driven forms, the **favorite color property** is always modified to its new value.
|
|
||||||
|
|
||||||
## Scalability
|
|
||||||
|
|
||||||
If forms are a central part of your application, scalability is very important. Being able to reuse form models across components is critical.
|
|
||||||
|
|
||||||
|
|
||||||
* **Reactive forms** provide access to low-level APIs and synchronous access to the form model, making creating large-scale forms easier.
|
|
||||||
|
|
||||||
* **Template-driven** forms focus on simple scenarios, are not as reusable, abstract away the low-level APIs, and provide asynchronous access to the form model. The abstraction with template-driven forms also surfaces in testing, where testing reactive forms requires less setup and no dependence on the change detection cycle when updating and validating the form and data models during testing.
|
|
||||||
|
|
||||||
## Final thoughts
|
|
||||||
|
|
||||||
Choosing a strategy begins with understanding the strengths and weaknesses of the options presented. Low-level API and form model access, predictability, mutability, straightforward validation and testing strategies, and scalability are all important considerations in choosing the infrastructure you use to build your forms in Angular. Template-driven forms are similar to patterns in AngularJS, but they have limitations given the criteria of many modern, large-scale Angular apps. Reactive forms minimize these limitations. Reactive forms integrate with reactive patterns already present in other areas of the Angular architecture, and complement those requirements well.
|
|
||||||
|
|
||||||
## Next steps
|
## Next steps
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
To learn more about reactive forms, see the following guides:
|
To learn more about reactive forms, see the following guides:
|
||||||
|
|
||||||
* [Reactive Forms](guide/reactive-forms)
|
* [Reactive forms](guide/reactive-forms)
|
||||||
* [Form Validation](guide/form-validation#reactive-form-validation)
|
* [Form validation](guide/form-validation#reactive-form-validation)
|
||||||
* [Dynamic Forms](guide/dynamic-form)
|
* [Dynamic forms](guide/dynamic-form)
|
||||||
|
|
||||||
To learn more about template-driven forms, see the following guides:
|
To learn more about template-driven forms, see the following guides:
|
||||||
|
|
||||||
* [Template-driven Forms](guide/forms#template-driven-forms)
|
* [Building a template-driven form](guide/forms) tutorial
|
||||||
* [Form Validation](guide/form-validation#template-driven-validation)
|
* [Form validation](guide/form-validation#template-driven-validation)
|
||||||
|
* `NgForm` directive API reference
|
||||||
|
@ -1,389 +1,234 @@
|
|||||||
# Template-driven forms
|
# Building a template-driven form
|
||||||
|
|
||||||
Forms are the mainstay of business applications.
|
|
||||||
You use forms to log in, submit a help request, place an order, book a flight,
|
|
||||||
schedule a meeting, and perform countless other data-entry tasks.
|
|
||||||
|
|
||||||
In developing a form, it's important to create a data-entry experience that guides the
|
|
||||||
user efficiently and effectively through the workflow.
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
|
||||||
|
|
||||||
For the sample app that this page describes, see the <live-example></live-example>.
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
## Introduction to Template-driven forms
|
|
||||||
|
|
||||||
Developing forms requires design skills (which are out of scope for this page), as well as framework support for
|
|
||||||
*two-way data binding, change tracking, validation, and error handling*,
|
|
||||||
which you'll learn about on this page.
|
|
||||||
|
|
||||||
This page shows you how to build a simple form from scratch. Along the way you'll learn how to:
|
|
||||||
|
|
||||||
* Build an Angular form with a component and template.
|
|
||||||
* Use `ngModel` to create two-way data bindings for reading and writing input-control values.
|
|
||||||
* Track state changes and the validity of form controls.
|
|
||||||
* Provide visual feedback using special CSS classes that track the state of the controls.
|
|
||||||
* Display validation errors to users and enable/disable form controls.
|
|
||||||
* Share information across HTML elements using template reference variables.
|
|
||||||
|
|
||||||
{@a template-driven}
|
{@a template-driven}
|
||||||
|
|
||||||
You can build forms by writing templates in the Angular [template syntax](guide/template-syntax) with
|
This tutorial shows you how to create a template-driven form whose control elements are bound to data properties, with input validation to maintain data integrity and styling to improve the user experience.
|
||||||
the form-specific directives and techniques described in this page.
|
|
||||||
|
Template-driven forms use [two-way data binding](guide/architecture-components#data-binding "Intro to 2-way data binding") to update the data model in the component as changes are made in the template and vice versa.
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
You can also use a reactive (or model-driven) approach to build forms.
|
Angular supports two design approaches for interactive forms. You can build forms by writing templates using Angular [template syntax and directives](guide/glossary#template "Definition of template terms") with the form-specific directives and techniques described in this tutorial, or you can use a reactive (or model-driven) approach to build forms.
|
||||||
However, this page focuses on template-driven forms.
|
|
||||||
|
Template-driven forms are suitable for small or simple forms, while reactive forms are more scalable and suitable for complex forms.
|
||||||
|
For a comparison of the two approaches, see [Introduction to Forms](guide/forms-overview "Overview of Angular forms.")
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
You can build almost any form with an Angular template—login forms, contact forms, and pretty much any business form.
|
You can build almost any kind of form with an Angular template—login forms, contact forms, and pretty much any business form.
|
||||||
You can lay out the controls creatively, bind them to data, specify validation rules and display validation errors,
|
You can lay out the controls creatively and bind them to the data in your object model.
|
||||||
|
You can specify validation rules and display validation errors,
|
||||||
conditionally enable or disable specific controls, trigger built-in visual feedback, and much more.
|
conditionally enable or disable specific controls, trigger built-in visual feedback, and much more.
|
||||||
|
|
||||||
Angular makes the process easy by handling many of the repetitive, boilerplate tasks you'd
|
This tutorial shows you how to build a form from scratch, using a simplified sample form like the one from the [Tour of Heroes tutorial](tutorial "Tour of Heroes") to illustrate the techniques.
|
||||||
otherwise wrestle with yourself.
|
|
||||||
|
|
||||||
You'll learn to build a template-driven form that looks like this:
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
Run or download the example app: <live-example></live-example>.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
## Objectives
|
||||||
|
|
||||||
|
This tutorial teaches you how to do the following:
|
||||||
|
|
||||||
|
* Build an Angular form with a component and template.
|
||||||
|
* Use `ngModel` to create two-way data bindings for reading and writing input-control values.
|
||||||
|
* Provide visual feedback using special CSS classes that track the state of the controls.
|
||||||
|
* Display validation errors to users and enable or disable form controls based on the form status.
|
||||||
|
* Share information across HTML elements using [template reference variables](guide/template-reference-variables).
|
||||||
|
|
||||||
|
## Prerequisites
|
||||||
|
|
||||||
|
Before going further into template-driven forms, you should have a basic understanding of the following.
|
||||||
|
|
||||||
|
* TypeScript and HTML5 programming.
|
||||||
|
* Angular app-design fundamentals, as described in [Angular Concepts](guide/architecture "Introduction to Angular concepts.").
|
||||||
|
* The basics of [Angular template syntax](guide/template-syntax "Template syntax guide").
|
||||||
|
* The form-design concepts that are presented in [Introduction to Forms](guide/forms-overview "Overview of Angular forms.").
|
||||||
|
|
||||||
|
{@a intro}
|
||||||
|
|
||||||
|
## Build a template-driven form
|
||||||
|
|
||||||
|
Template-driven forms rely on directives defined in the `FormsModule`.
|
||||||
|
|
||||||
|
* The `NgModel` directive reconciles value changes in the attached form element with changes in the data model, allowing you to respond to user input with input validation and error handling.
|
||||||
|
|
||||||
|
* The `NgForm` directive creates a top-level `FormGroup` instance and binds it to a `<form>` element to track aggregated form value and validation status.
|
||||||
|
As soon as you import `FormsModule`, this directive becomes active by default on all `<form>` tags. You don't need to add a special selector.
|
||||||
|
|
||||||
|
* The `NgModelGroup` directive creates and binds a `FormGroup` instance to a DOM element.
|
||||||
|
|
||||||
|
### The sample application
|
||||||
|
|
||||||
|
The sample form in this guide is used by the *Hero Employment Agency* to maintain personal information about heroes.
|
||||||
|
Every hero needs a job. This form helps the agency match the right hero with the right crisis.
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src="generated/images/guide/forms/hero-form-1.png" alt="Clean Form">
|
<img src="generated/images/guide/forms/hero-form-1.png" alt="Clean Form">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
The *Hero Employment Agency* uses this form to maintain personal information about heroes.
|
The form highlights some design features that make it easier to use. For instance, the two required fields have a green bar on the left to make them easy to spot. These fields have initial values, so the form is valid and the **Submit** button is enabled.
|
||||||
Every hero needs a job. It's the company mission to match the right hero with the right crisis.
|
|
||||||
|
|
||||||
Two of the three fields on this form are required. Required fields have a green bar on the left to make them easy to spot.
|
As you work with this form, you will learn how to include validation logic, how to customize the presentation with standard CSS, and how to handle error conditions to ensure valid input.
|
||||||
|
If the user deletes the hero name, for example, the form becomes invalid. The app detects the changed status, and displays a validation error in an attention-grabbing style.
|
||||||
If you delete the hero name, the form displays a validation error in an attention-grabbing style:
|
In addition, the **Submit** button is disabled, and the "required" bar to the left of the input control changes from green to red.
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src="generated/images/guide/forms/hero-form-2.png" alt="Invalid, Name Required">
|
<img src="generated/images/guide/forms/hero-form-2.png" alt="Invalid, Name Required">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
Note that the *Submit* button is disabled, and the "required" bar to the left of the input control changes from green to red.
|
### Step overview
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
In the course of this tutorial, you bind a sample form to data and handle user input using the following steps.
|
||||||
|
|
||||||
You can customize the colors and location of the "required" bar with standard CSS.
|
1. Build the basic form.
|
||||||
|
* Define a sample data model.
|
||||||
|
* Include required infrastructure such as the `FormsModule`.
|
||||||
|
2. Bind form controls to data properties using the `ngModel` directive and two-way data-binding syntax.
|
||||||
|
* Examine how `ngModel` reports control states using CSS classes.
|
||||||
|
* Name controls to make them accessible to `ngModel`.
|
||||||
|
3. Track input validity and control status using `ngModel`.
|
||||||
|
* Add custom CSS to provide visual feedback on the status.
|
||||||
|
* Show and hide validation-error messages.
|
||||||
|
4. Respond to a native HTML button-click event by adding to the model data.
|
||||||
|
5. Handle form submission using the [`ngSubmit`](api/forms/NgForm#properties) output property of the form.
|
||||||
|
* Disable the **Submit** button until the form is valid.
|
||||||
|
* After submit, swap out the finished form for different content on the page.
|
||||||
|
|
||||||
</div>
|
{@a step1}
|
||||||
|
|
||||||
You'll build this form in small steps:
|
## Build the form
|
||||||
|
|
||||||
1. Create the `Hero` model class.
|
You can recreate the sample application from the code provided here, or you can examine or download the <live-example></live-example>.
|
||||||
1. Create the component that controls the form.
|
|
||||||
1. Create a template with the initial form layout.
|
|
||||||
1. Bind data properties to each form control using the `ngModel` two-way data-binding syntax.
|
|
||||||
1. Add a `name` attribute to each form-input control.
|
|
||||||
1. Add custom CSS to provide visual feedback.
|
|
||||||
1. Show and hide validation-error messages.
|
|
||||||
1. Handle form submission with *ngSubmit*.
|
|
||||||
1. Disable the form’s *Submit* button until the form is valid.
|
|
||||||
|
|
||||||
## Setup
|
1. The provided sample application creates the `Hero` class which defines the data model reflected in the form.
|
||||||
|
|
||||||
Create a new project named <code>angular-forms</code>:
|
<code-example path="forms/src/app/hero.ts" header="src/app/hero.ts"></code-example>
|
||||||
|
|
||||||
<code-example language="sh" class="code-shell">
|
2. The form layout and details are defined in the `HeroFormComponent` class.
|
||||||
|
|
||||||
ng new angular-forms
|
<code-example path="forms/src/app/hero-form/hero-form.component.ts" header="src/app/hero-form/hero-form.component.ts (v1)" region="v1"></code-example>
|
||||||
|
|
||||||
</code-example>
|
The component's `selector` value of "app-hero-form" means you can drop this form in a parent
|
||||||
|
template using the `<app-hero-form>` tag.
|
||||||
|
|
||||||
## Create the Hero model class
|
3. The following code creates a new hero instance, so that the initial form can show an example hero.
|
||||||
|
|
||||||
As users enter form data, you'll capture their changes and update an instance of a model.
|
<code-example path="forms/src/app/hero-form/hero-form.component.ts" region="SkyDog"></code-example>
|
||||||
You can't lay out the form until you know what the model looks like.
|
|
||||||
|
|
||||||
A model can be as simple as a "property bag" that holds facts about a thing of application importance.
|
This demo uses dummy data for `model` and `powers`. In a real app, you would inject a data service to get and save real data, or expose these properties as inputs and outputs.
|
||||||
That describes well the `Hero` class with its three required fields (`id`, `name`, `power`)
|
|
||||||
and one optional field (`alterEgo`).
|
|
||||||
|
|
||||||
Using the Angular CLI command [`ng generate class`](cli/generate), generate a new class named `Hero`:
|
4. The application enables the Forms feature and registers the created form component.
|
||||||
|
|
||||||
<code-example language="sh" class="code-shell">
|
<code-example path="forms/src/app/app.module.ts" header="src/app/app.module.ts"></code-example>
|
||||||
|
|
||||||
ng generate class Hero
|
5. The form is displayed in the application layout defined by the root component's template.
|
||||||
|
|
||||||
</code-example>
|
<code-example path="forms/src/app/app.component.html" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
With this content:
|
The initial template defines the layout for a form with two form groups and a submit button.
|
||||||
|
The form groups correspond to two properties of the Hero data model, name and alterEgo. Each group has a label and a box for user input.
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero.ts" header="src/app/hero.ts"></code-example>
|
* The **Name** `<input>` control element has the HTML5 `required` attribute.
|
||||||
|
* The **Alter Ego** `<input>` control element does not because `alterEgo` is optional.
|
||||||
|
|
||||||
It's an anemic model with few requirements and no behavior. Perfect for the demo.
|
The **Submit** button has some classes on it for styling.
|
||||||
|
At this point, the form layout is all plain HTML5, with no bindings or directives.
|
||||||
|
|
||||||
The TypeScript compiler generates a public field for each `public` constructor parameter and
|
6. The sample form uses some style classes from [Twitter Bootstrap](http://getbootstrap.com/css/): `container`, `form-group`, `form-control`, and `btn`.
|
||||||
automatically assigns the parameter’s value to that field when you create heroes.
|
To use these styles, the app's style sheet imports the library.
|
||||||
|
|
||||||
The `alterEgo` is optional, so the constructor lets you omit it; note the question mark (?) in `alterEgo?`.
|
<code-example path="forms/src/styles.1.css" header="src/styles.css"></code-example>
|
||||||
|
|
||||||
You can create a new hero like this:
|
7. The form makes the hero applicant choose one superpower from a fixed list of agency-approved powers.
|
||||||
|
The predefined list of `powers` is part of the data model, maintained internally in `HeroFormComponent`.
|
||||||
|
The Angular [NgForOf directive](api/common/NgForOf "API reference") iterates over the data values to populate the `<select>` element.
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.ts" region="SkyDog"></code-example>
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (powers)" region="powers"></code-example>
|
||||||
|
|
||||||
## Create a form component
|
If you run the app right now, you see the list of powers in the selection control. The input elements are not yet bound to data values or events, so they are still blank and have no behavior.
|
||||||
|
|
||||||
An Angular form has two parts: an HTML-based _template_ and a component _class_
|
|
||||||
to handle data and user interactions programmatically.
|
|
||||||
Begin with the class because it states, in brief, what the hero editor can do.
|
|
||||||
|
|
||||||
Using the Angular CLI command [`ng generate component`](cli/generate), generate a new component named `HeroForm`:
|
|
||||||
|
|
||||||
<code-example language="sh" class="code-shell">
|
|
||||||
|
|
||||||
ng generate component HeroForm
|
|
||||||
|
|
||||||
</code-example>
|
|
||||||
|
|
||||||
With this content:
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.ts" header="src/app/hero-form/hero-form.component.ts (v1)" region="v1"></code-example>
|
|
||||||
|
|
||||||
There’s nothing special about this component, nothing form-specific,
|
|
||||||
nothing to distinguish it from any component you've written before.
|
|
||||||
|
|
||||||
Understanding this component requires only the Angular concepts covered in previous pages.
|
|
||||||
|
|
||||||
* The code imports the Angular core library and the `Hero` model you just created.
|
|
||||||
* The `@Component` selector value of "app-hero-form" means you can drop this form in a parent
|
|
||||||
template with a `<app-hero-form>` tag.
|
|
||||||
* The `templateUrl` property points to a separate file for the template HTML.
|
|
||||||
* You defined dummy data for `model` and `powers`, as befits a demo.
|
|
||||||
|
|
||||||
Down the road, you can inject a data service to get and save real data
|
|
||||||
or perhaps expose these properties as inputs and outputs
|
|
||||||
(see [Input and output properties](guide/template-syntax#inputs-outputs) on the
|
|
||||||
[Template Syntax](guide/template-syntax) page) for binding to a
|
|
||||||
parent component. This is not a concern now and these future changes won't affect the form.
|
|
||||||
|
|
||||||
* You added a `diagnostic` property to return a JSON representation of the model.
|
|
||||||
It'll help you see what you're doing during development; you've left yourself a cleanup note to discard it later.
|
|
||||||
|
|
||||||
## Revise *app.module.ts*
|
|
||||||
|
|
||||||
`app.module.ts` defines the application's root module. In it you identify the external modules you'll use in the application
|
|
||||||
and declare the components that belong to this module, such as the `HeroFormComponent`.
|
|
||||||
|
|
||||||
Because template-driven forms are in their own module, you need to add the `FormsModule` to the array of
|
|
||||||
`imports` for the application module before you can use forms.
|
|
||||||
|
|
||||||
Update it with the following:
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/app.module.ts" header="src/app/app.module.ts"></code-example>
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
|
||||||
|
|
||||||
There are two changes:
|
|
||||||
|
|
||||||
1. You import `FormsModule`.
|
|
||||||
|
|
||||||
1. You add the `FormsModule` to the list of `imports` defined in the `@NgModule` decorator. This gives the application
|
|
||||||
access to all of the template-driven forms features, including `ngModel`.
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="alert is-important">
|
|
||||||
|
|
||||||
If a component, directive, or pipe belongs to a module in the `imports` array, _don't_ re-declare it in the `declarations` array.
|
|
||||||
If you wrote it and it should belong to this module, _do_ declare it in the `declarations` array.
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
## Revise *app.component.html*
|
|
||||||
|
|
||||||
`AppComponent` is the application's root component. It will host the new `HeroFormComponent`.
|
|
||||||
|
|
||||||
Replace the contents of its template with the following:
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/app.component.html" header="src/app/app.component.html"></code-example>
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
|
||||||
|
|
||||||
There are only two changes.
|
|
||||||
The `template` is simply the new element tag identified by the component's `selector` property.
|
|
||||||
This displays the hero form when the application component is loaded.
|
|
||||||
Don't forget to remove the `name` field from the class body as well.
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
## Create an initial HTML form template
|
|
||||||
|
|
||||||
Update the template file with the following contents:
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" region="start" header="src/app/hero-form/hero-form.component.html"></code-example>
|
|
||||||
|
|
||||||
The language is simply HTML5. You're presenting two of the `Hero` fields, `name` and `alterEgo`, and
|
|
||||||
opening them up for user input in input boxes.
|
|
||||||
|
|
||||||
The *Name* `<input>` control has the HTML5 `required` attribute;
|
|
||||||
the *Alter Ego* `<input>` control does not because `alterEgo` is optional.
|
|
||||||
|
|
||||||
You added a *Submit* button at the bottom with some classes on it for styling.
|
|
||||||
|
|
||||||
*You're not using Angular yet*. There are no bindings or extra directives, just layout.
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
|
||||||
|
|
||||||
In template driven forms, if you've imported `FormsModule`, you don't have to do anything
|
|
||||||
to the `<form>` tag in order to make use of `FormsModule`. Continue on to see how this works.
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
The `container`, `form-group`, `form-control`, and `btn` classes
|
|
||||||
come from [Twitter Bootstrap](http://getbootstrap.com/css/). These classes are purely cosmetic.
|
|
||||||
Bootstrap gives the form a little style.
|
|
||||||
|
|
||||||
<div class="callout is-important">
|
|
||||||
|
|
||||||
<header>
|
|
||||||
Angular forms don't require a style library
|
|
||||||
</header>
|
|
||||||
|
|
||||||
Angular makes no use of the `container`, `form-group`, `form-control`, and `btn` classes or
|
|
||||||
the styles of any external library. Angular apps can use any CSS library or none at all.
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
To add the stylesheet, open `styles.css` and add the following import line at the top:
|
|
||||||
|
|
||||||
<code-example path="forms/src/styles.1.css" header="src/styles.css"></code-example>
|
|
||||||
|
|
||||||
## Add powers with _*ngFor_
|
|
||||||
|
|
||||||
The hero must choose one superpower from a fixed list of agency-approved powers.
|
|
||||||
You maintain that list internally (in `HeroFormComponent`).
|
|
||||||
|
|
||||||
You'll add a `select` to the
|
|
||||||
form and bind the options to the `powers` list using `ngFor`,
|
|
||||||
a technique seen previously in the [Displaying Data](guide/displaying-data) page.
|
|
||||||
|
|
||||||
Add the following HTML *immediately below* the *Alter Ego* group:
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (powers)" region="powers"></code-example>
|
|
||||||
|
|
||||||
This code repeats the `<option>` tag for each power in the list of powers.
|
|
||||||
The `pow` template input variable is a different power in each iteration;
|
|
||||||
you display its name using the interpolation syntax.
|
|
||||||
|
|
||||||
{@a ngModel}
|
|
||||||
|
|
||||||
## Two-way data binding with _ngModel_
|
|
||||||
|
|
||||||
Running the app right now would be disappointing.
|
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src="generated/images/guide/forms/hero-form-3.png" alt="Early form with no binding">
|
<img src="generated/images/guide/forms/hero-form-3.png" alt="Early form with no binding">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{@a ngModel}
|
||||||
|
|
||||||
You don't see hero data because you're not binding to the `Hero` yet.
|
## Bind input controls to data properties
|
||||||
You know how to do that from earlier pages.
|
|
||||||
[Displaying Data](guide/displaying-data) teaches property binding.
|
|
||||||
[User Input](guide/user-input) shows how to listen for DOM events with an
|
|
||||||
event binding and how to update a component property with the displayed value.
|
|
||||||
|
|
||||||
Now you need to display, listen, and extract at the same time.
|
The next step is to bind the input controls to the corresponding `Hero` properties with two-way data binding, so that they respond to user input by updating the data model, and also respond to programmatic changes in the data by updating the display.
|
||||||
|
|
||||||
You could use the techniques you already know, but
|
The `ngModel` directive declared in the `FormsModule` lets you bind controls in your template-driven form to properties in your data model.
|
||||||
instead you'll use the new `[(ngModel)]` syntax, which
|
When you include the directive using the syntax for two-way data binding, `[(ngModel)]`, Angular can track the value and user interaction of the control and keep the view synced with the model.
|
||||||
makes binding the form to the model easy.
|
|
||||||
|
|
||||||
Find the `<input>` tag for *Name* and update it like this:
|
1. Edit the template file `hero-form.component.html`.
|
||||||
|
|
||||||
|
2. Find the `<input>` tag next to the **Name** label.
|
||||||
|
|
||||||
|
3. Add the `ngModel` directive, using two-way data binding syntax `[(ngModel)]="..."`.
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="ngModelName-1"></code-example>
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="ngModelName-1"></code-example>
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
You added a diagnostic interpolation after the input tag
|
This example has a temporary diagnostic interpolation after each input tag, `{{model.name}}`, to show the current data value of the corresponding property.
|
||||||
so you can see what you're doing.
|
The note reminds you to remove the diagnostic lines when you have finished observing the two-way data binding at work.
|
||||||
You left yourself a note to throw it away when you're done.
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
Focus on the binding syntax: `[(ngModel)]="..."`.
|
{@a ngForm}
|
||||||
|
|
||||||
You need one more addition to display the data. Declare
|
### Access the overall form status
|
||||||
a template variable for the form. Update the `<form>` tag with
|
|
||||||
`#heroForm="ngForm"` as follows:
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="template-variable"></code-example>
|
When you imported the `FormsModule` in your component, Angular automatically created and attached an [NgForm](api/forms/NgForm "API reference for NgForm") directive to the `<form>` tag in the template (because `NgForm` has the selector `form` that matches `<form>` elements).
|
||||||
|
|
||||||
The variable `heroForm` is now a reference to the `NgForm` directive that governs the form as a whole.
|
To get access to the `NgForm` and the overall form status, declare a [template reference variable](guide/template-reference-variables).
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
1. Edit the template file `hero-form.component.html`.
|
||||||
|
|
||||||
{@a ngForm}
|
2. Update the `<form>` tag with a template reference variable, `#heroForm`, and set its value as follows.
|
||||||
|
|
||||||
### The _NgForm_ directive
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="template-variable"></code-example>
|
||||||
|
|
||||||
What `NgForm` directive?
|
The `heroForm` template variable is now a reference to the `NgForm` directive instance that governs the form as a whole.
|
||||||
You didn't add an [NgForm](api/forms/NgForm) directive.
|
|
||||||
|
|
||||||
Angular did. Angular automatically creates and attaches an `NgForm` directive to the `<form>` tag.
|
3. Run the app.
|
||||||
|
|
||||||
The `NgForm` directive supplements the `form` element with additional features.
|
4. Start typing in the **Name** input box.
|
||||||
It holds the controls you created for the elements with an `ngModel` directive
|
|
||||||
and `name` attribute, and monitors their properties, including their validity.
|
|
||||||
It also has its own `valid` property which is true only *if every contained
|
|
||||||
control* is valid.
|
|
||||||
|
|
||||||
</div>
|
As you add and delete characters, you can see them appear and disappear from the data model.
|
||||||
|
For example:
|
||||||
|
|
||||||
If you ran the app now and started typing in the *Name* input box,
|
<div class="lightbox">
|
||||||
adding and deleting characters, you'd see them appear and disappear
|
<img src="generated/images/guide/forms/ng-model-in-action.png" alt="ngModel in action">
|
||||||
from the interpolated text.
|
</div>
|
||||||
At some point it might look like this:
|
|
||||||
|
|
||||||
<div class="lightbox">
|
The diagnostic line that shows interpolated values demonstrates that values are really flowing from the input box to the model and back again.
|
||||||
<img src="generated/images/guide/forms/ng-model-in-action.png" alt="ngModel in action">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
The diagnostic is evidence that values really are flowing from the input box to the model and
|
### Naming control elements
|
||||||
back again.
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
When you use `[(ngModel)]` on an element, you must define a `name` attribute for that element.
|
||||||
|
Angular uses the assigned name to register the element with the `NgForm` directive attached to the parent `<form>` element.
|
||||||
|
|
||||||
That's *two-way data binding*.
|
The example added a `name` attribute to the `<input>` element and set it to "name",
|
||||||
For more information, see
|
which makes sense for the hero's name.
|
||||||
[Two-way binding with NgModel](guide/template-syntax#ngModel) on the
|
Any unique value will do, but using a descriptive name is helpful.
|
||||||
the [Template Syntax](guide/template-syntax) page.
|
|
||||||
|
|
||||||
</div>
|
1. Add similar `[(ngModel)]` bindings and `name` attributes to **Alter Ego** and **Hero Power**.
|
||||||
|
|
||||||
Notice that you also added a `name` attribute to the `<input>` tag and set it to "name",
|
2. You can now remove the diagnostic messages that show interpolated values.
|
||||||
which makes sense for the hero's name. Any unique value will do, but using a descriptive name is helpful.
|
|
||||||
Defining a `name` attribute is a requirement when using `[(ngModel)]` in combination with a form.
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
3. To confirm that two-way data binding works for the entire hero model, add a new binding at the top to the component's `diagnostic` property.
|
||||||
|
|
||||||
Internally, Angular creates `FormControl` instances and
|
After these revisions, the form template should look like the following:
|
||||||
registers them with an `NgForm` directive that Angular attached to the `<form>` tag.
|
|
||||||
Each `FormControl` is registered under the name you assigned to the `name` attribute.
|
|
||||||
Read more in the previous section, [The NgForm directive](guide/forms#ngForm).
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
Add similar `[(ngModel)]` bindings and `name` attributes to *Alter Ego* and *Hero Power*.
|
|
||||||
You'll ditch the input box binding message
|
|
||||||
and add a new binding (at the top) to the component's `diagnostic` property.
|
|
||||||
Then you can confirm that two-way data binding works *for the entire hero model*.
|
|
||||||
|
|
||||||
After revision, the core of the form should look like this:
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="ngModel-2"></code-example>
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="ngModel-2"></code-example>
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
* Notice that each `<input>` element has an `id` property. This is used by the `<label>` element's `for` attribute to match the label to its input control. This is a [standard HTML feature](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/label).
|
||||||
|
|
||||||
* Each input element has an `id` property that is used by the `label` element's `for` attribute
|
* Each `<input>` element also has the required `name` property that Angular uses to register the control with the form.
|
||||||
to match the label to its input control.
|
|
||||||
* Each input element has a `name` property that is required by Angular forms to register the control with the form.
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
If you run the app now and change every hero model property, the form might display like this:
|
If you run the app now and change every hero model property, the form might display like this:
|
||||||
|
|
||||||
@ -391,18 +236,15 @@ If you run the app now and change every hero model property, the form might disp
|
|||||||
<img src="generated/images/guide/forms/ng-model-in-action-2.png" alt="ngModel in action">
|
<img src="generated/images/guide/forms/ng-model-in-action-2.png" alt="ngModel in action">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
The diagnostic near the top of the form
|
The diagnostic near the top of the form confirms that all of your changes are reflected in the model.
|
||||||
confirms that all of your changes are reflected in the model.
|
|
||||||
|
|
||||||
*Delete* the `{{diagnostic}}` binding at the top as it has served its purpose.
|
4. When you have observed the effects, you can delete the `{{diagnostic}}` binding.
|
||||||
|
|
||||||
## Track control state and validity with _ngModel_
|
## Track control states
|
||||||
|
|
||||||
Using `ngModel` in a form gives you more than just two-way data binding. It also tells
|
The `NgModel` directive on a control tracks the state of that control.
|
||||||
you if the user touched the control, if the value changed, or if the value became invalid.
|
It tells you if the user touched the control, if the value changed, or if the value became invalid.
|
||||||
|
Angular sets special CSS classes on the control element to reflect the state, as shown in the following table.
|
||||||
The *NgModel* directive doesn't just track state; it updates the control with special Angular CSS classes that reflect the state.
|
|
||||||
You can leverage those class names to change the appearance of the control.
|
|
||||||
|
|
||||||
<table>
|
<table>
|
||||||
|
|
||||||
@ -472,38 +314,32 @@ You can leverage those class names to change the appearance of the control.
|
|||||||
|
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
Temporarily add a [template reference variable](guide/template-syntax#ref-vars) named `spy`
|
You use these CSS classes to define the styles for your control based on its status.
|
||||||
to the _Name_ `<input>` tag and use it to display the input's CSS classes.
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="ngModelName-2"></code-example>
|
### Observe control states
|
||||||
|
|
||||||
Now run the app and look at the _Name_ input box.
|
To see how the classes are added and removed by the framework, open the browser's developer tools and inspect the `<input>` element that represents the hero name.
|
||||||
Follow these steps *precisely*:
|
|
||||||
|
|
||||||
1. Look but don't touch.
|
1. Using your browser's developer tools, find the `<input>` element that corresponds to the **Name** input box.
|
||||||
1. Click inside the name box, then click outside it.
|
You can see that the element has multiple CSS classes in addition to "form-control".
|
||||||
1. Add slashes to the end of the name.
|
|
||||||
1. Erase the name.
|
|
||||||
|
|
||||||
The actions and effects are as follows:
|
2. When you first bring it up, the classes indicate that it has a valid value, that the value has not been changed since initialization or reset, and that the control has not been visited since initialization or reset.
|
||||||
|
|
||||||
<div class="lightbox">
|
```
|
||||||
<img src="generated/images/guide/forms/control-state-transitions-anim.gif" alt="Control State Transition">
|
<input ... class="form-control ng-untouched ng-pristine ng-valid" ...>
|
||||||
</div>
|
```
|
||||||
|
|
||||||
You should see the following transitions and class names:
|
3. Take the following actions on the **Name** `<input>` box, and observe which classes appear.
|
||||||
|
* Look but don't touch. The classes indicate that it is untouched, pristine, and valid.
|
||||||
|
* Click inside the name box, then click outside it. The control has now been visited, and the element has the `ng-touched` class instead of the `ng-untouched` class.
|
||||||
|
* Add slashes to the end of the name. It is now touched and dirty.
|
||||||
|
* Erase the name. This makes the value invalid, so the `ng-invalid` class replaces the `ng-valid` class.
|
||||||
|
|
||||||
<div class="lightbox">
|
### Create visual feedback for states
|
||||||
<img src="generated/images/guide/forms/ng-control-class-changes.png" alt="Control state transitions">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
The `ng-valid`/`ng-invalid` pair is the most interesting, because you want to send a
|
The `ng-valid`/`ng-invalid` pair is particularly interesting, because you want to send a
|
||||||
strong visual signal when the values are invalid. You also want to mark required fields.
|
strong visual signal when the values are invalid.
|
||||||
To create such visual feedback, add definitions for the `ng-*` CSS classes.
|
You also want to mark required fields.
|
||||||
|
|
||||||
*Delete* the `#spy` template reference variable and the `TODO` as they have served their purpose.
|
|
||||||
|
|
||||||
## Add custom CSS for visual feedback
|
|
||||||
|
|
||||||
You can mark required fields and invalid data at the same time with a colored bar
|
You can mark required fields and invalid data at the same time with a colored bar
|
||||||
on the left of the input box:
|
on the left of the input box:
|
||||||
@ -512,20 +348,25 @@ on the left of the input box:
|
|||||||
<img src="generated/images/guide/forms/validity-required-indicator.png" alt="Invalid Form">
|
<img src="generated/images/guide/forms/validity-required-indicator.png" alt="Invalid Form">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
You achieve this effect by adding these class definitions to a new `forms.css` file
|
To change the appearance in this way, take the following steps.
|
||||||
that you add to the project as a sibling to `index.html`:
|
|
||||||
|
|
||||||
<code-example path="forms/src/assets/forms.css" header="src/assets/forms.css"></code-example>
|
1. Add definitions for the `ng-*` CSS classes.
|
||||||
|
|
||||||
Update the `<head>` of `index.html` to include this style sheet:
|
2. Add these class definitions to a new `forms.css` file.
|
||||||
|
|
||||||
<code-example path="forms/src/index.html" header="src/index.html (styles)" region="styles"></code-example>
|
3. Add the new file to the project as a sibling to `index.html`:
|
||||||
|
|
||||||
## Show and hide validation error messages
|
<code-example path="forms/src/assets/forms.css" header="src/assets/forms.css"></code-example>
|
||||||
|
|
||||||
You can improve the form. The _Name_ input box is required and clearing it turns the bar red.
|
4. In the `index.html` file, update the `<head>` tag to include the new style sheet.
|
||||||
That says something is wrong but the user doesn't know *what* is wrong or what to do about it.
|
|
||||||
Leverage the control's state to reveal a helpful message.
|
<code-example path="forms/src/index.html" header="src/index.html (styles)" region="styles"></code-example>
|
||||||
|
|
||||||
|
### Show and hide validation error messages
|
||||||
|
|
||||||
|
The **Name** input box is required and clearing it turns the bar red.
|
||||||
|
That indicates that something is wrong, but the user doesn't know what is wrong or what to do about it.
|
||||||
|
You can provide a helpful message by checking for and responding to the control's state.
|
||||||
|
|
||||||
When the user deletes the name, the form should look like this:
|
When the user deletes the name, the form should look like this:
|
||||||
|
|
||||||
@ -533,166 +374,135 @@ When the user deletes the name, the form should look like this:
|
|||||||
<img src="generated/images/guide/forms/name-required-error.png" alt="Name required">
|
<img src="generated/images/guide/forms/name-required-error.png" alt="Name required">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
To achieve this effect, extend the `<input>` tag with the following:
|
The **Hero Power** select box is also required, but it doesn't need this kind of error handling because the selection box already constrains the selection to valid values.
|
||||||
|
|
||||||
* A [template reference variable](guide/template-syntax#ref-vars).
|
To define and show an error message when appropriate, take the following steps.
|
||||||
* The "*is required*" message in a nearby `<div>`, which you'll display only if the control is invalid.
|
|
||||||
|
|
||||||
Here's an example of an error message added to the _name_ input box:
|
1. Extend the `<input>` tag with a template reference variable that you can use to access the input box's Angular control from within the template. In the example, the variable is `#name="ngModel"`.
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="name-with-error-msg"></code-example>
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
You need a template reference variable to access the input box's Angular control from within the template.
|
The template reference variable (`#name`) is set to `"ngModel"` because that is the value of the [`NgModel.exportAs`](api/core/Directive#exportAs) property. This property tells Angular how to link a reference variable to a directive.
|
||||||
Here you created a variable called `name` and gave it the value "ngModel".
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
</div>
|
||||||
|
|
||||||
Why "ngModel"?
|
2. Add a `<div>` that contains a suitable error message.
|
||||||
A directive's [exportAs](api/core/Directive) property
|
3. Show or hide the error message by binding properties of the `name`
|
||||||
tells Angular how to link the reference variable to the directive.
|
|
||||||
You set `name` to `ngModel` because the `ngModel` directive's `exportAs` property happens to be "ngModel".
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
You control visibility of the name error message by binding properties of the `name`
|
|
||||||
control to the message `<div>` element's `hidden` property.
|
control to the message `<div>` element's `hidden` property.
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (hidden-error-msg)" region="hidden-error-msg"></code-example>
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (hidden-error-msg)" region="hidden-error-msg"></code-example>
|
||||||
|
|
||||||
In this example, you hide the message when the control is valid or pristine;
|
4. Add a conditional error message to the _name_ input box, as in the following example.
|
||||||
"pristine" means the user hasn't changed the value since it was displayed in this form.
|
|
||||||
|
|
||||||
This user experience is the developer's choice. Some developers want the message to display at all times.
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="name-with-error-msg"></code-example>
|
||||||
|
|
||||||
|
<div class="callout is-helpful">
|
||||||
|
|
||||||
|
<header>Illustrating the "pristine" state</header>
|
||||||
|
|
||||||
|
In this example, you hide the message when the control is either valid or *pristine*.
|
||||||
|
Pristine means the user hasn't changed the value since it was displayed in this form.
|
||||||
If you ignore the `pristine` state, you would hide the message only when the value is valid.
|
If you ignore the `pristine` state, you would hide the message only when the value is valid.
|
||||||
If you arrive in this component with a new (blank) hero or an invalid hero,
|
If you arrive in this component with a new (blank) hero or an invalid hero,
|
||||||
you'll see the error message immediately, before you've done anything.
|
you'll see the error message immediately, before you've done anything.
|
||||||
|
|
||||||
Some developers want the message to display only when the user makes an invalid change.
|
You might want the message to display only when the user makes an invalid change.
|
||||||
Hiding the message while the control is "pristine" achieves that goal.
|
Hiding the message while the control is in the `pristine` state achieves that goal.
|
||||||
You'll see the significance of this choice when you add a new hero to the form.
|
You'll see the significance of this choice when you add a new hero to the form in the next step.
|
||||||
|
|
||||||
The hero *Alter Ego* is optional so you can leave that be.
|
</div>
|
||||||
|
|
||||||
Hero *Power* selection is required.
|
## Add a new hero
|
||||||
You can add the same kind of error handling to the `<select>` if you want,
|
|
||||||
but it's not imperative because the selection box already constrains the
|
|
||||||
power to valid values.
|
|
||||||
|
|
||||||
Now you'll add a new hero in this form.
|
This exercise shows how you can respond to a native HTML button-click event by adding to the model data.
|
||||||
Place a *New Hero* button at the bottom of the form and bind its click event to a `newHero` component method.
|
To let form users add a new hero, you will add a **New Hero** button that responds to a click event.
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" region="new-hero-button-no-reset" header="src/app/hero-form/hero-form.component.html (New Hero button)"></code-example>
|
1. In the template, place a "New Hero" `<button>` element at the bottom of the form.
|
||||||
|
2. In the component file, add the hero-creation method to the hero data model.
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.ts" region="new-hero" header="src/app/hero-form/hero-form.component.ts (New Hero method)"></code-example>
|
<code-example path="forms/src/app/hero-form/hero-form.component.ts" region="new-hero" header="src/app/hero-form/hero-form.component.ts (New Hero method)"></code-example>
|
||||||
|
|
||||||
Run the application again, click the *New Hero* button, and the form clears.
|
3. Bind the button's click event to a hero-creation method, `newHero()`.
|
||||||
The *required* bars to the left of the input box are red, indicating invalid `name` and `power` properties.
|
|
||||||
That's understandable as these are required fields.
|
|
||||||
The error messages are hidden because the form is pristine; you haven't changed anything yet.
|
|
||||||
|
|
||||||
Enter a name and click *New Hero* again.
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" region="new-hero-button-no-reset" header="src/app/hero-form/hero-form.component.html (New Hero button)"></code-example>
|
||||||
The app displays a _Name is required_ error message.
|
|
||||||
You don't want error messages when you create a new (empty) hero.
|
|
||||||
Why are you getting one now?
|
|
||||||
|
|
||||||
Inspecting the element in the browser tools reveals that the *name* input box is _no longer pristine_.
|
4. Run the application again and click the **New Hero** button.
|
||||||
The form remembers that you entered a name before clicking *New Hero*.
|
|
||||||
Replacing the hero object *did not restore the pristine state* of the form controls.
|
|
||||||
|
|
||||||
You have to clear all of the flags imperatively, which you can do
|
The form clears, and the *required* bars to the left of the input box are red, indicating invalid `name` and `power` properties.
|
||||||
by calling the form's `reset()` method after calling the `newHero()` method.
|
Notice that the error messages are hidden. This is because the form is pristine; you haven't changed anything yet.
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" region="new-hero-button-form-reset" header="src/app/hero-form/hero-form.component.html (Reset the form)"></code-example>
|
5. Enter a name and click **New Hero** again.
|
||||||
|
|
||||||
Now clicking "New Hero" resets both the form and its control flags.
|
Now the app displays a _Name is required_ error message, because the input box is no longer pristine.
|
||||||
|
The form remembers that you entered a name before clicking **New Hero**.
|
||||||
|
|
||||||
|
6. To restore the pristine state of the form controls, clear all of the flags imperatively by calling the form's `reset()` method after calling the `newHero()` method.
|
||||||
|
|
||||||
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" region="new-hero-button-form-reset" header="src/app/hero-form/hero-form.component.html (Reset the form)"></code-example>
|
||||||
|
|
||||||
|
Now clicking **New Hero** resets both the form and its control flags.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
See the [User Input](guide/user-input) guide for more information about listening for DOM events with an event binding and updating a corresponding component property.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
## Submit the form with _ngSubmit_
|
## Submit the form with _ngSubmit_
|
||||||
|
|
||||||
The user should be able to submit this form after filling it in.
|
The user should be able to submit this form after filling it in.
|
||||||
The *Submit* button at the bottom of the form
|
The **Submit** button at the bottom of the form does nothing on its own, but it does
|
||||||
does nothing on its own, but it will
|
trigger a form-submit event because of its type (`type="submit"`).
|
||||||
trigger a form submit because of its type (`type="submit"`).
|
To respond to this event, take the following steps.
|
||||||
|
|
||||||
A "form submit" is useless at the moment.
|
1. Bind the form's [`ngSubmit`](api/forms/NgForm#properties) event property to the hero-form component's `onSubmit()` method.
|
||||||
To make it useful, bind the form's `ngSubmit` event property
|
|
||||||
to the hero form component's `onSubmit()` method:
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (ngSubmit)" region="ngSubmit"></code-example>
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (ngSubmit)" region="ngSubmit"></code-example>
|
||||||
|
|
||||||
You'd already defined a template reference variable,
|
2. Use the template reference variable, `#heroForm` to access the form that contains the **Submit** button and create an event binding.
|
||||||
`#heroForm`, and initialized it with the value "ngForm".
|
You will bind the form property that indicates its overall validity to the **Submit** button's `disabled` property.
|
||||||
Now, use that variable to access the form with the Submit button.
|
|
||||||
|
|
||||||
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (submit-button)" region="submit-button"></code-example>
|
||||||
|
|
||||||
You'll bind the form's overall validity via
|
3. Run the application now. Notice that the button is enabled—although
|
||||||
the `heroForm` variable to the button's `disabled` property
|
|
||||||
using an event binding. Here's the code:
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (submit-button)" region="submit-button"></code-example>
|
|
||||||
|
|
||||||
If you run the application now, you find that the button is enabled—although
|
|
||||||
it doesn't do anything useful yet.
|
it doesn't do anything useful yet.
|
||||||
|
|
||||||
Now if you delete the Name, you violate the "required" rule, which
|
4. Delete the **Name** value. This violates the "required" rule, so it displays the error message—and notice that it also disables the **Submit** button.
|
||||||
is duly noted in the error message.
|
|
||||||
The *Submit* button is also disabled.
|
|
||||||
|
|
||||||
Not impressed? Think about it for a moment. What would you have to do to
|
|
||||||
wire the button's enable/disabled state to the form's validity without Angular's help?
|
|
||||||
|
|
||||||
For you, it was as simple as this:
|
You didn't have to explicitly wire the button's enabled state to the form's validity.
|
||||||
|
The `FormsModule` did this automatically when you defined a template reference variable on the enhanced form element, then referred to that variable in the button control.
|
||||||
|
|
||||||
1. Define a template reference variable on the (enhanced) form element.
|
### Respond to form submission
|
||||||
2. Refer to that variable in a button many lines away.
|
|
||||||
|
|
||||||
## Toggle two form regions (extra credit)
|
To show a response to form submission, you can hide the data entry area and display something else in its place.
|
||||||
|
|
||||||
Submitting the form isn't terribly dramatic at the moment.
|
1. Wrap the entire form in a `<div>` and bind
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
|
||||||
|
|
||||||
An unsurprising observation for a demo. To be honest,
|
|
||||||
jazzing it up won't teach you anything new about forms.
|
|
||||||
But this is an opportunity to exercise some of your newly won
|
|
||||||
binding skills.
|
|
||||||
If you aren't interested, skip to this page's conclusion.
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
For a more strikingly visual effect,
|
|
||||||
hide the data entry area and display something else.
|
|
||||||
|
|
||||||
Wrap the form in a `<div>` and bind
|
|
||||||
its `hidden` property to the `HeroFormComponent.submitted` property.
|
its `hidden` property to the `HeroFormComponent.submitted` property.
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="edit-div"></code-example>
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="edit-div"></code-example>
|
||||||
|
|
||||||
The main form is visible from the start because the
|
* The main form is visible from the start because the `submitted` property is false until you submit the form, as this fragment from the `HeroFormComponent` shows:
|
||||||
`submitted` property is false until you submit the form,
|
|
||||||
as this fragment from the `HeroFormComponent` shows:
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.ts" header="src/app/hero-form/hero-form.component.ts (submitted)" region="submitted"></code-example>
|
<code-example path="forms/src/app/hero-form/hero-form.component.ts" header="src/app/hero-form/hero-form.component.ts (submitted)" region="submitted"></code-example>
|
||||||
|
|
||||||
When you click the *Submit* button, the `submitted` flag becomes true and the form disappears
|
* When you click the **Submit** button, the `submitted` flag becomes true and the form disappears.
|
||||||
as planned.
|
|
||||||
|
|
||||||
Now the app needs to show something else while the form is in the submitted state.
|
2. To show something else while the form is in the submitted state, add the following HTML below the new `<div>` wrapper.
|
||||||
Add the following HTML below the `<div>` wrapper you just wrote:
|
|
||||||
|
|
||||||
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="submitted"></code-example>
|
<code-example path="forms/src/app/hero-form/hero-form.component.html" header="src/app/hero-form/hero-form.component.html (excerpt)" region="submitted"></code-example>
|
||||||
|
|
||||||
There's the hero again, displayed read-only with interpolation bindings.
|
This `<div>`, which shows a read-only hero with interpolation bindings, appears only while the component is in the submitted state.
|
||||||
This `<div>` appears only while the component is in the submitted state.
|
|
||||||
|
|
||||||
The HTML includes an *Edit* button whose click event is bound to an expression
|
The alternative display includes an *Edit* button whose click event is bound to an expression
|
||||||
that clears the `submitted` flag.
|
that clears the `submitted` flag.
|
||||||
|
|
||||||
When you click the *Edit* button, this block disappears and the editable form reappears.
|
3. Click the *Edit* button to switch the display back to the editable form.
|
||||||
|
|
||||||
## Summary
|
## Summary
|
||||||
|
|
||||||
The Angular form discussed in this page takes advantage of the following
|
The Angular form discussed in this page takes advantage of the following
|
||||||
framework features to provide support for data modification, validation, and more:
|
framework features to provide support for data modification, validation, and more.
|
||||||
|
|
||||||
* An Angular HTML form template.
|
* An Angular HTML form template.
|
||||||
* A form component class with a `@Component` decorator.
|
* A form component class with a `@Component` decorator.
|
||||||
@ -700,8 +510,8 @@ framework features to provide support for data modification, validation, and mor
|
|||||||
* Template-reference variables such as `#heroForm` and `#name`.
|
* Template-reference variables such as `#heroForm` and `#name`.
|
||||||
* `[(ngModel)]` syntax for two-way data binding.
|
* `[(ngModel)]` syntax for two-way data binding.
|
||||||
* The use of `name` attributes for validation and form-element change tracking.
|
* The use of `name` attributes for validation and form-element change tracking.
|
||||||
* The reference variable’s `valid` property on input controls to check if a control is valid and show/hide error messages.
|
* The reference variable’s `valid` property on input controls to check if a control is valid and show or hide error messages.
|
||||||
* Controlling the *Submit* button's enabled state by binding to `NgForm` validity.
|
* Controlling the **Submit** button's enabled state by binding to `NgForm` validity.
|
||||||
* Custom CSS classes that provide visual feedback to users about invalid controls.
|
* Custom CSS classes that provide visual feedback to users about invalid controls.
|
||||||
|
|
||||||
Here’s the code for the final version of the application:
|
Here’s the code for the final version of the application:
|
||||||
@ -741,4 +551,3 @@ Here’s the code for the final version of the application:
|
|||||||
</code-pane>
|
</code-pane>
|
||||||
|
|
||||||
</code-tabs>
|
</code-tabs>
|
||||||
|
|
||||||
|
@ -210,6 +210,7 @@ An Angular component class is responsible for exposing data and handling most of
|
|||||||
|
|
||||||
Read more about component classes, templates, and views in [Introduction to Angular concepts](guide/architecture).
|
Read more about component classes, templates, and views in [Introduction to Angular concepts](guide/architecture).
|
||||||
|
|
||||||
|
|
||||||
## configuration
|
## configuration
|
||||||
|
|
||||||
See [workspace configuration](#cli-config)
|
See [workspace configuration](#cli-config)
|
||||||
@ -252,15 +253,15 @@ Data binding is an alternative to manually pushing application data values into
|
|||||||
event listeners, pulling changed values from the screen, and
|
event listeners, pulling changed values from the screen, and
|
||||||
updating application data values.
|
updating application data values.
|
||||||
|
|
||||||
Read about the following forms of binding in [Template Syntax](guide/template-syntax):
|
Read about the following forms of binding in Angular's [Template Syntax](guide/template-syntax):
|
||||||
|
|
||||||
* [Interpolation](guide/template-syntax#interpolation)
|
* [Interpolation](guide/interpolation)
|
||||||
* [Property binding](guide/template-syntax#property-binding)
|
* [Property binding](guide/property-binding)
|
||||||
* [Event binding](guide/template-syntax#event-binding)
|
* [Event binding](guide/event-binding)
|
||||||
* [Attribute binding](guide/template-syntax#attribute-binding)
|
* [Attribute binding](guide/attribute-binding)
|
||||||
* [Class binding](guide/template-syntax#class-binding)
|
* [Class binding](guide/attribute-binding#class-binding)
|
||||||
* [Style binding](guide/template-syntax#style-binding)
|
* [Style binding](guide/attribute-binding#style-binding)
|
||||||
* [Two-way data binding with ngModel](guide/template-syntax#ngModel)
|
* [Two-way data binding with ngModel](guide/built-in-directives#ngModel)
|
||||||
|
|
||||||
{@a declarable}
|
{@a declarable}
|
||||||
|
|
||||||
@ -472,11 +473,11 @@ Learn more about the injector hierarchy in [Hierarchical Dependency Injectors](g
|
|||||||
## input
|
## input
|
||||||
|
|
||||||
When defining a [directive](#directive), the `@Input()` decorator on a directive property
|
When defining a [directive](#directive), the `@Input()` decorator on a directive property
|
||||||
makes that property available as a *target* of a [property binding](guide/template-syntax#property-binding).
|
makes that property available as a *target* of a [property binding](guide/property-binding).
|
||||||
Data values flow into an input property from the data source identified
|
Data values flow into an input property from the data source identified
|
||||||
in the [template expression](#template-expression) to the right of the equal sign.
|
in the [template expression](#template-expression) to the right of the equal sign.
|
||||||
|
|
||||||
To learn more, see [input and output properties](guide/template-syntax#inputs-outputs).
|
To learn more, see [input and output properties](guide/inputs-outputs).
|
||||||
|
|
||||||
{@a interpolation}
|
{@a interpolation}
|
||||||
|
|
||||||
@ -491,7 +492,7 @@ or displayed between element tags, as in this example.
|
|||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
Read more about [interpolation](guide/template-syntax#interpolation) in [Template Syntax](guide/template-syntax).
|
Read more in the [Interpolation](guide/interpolation) guide.
|
||||||
|
|
||||||
{@a ivy}
|
{@a ivy}
|
||||||
|
|
||||||
@ -578,7 +579,6 @@ Angular calls these hook methods in the following order:
|
|||||||
|
|
||||||
To learn more, see [Lifecycle Hooks](guide/lifecycle-hooks).
|
To learn more, see [Lifecycle Hooks](guide/lifecycle-hooks).
|
||||||
|
|
||||||
|
|
||||||
{@a M}
|
{@a M}
|
||||||
|
|
||||||
{@a module}
|
{@a module}
|
||||||
@ -653,11 +653,11 @@ An object passed to the `subscribe()` method for an [observable](#observable). T
|
|||||||
## output
|
## output
|
||||||
|
|
||||||
When defining a [directive](#directive), the `@Output{}` decorator on a directive property
|
When defining a [directive](#directive), the `@Output{}` decorator on a directive property
|
||||||
makes that property available as a *target* of [event binding](guide/template-syntax#event-binding).
|
makes that property available as a *target* of [event binding](guide/event-binding).
|
||||||
Events stream *out* of this property to the receiver identified
|
Events stream *out* of this property to the receiver identified
|
||||||
in the [template expression](#template-expression) to the right of the equal sign.
|
in the [template expression](#template-expression) to the right of the equal sign.
|
||||||
|
|
||||||
To learn more, see [Input and Output Properties](guide/template-syntax#inputs-outputs).
|
To learn more, see [Input and Output Properties](guide/inputs-outputs).
|
||||||
|
|
||||||
|
|
||||||
{@a P}
|
{@a P}
|
||||||
@ -732,13 +732,33 @@ The alternative is a [template-driven form](#template-driven-forms).
|
|||||||
When using reactive forms:
|
When using reactive forms:
|
||||||
|
|
||||||
* The "source of truth", the form model, is defined in the component class.
|
* The "source of truth", the form model, is defined in the component class.
|
||||||
* Validation is set up through validation functions rather than valdation directives.
|
* Validation is set up through validation functions rather than validation directives.
|
||||||
* Each control is explicitly created in the component class by creating a `FormControl` instance manually or with `FormBuilder`.
|
* Each control is explicitly created in the component class by creating a `FormControl` instance manually or with `FormBuilder`.
|
||||||
* The template input elements do *not* use `ngModel`.
|
* The template input elements do *not* use `ngModel`.
|
||||||
* The associated Angular directives are prefixed with `form`, such as `formControl`, `formGroup`, and `formControlName`.
|
* The associated Angular directives are prefixed with `form`, such as `formControl`, `formGroup`, and `formControlName`.
|
||||||
|
|
||||||
The alternative is a template-driven form. For an introduction and comparison of both forms approaches, see [Introduction to Angular Forms](guide/forms-overview).
|
The alternative is a template-driven form. For an introduction and comparison of both forms approaches, see [Introduction to Angular Forms](guide/forms-overview).
|
||||||
|
|
||||||
|
{@a resolver}
|
||||||
|
|
||||||
|
## resolver
|
||||||
|
|
||||||
|
A class that implements the [Resolve](api/router/Resolve "API reference") interface (or a function with the same signature as the [resolve() method](api/router/Resolve#resolve "API reference")) that you use to produce or retrieve data that is needed before navigation to a requested route can be completed.
|
||||||
|
|
||||||
|
Resolvers run after all [route guards](#route-guard "Definition") for a route tree have been executed and have succeeded.
|
||||||
|
|
||||||
|
See an example of using a [resolve guard](guide/router-tutorial-toh#resolve-guard "Routing techniques tutorial") to retrieve dynamic data.
|
||||||
|
|
||||||
|
{@a route-guard}
|
||||||
|
|
||||||
|
## route guard
|
||||||
|
|
||||||
|
A method that controls navigation to a requested route in a routing application.
|
||||||
|
Guards determine whether a route can be activated or deactivated, and whether a lazy-loaded module can be loaded.
|
||||||
|
|
||||||
|
Learn more in the [Routing and Navigation](guide/router#preventing-unauthorized-access "Examples") guide.
|
||||||
|
|
||||||
|
|
||||||
{@a router}
|
{@a router}
|
||||||
{@a router-module}
|
{@a router-module}
|
||||||
|
|
||||||
@ -917,7 +937,20 @@ The alternative is a reactive form. For an introduction and comparison of both f
|
|||||||
|
|
||||||
A TypeScript-like syntax that Angular evaluates within a [data binding](#data-binding).
|
A TypeScript-like syntax that Angular evaluates within a [data binding](#data-binding).
|
||||||
|
|
||||||
Read about how to write template expressions in [Template expressions](guide/template-syntax#template-expressions).
|
Read about how to write template expressions in the [template expressions](guide/interpolation#template-expressions) section of the [Interpolation](guide/interpolation) guide.
|
||||||
|
|
||||||
|
{@a template-reference-variable}
|
||||||
|
|
||||||
|
## template reference variable
|
||||||
|
|
||||||
|
A variable defined in a template that references an instance associated with an element, such as a directive instance, component instance, template as in `TemplateRef`, or DOM element.
|
||||||
|
After declaring a template reference variable on an element in a template,
|
||||||
|
you can access values from that variable elsewhere within the same template.
|
||||||
|
The following example defines a template reference variable named `#phone`.
|
||||||
|
|
||||||
|
<code-example path="template-reference-variables/src/app/app.component.html" region="ref-var" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
For more information, see the [Template reference variable](guide/template-reference-variables) guide.
|
||||||
|
|
||||||
{@a token}
|
{@a token}
|
||||||
|
|
||||||
@ -950,6 +983,10 @@ Many code editors and IDEs support TypeScript either natively or with plug-ins.
|
|||||||
TypeScript is the preferred language for Angular development.
|
TypeScript is the preferred language for Angular development.
|
||||||
Read more about TypeScript at [typescriptlang.org](http://www.typescriptlang.org/).
|
Read more about TypeScript at [typescriptlang.org](http://www.typescriptlang.org/).
|
||||||
|
|
||||||
|
## TypeScript configuration file
|
||||||
|
|
||||||
|
A file specifies the root files and the compiler options required to compile a TypeScript project. For more information, see [TypeScript configuration](/guide/typescript-configuration).
|
||||||
|
|
||||||
|
|
||||||
{@a U}
|
{@a U}
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Communicating with backend services using HTTP
|
# Communicating with backend services using HTTP
|
||||||
|
|
||||||
Most front-end applications need to communicate with a server over the HTTP protocol, in order to download or upload data and accesss other back-end services.
|
Most front-end applications need to communicate with a server over the HTTP protocol, in order to download or upload data and access other back-end services.
|
||||||
Angular provides a simplified client HTTP API for Angular applications, the `HttpClient` service class in `@angular/common/http`.
|
Angular provides a simplified client HTTP API for Angular applications, the `HttpClient` service class in `@angular/common/http`.
|
||||||
|
|
||||||
The HTTP client service offers the following major features.
|
The HTTP client service offers the following major features.
|
||||||
@ -63,7 +63,7 @@ Look at the `AppModule` _imports_ to see how it is configured.
|
|||||||
## Requesting data from a server
|
## Requesting data from a server
|
||||||
|
|
||||||
Use the [`HTTPClient.get()`](api/common/http/HttpClient#get) method to fetch data from a server.
|
Use the [`HTTPClient.get()`](api/common/http/HttpClient#get) method to fetch data from a server.
|
||||||
The aynchronous method sends an HTTP request, and returns an Observable that emits the requested data when the response is received.
|
The asynchronous method sends an HTTP request, and returns an Observable that emits the requested data when the response is received.
|
||||||
The return type varies based on the `observe` and `responseType` values that you pass to the call.
|
The return type varies based on the `observe` and `responseType` values that you pass to the call.
|
||||||
|
|
||||||
The `get()` method takes two arguments; the endpoint URL from which to fetch, and an *options* object that you can use to configure the request.
|
The `get()` method takes two arguments; the endpoint URL from which to fetch, and an *options* object that you can use to configure the request.
|
||||||
@ -277,7 +277,7 @@ searchHeroes(term: string): Observable {
|
|||||||
return this.http.jsonp(heroesUrl, 'callback').pipe(
|
return this.http.jsonp(heroesUrl, 'callback').pipe(
|
||||||
catchError(this.handleError('searchHeroes', [])) // then handle the error
|
catchError(this.handleError('searchHeroes', [])) // then handle the error
|
||||||
);
|
);
|
||||||
};
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
This request passes the `heroesURL` as the first parameter and the callback function name as the second parameter.
|
This request passes the `heroesURL` as the first parameter and the callback function name as the second parameter.
|
||||||
@ -805,16 +805,16 @@ The `CachingInterceptor` in the following example demonstrates this approach.
|
|||||||
header="app/http-interceptors/caching-interceptor.ts)">
|
header="app/http-interceptors/caching-interceptor.ts)">
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
* The `isCachable()` function determines if the request is cachable.
|
* The `isCacheable()` function determines if the request is cacheable.
|
||||||
In this sample, only GET requests to the npm package search api are cachable.
|
In this sample, only GET requests to the npm package search api are cacheable.
|
||||||
|
|
||||||
* If the request is not cachable, the interceptor simply forwards the request
|
* If the request is not cacheable, the interceptor simply forwards the request
|
||||||
to the next handler in the chain.
|
to the next handler in the chain.
|
||||||
|
|
||||||
* If a cachable request is found in the cache, the interceptor returns an `of()` _observable_ with
|
* If a cacheable request is found in the cache, the interceptor returns an `of()` _observable_ with
|
||||||
the cached response, by-passing the `next` handler (and all other interceptors downstream).
|
the cached response, by-passing the `next` handler (and all other interceptors downstream).
|
||||||
|
|
||||||
* If a cachable request is not in cache, the code calls `sendRequest()`.
|
* If a cacheable request is not in cache, the code calls `sendRequest()`.
|
||||||
This function creates a [request clone](#immutability) without headers, because the npm API forbids them.
|
This function creates a [request clone](#immutability) without headers, because the npm API forbids them.
|
||||||
The function then forwards the clone of the request to `next.handle()` which ultimately calls the server and returns the server's response.
|
The function then forwards the clone of the request to `next.handle()` which ultimately calls the server and returns the server's response.
|
||||||
|
|
||||||
|
File diff suppressed because it is too large
Load Diff
342
aio/content/guide/inputs-outputs.md
Normal file
342
aio/content/guide/inputs-outputs.md
Normal file
@ -0,0 +1,342 @@
|
|||||||
|
# `@Input()` and `@Output()` properties
|
||||||
|
|
||||||
|
`@Input()` and `@Output()` allow Angular to share data between the parent context
|
||||||
|
and child directives or components. An `@Input()` property is writable
|
||||||
|
while an `@Output()` property is observable.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
See the <live-example></live-example> for a working example containing the code snippets in this guide.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
Consider this example of a child/parent relationship:
|
||||||
|
|
||||||
|
```html
|
||||||
|
<parent-component>
|
||||||
|
<child-component></child-component>
|
||||||
|
</parent-component>
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
Here, the `<child-component>` selector, or child directive, is embedded
|
||||||
|
within a `<parent-component>`, which serves as the child's context.
|
||||||
|
|
||||||
|
`@Input()` and `@Output()` act as
|
||||||
|
the API, or application programming interface, of the child
|
||||||
|
component in that they allow the child to
|
||||||
|
communicate with the parent. Think of `@Input()` and `@Output()` like ports
|
||||||
|
or doorways—`@Input()` is the doorway into the component allowing data
|
||||||
|
to flow in while `@Output()` is the doorway out of the component, allowing the
|
||||||
|
child component to send data out.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
#### `@Input()` and `@Output()` are independent
|
||||||
|
|
||||||
|
Though `@Input()` and `@Output()` often appear together in apps, you can use
|
||||||
|
them separately. If the nested
|
||||||
|
component is such that it only needs to send data to its parent, you wouldn't
|
||||||
|
need an `@Input()`, only an `@Output()`. The reverse is also true in that if the
|
||||||
|
child only needs to receive data from the parent, you'd only need `@Input()`.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{@a input}
|
||||||
|
|
||||||
|
## How to use `@Input()`
|
||||||
|
|
||||||
|
Use the `@Input()` decorator in a child component or directive to let Angular know
|
||||||
|
that a property in that component can receive its value from its parent component.
|
||||||
|
It helps to remember that the data flow is from the perspective of the
|
||||||
|
child component. So an `@Input()` allows data to be input _into_ the
|
||||||
|
child component from the parent component.
|
||||||
|
|
||||||
|
|
||||||
|
<div class="lightbox">
|
||||||
|
<img src="generated/images/guide/inputs-outputs/input.svg" alt="Input data flow diagram">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
To illustrate the use of `@Input()`, edit these parts of your app:
|
||||||
|
|
||||||
|
* The child component class and template
|
||||||
|
* The parent component class and template
|
||||||
|
|
||||||
|
|
||||||
|
### In the child
|
||||||
|
|
||||||
|
To use the `@Input()` decorator in a child component class, first import
|
||||||
|
`Input` and then decorate the property with `@Input()`:
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/item-detail/item-detail.component.ts" region="use-input" header="src/app/item-detail/item-detail.component.ts"></code-example>
|
||||||
|
|
||||||
|
|
||||||
|
In this case, `@Input()` decorates the property <code class="no-auto-link">item</code>, which has
|
||||||
|
a type of `string`, however, `@Input()` properties can have any type, such as
|
||||||
|
`number`, `string`, `boolean`, or `object`. The value for `item` will come from the parent component, which the next section covers.
|
||||||
|
|
||||||
|
Next, in the child component template, add the following:
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/item-detail/item-detail.component.html" region="property-in-template" header="src/app/item-detail/item-detail.component.html"></code-example>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### In the parent
|
||||||
|
|
||||||
|
The next step is to bind the property in the parent component's template.
|
||||||
|
In this example, the parent component template is `app.component.html`.
|
||||||
|
|
||||||
|
First, use the child's selector, here `<app-item-detail>`, as a directive within the
|
||||||
|
parent component template. Then, use [property binding](guide/property-binding)
|
||||||
|
to bind the property in the child to the property of the parent.
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/app.component.html" region="input-parent" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Next, in the parent component class, `app.component.ts`, designate a value for `currentItem`:
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/app.component.ts" region="parent-property" header="src/app/app.component.ts"></code-example>
|
||||||
|
|
||||||
|
With `@Input()`, Angular passes the value for `currentItem` to the child so that `item` renders as `Television`.
|
||||||
|
|
||||||
|
The following diagram shows this structure:
|
||||||
|
|
||||||
|
<div class="lightbox">
|
||||||
|
<img src="generated/images/guide/inputs-outputs/input-diagram-target-source.svg" alt="Property binding diagram">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
The target in the square brackets, `[]`, is the property you decorate
|
||||||
|
with `@Input()` in the child component. The binding source, the part
|
||||||
|
to the right of the equal sign, is the data that the parent
|
||||||
|
component passes to the nested component.
|
||||||
|
|
||||||
|
The key takeaway is that when binding to a child component's property in a parent component—that is, what's
|
||||||
|
in square brackets—you must
|
||||||
|
decorate the property with `@Input()` in the child component.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
### `OnChanges` and `@Input()`
|
||||||
|
|
||||||
|
To watch for changes on an `@Input()` property, use
|
||||||
|
`OnChanges`, one of Angular's [lifecycle hooks](guide/lifecycle-hooks#onchanges).
|
||||||
|
`OnChanges` is specifically designed to work with properties that have the
|
||||||
|
`@Input()` decorator. See the [`OnChanges`](guide/lifecycle-hooks#onchanges) section of the [Lifecycle Hooks](guide/lifecycle-hooks) guide for more details and examples.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{@a output}
|
||||||
|
|
||||||
|
## How to use `@Output()`
|
||||||
|
|
||||||
|
Use the `@Output()` decorator in the child component or directive to allow data to flow from
|
||||||
|
the child _out_ to the parent.
|
||||||
|
|
||||||
|
An `@Output()` property should normally be initialized to an Angular [`EventEmitter`](api/core/EventEmitter) with values flowing out of the component as [events](guide/event-binding).
|
||||||
|
|
||||||
|
|
||||||
|
<div class="lightbox">
|
||||||
|
<img src="generated/images/guide/inputs-outputs/output.svg" alt="Output diagram">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
Just like with `@Input()`, you can use `@Output()`
|
||||||
|
on a property of the child component but its type should be
|
||||||
|
`EventEmitter`.
|
||||||
|
|
||||||
|
`@Output()` marks a property in a child component as a doorway
|
||||||
|
through which data can travel from the child to the parent.
|
||||||
|
The child component then has to raise an event so the
|
||||||
|
parent knows something has changed. To raise an event,
|
||||||
|
`@Output()` works hand in hand with `EventEmitter`,
|
||||||
|
which is a class in `@angular/core` that you
|
||||||
|
use to emit custom events.
|
||||||
|
|
||||||
|
When you use `@Output()`, edit these parts of your app:
|
||||||
|
|
||||||
|
* The child component class and template
|
||||||
|
* The parent component class and template
|
||||||
|
|
||||||
|
|
||||||
|
The following example shows how to set up an `@Output()` in a child
|
||||||
|
component that pushes data you enter in an HTML `<input>` to an array in the
|
||||||
|
parent component.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
The HTML element `<input>` and the Angular decorator `@Input()`
|
||||||
|
are different. This documentation is about component communication in Angular as it pertains to `@Input()` and `@Output()`. For more information on the HTML element `<input>`, see the [W3C Recommendation](https://www.w3.org/TR/html5/sec-forms.html#the-input-element).
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
## In the child
|
||||||
|
|
||||||
|
This example features an `<input>` where a user can enter a value and click a `<button>` that raises an event. The `EventEmitter` then relays the data to the parent component.
|
||||||
|
|
||||||
|
First, be sure to import `Output` and `EventEmitter`
|
||||||
|
in the child component class:
|
||||||
|
|
||||||
|
```js
|
||||||
|
import { Output, EventEmitter } from '@angular/core';
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
Next, still in the child, decorate a property with `@Output()` in the component class.
|
||||||
|
The following example `@Output()` is called `newItemEvent` and its type is
|
||||||
|
`EventEmitter`, which means it's an event.
|
||||||
|
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/item-output/item-output.component.ts" region="item-output" header="src/app/item-output/item-output.component.ts"></code-example>
|
||||||
|
|
||||||
|
The different parts of the above declaration are as follows:
|
||||||
|
|
||||||
|
* `@Output()`—a decorator function marking the property as a way for data to go from the child to the parent
|
||||||
|
* `newItemEvent`—the name of the `@Output()`
|
||||||
|
* `EventEmitter<string>`—the `@Output()`'s type
|
||||||
|
* `new EventEmitter<string>()`—tells Angular to create a new event emitter and that the data it emits is of type string. The type could be any type, such as `number`, `boolean`, and so on. For more information on `EventEmitter`, see the [EventEmitter API documentation](api/core/EventEmitter).
|
||||||
|
|
||||||
|
Next, create an `addNewItem()` method in the same component class:
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/item-output/item-output.component.ts" region="item-output-class" header="src/app/item-output/item-output.component.ts"></code-example>
|
||||||
|
|
||||||
|
The `addNewItem()` function uses the `@Output()`, `newItemEvent`,
|
||||||
|
to raise an event in which it emits the value the user
|
||||||
|
types into the `<input>`. In other words, when
|
||||||
|
the user clicks the add button in the UI, the child lets the parent know
|
||||||
|
about the event and gives that data to the parent.
|
||||||
|
|
||||||
|
### In the child's template
|
||||||
|
|
||||||
|
The child's template has two controls. The first is an HTML `<input>` with a
|
||||||
|
[template reference variable](guide/template-reference-variables) , `#newItem`,
|
||||||
|
where the user types in an item name. Whatever the user types
|
||||||
|
into the `<input>` gets stored in the `#newItem` variable.
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/item-output/item-output.component.html" region="child-output" header="src/app/item-output/item-output.component.html"></code-example>
|
||||||
|
|
||||||
|
The second element is a `<button>`
|
||||||
|
with an [event binding](guide/event-binding). You know it's
|
||||||
|
an event binding because the part to the left of the equal
|
||||||
|
sign is in parentheses, `(click)`.
|
||||||
|
|
||||||
|
The `(click)` event is bound to the `addNewItem()` method in the child component class which
|
||||||
|
takes as its argument whatever the value of `#newItem` is.
|
||||||
|
|
||||||
|
Now the child component has an `@Output()`
|
||||||
|
for sending data to the parent and a method for raising an event.
|
||||||
|
The next step is in the parent.
|
||||||
|
|
||||||
|
## In the parent
|
||||||
|
|
||||||
|
In this example, the parent component is `AppComponent`, but you could use
|
||||||
|
any component in which you could nest the child.
|
||||||
|
|
||||||
|
The `AppComponent` in this example features a list of `items`
|
||||||
|
in an array and a method for adding more items to the array.
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/app.component.ts" region="add-new-item" header="src/app/app.component.ts"></code-example>
|
||||||
|
|
||||||
|
The `addItem()` method takes an argument in the form of a string
|
||||||
|
and then pushes, or adds, that string to the `items` array.
|
||||||
|
|
||||||
|
### In the parent's template
|
||||||
|
|
||||||
|
Next, in the parent's template, bind the parent's
|
||||||
|
method to the child's event. Put the child selector, here `<app-item-output>`,
|
||||||
|
within the parent component's
|
||||||
|
template, `app.component.html`.
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/app.component.html" region="output-parent" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
The event binding, `(newItemEvent)='addItem($event)'`, tells
|
||||||
|
Angular to connect the event in the child, `newItemEvent`, to
|
||||||
|
the method in the parent, `addItem()`, and that the event that the child
|
||||||
|
is notifying the parent about is to be the argument of `addItem()`.
|
||||||
|
In other words, this is where the actual hand off of data takes place.
|
||||||
|
The `$event` contains the data that the user types into the `<input>`
|
||||||
|
in the child template UI.
|
||||||
|
|
||||||
|
Now, in order to see the `@Output()` working, add the following to the parent's template:
|
||||||
|
|
||||||
|
```html
|
||||||
|
<ul>
|
||||||
|
<li *ngFor="let item of items">{{item}}</li>
|
||||||
|
</ul>
|
||||||
|
```
|
||||||
|
|
||||||
|
The `*ngFor` iterates over the items in the `items` array. When you enter a value in the child's `<input>` and click the button, the child emits the event and the parent's `addItem()` method pushes the value to the `items` array and it renders in the list.
|
||||||
|
|
||||||
|
|
||||||
|
## `@Input()` and `@Output()` together
|
||||||
|
|
||||||
|
You can use `@Input()` and `@Output()` on the same child component as in the following:
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/app.component.html" region="together" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
The target, `item`, which is an `@Input()` property in the child component class, receives its value from the parent's property, `currentItem`. When you click delete, the child component raises an event, `deleteRequest`, which is the argument for the parent's `crossOffItem()` method.
|
||||||
|
|
||||||
|
The following diagram is of an `@Input()` and an `@Output()` on the same
|
||||||
|
child component and shows the different parts of each:
|
||||||
|
|
||||||
|
<div class="lightbox">
|
||||||
|
<img src="generated/images/guide/inputs-outputs/input-output-diagram.svg" alt="Input/Output diagram">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
As the diagram shows, use inputs and outputs together in the same manner as using them separately. Here, the child selector is `<app-input-output>` with `item` and `deleteRequest` being `@Input()` and `@Output()`
|
||||||
|
properties in the child component class. The property `currentItem` and the method `crossOffItem()` are both in the parent component class.
|
||||||
|
|
||||||
|
To combine property and event bindings using the banana-in-a-box
|
||||||
|
syntax, `[()]`, see [Two-way Binding](guide/two-way-binding).
|
||||||
|
|
||||||
|
## `@Input()` and `@Output()` declarations
|
||||||
|
|
||||||
|
Instead of using the `@Input()` and `@Output()` decorators
|
||||||
|
to declare inputs and outputs, you can identify
|
||||||
|
members in the `inputs` and `outputs` arrays
|
||||||
|
of the directive metadata, as in this example:
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/in-the-metadata/in-the-metadata.component.ts" region="metadata" header="src/app/in-the-metadata/in-the-metadata.component.ts"></code-example>
|
||||||
|
|
||||||
|
While declaring `inputs` and `outputs` in the `@Directive` and `@Component`
|
||||||
|
metadata is possible, it is a better practice to use the `@Input()` and `@Output()`
|
||||||
|
class decorators instead, as follows:
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/input-output/input-output.component.ts" region="input-output" header="src/app/input-output/input-output.component.ts"></code-example>
|
||||||
|
|
||||||
|
See the [Decorate input and output properties](guide/styleguide#decorate-input-and-output-properties) section of the
|
||||||
|
[Style Guide](guide/styleguide) for details.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
If you get a template parse error when trying to use inputs or outputs, but you know that the
|
||||||
|
properties do indeed exist, double check
|
||||||
|
that your properties are annotated with `@Input()` / `@Output()` or that you've declared
|
||||||
|
them in an `inputs`/`outputs` array:
|
||||||
|
|
||||||
|
<code-example language="bash">
|
||||||
|
Uncaught Error: Template parse errors:
|
||||||
|
Can't bind to 'item' since it isn't a known property of 'app-item-detail'
|
||||||
|
</code-example>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{@a aliasing-io}
|
||||||
|
|
||||||
|
## Aliasing inputs and outputs
|
||||||
|
|
||||||
|
Sometimes the public name of an input/output property should be different from the internal name. While it is a best practice to avoid this situation, Angular does
|
||||||
|
offer a solution.
|
||||||
|
|
||||||
|
### Aliasing in the metadata
|
||||||
|
|
||||||
|
Alias inputs and outputs in the metadata using a colon-delimited (`:`) string with
|
||||||
|
the directive property name on the left and the public alias on the right:
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/aliasing/aliasing.component.ts" region="alias" header="src/app/aliasing/aliasing.component.ts"></code-example>
|
||||||
|
|
||||||
|
|
||||||
|
### Aliasing with the `@Input()`/`@Output()` decorator
|
||||||
|
|
||||||
|
You can specify the alias for the property name by passing the alias name to the `@Input()`/`@Output()` decorator. The internal name remains as usual.
|
||||||
|
|
||||||
|
<code-example path="inputs-outputs/src/app/aliasing/aliasing.component.ts" region="alias-input-output" header="src/app/aliasing/aliasing.component.ts"></code-example>
|
175
aio/content/guide/interpolation.md
Normal file
175
aio/content/guide/interpolation.md
Normal file
@ -0,0 +1,175 @@
|
|||||||
|
# Interpolation and template expressions
|
||||||
|
|
||||||
|
Interpolation allows you to incorporate calculated strings into the text
|
||||||
|
between HTML element tags and within attribute assignments. Template
|
||||||
|
expressions are what you use to calculate those strings.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
See the <live-example></live-example> for all of
|
||||||
|
the syntax and code snippets in this guide.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
## Interpolation `{{...}}`
|
||||||
|
|
||||||
|
Interpolation refers to embedding expressions into marked up text.
|
||||||
|
By default, interpolation uses as its delimiter the double curly braces, `{{` and `}}`.
|
||||||
|
|
||||||
|
In the following snippet, `{{ currentCustomer }}` is an example of interpolation.
|
||||||
|
|
||||||
|
<code-example path="interpolation/src/app/app.component.html" region="interpolation-example1" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
The text between the braces is often the name of a component
|
||||||
|
property. Angular replaces that name with the
|
||||||
|
string value of the corresponding component property.
|
||||||
|
|
||||||
|
<code-example path="interpolation/src/app/app.component.html" region="component-property" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
In the example above, Angular evaluates the `title` and `itemImageUrl` properties
|
||||||
|
and fills in the blanks, first displaying some title text and then an image.
|
||||||
|
|
||||||
|
More generally, the text between the braces is a **template expression**
|
||||||
|
that Angular first **evaluates** and then **converts to a string**.
|
||||||
|
The following interpolation illustrates the point by adding two numbers:
|
||||||
|
|
||||||
|
<code-example path="interpolation/src/app/app.component.html" region="convert-string" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
The expression can invoke methods of the host component such as `getVal()` in
|
||||||
|
the following example:
|
||||||
|
|
||||||
|
<code-example path="interpolation/src/app/app.component.html" region="invoke-method" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Angular evaluates all expressions in double curly braces,
|
||||||
|
converts the expression results to strings, and links them with neighboring literal strings. Finally,
|
||||||
|
it assigns this composite interpolated result to an **element or directive property**.
|
||||||
|
|
||||||
|
You appear to be inserting the result between element tags and assigning it to attributes.
|
||||||
|
However, interpolation is a special syntax that Angular converts into a *property binding*.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
If you'd like to use something other than `{{` and `}}`, you can
|
||||||
|
configure the interpolation delimiter via the
|
||||||
|
[interpolation](api/core/Component#interpolation)
|
||||||
|
option in the `Component` metadata.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
## Template expressions
|
||||||
|
|
||||||
|
A template **expression** produces a value and appears within the double
|
||||||
|
curly braces, `{{ }}`.
|
||||||
|
Angular executes the expression and assigns it to a property of a binding target;
|
||||||
|
the target could be an HTML element, a component, or a directive.
|
||||||
|
|
||||||
|
The interpolation braces in `{{1 + 1}}` surround the template expression `1 + 1`.
|
||||||
|
In the property binding,
|
||||||
|
a template expression appears in quotes to the right of the `=` symbol as in `[property]="expression"`.
|
||||||
|
|
||||||
|
In terms of syntax, template expressions are similar to JavaScript.
|
||||||
|
Many JavaScript expressions are legal template expressions, with a few exceptions.
|
||||||
|
|
||||||
|
You can't use JavaScript expressions that have or promote side effects,
|
||||||
|
including:
|
||||||
|
|
||||||
|
* Assignments (`=`, `+=`, `-=`, `...`)
|
||||||
|
* Operators such as `new`, `typeof`, `instanceof`, etc.
|
||||||
|
* Chaining expressions with <code>;</code> or <code>,</code>
|
||||||
|
* The increment and decrement operators `++` and `--`
|
||||||
|
* Some of the ES2015+ operators
|
||||||
|
|
||||||
|
Other notable differences from JavaScript syntax include:
|
||||||
|
|
||||||
|
* No support for the bitwise operators such as `|` and `&`
|
||||||
|
* New [template expression operators](guide/template-expression-operators), such as `|`, `?.` and `!`
|
||||||
|
|
||||||
|
|
||||||
|
## Expression context
|
||||||
|
|
||||||
|
The *expression context* is typically the _component_ instance.
|
||||||
|
In the following snippets, the `recommended` within double curly braces and the
|
||||||
|
`itemImageUrl2` in quotes refer to properties of the `AppComponent`.
|
||||||
|
|
||||||
|
<code-example path="interpolation/src/app/app.component.html" region="component-context" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
An expression may also refer to properties of the _template's_ context
|
||||||
|
such as a template input variable,
|
||||||
|
<!-- link to built-in-directives#template-input-variables -->
|
||||||
|
`let customer`, or a template reference variable, `#customerInput`.
|
||||||
|
<!-- link to guide/template-ref-variables -->
|
||||||
|
|
||||||
|
<code-example path="interpolation/src/app/app.component.html" region="template-input-variable" header="src/app/app.component.html (template input variable)"></code-example>
|
||||||
|
|
||||||
|
<code-example path="interpolation/src/app/app.component.html" region="template-reference-variable" header="src/app/app.component.html (template reference variable)"></code-example>
|
||||||
|
|
||||||
|
The context for terms in an expression is a blend of the _template variables_,
|
||||||
|
the directive's _context_ object (if it has one), and the component's _members_.
|
||||||
|
If you reference a name that belongs to more than one of these namespaces,
|
||||||
|
the template variable name takes precedence, followed by a name in the directive's _context_,
|
||||||
|
and, lastly, the component's member names.
|
||||||
|
|
||||||
|
The previous example presents such a name collision. The component has a `customer`
|
||||||
|
property and the `*ngFor` defines a `customer` template variable.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
The `customer` in `{{customer.name}}`
|
||||||
|
refers to the template input variable, not the component's property.
|
||||||
|
|
||||||
|
Template expressions cannot refer to anything in
|
||||||
|
the global namespace, except `undefined`. They can't refer to
|
||||||
|
`window` or `document`. Additionally, they
|
||||||
|
can't call `console.log()` or `Math.max()` and they are restricted to referencing
|
||||||
|
members of the expression context.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
## Expression guidelines
|
||||||
|
|
||||||
|
When using template expressions follow these guidelines:
|
||||||
|
|
||||||
|
* [Simplicity](guide/interpolation#simplicity)
|
||||||
|
* [Quick execution](guide/interpolation#quick-execution)
|
||||||
|
* [No visible side effects](guide/interpolation#no-visible-side-effects)
|
||||||
|
|
||||||
|
### Simplicity
|
||||||
|
|
||||||
|
Although it's possible to write complex template expressions, it's a better
|
||||||
|
practice to avoid them.
|
||||||
|
|
||||||
|
A property name or method call should be the norm, but an occasional Boolean negation, `!`, is OK.
|
||||||
|
Otherwise, confine application and business logic to the component,
|
||||||
|
where it is easier to develop and test.
|
||||||
|
|
||||||
|
### Quick execution
|
||||||
|
|
||||||
|
Angular executes template expressions after every change detection cycle.
|
||||||
|
Change detection cycles are triggered by many asynchronous activities such as
|
||||||
|
promise resolutions, HTTP results, timer events, key presses and mouse moves.
|
||||||
|
|
||||||
|
Expressions should finish quickly or the user experience may drag, especially on slower devices.
|
||||||
|
Consider caching values when their computation is expensive.
|
||||||
|
|
||||||
|
### No visible side effects
|
||||||
|
|
||||||
|
A template expression should not change any application state other than the value of the
|
||||||
|
target property.
|
||||||
|
|
||||||
|
This rule is essential to Angular's "unidirectional data flow" policy.
|
||||||
|
You should never worry that reading a component value might change some other displayed value.
|
||||||
|
The view should be stable throughout a single rendering pass.
|
||||||
|
|
||||||
|
An [idempotent](https://en.wikipedia.org/wiki/Idempotence) expression is ideal because
|
||||||
|
it is free of side effects and improves Angular's change detection performance.
|
||||||
|
In Angular terms, an idempotent expression always returns
|
||||||
|
*exactly the same thing* until one of its dependent values changes.
|
||||||
|
|
||||||
|
Dependent values should not change during a single turn of the event loop.
|
||||||
|
If an idempotent expression returns a string or a number, it returns the same string or number when called twice in a row. If the expression returns an object, including an `array`, it returns the same object *reference* when called twice in a row.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
There is one exception to this behavior that applies to `*ngFor`. `*ngFor` has `trackBy` functionality that can deal with referential inequality of objects when iterating over them. See [*ngFor with `trackBy`](guide/built-in-directives#ngfor-with-trackby) for details.
|
||||||
|
|
||||||
|
</div>
|
@ -2,20 +2,20 @@
|
|||||||
|
|
||||||
The Angular team has worked hard to ensure Ivy is as backwards-compatible with the previous rendering engine ("View Engine") as possible.
|
The Angular team has worked hard to ensure Ivy is as backwards-compatible with the previous rendering engine ("View Engine") as possible.
|
||||||
However, in rare cases, minor changes were necessary to ensure that the Angular's behavior was predictable and consistent, correcting issues in the View Engine implementation.
|
However, in rare cases, minor changes were necessary to ensure that the Angular's behavior was predictable and consistent, correcting issues in the View Engine implementation.
|
||||||
In order to smooth the transition, we have provided [automated migrations](guide/updating-to-version-9#migrations) wherever possible so your application and library code is migrated automatically by the CLI.
|
In order to smooth the transition, we have provided [automated migrations](guide/updating-to-version-10#migrations) wherever possible so your application and library code is migrated automatically by the CLI.
|
||||||
That said, some applications will likely need to apply some manual updates.
|
That said, some applications will likely need to apply some manual updates.
|
||||||
|
|
||||||
{@a debugging}
|
{@a debugging}
|
||||||
## How to debug errors with Ivy
|
## How to debug errors with Ivy
|
||||||
|
|
||||||
In version 9, [a few deprecated APIs have been removed](guide/updating-to-version-9#removals) and there are a [few breaking changes](guide/updating-to-version-9#breaking-changes) unrelated to Ivy.
|
In version 10, [a few deprecated APIs have been removed](guide/updating-to-version-10#removals) and there are a [few breaking changes](guide/updating-to-version-10#breaking-changes) unrelated to Ivy.
|
||||||
If you're seeing errors after updating to version 9, you'll first want to rule those changes out.
|
If you're seeing errors after updating to version 9, you'll first want to rule those changes out.
|
||||||
|
|
||||||
To do so, temporarily [turn off Ivy](guide/ivy#opting-out-of-angular-ivy) in your `tsconfig.json` and re-start your app.
|
To do so, temporarily [turn off Ivy](guide/ivy#opting-out-of-angular-ivy) in your `tsconfig.base.json` and re-start your app.
|
||||||
|
|
||||||
If you're still seeing the errors, they are not specific to Ivy. In this case, you may want to consult the [general version 9 guide](guide/updating-to-version-9). If you've opted into any of the stricter type-checking settings that are new with v9, you may also want to check out the [template type-checking guide](guide/template-typecheck).
|
If you're still seeing the errors, they are not specific to Ivy. In this case, you may want to consult the [general version 10 guide](guide/updating-to-version-10). If you've opted into any of the new, stricter type-checking settings, you may also want to check out the [template type-checking guide](guide/template-typecheck).
|
||||||
|
|
||||||
If the errors are gone, switch back to Ivy by removing the changes to the `tsconfig.json` and review the list of expected changes below.
|
If the errors are gone, switch back to Ivy by removing the changes to the `tsconfig.base.json` and review the list of expected changes below.
|
||||||
|
|
||||||
{@a payload-size-debugging}
|
{@a payload-size-debugging}
|
||||||
### Payload size debugging
|
### Payload size debugging
|
||||||
|
@ -311,7 +311,7 @@ ngOnInit() {
|
|||||||
}
|
}
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
For more information with a working example, see the [routing tutorial section on preloading](guide/router#preloading-background-loading-of-feature-areas).
|
For more information with a working example, see the [routing tutorial section on preloading](guide/router-tutorial-toh#preloading-background-loading-of-feature-areas).
|
||||||
|
|
||||||
|
|
||||||
<hr>
|
<hr>
|
||||||
|
@ -495,7 +495,7 @@ for one turn of the browser's JavaScript cycle, which triggers a new change-dete
|
|||||||
|
|
||||||
#### Write lean hook methods to avoid performance problems
|
#### Write lean hook methods to avoid performance problems
|
||||||
|
|
||||||
When you run the *AfterView* sample, notice how frequently Angular calls `AfterViewChecked()`$emdash;often when there are no changes of interest.
|
When you run the *AfterView* sample, notice how frequently Angular calls `AfterViewChecked()`-often when there are no changes of interest.
|
||||||
Be very careful about how much logic or computation you put into one of these methods.
|
Be very careful about how much logic or computation you put into one of these methods.
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
@ -569,7 +569,7 @@ which can only be reached by querying for them via the property decorated with
|
|||||||
|
|
||||||
{@a no-unidirectional-flow-worries}
|
{@a no-unidirectional-flow-worries}
|
||||||
|
|
||||||
<div class="alert is-helpful>
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
<header>No need to wait for content updates</header>
|
<header>No need to wait for content updates</header>
|
||||||
|
|
||||||
|
55
aio/content/guide/migration-solution-style-tsconfig.md
Normal file
55
aio/content/guide/migration-solution-style-tsconfig.md
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
# Solution-style `tsconfig.json` migration
|
||||||
|
|
||||||
|
## What does this migration do?
|
||||||
|
|
||||||
|
This migration adds support to existing projects for TypeScript's new ["solution-style" tsconfig feature](https://devblogs.microsoft.com/typescript/announcing-typescript-3-9/#solution-style-tsconfig).
|
||||||
|
|
||||||
|
Support is added by making two changes:
|
||||||
|
|
||||||
|
1. Renaming the workspace-level `tsconfig.json` to `tsconfig.base.json`.
|
||||||
|
All project [TypeScript configuration files](guide/typescript-configuration) will extend from this base which contains the common options used throughout the workspace.
|
||||||
|
|
||||||
|
2. Adding the solution `tsconfig.json` file at the root of the workspace.
|
||||||
|
This `tsconfig.json` file will only contain references to project-level TypeScript configuration files and is only used by editors/IDEs.
|
||||||
|
|
||||||
|
As an example, the solution `tsconfig.json` for a new project is as follows:
|
||||||
|
```json
|
||||||
|
// This is a "Solution Style" tsconfig.json file, and is used by editors and TypeScript’s language server to improve development experience.
|
||||||
|
// It is not intended to be used to perform a compilation.
|
||||||
|
{
|
||||||
|
"files": [],
|
||||||
|
"references": [
|
||||||
|
{
|
||||||
|
"path": "./tsconfig.app.json"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "./tsconfig.spec.json"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "./e2e/tsconfig.json"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Why is this migration necessary?
|
||||||
|
|
||||||
|
Solution-style `tsconfig.json` files provide an improved editing experience and fix several long-standing defects when editing files in an IDE.
|
||||||
|
IDEs that leverage the TypeScript language service (for example, [Visual Studio Code](https://code.visualstudio.com)), will only use TypeScript configuration files that are named `tsconfig.json`.
|
||||||
|
In complex projects, there may be more than one compilation unit and each of these units may have different settings and options.
|
||||||
|
|
||||||
|
With the Angular CLI, a project will have application code that will target a browser.
|
||||||
|
It will also have unit tests that should not be included within the built application and that also need additional type information present (`jasmine` in this case).
|
||||||
|
Both parts of the project also share some but not all of the code within the project.
|
||||||
|
As a result, two separate TypeScript configuration files (`tsconfig.app.json` and `tsconfig.spec.json`) are needed to ensure that each part of the application is configured properly and that the right types are used for each part.
|
||||||
|
Also if web workers are used within a project, an additional tsconfig (`tsconfig.worker.json`) is needed.
|
||||||
|
Web workers use similar but incompatible types to the main browser application.
|
||||||
|
This requires the additional configuration file to ensure that the web worker files use the appropriate types and will build successfully.
|
||||||
|
|
||||||
|
While the Angular build system knows about all of these TypeScript configuration files, an IDE using TypeScript's language service does not.
|
||||||
|
Because of this, an IDE will not be able to properly analyze the code from each part of the project and may generate false errors or make suggestions that are incorrect for certain files.
|
||||||
|
By leveraging the new solution-style tsconfig, the IDE can now be aware of the configuration of each part of a project.
|
||||||
|
This allows each file to be treated appropriately based on its tsconfig.
|
||||||
|
IDE features such as error/warning reporting and auto-suggestion will operate more effectively as well.
|
||||||
|
|
||||||
|
The TypeScript 3.9 release [blog post](https://devblogs.microsoft.com/typescript/announcing-typescript-3-9/#solution-style-tsconfig) also contains some additional information regarding this new feature.
|
52
aio/content/guide/migration-update-libraries-tslib.md
Normal file
52
aio/content/guide/migration-update-libraries-tslib.md
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
# `tslib` direct dependency migration
|
||||||
|
|
||||||
|
## What does this migration do?
|
||||||
|
|
||||||
|
If you have any libraries within your workspace, this migration will convert `tslib` peer dependencies to direct dependencies for the libraries.
|
||||||
|
TypeScript uses the `tslib` package to provide common helper functions used in compiled TypeScript code.
|
||||||
|
The `tslib` version is also updated to `2.0.0` to support TypeScript 3.9.
|
||||||
|
|
||||||
|
Before:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "my-lib",
|
||||||
|
"version": "0.0.1",
|
||||||
|
"peerDependencies": {
|
||||||
|
"@angular/common": "^9.0.0",
|
||||||
|
"@angular/core": "^9.0.0",
|
||||||
|
"tslib": "^1.12.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
After:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "my-lib",
|
||||||
|
"version": "0.0.1",
|
||||||
|
"peerDependencies": {
|
||||||
|
"@angular/common": "^9.0.0",
|
||||||
|
"@angular/core": "^9.0.0"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"tslib": "^2.0.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Why is this migration necessary?
|
||||||
|
|
||||||
|
The [`tslib`](https://github.com/Microsoft/tslib) is a runtime library for Typescript.
|
||||||
|
The version of this library is bound to the version of the TypeScript compiler used to compile a library.
|
||||||
|
Peer dependencies do not accurately represent this relationship between the runtime and the compiler.
|
||||||
|
If `tslib` remained declared as a library peer dependency, it would be possible for some Angular workspaces to get into a state where the workspace could not satisfy `tslib` peer dependency requirements for multiple libraries, resulting in build-time or run-time errors.
|
||||||
|
|
||||||
|
As of TypeScript 3.9 (used by Angular v10), `tslib` version of 2.x is required to build new applications.
|
||||||
|
However, older libraries built with previous version of TypeScript and already published to npm might need `tslib` 1.x.
|
||||||
|
This migration makes it possible for code depending on incompatible versions of the `tslib` runtime library to remain interoperable.
|
||||||
|
|
||||||
|
|
||||||
|
## Do I still need `tslib` as a dependency in my workspace `package.json`?
|
||||||
|
|
||||||
|
Yes.
|
||||||
|
The `tslib` dependency declared in the `package.json` file of the workspace is used to build applications within this workspace, as well as run unit tests for workspace libraries, and is required.
|
@ -0,0 +1,33 @@
|
|||||||
|
# Update `module` and `target` compiler options migration
|
||||||
|
|
||||||
|
## What does this migration do?
|
||||||
|
|
||||||
|
This migration adjusts the [`target`](https://www.typescriptlang.org/v2/en/tsconfig#target) and [`module`](https://www.typescriptlang.org/v2/en/tsconfig#module) settings within the [TypeScript configuration files](guide/typescript-configuration) for the workspace.
|
||||||
|
The changes to each option vary based on the builder or command that uses the TypeScript configuration file.
|
||||||
|
Unless otherwise noted, changes are only made if the existing value was not changed since the project was created.
|
||||||
|
This process helps ensure that intentional changes to the options are kept in place.
|
||||||
|
|
||||||
|
TypeScript Configuration File(s) | Changed Property | Existing Value | New Value
|
||||||
|
------------- | ------------- | ------------- | ------------- | -------------
|
||||||
|
`<workspace base>/tsconfig.base.json` | `"module"` | `"esnext"` | `"es2020"`
|
||||||
|
Used in `browser` builder options (`ng build` for applications) | `"module"` | `"esnext"` | `"es2020"`
|
||||||
|
Used in `ng-packgr` builder options (`ng build` for libraries) | `"module"` | `"esnext"` | `"es2020"`
|
||||||
|
Used in `karma` builder options (`ng test` for applications) | `"module"` | `"esnext"` | `"es2020"`
|
||||||
|
Used in `server` builder options (universal) | `"module"` | `"commonjs"` | _removed_
|
||||||
|
Used in `server` builder options (universal) | `"target"` | _any_ | `"es2016"`
|
||||||
|
Used in `protractor` builder options (`ng e2e` for applications) | `"target"` | `"es5"` | `"es2018"`
|
||||||
|
|
||||||
|
## Why is this migration necessary?
|
||||||
|
|
||||||
|
This migration provides improvements to the long-term supportability of projects by updating the projects to use recommended best practice compilation options.
|
||||||
|
|
||||||
|
For the functionality that executes on Node.js, such as Universal and Protractor, the new settings provide performance and troubleshooting benefits as well.
|
||||||
|
The minimum Node.js version for the Angular CLI (v10.13) supports features in ES2018 and earlier.
|
||||||
|
By targeting later ES versions, the compiler transforms less code and can use newer features directly.
|
||||||
|
Since zone.js does not support native `async` and `await`, the universal builds still target ES2016.
|
||||||
|
|
||||||
|
## Why `"es2020"` instead of `"esnext"`?
|
||||||
|
|
||||||
|
In TypeScript 3.9, the behavior of the TypeScript compiler controlled by `module` is the same with both `"esnext"` and `"es2020"` values.
|
||||||
|
This behavior can change in the future, because the `"esnext"` option could evolve in a backwards incompatible ways, resulting in build-time or run-time errors during a TypeScript update.
|
||||||
|
As a result, code can become unstable. Using the `"es2020"` option mitigates this risk.
|
@ -1,116 +1,41 @@
|
|||||||
# Types of feature modules
|
# Guidelines for creating NgModules
|
||||||
|
|
||||||
There are five general categories of feature modules which
|
This topic provides a conceptual overview of the different categories of [NgModules](guide/glossary#ngmodule "Definition of NgModule") you can create in order to organize your code in a modular structure.
|
||||||
tend to fall into the following groups:
|
These categories are not cast in stone—they are suggestions.
|
||||||
|
You may want to create NgModules for other purposes, or combine the characteristics of some of these categories.
|
||||||
|
|
||||||
* Domain feature modules.
|
NgModules are a great way to organize an app and keep code related to a specific functionality or feature separate from other code.
|
||||||
* Routed feature modules.
|
Use NgModules to consolidate [components](guide/glossary#component "Definition of component"), [directives](guide/glossary#directive "Definition of directive"), and [pipes](guide/glossary#pipe "Definition of pipe)") into cohesive blocks of functionality.
|
||||||
* Routing modules.
|
Focus each block on a feature or business domain, a workflow or navigation flow, a common collection of utilities, or one or more [providers](guide/glossary#provider "Definition of provider") for [services](guide/glossary#service "Definition of service").
|
||||||
* Service feature modules.
|
|
||||||
* Widget feature modules.
|
|
||||||
|
|
||||||
While the following guidelines describe the use of each type and their
|
For more about NgModules, see [Organizing your app with NgModules](guide/ngmodules "Organizing your app with NgModules").
|
||||||
typical characteristics, in real world apps, you may see hybrids.
|
|
||||||
|
|
||||||
<table>
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
<tr>
|
For the example app used in NgModules-related topics, see the <live-example name="ngmodules"></live-example>.
|
||||||
<th style="vertical-align: top">
|
|
||||||
Feature Module
|
|
||||||
</th>
|
|
||||||
|
|
||||||
<th style="vertical-align: top">
|
</div>
|
||||||
Guidelines
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
|
|
||||||
<tr>
|
## Summary of NgModule categories
|
||||||
<td>Domain</td>
|
|
||||||
<td>
|
|
||||||
Domain feature modules deliver a user experience dedicated to a particular application domain like editing a customer or placing an order.
|
|
||||||
|
|
||||||
They typically have a top component that acts as the feature root and private, supporting sub-components descend from it.
|
All apps start by [bootstrapping a root NgModule](guide/bootstrapping "Launching an app with a root NgModule").
|
||||||
|
You can organize your other NgModules any way you wish.
|
||||||
|
|
||||||
Domain feature modules consist mostly of declarations. Only the top component is exported.
|
This topic provides some guidelines for the following general categories of NgModules:
|
||||||
|
|
||||||
Domain feature modules rarely have providers. When they do, the lifetime of the provided services should be the same as the lifetime of the module.
|
* [Domain](#domain): A domain NgModule is organized around a feature, business domain, or user experience.
|
||||||
|
* [Routed](#routed): The top component of the NgModule acts as the destination of a [router](guide/glossary#router "Definition of router") navigation route.
|
||||||
|
* [Routing](#routing): A routing NgModule provides the routing configuration for another NgModule.
|
||||||
|
* [Service](#service): A service NgModule provides utility services such as data access and messaging.
|
||||||
|
* [Widget](#widget): A widget NgModule makes a component, directive, or pipe available to other NgModules.
|
||||||
|
* [Shared](#shared): A shared NgModule makes a set of components, directives, and pipes available to other NgModules.
|
||||||
|
|
||||||
Domain feature modules are typically imported exactly once by a larger feature module.
|
The following table summarizes the key characteristics of each category.
|
||||||
|
|
||||||
They might be imported by the root `AppModule` of a small application that lacks routing.
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>Routed</td>
|
|
||||||
<td>
|
|
||||||
Routed feature modules are domain feature modules whose top components are the targets of router navigation routes.
|
|
||||||
|
|
||||||
All lazy-loaded modules are routed feature modules by definition.
|
|
||||||
|
|
||||||
Routed feature modules don’t export anything because their components never appear in the template of an external component.
|
|
||||||
|
|
||||||
A lazy-loaded routed feature module should not be imported by any module. Doing so would trigger an eager load, defeating the purpose of lazy loading.That means you won’t see them mentioned among the `AppModule` imports. An eager loaded routed feature module must be imported by another module so that the compiler learns about its components.
|
|
||||||
|
|
||||||
Routed feature modules rarely have providers for reasons explained in [Lazy Loading Feature Modules](/guide/lazy-loading-ngmodules). When they do, the lifetime of the provided services should be the same as the lifetime of the module. Don't provide application-wide singleton services in a routed feature module or in a module that the routed module imports.
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
|
|
||||||
<tr>
|
|
||||||
<td>Routing</td>
|
|
||||||
<td>
|
|
||||||
|
|
||||||
A routing module provides routing configuration for another module and separates routing concerns from its companion module.
|
|
||||||
|
|
||||||
A routing module typically does the following:
|
|
||||||
|
|
||||||
<ul>
|
|
||||||
<li>Defines routes.</li>
|
|
||||||
<li>Adds router configuration to the module's imports.</li>
|
|
||||||
<li>Adds guard and resolver service providers to the module's providers.</li>
|
|
||||||
<li>The name of the routing module should parallel the name of its companion module, using the suffix "Routing". For example, <code>FooModule</code> in <code>foo.module.ts</code> has a routing module named <code>FooRoutingModule</code> in <code>foo-routing.module.ts</code>. If the companion module is the root <code>AppModule</code>, the <code>AppRoutingModule</code> adds router configuration to its imports with <code>RouterModule.forRoot(routes)</code>. All other routing modules are children that import <code>RouterModule.forChild(routes)</code>.</li>
|
|
||||||
<li>A routing module re-exports the <code>RouterModule</code> as a convenience so that components of the companion module have access to router directives such as <code>RouterLink</code> and <code>RouterOutlet</code>.</li>
|
|
||||||
<li>A routing module does not have its own declarations. Components, directives, and pipes are the responsibility of the feature module, not the routing module.</li>
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
A routing module should only be imported by its companion module.
|
|
||||||
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
|
|
||||||
<tr>
|
|
||||||
<td>Service</td>
|
|
||||||
<td>
|
|
||||||
|
|
||||||
Service modules provide utility services such as data access and messaging. Ideally, they consist entirely of providers and have no declarations. Angular's `HttpClientModule` is a good example of a service module.
|
|
||||||
|
|
||||||
The root `AppModule` is the only module that should import service modules.
|
|
||||||
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
|
|
||||||
<tr>
|
|
||||||
<td>Widget</td>
|
|
||||||
<td>
|
|
||||||
|
|
||||||
A widget module makes components, directives, and pipes available to external modules. Many third-party UI component libraries are widget modules.
|
|
||||||
|
|
||||||
A widget module should consist entirely of declarations, most of them exported.
|
|
||||||
|
|
||||||
A widget module should rarely have providers.
|
|
||||||
|
|
||||||
Import widget modules in any module whose component templates need the widgets.
|
|
||||||
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
|
|
||||||
</table>
|
|
||||||
|
|
||||||
The following table summarizes the key characteristics of each feature module group.
|
|
||||||
|
|
||||||
<table>
|
<table>
|
||||||
<tr>
|
<tr>
|
||||||
<th style="vertical-align: top">
|
<th style="vertical-align: top">
|
||||||
Feature Module
|
NgModule
|
||||||
</th>
|
</th>
|
||||||
|
|
||||||
<th style="vertical-align: top">
|
<th style="vertical-align: top">
|
||||||
@ -135,7 +60,7 @@ The following table summarizes the key characteristics of each feature module gr
|
|||||||
<td>Yes</td>
|
<td>Yes</td>
|
||||||
<td>Rare</td>
|
<td>Rare</td>
|
||||||
<td>Top component</td>
|
<td>Top component</td>
|
||||||
<td>Feature, AppModule</td>
|
<td>Another domain, AppModule</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
@ -151,7 +76,7 @@ The following table summarizes the key characteristics of each feature module gr
|
|||||||
<td>No</td>
|
<td>No</td>
|
||||||
<td>Yes (Guards)</td>
|
<td>Yes (Guards)</td>
|
||||||
<td>RouterModule</td>
|
<td>RouterModule</td>
|
||||||
<td>Feature (for routing)</td>
|
<td>Another domain (for routing)</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
@ -167,14 +92,137 @@ The following table summarizes the key characteristics of each feature module gr
|
|||||||
<td>Yes</td>
|
<td>Yes</td>
|
||||||
<td>Rare</td>
|
<td>Rare</td>
|
||||||
<td>Yes</td>
|
<td>Yes</td>
|
||||||
<td>Feature</td>
|
<td>Another domain</td>
|
||||||
|
</tr>
|
||||||
|
|
||||||
|
<tr>
|
||||||
|
<td>Shared</td>
|
||||||
|
<td>Yes</td>
|
||||||
|
<td>No</td>
|
||||||
|
<td>Yes</td>
|
||||||
|
<td>Another domain</td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
<hr />
|
{@a domain}
|
||||||
|
|
||||||
## More on NgModules
|
## Domain NgModules
|
||||||
|
|
||||||
|
Use a domain NgModule to deliver a user experience dedicated to a particular feature or app domain, such as editing a customer or placing an order.
|
||||||
|
One example is `ContactModule` in the <live-example name="ngmodules"></live-example>.
|
||||||
|
|
||||||
|
A domain NgModule organizes the code related to a certain function, containing all of the components, routing, and templates that make up the function.
|
||||||
|
Your top component in the domain NgModule acts as the feature or domain's root, and is the only component you export.
|
||||||
|
Private supporting subcomponents descend from it.
|
||||||
|
|
||||||
|
Import a domain NgModule exactly once into another NgModule, such as a domain NgModule, or into the root NgModule (`AppModule`) of an app that contains only a few NgModules.
|
||||||
|
|
||||||
|
Domain NgModules consist mostly of declarations.
|
||||||
|
You rarely include providers.
|
||||||
|
If you do, the lifetime of the provided services should be the same as the lifetime of the NgModule.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
For more information about lifecycles, see [Hooking into the component lifecycle](guide/lifecycle-hooks "Hooking into the component lifecycle").
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{@a routed}
|
||||||
|
|
||||||
|
## Routed NgModules
|
||||||
|
|
||||||
|
Use a routed NgModule for all [lazy-loaded NgModules](guide/lazy-loading-ngmodules "Lazy-loading an NgModule").
|
||||||
|
Use the top component of the NgModule as the destination of a router navigation route.
|
||||||
|
Routed NgModules don’t export anything because their components never appear in the template of an external component.
|
||||||
|
|
||||||
|
Don't import a lazy-loaded routed NgModule into another NgModule, as this would trigger an eager load, defeating the purpose of lazy loading.
|
||||||
|
|
||||||
|
Routed NgModules rarely have providers because you load a routed NgModule only when needed (such as for routing).
|
||||||
|
Services listed in the NgModules' `provider` array would not be available because the root injector wouldn’t know about the lazy-loaded NgModule.
|
||||||
|
If you include providers, the lifetime of the provided services should be the same as the lifetime of the NgModule.
|
||||||
|
Don't provide app-wide [singleton services](guide/singleton-services) in a routed NgModule or in an NgModule that the routed NgModule imports.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
For more information about providers and lazy-loaded routed NgModules, see [Limiting provider scope](guide/providers#limiting-provider-scope-by-lazy-loading-modules "Providing dependencies: Limiting provider scope").
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{@a routing}
|
||||||
|
|
||||||
|
## Routing NgModules
|
||||||
|
|
||||||
|
Use a routing NgModule to provide the routing configuration for a domain NgModule, thereby separating routing concerns from its companion domain NgModule.
|
||||||
|
One example is `ContactRoutingModule` in the <live-example name="ngmodules"></live-example>, which provides the routing for its companion domain NgModule `ContactModule`.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
For an overview and details about routing, see [In-app navigation: routing to views](guide/router "In-app navigation: routing to views").
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
Use a routing NgModule to do the following tasks:
|
||||||
|
|
||||||
|
* Define routes.
|
||||||
|
* Add router configuration to the NgModule's import.
|
||||||
|
* Add guard and resolver service providers to the NgModule's providers.
|
||||||
|
|
||||||
|
The name of the routing NgModule should parallel the name of its companion NgModule, using the suffix `Routing`.
|
||||||
|
For example, <code>ContactModule</code> in <code>contact.module.ts</code> has a routing NgModule named <code>ContactRoutingModule</code> in <code>contact-routing.module.ts</code>.
|
||||||
|
|
||||||
|
Import a routing NgModule only into its companion NgModule.
|
||||||
|
If the companion NgModule is the root <code>AppModule</code>, the <code>AppRoutingModule</code> adds router configuration to its imports with <code>RouterModule.forRoot(routes)</code>.
|
||||||
|
All other routing NgModules are children that import <code>RouterModule.forChild(routes)</code>.
|
||||||
|
|
||||||
|
In your routing NgModule, re-export the <code>RouterModule</code> as a convenience so that components of the companion NgModule have access to router directives such as <code>RouterLink</code> and <code>RouterOutlet</code>.
|
||||||
|
|
||||||
|
Don't use declarations in a routing NgModule.
|
||||||
|
Components, directives, and pipes are the responsibility of the companion domain NgModule, not the routing NgModule.
|
||||||
|
|
||||||
|
{@a service}
|
||||||
|
|
||||||
|
## Service NgModules
|
||||||
|
|
||||||
|
Use a service NgModule to provide a utility service such as data access or messaging.
|
||||||
|
Ideal service NgModules consist entirely of providers and have no declarations.
|
||||||
|
Angular's `HttpClientModule` is a good example of a service NgModule.
|
||||||
|
|
||||||
|
Use only the root `AppModule` to import service NgModules.
|
||||||
|
|
||||||
|
{@a widget}
|
||||||
|
|
||||||
|
## Widget NgModules
|
||||||
|
|
||||||
|
Use a widget NgModule to make a component, directive, or pipe available to external NgModules.
|
||||||
|
Import widget NgModules into any NgModules that need the widgets in their templates.
|
||||||
|
Many third-party UI component libraries are provided as widget NgModules.
|
||||||
|
|
||||||
|
A widget NgModule should consist entirely of declarations, most of them exported.
|
||||||
|
It would rarely have providers.
|
||||||
|
|
||||||
|
{@a shared}
|
||||||
|
|
||||||
|
## Shared NgModules
|
||||||
|
|
||||||
|
Put commonly used directives, pipes, and components into one NgModule, typically named `SharedModule`, and then import just that NgModule wherever you need it in other parts of your app.
|
||||||
|
You can import the shared NgModule in your domain NgModules, including [lazy-loaded NgModules](guide/lazy-loading-ngmodules "Lazy-loading an NgModule").
|
||||||
|
One example is `SharedModule` in the <live-example name="ngmodules"></live-example>, which provides the `AwesomePipe` custom pipe and `HighlightDirective` directive.
|
||||||
|
|
||||||
|
Shared NgModules should not include providers, nor should any of its imported or re-exported NgModules include providers.
|
||||||
|
|
||||||
|
To learn how to use shared modules to organize and streamline your code, see [Sharing NgModules in an app](guide/sharing-ngmodules "Sharing NgModules in an app").
|
||||||
|
|
||||||
|
## Next steps
|
||||||
|
|
||||||
You may also be interested in the following:
|
You may also be interested in the following:
|
||||||
* [Lazy Loading Modules with the Angular Router](guide/lazy-loading-ngmodules).
|
|
||||||
* [Providers](guide/providers).
|
* For more about NgModules, see [Organizing your app with NgModules](guide/ngmodules "Organizing your app with NgModules").
|
||||||
|
* To learn more about the root NgModule, see [Launching an app with a root NgModule](guide/bootstrapping "Launching an app with a root NgModule").
|
||||||
|
* To learn about frequently used Angular NgModules and how to import them into your app, see [Frequently-used modules](guide/frequent-ngmodules "Frequently-used modules").
|
||||||
|
* For a complete description of the NgModule metadata properties, see [Using the NgModule metadata](guide/ngmodule-api "Using the NgModule metadata").
|
||||||
|
|
||||||
|
If you want to manage NgModule loading and the use of dependencies and services, see the following:
|
||||||
|
|
||||||
|
* To learn about loading NgModules eagerly when the app starts, or lazy-loading NgModules asynchronously by the router, see [Lazy-loading feature modules](guide/lazy-loading-ngmodules).
|
||||||
|
* To understand how to provide a service or other dependency for your app, see [Providing Dependencies for an NgModule](guide/providers "Providing Dependencies for an NgModule").
|
||||||
|
* To learn how to create a singleton service to use in NgModules, see [Making a service a singleton](guide/singleton-services "Making a service a singleton").
|
||||||
|
@ -1,72 +1,82 @@
|
|||||||
# JavaScript modules vs. NgModules
|
# JavaScript modules vs. NgModules
|
||||||
|
|
||||||
JavaScript and Angular use modules to organize code, and
|
JavaScript modules and NgModules can help you modularize your code, but they are very different.
|
||||||
though they organize it differently, Angular apps rely on both.
|
Angular apps rely on both kinds of modules.
|
||||||
|
|
||||||
|
## JavaScript modules: Files containing code
|
||||||
|
|
||||||
## JavaScript modules
|
A [JavaScript module](https://javascript.info/modules "JavaScript.Info - Modules") is an individual file with JavaScript code, usually containing a class or a library of functions for a specific purpose within your app.
|
||||||
|
JavaScript modules let you spread your work across multiple files.
|
||||||
|
|
||||||
In JavaScript, modules are individual files with JavaScript code in them. To make what’s in them available, you write an export statement, usually after the relevant code, like this:
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
To learn more about JavaScript modules, see [ES6 In Depth: Modules](https://hacks.mozilla.org/2015/08/es6-in-depth-modules/).
|
||||||
|
For the module specification, see the [6th Edition of the ECMAScript standard](http://www.ecma-international.org/ecma-262/6.0/#sec-modules).
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
To make the code in a JavaScript module available to other modules, use an `export` statement at the end of the relevant code in the module, such as the following:
|
||||||
|
|
||||||
```typescript
|
```typescript
|
||||||
export class AppComponent { ... }
|
export class AppComponent { ... }
|
||||||
```
|
```
|
||||||
|
|
||||||
Then, when you need that file’s code in another file, you import it like this:
|
When you need that module’s code in another module, use an `import` statement as follows:
|
||||||
|
|
||||||
```typescript
|
```typescript
|
||||||
import { AppComponent } from './app.component';
|
import { AppComponent } from './app.component';
|
||||||
```
|
```
|
||||||
|
|
||||||
JavaScript modules help you namespace, preventing accidental global variables.
|
Each module has its own top-level scope.
|
||||||
|
In other words, top-level variables and functions in a module are not seen in other scripts or modules.
|
||||||
|
Each module provides a namespace for identifiers to prevent them from clashing with identifiers in other modules.
|
||||||
|
With multiple modules, you can prevent accidental global variables by creating a single global namespace and adding sub-modules to it.
|
||||||
|
|
||||||
For more information on JavaScript modules, see [JavaScript/ECMAScript modules](https://hacks.mozilla.org/2015/08/es6-in-depth-modules/).
|
The Angular framework itself is loaded as a set of JavaScript modules.
|
||||||
|
|
||||||
## NgModules
|
## NgModules: Classes with metadata for compiling
|
||||||
|
|
||||||
<!-- KW-- perMisko: let's discuss. This does not answer the question why it is different. Also, last sentence is confusing.-->
|
An [NgModule](guide/glossary#ngmodule "Definition of NgModule") is a class marked by the `@NgModule` decorator with a metadata object that describes how that particular part of the app fits together with the other parts.
|
||||||
NgModules are classes decorated with `@NgModule`. The `@NgModule` decorator’s `imports` array tells Angular what other NgModules the current module needs. The modules in the `imports` array are different than JavaScript modules because they are NgModules rather than regular JavaScript modules. Classes with an `@NgModule` decorator are by convention kept in their own files, but what makes them an `NgModule` isn’t being in their own file, like JavaScript modules; it’s the presence of `@NgModule` and its metadata.
|
NgModules are specific to Angular.
|
||||||
|
While classes with an `@NgModule` decorator are by convention kept in their own files, they differ from JavaScript modules because they include this metadata.
|
||||||
|
|
||||||
The `AppModule` generated from the [Angular CLI](cli) demonstrates both kinds of modules in action:
|
The `@NgModule` metadata plays an important role in guiding the Angular compilation process that converts the app code you write into highly performant JavaScript code.
|
||||||
|
The metadata describes how to compile a component's template and how to create an [injector](guide/glossary#injector "Definition of injector") at runtime.
|
||||||
|
It identifies the NgModule's [components](guide/glossary#component "Definition of component"), [directives](guide/glossary#directive "Definition of directive"), and [pipes](guide/glossary#pipe "Definition of pipe)"),
|
||||||
|
and makes some of them public through the `exports` property so that external components can use them.
|
||||||
|
You can also use an NgModule to add [providers](guide/glossary#provider "Definition of provider") for [services](guide/glossary#service "Definition of a service"), so that the services are available elsewhere in your app.
|
||||||
|
|
||||||
```typescript
|
Rather than defining all member classes in one giant file as a JavaScript module, declare which components, directives, and pipes belong to the NgModule in the `@NgModule.declarations` list.
|
||||||
/* These are JavaScript import statements. Angular doesn’t know anything about these. */
|
These classes are called [declarables](guide/glossary#declarable "Definition of a declarable").
|
||||||
import { BrowserModule } from '@angular/platform-browser';
|
An NgModule can export only the declarable classes it owns or imports from other NgModules.
|
||||||
import { NgModule } from '@angular/core';
|
It doesn't declare or export any other kind of class.
|
||||||
|
Declarables are the only classes that matter to the Angular compilation process.
|
||||||
|
|
||||||
import { AppComponent } from './app.component';
|
For a complete description of the NgModule metadata properties, see [Using the NgModule metadata](guide/ngmodule-api "Using the NgModule metadata").
|
||||||
|
|
||||||
/* The @NgModule decorator lets Angular know that this is an NgModule. */
|
## An example that uses both
|
||||||
@NgModule({
|
|
||||||
declarations: [
|
|
||||||
AppComponent
|
|
||||||
],
|
|
||||||
imports: [ /* These are NgModule imports. */
|
|
||||||
BrowserModule
|
|
||||||
],
|
|
||||||
providers: [],
|
|
||||||
bootstrap: [AppComponent]
|
|
||||||
})
|
|
||||||
export class AppModule { }
|
|
||||||
```
|
|
||||||
|
|
||||||
|
The root NgModule `AppModule` generated by the [Angular CLI](cli) for a new app project demonstrates how you use both kinds of modules:
|
||||||
|
|
||||||
The NgModule classes differ from JavaScript module in the following key ways:
|
<code-example path="ngmodules/src/app/app.module.1.ts" header="src/app/app.module.ts (default AppModule)"></code-example>
|
||||||
|
|
||||||
* An NgModule bounds [declarable classes](guide/ngmodule-faq#q-declarable) only.
|
The root NgModule starts with `import` statements to import JavaScript modules.
|
||||||
Declarables are the only classes that matter to the [Angular compiler](guide/ngmodule-faq#q-angular-compiler).
|
It then configures the `@NgModule` with the following arrays:
|
||||||
* Instead of defining all member classes in one giant file as in a JavaScript module,
|
|
||||||
you list the module's classes in the `@NgModule.declarations` list.
|
|
||||||
* An NgModule can only export the [declarable classes](guide/ngmodule-faq#q-declarable)
|
|
||||||
it owns or imports from other modules. It doesn't declare or export any other kind of class.
|
|
||||||
* Unlike JavaScript modules, an NgModule can extend the _entire_ application with services
|
|
||||||
by adding providers to the `@NgModule.providers` list.
|
|
||||||
|
|
||||||
<hr />
|
* `declarations`: The components, directives, and pipes that belong to the NgModule.
|
||||||
|
A new app project's root NgModule has only one component, called `AppComponent`.
|
||||||
|
|
||||||
## More on NgModules
|
* `imports`: Other NgModules you are using, so that you can use their declarables.
|
||||||
|
The newly generated root NgModule imports [`BrowserModule`](api/platform-browser/BrowserModule "BrowserModule NgModule") in order to use browser-specific services such as [DOM](https://www.w3.org/TR/DOM-Level-2-Core/introduction.html "Definition of Document Object Model") rendering, sanitization, and location.
|
||||||
|
|
||||||
For more information on NgModules, see:
|
* `providers`: Providers of services that components in other NgModules can use.
|
||||||
* [Bootstrapping](guide/bootstrapping).
|
There are no providers in a newly generated root NgModule.
|
||||||
* [Frequently used modules](guide/frequent-ngmodules).
|
|
||||||
* [Providers](guide/providers).
|
* `bootstrap`: The [entry component](guide/entry-components "Specifying an entry component") that Angular creates and inserts into the `index.html` host web page, thereby bootstrapping the app.
|
||||||
|
This entry component, `AppComponent`, appears in both the `declarations` and the `bootstrap` arrays.
|
||||||
|
|
||||||
|
## Next steps
|
||||||
|
|
||||||
|
* For more about NgModules, see [Organizing your app with NgModules](guide/ngmodules "Organizing your app with NgModules").
|
||||||
|
* To learn more about the root NgModule, see [Launching an app with a root NgModule](guide/bootstrapping "Launching an app with a root NgModule").
|
||||||
|
* To learn about frequently used Angular NgModules and how to import them into your app, see [Frequently-used modules](guide/frequent-ngmodules "Frequently-used modules").
|
||||||
|
@ -118,7 +118,6 @@ Package name | Description
|
|||||||
[**@angular‑devkit/<br />build‑angular**](https://github.com/angular/angular-cli/) | The Angular build tools.
|
[**@angular‑devkit/<br />build‑angular**](https://github.com/angular/angular-cli/) | The Angular build tools.
|
||||||
[**@angular/cli**](https://github.com/angular/angular-cli/) | The Angular CLI tools.
|
[**@angular/cli**](https://github.com/angular/angular-cli/) | The Angular CLI tools.
|
||||||
**@angular/<br />compiler‑cli** | The Angular compiler, which is invoked by the Angular CLI's `ng build` and `ng serve` commands.
|
**@angular/<br />compiler‑cli** | The Angular compiler, which is invoked by the Angular CLI's `ng build` and `ng serve` commands.
|
||||||
**@angular/<br />language‑service** | The [Angular language service](guide/language-service) analyzes component templates and provides type and error information that TypeScript-aware editors can use to improve the developer's experience. For example, see the [Angular language service extension for VS Code](https://marketplace.visualstudio.com/items?itemName=Angular.ng-template).
|
|
||||||
**@types/... ** | TypeScript definition files for 3rd party libraries such as Jasmine and Node.js.
|
**@types/... ** | TypeScript definition files for 3rd party libraries such as Jasmine and Node.js.
|
||||||
[**codelyzer**](https://www.npmjs.com/package/codelyzer) | A linter for Angular apps whose rules conform to the Angular [style guide](guide/styleguide).
|
[**codelyzer**](https://www.npmjs.com/package/codelyzer) | A linter for Angular apps whose rules conform to the Angular [style guide](guide/styleguide).
|
||||||
**jasmine/... ** | Packages to support the [Jasmine](https://jasmine.github.io/) test library.
|
**jasmine/... ** | Packages to support the [Jasmine](https://jasmine.github.io/) test library.
|
||||||
@ -135,3 +134,4 @@ Package name | Description
|
|||||||
|
|
||||||
* [Building and serving](guide/build) describes how packages come together to create a development build.
|
* [Building and serving](guide/build) describes how packages come together to create a development build.
|
||||||
* [Deployment](guide/deployment) describes how packages come together to create a production build.
|
* [Deployment](guide/deployment) describes how packages come together to create a production build.
|
||||||
|
|
@ -2,13 +2,13 @@
|
|||||||
|
|
||||||
Angular makes use of observables as an interface to handle a variety of common asynchronous operations. For example:
|
Angular makes use of observables as an interface to handle a variety of common asynchronous operations. For example:
|
||||||
|
|
||||||
* You can define [custom events](guide/template-syntax#custom-events-with-eventemitter) that send observable output data from a child to a parent component.
|
* You can define [custom events](guide/event-binding#custom-events-with-eventemitter) that send observable output data from a child to a parent component.
|
||||||
* The HTTP module uses observables to handle AJAX requests and responses.
|
* The HTTP module uses observables to handle AJAX requests and responses.
|
||||||
* The Router and Forms modules use observables to listen for and respond to user-input events.
|
* The Router and Forms modules use observables to listen for and respond to user-input events.
|
||||||
|
|
||||||
## Transmitting data between components
|
## Transmitting data between components
|
||||||
|
|
||||||
Angular provides an `EventEmitter` class that is used when publishing values from a component through the [`@Output()` decorator](guide/template-syntax#how-to-use-output).
|
Angular provides an `EventEmitter` class that is used when publishing values from a component through the [`@Output()` decorator](guide/inputs-outputs#how-to-use-output).
|
||||||
`EventEmitter` extends [RxJS `Subject`](https://rxjs.dev/api/index/class/Subject), adding an `emit()` method so it can send arbitrary values.
|
`EventEmitter` extends [RxJS `Subject`](https://rxjs.dev/api/index/class/Subject), adding an `emit()` method so it can send arbitrary values.
|
||||||
When you call `emit()`, it passes the emitted value to the `next()` method of any subscribed observer.
|
When you call `emit()`, it passes the emitted value to the `next()` method of any subscribed observer.
|
||||||
|
|
||||||
|
@ -112,7 +112,7 @@ Because observables produce values asynchronously, try/catch will not effectivel
|
|||||||
<code-example>
|
<code-example>
|
||||||
myObservable.subscribe({
|
myObservable.subscribe({
|
||||||
next(num) { console.log('Next num: ' + num)},
|
next(num) { console.log('Next num: ' + num)},
|
||||||
error(err) { console.log('Received an errror: ' + err)}
|
error(err) { console.log('Received an error: ' + err)}
|
||||||
});
|
});
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
|
@ -1,607 +1,440 @@
|
|||||||
# Pipes
|
# Transforming Data Using Pipes
|
||||||
|
|
||||||
Every application starts out with what seems like a simple task: get data, transform them, and show them to users.
|
Use [pipes](guide/glossary#pipe "Definition of a pipe") to transform strings, currency amounts, dates, and other data for display.
|
||||||
Getting data could be as simple as creating a local variable or as complex as streaming data over a WebSocket.
|
Pipes are simple functions you can use in [template expressions](/guide/glossary#template-expression "Definition of template expression") to accept an input value and return a transformed value. Pipes are useful because you can use them throughout your application, while only declaring each pipe once.
|
||||||
|
For example, you would use a pipe to show a date as **April 15, 1988** rather than the raw string format.
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
For the sample app that this page describes, see the <live-example></live-example>.
|
For the sample app used in this topic, see the <live-example></live-example>.
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
Once data arrives, you could push their raw `toString` values directly to the view,
|
Angular provides built-in pipes for typical data transformations, including transformations for internationalization (i18n), which use locale information to format data.
|
||||||
but that rarely makes for a good user experience.
|
The following are commonly used built-in pipes for data formatting:
|
||||||
For example, in most use cases, users prefer to see a date in a simple format like
|
|
||||||
<samp>April 15, 1988</samp> rather than the raw string format
|
|
||||||
<samp>Fri Apr 15 1988 00:00:00 GMT-0700 (Pacific Daylight Time)</samp>.
|
|
||||||
|
|
||||||
Clearly, some values benefit from a bit of editing. You may notice that you
|
|
||||||
desire many of the same transformations repeatedly, both within and across many applications.
|
|
||||||
You can almost think of them as styles.
|
|
||||||
In fact, you might like to apply them in your HTML templates as you do styles.
|
|
||||||
|
|
||||||
Introducing Angular pipes, a way to write display-value transformations that you can declare in your HTML.
|
|
||||||
|
|
||||||
|
|
||||||
## Using pipes
|
|
||||||
|
|
||||||
A pipe takes in data as input and transforms it to a desired output.
|
|
||||||
In this page, you'll use pipes to transform a component's birthday property into
|
|
||||||
a human-friendly date.
|
|
||||||
|
|
||||||
|
|
||||||
<code-example path="pipes/src/app/hero-birthday1.component.ts" header="src/app/hero-birthday1.component.ts"></code-example>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Focus on the component's template.
|
|
||||||
|
|
||||||
|
|
||||||
<code-example path="pipes/src/app/app.component.html" region="hero-birthday-template" header="src/app/app.component.html"></code-example>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Inside the interpolation expression, you flow the component's `birthday` value through the
|
|
||||||
[pipe operator](guide/template-syntax#pipe) ( | ) to the [Date pipe](api/common/DatePipe)
|
|
||||||
function on the right. All pipes work this way.
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## Built-in pipes
|
|
||||||
Angular comes with a stock of pipes such as
|
|
||||||
`DatePipe`, `UpperCasePipe`, `LowerCasePipe`, `CurrencyPipe`, and `PercentPipe`.
|
|
||||||
They are all available for use in any template.
|
|
||||||
|
|
||||||
|
* [`DatePipe`](api/common/DatePipe): Formats a date value according to locale rules.
|
||||||
|
* [`UpperCasePipe`](api/common/UpperCasePipe): Transforms text to all upper case.
|
||||||
|
* [`LowerCasePipe`](api/common/LowerCasePipe): Transforms text to all lower case.
|
||||||
|
* [`CurrencyPipe`](api/common/CurrencyPipe): Transforms a number to a currency string, formatted according to locale rules.
|
||||||
|
* [`DecimalPipe`](/api/common/DecimalPipe): Transforms a number into a string with a decimal point, formatted according to locale rules.
|
||||||
|
* [`PercentPipe`](api/common/PercentPipe): Transforms a number to a percentage string, formatted according to locale rules.
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
* For a complete list of built-in pipes, see the [pipes API documentation](/api/common#pipes "Pipes API reference summary").
|
||||||
|
* To learn more about using pipes for internationalization (i18n) efforts, see [formatting data based on locale](/guide/i18n#i18n-pipes "Formatting data based on locale").
|
||||||
Read more about these and many other built-in pipes in the [pipes topics](api?type=pipe) of the
|
|
||||||
[API Reference](api); filter for entries that include the word "pipe".
|
|
||||||
|
|
||||||
Angular doesn't have a `FilterPipe` or an `OrderByPipe` for reasons explained in the [Appendix](guide/pipes#no-filter-pipe) of this page.
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
You can also create pipes to encapsulate custom transformations and use your custom pipes in template expressions.
|
||||||
|
|
||||||
|
## Prerequisites
|
||||||
|
|
||||||
|
To use pipes you should have a basic understanding of the following:
|
||||||
|
|
||||||
## Parameterizing a pipe
|
* [Typescript](guide/glossary#typescript "Definition of Typescript") and HTML5 programming
|
||||||
|
* [Templates](guide/glossary#template "Definition of a template") in HTML with CSS styles
|
||||||
|
* [Components](guide/glossary#component "Definition of a component")
|
||||||
|
|
||||||
A pipe can accept any number of optional parameters to fine-tune its output.
|
## Using a pipe in a template
|
||||||
To add parameters to a pipe, follow the pipe name with a colon ( : ) and then the parameter value
|
|
||||||
(such as `currency:'EUR'`). If the pipe accepts multiple parameters, separate the values with colons (such as `slice:1:5`)
|
|
||||||
|
|
||||||
Modify the birthday template to give the date pipe a format parameter.
|
To apply a pipe, use the pipe operator (`|`) within a template expression as shown in the following code example, along with the *name* of the pipe, which is `date` for the built-in [`DatePipe`](api/common/DatePipe).
|
||||||
After formatting the hero's April 15th birthday, it renders as **<samp>04/15/88</samp>**:
|
The tabs in the example show the following:
|
||||||
|
|
||||||
|
* `app.component.html` uses `date` in a separate template to display a birthday.
|
||||||
|
* `hero-birthday1.component.ts` uses the same pipe as part of an in-line template in a component that also sets the birthday value.
|
||||||
|
|
||||||
<code-example path="pipes/src/app/app.component.html" region="format-birthday" header="src/app/app.component.html"></code-example>
|
<code-tabs>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/app.component.html"
|
||||||
|
region="hero-birthday-template"
|
||||||
|
path="pipes/src/app/app.component.html">
|
||||||
|
</code-pane>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/hero-birthday1.component.ts"
|
||||||
|
path="pipes/src/app/hero-birthday1.component.ts">
|
||||||
|
</code-pane>
|
||||||
|
</code-tabs>
|
||||||
|
|
||||||
|
The component's `birthday` value flows through the
|
||||||
|
[pipe operator](guide/template-expression-operators#pipe) ( | ) to the [`date`](api/common/DatePipe)
|
||||||
|
function.
|
||||||
|
|
||||||
|
{@a parameterizing-a-pipe}
|
||||||
|
|
||||||
The parameter value can be any valid template expression,
|
## Transforming data with parameters and chained pipes
|
||||||
(see the [Template expressions](guide/template-syntax#template-expressions) section of the
|
|
||||||
[Template Syntax](guide/template-syntax) page)
|
|
||||||
such as a string literal or a component property.
|
|
||||||
In other words, you can control the format through a binding the same way you control the birthday value through a binding.
|
|
||||||
|
|
||||||
Write a second component that *binds* the pipe's format parameter
|
Use optional parameters to fine-tune a pipe's output.
|
||||||
to the component's `format` property. Here's the template for that component:
|
For example, you can use the [`CurrencyPipe`](api/common/CurrencyPipe "API reference") with a country code such as EUR as a parameter.
|
||||||
|
The template expression `{{ amount | currency:'EUR' }}` transforms the `amount` to currency in euros.
|
||||||
|
Follow the pipe name (`currency`) with a colon (`:`) and the parameter value (`'EUR'`).
|
||||||
|
|
||||||
|
If the pipe accepts multiple parameters, separate the values with colons.
|
||||||
|
For example, `{{ amount | currency:'EUR':'Euros '}}` adds the second parameter, the string literal `'Euros '`, to the output string. You can use any valid template expression as a parameter, such as a string literal or a component property.
|
||||||
|
|
||||||
<code-example path="pipes/src/app/hero-birthday2.component.ts" region="template" header="src/app/hero-birthday2.component.ts (template)"></code-example>
|
Some pipes require at least one parameter and allow more optional parameters, such as [`SlicePipe`](/api/common/SlicePipe "API reference for SlicePipe"). For example, `{{ slice:1:5 }}` creates a new array or string containing a subset of the elements starting with element `1` and ending with element `5`.
|
||||||
|
|
||||||
|
### Example: Formatting a date
|
||||||
|
|
||||||
|
The tabs in the following example demonstrates toggling between two different formats (`'shortDate'` and `'fullDate'`):
|
||||||
|
|
||||||
You also added a button to the template and bound its click event to the component's `toggleFormat()` method.
|
* The `app.component.html` template uses a format parameter for the [`DatePipe`](api/common/DatePipe) (named `date`) to show the date as **04/15/88**.
|
||||||
That method toggles the component's `format` property between a short form
|
* The `hero-birthday2.component.ts` component binds the pipe's format parameter to the component's `format` property in the `template` section, and adds a button for a click event bound to the component's `toggleFormat()` method.
|
||||||
|
* The `hero-birthday2.component.ts` component's `toggleFormat()` method toggles the component's `format` property between a short form
|
||||||
(`'shortDate'`) and a longer form (`'fullDate'`).
|
(`'shortDate'`) and a longer form (`'fullDate'`).
|
||||||
|
|
||||||
|
<code-tabs>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/app.component.html"
|
||||||
|
region="format-birthday"
|
||||||
|
path="pipes/src/app/app.component.html">
|
||||||
|
</code-pane>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/hero-birthday2.component.ts (template)"
|
||||||
|
region="template"
|
||||||
|
path="pipes/src/app/hero-birthday2.component.ts">
|
||||||
|
</code-pane>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/hero-birthday2.component.ts (class)"
|
||||||
|
region="class"
|
||||||
|
path="pipes/src/app/hero-birthday2.component.ts">
|
||||||
|
</code-pane>
|
||||||
|
</code-tabs>
|
||||||
|
|
||||||
<code-example path="pipes/src/app/hero-birthday2.component.ts" region="class" header="src/app/hero-birthday2.component.ts (class)"></code-example>
|
Clicking the **Toggle Format** button alternates the date format between **04/15/1988** and **Friday, April 15, 1988** as shown in Figure 1.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
As you click the button, the displayed date alternates between
|
|
||||||
"**<samp>04/15/1988</samp>**" and
|
|
||||||
"**<samp>Friday, April 15, 1988</samp>**".
|
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src='generated/images/guide/pipes/date-format-toggle-anim.gif' alt="Date Format Toggle">
|
<img src='generated/images/guide/pipes/date-format-toggle-anim.gif' alt="Date Format Toggle">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
**Figure 1.** Clicking the button toggles the date format
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
For `date` pipe format options, see [DatePipe](api/common/DatePipe "DatePipe API Reference page").
|
||||||
|
|
||||||
Read more about the `DatePipe` format options in the [Date Pipe](api/common/DatePipe)
|
|
||||||
API Reference page.
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
### Example: Applying two formats by chaining pipes
|
||||||
|
|
||||||
|
You can chain pipes so that the output of one pipe becomes the input to the next.
|
||||||
|
|
||||||
## Chaining pipes
|
In the following example, chained pipes first apply a format to a date value, then convert the formatted date to uppercase characters.
|
||||||
|
The first tab for the `src/app/app.component.html` template chains `DatePipe` and `UpperCasePipe` to display the birthday as **APR 15, 1988**.
|
||||||
|
The second tab for the `src/app/app.component.html` template passes the `fullDate` parameter to `date` before chaining to `uppercase`, which produces **FRIDAY, APRIL 15, 1988**.
|
||||||
|
|
||||||
You can chain pipes together in potentially useful combinations.
|
<code-tabs>
|
||||||
In the following example, to display the birthday in uppercase,
|
<code-pane
|
||||||
the birthday is chained to the `DatePipe` and on to the `UpperCasePipe`.
|
header="src/app/app.component.html (1)"
|
||||||
The birthday displays as **<samp>APR 15, 1988</samp>**.
|
region="chained-birthday"
|
||||||
|
path="pipes/src/app/app.component.html">
|
||||||
|
</code-pane>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/app.component.html (2)"
|
||||||
|
region="chained-parameter-birthday"
|
||||||
|
path="pipes/src/app/app.component.html">
|
||||||
|
</code-pane>
|
||||||
|
</code-tabs>
|
||||||
|
|
||||||
|
{@a Custom-pipes}
|
||||||
|
|
||||||
<code-example path="pipes/src/app/app.component.html" region="chained-birthday" header="src/app/app.component.html"></code-example>
|
## Creating pipes for custom data transformations
|
||||||
|
|
||||||
|
Create custom pipes to encapsulate transformations that are not provided with the built-in pipes.
|
||||||
|
You can then use your custom pipe in template expressions, the same way you use built-in pipes—to transform input values to output values for display.
|
||||||
|
|
||||||
|
### Marking a class as a pipe
|
||||||
|
|
||||||
This example—which displays **<samp>FRIDAY, APRIL 15, 1988</samp>**—chains
|
To mark a class as a pipe and supply configuration metadata, apply the [`@Pipe`](/api/core/Pipe "API reference for Pipe") [decorator](/guide/glossary#decorator--decoration "Definition for decorator") to the class.
|
||||||
the same pipes as above, but passes in a parameter to `date` as well.
|
Use [UpperCamelCase](guide/glossary#case-types "Definition of case types") (the general convention for class names) for the pipe class name, and [camelCase](guide/glossary#case-types "Definition of case types") for the corresponding `name` string.
|
||||||
|
Do not use hyphens in the `name`.
|
||||||
|
For details and more examples, see [Pipe names](guide/styleguide#pipe-names "Pipe names in the Angular coding style guide").
|
||||||
|
|
||||||
|
Use `name` in template expressions as you would for a built-in pipe.
|
||||||
|
|
||||||
<code-example path="pipes/src/app/app.component.html" region="chained-parameter-birthday" header="src/app/app.component.html"></code-example>
|
<div class="alert is-important">
|
||||||
|
|
||||||
|
* Include your pipe in the `declarations` field of the `NgModule` metadata in order for it to be available to a template. See the `app.module.ts` file in the example app (<live-example></live-example>). For details, see [NgModules](guide/ngmodules "NgModules introduction").
|
||||||
|
* Register your custom pipes. The [Angular CLI](cli "CLI Overview and Command Reference") [`ng generate pipe`](cli/generate#pipe "ng generate pipe in the CLI Command Reference") command registers the pipe automatically.
|
||||||
|
|
||||||
## Custom pipes
|
|
||||||
|
|
||||||
You can write your own custom pipes.
|
|
||||||
Here's a custom pipe named `ExponentialStrengthPipe` that can boost a hero's powers:
|
|
||||||
|
|
||||||
|
|
||||||
<code-example path="pipes/src/app/exponential-strength.pipe.ts" header="src/app/exponential-strength.pipe.ts"></code-example>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
This pipe definition reveals the following key points:
|
|
||||||
|
|
||||||
* A pipe is a class decorated with pipe metadata.
|
|
||||||
* The pipe class implements the `PipeTransform` interface's `transform` method that
|
|
||||||
accepts an input value followed by optional parameters and returns the transformed value.
|
|
||||||
* There will be one additional argument to the `transform` method for each parameter passed to the pipe.
|
|
||||||
Your pipe has one such parameter: the `exponent`.
|
|
||||||
* To tell Angular that this is a pipe, you apply the
|
|
||||||
`@Pipe` decorator, which you import from the core Angular library.
|
|
||||||
* The `@Pipe` decorator allows you to define the
|
|
||||||
pipe name that you'll use within template expressions. It must be a valid JavaScript identifier.
|
|
||||||
Your pipe's name is `exponentialStrength`.
|
|
||||||
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## The *PipeTransform* interface
|
|
||||||
|
|
||||||
The `transform` method is essential to a pipe.
|
|
||||||
The `PipeTransform` *interface* defines that method and guides both tooling and the compiler.
|
|
||||||
Technically, it's optional; Angular looks for and executes the `transform` method regardless.
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
Now you need a component to demonstrate the pipe.
|
### Using the PipeTransform interface
|
||||||
|
|
||||||
<code-example path="pipes/src/app/power-booster.component.ts" header="src/app/power-booster.component.ts"></code-example>
|
Implement the [`PipeTransform`](/api/core/PipeTransform "API reference for PipeTransform") interface in your custom pipe class to perform the transformation.
|
||||||
|
|
||||||
|
Angular invokes the `transform` method with the value of a binding as the first argument, and any parameters as the second argument in list form, and returns the transformed value.
|
||||||
|
|
||||||
|
### Example: Transforming a value exponentially
|
||||||
|
|
||||||
|
In a game, you may want to implement a transformation that raises a value exponentially to increase a hero's power.
|
||||||
|
For example, if the hero's score is 2, boosting the hero's power exponentially by 10 produces a score of 1024.
|
||||||
|
You can use a custom pipe for this transformation.
|
||||||
|
|
||||||
|
The following code example shows two component definitions:
|
||||||
|
|
||||||
|
* The `exponential-strength.pipe.ts` component defines a custom pipe named `exponentialStrength` with the `transform` method that performs the transformation.
|
||||||
|
It defines an argument to the `transform` method (`exponent`) for a parameter passed to the pipe.
|
||||||
|
|
||||||
|
* The `power-booster.component.ts` component demonstrates how to use the pipe, specifying a value (`2`) and the exponent parameter (`10`).
|
||||||
|
Figure 2 shows the output.
|
||||||
|
|
||||||
|
<code-tabs>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/exponential-strength.pipe.ts"
|
||||||
|
path="pipes/src/app/exponential-strength.pipe.ts">
|
||||||
|
</code-pane>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/power-booster.component.ts"
|
||||||
|
path="pipes/src/app/power-booster.component.ts">
|
||||||
|
</code-pane>
|
||||||
|
</code-tabs>
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src='generated/images/guide/pipes/power-booster.png' alt="Power Booster">
|
<img src='generated/images/guide/pipes/power-booster.png' alt="Power Booster">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
**Figure 2.** Output from the `exponentialStrength` pipe
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
Note the following:
|
To examine the behavior the `exponentialStrength` pipe in the <live-example></live-example>, change the value and optional exponent in the template.
|
||||||
|
|
||||||
* You use your custom pipe the same way you use built-in pipes.
|
|
||||||
* You must include your pipe in the `declarations` array of the `AppModule`
|
|
||||||
* If you choose to inject your pipe into a class, you must provide it in the `providers` array of your `NgModule`.
|
|
||||||
|
|
||||||
<div class="callout is-helpful">
|
|
||||||
|
|
||||||
<header>
|
|
||||||
Remember the declarations array
|
|
||||||
</header>
|
|
||||||
|
|
||||||
|
|
||||||
You must register custom pipes.
|
|
||||||
If you don't, Angular reports an error.
|
|
||||||
The [Angular CLI's](cli) generator registers the pipe automatically.
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{@a change-detection}
|
||||||
|
|
||||||
|
## Detecting changes with data binding in pipes
|
||||||
|
|
||||||
To probe the behavior in the <live-example></live-example>,
|
You use [data binding](/guide/glossary#data-binding "Definition of data binding") with a pipe to display values and respond to user actions.
|
||||||
change the value and optional exponent in the template.
|
If the data is a primitive input value, such as `String` or `Number`, or an object reference as input, such as `Date` or `Array`, Angular executes the pipe whenever it detects a change for the input value or reference.
|
||||||
|
|
||||||
## Power Boost Calculator
|
|
||||||
|
|
||||||
It's not much fun updating the template to test the custom pipe.
|
|
||||||
Upgrade the example to a "Power Boost Calculator" that combines
|
|
||||||
your pipe and two-way data binding with `ngModel`.
|
|
||||||
|
|
||||||
|
For example, you could change the previous custom pipe example to use two-way data binding with `ngModel` to input the amount and boost factor, as shown in the following code example.
|
||||||
|
|
||||||
<code-example path="pipes/src/app/power-boost-calculator.component.ts" header="src/app/power-boost-calculator.component.ts">
|
<code-example path="pipes/src/app/power-boost-calculator.component.ts" header="src/app/power-boost-calculator.component.ts">
|
||||||
|
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
|
The `exponentialStrength` pipe executes every time the user changes the "normal power" value or the "boost factor", as shown in Figure 3.
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src='generated/images/guide/pipes/power-boost-calculator-anim.gif' alt="Power Boost Calculator">
|
<img src='generated/images/guide/pipes/power-boost-calculator-anim.gif' alt="Power Boost Calculator">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
**Figure 3.** Changing the amount and boost factor for the `exponentialStrength` pipe
|
||||||
|
|
||||||
|
Angular detects each change and immediately runs the pipe.
|
||||||
|
This is fine for primitive input values.
|
||||||
|
However, if you change something *inside* a composite object (such as the month of a date, an element of an array, or an object property), you need to understand how change detection works, and how to use an `impure` pipe.
|
||||||
|
|
||||||
|
### How change detection works
|
||||||
|
|
||||||
{@a change-detection}
|
Angular looks for changes to data-bound values in a [change detection](guide/glossary#change-detection "Definition of change detection") process that runs after every DOM event: every keystroke, mouse move, timer tick, and server response.
|
||||||
|
The following example, which doesn't use a pipe, demonstrates how Angular uses its default change detection strategy to monitor and update its display of every hero in the `heroes` array.
|
||||||
|
The example tabs show the following:
|
||||||
|
|
||||||
|
* In the `flying-heroes.component.html (v1)` template, the `*ngFor` repeater displays the hero names.
|
||||||
|
* Its companion component class `flying-heroes.component.ts (v1)` provides heroes, adds heroes into the array, and resets the array.
|
||||||
|
|
||||||
## Pipes and change detection
|
<code-tabs>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/flying-heroes.component.html (v1)"
|
||||||
|
region="template-1"
|
||||||
|
path="pipes/src/app/flying-heroes.component.html">
|
||||||
|
</code-pane>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/flying-heroes.component.ts (v1)"
|
||||||
|
region="v1"
|
||||||
|
path="pipes/src/app/flying-heroes.component.ts">
|
||||||
|
</code-pane>
|
||||||
|
</code-tabs>
|
||||||
|
|
||||||
Angular looks for changes to data-bound values through a *change detection* process that runs after every DOM event:
|
Angular updates the display every time the user adds a hero.
|
||||||
every keystroke, mouse move, timer tick, and server response. This could be expensive.
|
If the user clicks the **Reset** button, Angular replaces `heroes` with a new array of the original heroes and updates the display.
|
||||||
Angular strives to lower the cost whenever possible and appropriate.
|
If you add the ability to remove or change a hero, Angular would detect those changes and update the display as well.
|
||||||
|
|
||||||
Angular picks a simpler, faster change detection algorithm when you use a pipe.
|
However, executing a pipe to update the display with every change would slow down your app's performance.
|
||||||
|
So Angular uses a faster change-detection algorithm for executing a pipe, as described in the next section.
|
||||||
|
|
||||||
<h3 class="no-toc">No pipe</h3>
|
{@a pure-and-impure-pipes}
|
||||||
|
|
||||||
In the next example, the component uses the default, aggressive change detection strategy to monitor and update
|
### Detecting pure changes to primitives and object references
|
||||||
its display of every hero in the `heroes` array. Here's the template:
|
|
||||||
|
|
||||||
|
By default, pipes are defined as *pure* so that Angular executes the pipe only when it detects a *pure change* to the input value.
|
||||||
|
A pure change is either a change to a primitive input value (such as `String`, `Number`, `Boolean`, or `Symbol`), or a changed object reference (such as `Date`, `Array`, `Function`, or `Object`).
|
||||||
|
|
||||||
<code-example path="pipes/src/app/flying-heroes.component.html" region="template-1" header="src/app/flying-heroes.component.html (v1)"></code-example>
|
{@a pure-pipe-pure-fn}
|
||||||
|
|
||||||
|
A pure pipe must use a pure function, which is one that processes inputs and returns values without side effects.
|
||||||
|
In other words, given the same input, a pure function should always return the same output.
|
||||||
|
|
||||||
|
With a pure pipe, Angular ignores changes within composite objects, such as a newly added element of an existing array, because checking a primitive value or object reference is much faster than performing a deep check for differences within objects.
|
||||||
|
Angular can quickly determine if it can skip executing the pipe and updating the view.
|
||||||
|
|
||||||
The companion component class provides heroes, adds heroes into the array, and can reset the array.
|
However, a pure pipe with an array as input may not work the way you want.
|
||||||
|
To demonstrate this issue, change the previous example to filter the list of heroes to just those heroes who can fly.
|
||||||
|
Use the `FlyingHeroesPipe` in the `*ngFor` repeater as shown in the following code.
|
||||||
|
The tabs for the example show the following:
|
||||||
|
|
||||||
<code-example path="pipes/src/app/flying-heroes.component.ts" region="v1" header="src/app/flying-heroes.component.ts (v1)"></code-example>
|
* The template (`flying-heroes.component.html (flyers)`) with the new pipe.
|
||||||
|
* The `FlyingHeroesPipe` custom pipe implementation (`flying-heroes.pipe.ts`).
|
||||||
|
|
||||||
|
<code-tabs>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/flying-heroes.component.html (flyers)"
|
||||||
|
region="template-flying-heroes"
|
||||||
|
path="pipes/src/app/flying-heroes.component.html">
|
||||||
|
</code-pane>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/flying-heroes.pipe.ts"
|
||||||
|
region="pure"
|
||||||
|
path="pipes/src/app/flying-heroes.pipe.ts">
|
||||||
|
</code-pane>
|
||||||
|
</code-tabs>
|
||||||
|
|
||||||
|
The app now shows unexpected behavior: When the user adds flying heroes, none of them appear under "Heroes who fly."
|
||||||
You can add heroes and Angular updates the display when you do.
|
This happens because the code that adds a hero does so by pushing it onto the `heroes` array:
|
||||||
If you click the `reset` button, Angular replaces `heroes` with a new array of the original heroes and updates the display.
|
|
||||||
If you added the ability to remove or change a hero, Angular would detect those changes and update the display as well.
|
|
||||||
|
|
||||||
<h3 class="no-toc"><i>FlyingHeroesPipe</i></h3>
|
|
||||||
|
|
||||||
Add a `FlyingHeroesPipe` to the `*ngFor` repeater that filters the list of heroes to just those heroes who can fly.
|
|
||||||
|
|
||||||
<code-example path="pipes/src/app/flying-heroes.component.html" region="template-flying-heroes" header="src/app/flying-heroes.component.html (flyers)"></code-example>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Here's the `FlyingHeroesPipe` implementation, which follows the pattern for custom pipes described earlier.
|
|
||||||
|
|
||||||
<code-example path="pipes/src/app/flying-heroes.pipe.ts" region="pure" header="src/app/flying-heroes.pipe.ts"></code-example>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Notice the odd behavior in the <live-example></live-example>:
|
|
||||||
when you add flying heroes, none of them are displayed under "Heroes who fly."
|
|
||||||
|
|
||||||
Although you're not getting the behavior you want, Angular isn't broken.
|
|
||||||
It's just using a different change-detection algorithm that ignores changes to the list or any of its items.
|
|
||||||
|
|
||||||
Notice how a hero is added:
|
|
||||||
|
|
||||||
<code-example path="pipes/src/app/flying-heroes.component.ts" region="push" header="src/app/flying-heroes.component.ts"></code-example>
|
<code-example path="pipes/src/app/flying-heroes.component.ts" region="push" header="src/app/flying-heroes.component.ts"></code-example>
|
||||||
|
|
||||||
|
The change detector ignores changes to elements of an array, so the pipe doesn't run.
|
||||||
|
|
||||||
|
The reason Angular ignores the changed array element is that the *reference* to the array hasn't changed.
|
||||||
|
Since the array is the same, Angular does not update the display.
|
||||||
|
|
||||||
You add the hero into the `heroes` array. The reference to the array hasn't changed.
|
One way to get the behavior you want is to change the object reference itself.
|
||||||
It's the same array. That's all Angular cares about. From its perspective, *same array, no change, no display update*.
|
You can replace the array with a new array containing the newly changed elements, and then input the new array to the pipe.
|
||||||
|
In the above example, you can create an array with the new hero appended, and assign that to `heroes`. Angular detects the change in the array reference and executes the pipe.
|
||||||
To fix that, create an array with the new hero appended and assign that to `heroes`.
|
|
||||||
This time Angular detects that the array reference has changed.
|
|
||||||
It executes the pipe and updates the display with the new array, which includes the new flying hero.
|
|
||||||
|
|
||||||
If you *mutate* the array, no pipe is invoked and the display isn't updated;
|
|
||||||
if you *replace* the array, the pipe executes and the display is updated.
|
|
||||||
The Flying Heroes application extends the
|
|
||||||
code with checkbox switches and additional displays to help you experience these effects.
|
|
||||||
|
|
||||||
|
To summarize, if you mutate the input array, the pure pipe doesn't execute.
|
||||||
|
If you *replace* the input array, the pipe executes and the display is updated, as shown in Figure 4.
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src='generated/images/guide/pipes/flying-heroes-anim.gif' alt="Flying Heroes">
|
<img src='generated/images/guide/pipes/flying-heroes-anim.gif' alt="Flying Heroes">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
**Figure 4.** The `flyingHeroes` pipe filtering the display to flying heroes
|
||||||
|
|
||||||
|
The above example demonstrates changing a component's code to accommodate a pipe.
|
||||||
|
|
||||||
Replacing the array is an efficient way to signal Angular to update the display.
|
To keep your component simpler and independent of HTML templates that use pipes, you can, as an alternative, use an *impure* pipe to detect changes within composite objects such as arrays, as described in the next section.
|
||||||
When do you replace the array? When the data changes.
|
|
||||||
That's an easy rule to follow in *this* example
|
|
||||||
where the only way to change the data is by adding a hero.
|
|
||||||
|
|
||||||
More often, you don't know when the data has changed,
|
|
||||||
especially in applications that mutate data in many ways,
|
|
||||||
perhaps in application locations far away.
|
|
||||||
A component in such an application usually can't know about those changes.
|
|
||||||
Moreover, it's unwise to distort the component design to accommodate a pipe.
|
|
||||||
Strive to keep the component class independent of the HTML.
|
|
||||||
The component should be unaware of pipes.
|
|
||||||
|
|
||||||
For filtering flying heroes, consider an *impure pipe*.
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## Pure and impure pipes
|
|
||||||
|
|
||||||
There are two categories of pipes: *pure* and *impure*.
|
|
||||||
Pipes are pure by default. Every pipe you've seen so far has been pure.
|
|
||||||
You make a pipe impure by setting its pure flag to false. You could make the `FlyingHeroesPipe`
|
|
||||||
impure like this:
|
|
||||||
|
|
||||||
|
|
||||||
<code-example path="pipes/src/app/flying-heroes.pipe.ts" region="pipe-decorator" header="src/app/flying-heroes.pipe.ts"></code-example>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Before doing that, understand the difference between pure and impure, starting with a pure pipe.
|
|
||||||
|
|
||||||
<h3 class="no-toc">Pure pipes</h3>
|
|
||||||
|
|
||||||
Angular executes a *pure pipe* only when it detects a *pure change* to the input value.
|
|
||||||
A pure change is either a change to a primitive input value (`String`, `Number`, `Boolean`, `Symbol`)
|
|
||||||
or a changed object reference (`Date`, `Array`, `Function`, `Object`).
|
|
||||||
|
|
||||||
Angular ignores changes within (composite) objects.
|
|
||||||
It won't call a pure pipe if you change an input month, add to an input array, or update an input object property.
|
|
||||||
|
|
||||||
This may seem restrictive but it's also fast.
|
|
||||||
An object reference check is fast—much faster than a deep check for
|
|
||||||
differences—so Angular can quickly determine if it can skip both the
|
|
||||||
pipe execution and a view update.
|
|
||||||
|
|
||||||
For this reason, a pure pipe is preferable when you can live with the change detection strategy.
|
|
||||||
When you can't, you *can* use the impure pipe.
|
|
||||||
|
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Or you might not use a pipe at all.
|
|
||||||
It may be better to pursue the pipe's purpose with a property of the component,
|
|
||||||
a point that's discussed later in this page.
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<h3 class="no-toc">Impure pipes</h3>
|
|
||||||
|
|
||||||
Angular executes an *impure pipe* during every component change detection cycle.
|
|
||||||
An impure pipe is called often, as often as every keystroke or mouse-move.
|
|
||||||
|
|
||||||
With that concern in mind, implement an impure pipe with great care.
|
|
||||||
An expensive, long-running pipe could destroy the user experience.
|
|
||||||
|
|
||||||
|
|
||||||
{@a impure-flying-heroes}
|
{@a impure-flying-heroes}
|
||||||
|
|
||||||
|
### Detecting impure changes within composite objects
|
||||||
|
|
||||||
<h3 class="no-toc">An impure <i>FlyingHeroesPipe</i></h3>
|
To execute a custom pipe after a change *within* a composite object, such as a change to an element of an array, you need to define your pipe as `impure` to detect impure changes.
|
||||||
|
Angular executes an impure pipe every time it detects a change with every keystroke or mouse movement.
|
||||||
|
|
||||||
A flip of the switch turns the `FlyingHeroesPipe` into a `FlyingHeroesImpurePipe`.
|
<div class="alert is-important">
|
||||||
The complete implementation is as follows:
|
|
||||||
|
While an impure pipe can be useful, be careful using one. A long-running impure pipe could dramatically slow down your app.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
Make a pipe impure by setting its `pure` flag to `false`:
|
||||||
|
|
||||||
|
<code-example path="pipes/src/app/flying-heroes.pipe.ts" region="pipe-decorator" header="src/app/flying-heroes.pipe.ts"></code-example>
|
||||||
|
|
||||||
|
The following code shows the complete implementation of `FlyingHeroesImpurePipe`, which extends `FlyingHeroesPipe` to inherit its characteristics.
|
||||||
|
The example shows that you don't have to change anything else—the only difference is setting the `pure` flag as `false` in the pipe metadata.
|
||||||
|
|
||||||
<code-tabs>
|
<code-tabs>
|
||||||
|
<code-pane
|
||||||
<code-pane header="FlyingHeroesImpurePipe" path="pipes/src/app/flying-heroes.pipe.ts" region="impure">
|
header="src/app/flying-heroes.pipe.ts (FlyingHeroesImpurePipe)"
|
||||||
|
region="impure"
|
||||||
|
path="pipes/src/app/flying-heroes.pipe.ts">
|
||||||
</code-pane>
|
</code-pane>
|
||||||
|
<code-pane
|
||||||
<code-pane header="FlyingHeroesPipe" path="pipes/src/app/flying-heroes.pipe.ts" region="pure">
|
header="src/app/flying-heroes.pipe.ts (FlyingHeroesPipe)"
|
||||||
|
region="pure"
|
||||||
|
path="pipes/src/app/flying-heroes.pipe.ts">
|
||||||
</code-pane>
|
</code-pane>
|
||||||
|
|
||||||
</code-tabs>
|
</code-tabs>
|
||||||
|
|
||||||
|
`FlyingHeroesImpurePipe` is a good candidate for an impure pipe because the `transform` function is trivial and fast:
|
||||||
|
|
||||||
You inherit from `FlyingHeroesPipe` to prove the point that nothing changed internally.
|
|
||||||
The only difference is the `pure` flag in the pipe metadata.
|
|
||||||
|
|
||||||
This is a good candidate for an impure pipe because the `transform` function is trivial and fast.
|
|
||||||
|
|
||||||
|
|
||||||
<code-example path="pipes/src/app/flying-heroes.pipe.ts" header="src/app/flying-heroes.pipe.ts (filter)" region="filter"></code-example>
|
<code-example path="pipes/src/app/flying-heroes.pipe.ts" header="src/app/flying-heroes.pipe.ts (filter)" region="filter"></code-example>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
You can derive a `FlyingHeroesImpureComponent` from `FlyingHeroesComponent`.
|
You can derive a `FlyingHeroesImpureComponent` from `FlyingHeroesComponent`.
|
||||||
|
As shown in the code below, only the pipe in the template changes.
|
||||||
|
|
||||||
<code-example path="pipes/src/app/flying-heroes-impure.component.html" header="src/app/flying-heroes-impure.component.html (excerpt)" region="template-flying-heroes"></code-example>
|
<code-example path="pipes/src/app/flying-heroes-impure.component.html" header="src/app/flying-heroes-impure.component.html (excerpt)" region="template-flying-heroes"></code-example>
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
To confirm that the display updates as the user adds heroes, see the <live-example></live-example>.
|
||||||
|
|
||||||
The only substantive change is the pipe in the template.
|
</div>
|
||||||
You can confirm in the <live-example></live-example> that the _flying heroes_
|
|
||||||
display updates as you add heroes, even when you mutate the `heroes` array.
|
|
||||||
|
|
||||||
|
|
||||||
{@a async-pipe}
|
{@a async-pipe}
|
||||||
<h3 class="no-toc">The impure <i>AsyncPipe</i></h3>
|
|
||||||
|
|
||||||
|
## Unwrapping data from an observable
|
||||||
|
|
||||||
The Angular `AsyncPipe` is an interesting example of an impure pipe.
|
[Observables](/guide/glossary#observable "Definition of observable") let you pass messages between parts of your application.
|
||||||
The `AsyncPipe` accepts a `Promise` or `Observable` as input
|
Observables are recommended for event handling, asynchronous programming, and handling multiple values.
|
||||||
and subscribes to the input automatically, eventually returning the emitted values.
|
Observables can deliver single or multiple values of any type, either synchronously (as a function delivers a value to its caller) or asynchronously on a schedule.
|
||||||
|
|
||||||
The `AsyncPipe` is also stateful.
|
<div class="alert is-helpful">
|
||||||
The pipe maintains a subscription to the input `Observable` and
|
|
||||||
keeps delivering values from that `Observable` as they arrive.
|
|
||||||
|
|
||||||
This next example binds an `Observable` of message strings
|
For details and examples of observables, see the [Observables Overview](/guide/observables#using-observables-to-pass-values "Using observables to pass values"").
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
Use the built-in [`AsyncPipe`](/api/common/AsyncPipe "API description of AsyncPipe") to accept an observable as input and subscribe to the input automatically.
|
||||||
|
Without this pipe, your component code would have to subscribe to the observable to consume its values, extract the resolved values, expose them for binding, and unsubscribe when the observable is destroyed in order to prevent memory leaks. `AsyncPipe` is an impure pipe that saves boilerplate code in your component to maintain the subscription and keep delivering values from that observable as they arrive.
|
||||||
|
|
||||||
|
The following code example binds an observable of message strings
|
||||||
(`message$`) to a view with the `async` pipe.
|
(`message$`) to a view with the `async` pipe.
|
||||||
|
|
||||||
|
|
||||||
<code-example path="pipes/src/app/hero-async-message.component.ts" header="src/app/hero-async-message.component.ts">
|
<code-example path="pipes/src/app/hero-async-message.component.ts" header="src/app/hero-async-message.component.ts">
|
||||||
|
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
|
{@a no-filter-pipe}
|
||||||
|
|
||||||
|
## Caching HTTP requests
|
||||||
|
|
||||||
The Async pipe saves boilerplate in the component code.
|
To [communicate with backend services using HTTP](/guide/http "Communicating with backend services using HTTP"), the `HttpClient` service uses observables and offers the `HTTPClient.get()` method to fetch data from a server.
|
||||||
The component doesn't have to subscribe to the async data source,
|
The aynchronous method sends an HTTP request, and returns an observable that emits the requested data for the response.
|
||||||
extract the resolved values and expose them for binding,
|
|
||||||
and have to unsubscribe when it's destroyed
|
|
||||||
(a potent source of memory leaks).
|
|
||||||
|
|
||||||
<h3 class="no-toc">An impure caching pipe</h3>
|
As shown in the previous section, you can use the impure `AsyncPipe` to accept an observable as input and subscribe to the input automatically.
|
||||||
|
You can also create an impure pipe to make and cache an HTTP request.
|
||||||
|
|
||||||
Write one more impure pipe, a pipe that makes an HTTP request.
|
Impure pipes are called whenever change detection runs for a component, which could be every few milliseconds for `CheckAlways`.
|
||||||
|
To avoid performance problems, call the server only when the requested URL changes, as shown in the following example, and use the pipe to cache the server response.
|
||||||
|
The tabs show the following:
|
||||||
|
|
||||||
Remember that impure pipes are called every few milliseconds.
|
* The `fetch` pipe (`fetch-json.pipe.ts`).
|
||||||
If you're not careful, this pipe will punish the server with requests.
|
* A harness component (`hero-list.component.ts`) for demonstrating the request, using a template that defines two bindings to the pipe requesting the heroes from the `heroes.json` file. The second binding chains the `fetch` pipe with the built-in `JsonPipe` to display the same hero data in JSON format.
|
||||||
|
|
||||||
In the following code, the pipe only calls the server when the requested URL changes and it caches the server response.
|
<code-tabs>
|
||||||
The code uses the [Angular http](guide/http) client to retrieve data:
|
<code-pane
|
||||||
|
header="src/app/fetch-json.pipe.ts"
|
||||||
|
path="pipes/src/app/fetch-json.pipe.ts">
|
||||||
|
</code-pane>
|
||||||
|
<code-pane
|
||||||
|
header="src/app/hero-list.component.ts"
|
||||||
|
path="pipes/src/app/hero-list.component.ts">
|
||||||
|
</code-pane>
|
||||||
|
</code-tabs>
|
||||||
|
|
||||||
|
In the above example, a breakpoint on the pipe's request for data shows the following:
|
||||||
|
|
||||||
<code-example path="pipes/src/app/fetch-json.pipe.ts" header="src/app/fetch-json.pipe.ts">
|
* Each binding gets its own pipe instance.
|
||||||
|
* Each pipe instance caches its own URL and data and calls the server only once.
|
||||||
</code-example>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Now demonstrate it in a harness component whose template defines two bindings to this pipe,
|
|
||||||
both requesting the heroes from the `heroes.json` file.
|
|
||||||
|
|
||||||
|
|
||||||
<code-example path="pipes/src/app/hero-list.component.ts" header="src/app/hero-list.component.ts">
|
|
||||||
|
|
||||||
</code-example>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
The component renders as the following:
|
|
||||||
|
|
||||||
|
The `fetch` and `fetch-json` pipes display the heroes as shown in Figure 5.
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
<img src='generated/images/guide/pipes/hero-list.png' alt="Hero List">
|
<img src='generated/images/guide/pipes/hero-list.png' alt="Hero List">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
**Figure 5.** The `fetch` and `fetch-json` pipes displaying the heroes
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
A breakpoint on the pipe's request for data shows the following:
|
The built-in [JsonPipe](api/common/JsonPipe "API description for JsonPipe") provides a way to diagnose a mysteriously failing data binding or to inspect an object for future binding.
|
||||||
|
|
||||||
* Each binding gets its own pipe instance.
|
|
||||||
* Each pipe instance caches its own URL and data.
|
|
||||||
* Each pipe instance only calls the server once.
|
|
||||||
|
|
||||||
<h3 class="no-toc"><i>JsonPipe</i></h3>
|
|
||||||
|
|
||||||
In the previous code sample, the second `fetch` pipe binding demonstrates more pipe chaining.
|
|
||||||
It displays the same hero data in JSON format by chaining through to the built-in `JsonPipe`.
|
|
||||||
|
|
||||||
|
|
||||||
<div class="callout is-helpful">
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<header>
|
|
||||||
Debugging with the json pipe
|
|
||||||
</header>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
The [JsonPipe](api/common/JsonPipe)
|
|
||||||
provides an easy way to diagnose a mysteriously failing data binding or
|
|
||||||
inspect an object for future binding.
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
{@a pure-pipe-pure-fn}
|
|
||||||
|
|
||||||
|
|
||||||
<h3 class="no-toc">Pure pipes and pure functions</h3>
|
|
||||||
|
|
||||||
A pure pipe uses pure functions.
|
|
||||||
Pure functions process inputs and return values without detectable side effects.
|
|
||||||
Given the same input, they should always return the same output.
|
|
||||||
|
|
||||||
The pipes discussed earlier in this page are implemented with pure functions.
|
|
||||||
The built-in `DatePipe` is a pure pipe with a pure function implementation.
|
|
||||||
So are the `ExponentialStrengthPipe` and `FlyingHeroesPipe`.
|
|
||||||
A few steps back, you reviewed the `FlyingHeroesImpurePipe`—an impure pipe with a pure function.
|
|
||||||
|
|
||||||
But always implement a *pure pipe* with a *pure function*.
|
|
||||||
Otherwise, you'll see many console errors regarding expressions that changed after they were checked.
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## Next steps
|
|
||||||
|
|
||||||
Pipes are a great way to encapsulate and share common display-value
|
|
||||||
transformations. Use them like styles, dropping them
|
|
||||||
into your template's expressions to enrich the appeal and usability
|
|
||||||
of your views.
|
|
||||||
|
|
||||||
Explore Angular's inventory of built-in pipes in the [API Reference](api?type=pipe).
|
|
||||||
Try writing a custom pipe and perhaps contributing it to the community.
|
|
||||||
|
|
||||||
|
|
||||||
{@a no-filter-pipe}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## Appendix: No *FilterPipe* or *OrderByPipe*
|
|
||||||
|
|
||||||
Angular doesn't provide pipes for filtering or sorting lists.
|
|
||||||
Developers familiar with AngularJS know these as `filter` and `orderBy`.
|
|
||||||
There are no equivalents in Angular.
|
|
||||||
|
|
||||||
This isn't an oversight. Angular doesn't offer such pipes because
|
|
||||||
they perform poorly and prevent aggressive minification.
|
|
||||||
Both `filter` and `orderBy` require parameters that reference object properties.
|
|
||||||
Earlier in this page, you learned that such pipes must be [impure](guide/pipes#pure-and-impure-pipes) and that
|
|
||||||
Angular calls impure pipes in almost every change-detection cycle.
|
|
||||||
|
|
||||||
Filtering and especially sorting are expensive operations.
|
|
||||||
The user experience can degrade severely for even moderate-sized lists when Angular calls these pipe methods many times per second.
|
|
||||||
`filter` and `orderBy` have often been abused in AngularJS apps, leading to complaints that Angular itself is slow.
|
|
||||||
That charge is fair in the indirect sense that AngularJS prepared this performance trap
|
|
||||||
by offering `filter` and `orderBy` in the first place.
|
|
||||||
|
|
||||||
The minification hazard is also compelling, if less obvious. Imagine a sorting pipe applied to a list of heroes.
|
|
||||||
The list might be sorted by hero `name` and `planet` of origin properties in the following way:
|
|
||||||
|
|
||||||
<code-example language="html">
|
|
||||||
<!-- NOT REAL CODE! -->
|
|
||||||
<div *ngFor="let hero of heroes | orderBy:'name,planet'"></div>
|
|
||||||
</code-example>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
You identify the sort fields by text strings, expecting the pipe to reference a property value by indexing
|
|
||||||
(such as `hero['name']`).
|
|
||||||
Unfortunately, aggressive minification manipulates the `Hero` property names so that `Hero.name` and `Hero.planet`
|
|
||||||
become something like `Hero.a` and `Hero.b`. Clearly `hero['name']` doesn't work.
|
|
||||||
|
|
||||||
While some may not care to minify this aggressively,
|
|
||||||
the Angular product shouldn't prevent anyone from minifying aggressively.
|
|
||||||
Therefore, the Angular team decided that everything Angular provides will minify safely.
|
|
||||||
|
|
||||||
The Angular team and many experienced Angular developers strongly recommend moving
|
|
||||||
filtering and sorting logic into the component itself.
|
|
||||||
The component can expose a `filteredHeroes` or `sortedHeroes` property and take control
|
|
||||||
over when and how often to execute the supporting logic.
|
|
||||||
Any capabilities that you would have put in a pipe and shared across the app can be
|
|
||||||
written in a filtering/sorting service and injected into the component.
|
|
||||||
|
|
||||||
If these performance and minification considerations don't apply to you, you can always create your own such pipes
|
|
||||||
(similar to the [FlyingHeroesPipe](guide/pipes#impure-flying-heroes)) or find them in the community.
|
|
||||||
|
228
aio/content/guide/property-binding.md
Normal file
228
aio/content/guide/property-binding.md
Normal file
@ -0,0 +1,228 @@
|
|||||||
|
|
||||||
|
# Property binding `[property]`
|
||||||
|
|
||||||
|
Use property binding to _set_ properties of target elements or
|
||||||
|
directive `@Input()` decorators.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
See the <live-example></live-example> for a working example containing the code snippets in this guide.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
## One-way in
|
||||||
|
|
||||||
|
Property binding flows a value in one direction,
|
||||||
|
from a component's property into a target element property.
|
||||||
|
|
||||||
|
You can't use property
|
||||||
|
binding to read or pull values out of target elements. Similarly, you cannot use
|
||||||
|
property binding to call a method on the target element.
|
||||||
|
If the element raises events, you can listen to them with an [event binding](guide/event-binding).
|
||||||
|
|
||||||
|
If you must read a target element property or call one of its methods,
|
||||||
|
see the API reference for [ViewChild](api/core/ViewChild) and
|
||||||
|
[ContentChild](api/core/ContentChild).
|
||||||
|
|
||||||
|
## Examples
|
||||||
|
|
||||||
|
The most common property binding sets an element property to a component
|
||||||
|
property value. An example is
|
||||||
|
binding the `src` property of an image element to a component's `itemImageUrl` property:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="property-binding" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Here's an example of binding to the `colSpan` property. Notice that it's not `colspan`,
|
||||||
|
which is the attribute, spelled with a lowercase `s`.
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="colSpan" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
For more details, see the [MDN HTMLTableCellElement](https://developer.mozilla.org/en-US/docs/Web/API/HTMLTableCellElement) documentation.
|
||||||
|
|
||||||
|
For more information about `colSpan` and `colspan`, see the [Attribute binding](guide/attribute-binding#colspan) guide.
|
||||||
|
|
||||||
|
Another example is disabling a button when the component says that it `isUnchanged`:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="disabled-button" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Another is setting a property of a directive:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="class-binding" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Yet another is setting the model property of a custom component—a great way
|
||||||
|
for parent and child components to communicate:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="model-property-binding" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
## Binding targets
|
||||||
|
|
||||||
|
An element property between enclosing square brackets identifies the target property.
|
||||||
|
The target property in the following code is the image element's `src` property.
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="property-binding" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
There's also the `bind-` prefix alternative:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="bind-prefix" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
|
||||||
|
In most cases, the target name is the name of a property, even
|
||||||
|
when it appears to be the name of an attribute.
|
||||||
|
So in this case, `src` is the name of the `<img>` element property.
|
||||||
|
|
||||||
|
Element properties may be the more common targets,
|
||||||
|
but Angular looks first to see if the name is a property of a known directive,
|
||||||
|
as it is in the following example:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="class-binding" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Technically, Angular is matching the name to a directive `@Input()`,
|
||||||
|
one of the property names listed in the directive's `inputs` array
|
||||||
|
or a property decorated with `@Input()`.
|
||||||
|
Such inputs map to the directive's own properties.
|
||||||
|
|
||||||
|
If the name fails to match a property of a known directive or element, Angular reports an “unknown directive” error.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
Though the target name is usually the name of a property,
|
||||||
|
there is an automatic attribute-to-property mapping in Angular for
|
||||||
|
several common attributes. These include `class`/`className`, `innerHtml`/`innerHTML`, and
|
||||||
|
`tabindex`/`tabIndex`.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
## Avoid side effects
|
||||||
|
|
||||||
|
Evaluation of a template expression should have no visible side effects.
|
||||||
|
The expression language itself, or the way you write template expressions,
|
||||||
|
helps to a certain extent;
|
||||||
|
you can't assign a value to anything in a property binding expression
|
||||||
|
nor use the increment and decrement operators.
|
||||||
|
|
||||||
|
For example, you could have an expression that invoked a property or method that had
|
||||||
|
side effects. The expression could call something like `getFoo()` where only you
|
||||||
|
know what `getFoo()` does. If `getFoo()` changes something
|
||||||
|
and you happen to be binding to that something,
|
||||||
|
Angular may or may not display the changed value. Angular may detect the
|
||||||
|
change and throw a warning error.
|
||||||
|
As a best practice, stick to properties and to methods that return
|
||||||
|
values and avoid side effects.
|
||||||
|
|
||||||
|
## Return the proper type
|
||||||
|
|
||||||
|
The template expression should evaluate to the type of value
|
||||||
|
that the target property expects.
|
||||||
|
Return a string if the target property expects a string, a number if it
|
||||||
|
expects a number, an object if it expects an object, and so on.
|
||||||
|
|
||||||
|
In the following example, the `childItem` property of the `ItemDetailComponent` expects a string, which is exactly what you're sending in the property binding:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="model-property-binding" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
You can confirm this by looking in the `ItemDetailComponent` where the `@Input` type is set to a string:
|
||||||
|
<code-example path="property-binding/src/app/item-detail/item-detail.component.ts" region="input-type" header="src/app/item-detail/item-detail.component.ts (setting the @Input() type)"></code-example>
|
||||||
|
|
||||||
|
As you can see here, the `parentItem` in `AppComponent` is a string, which the `ItemDetailComponent` expects:
|
||||||
|
<code-example path="property-binding/src/app/app.component.ts" region="parent-data-type" header="src/app/app.component.ts"></code-example>
|
||||||
|
|
||||||
|
### Passing in an object
|
||||||
|
|
||||||
|
The previous simple example showed passing in a string. To pass in an object,
|
||||||
|
the syntax and thinking are the same.
|
||||||
|
|
||||||
|
In this scenario, `ItemListComponent` is nested within `AppComponent` and the `items` property expects an array of objects.
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="pass-object" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
The `items` property is declared in the `ItemListComponent` with a type of `Item` and decorated with `@Input()`:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/item-list/item-list.component.ts" region="item-input" header="src/app/item-list.component.ts"></code-example>
|
||||||
|
|
||||||
|
In this sample app, an `Item` is an object that has two properties; an `id` and a `name`.
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/item.ts" region="item-class" header="src/app/item.ts"></code-example>
|
||||||
|
|
||||||
|
While a list of items exists in another file, `mock-items.ts`, you can
|
||||||
|
specify a different item in `app.component.ts` so that the new item will render:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.ts" region="pass-object" header="src/app.component.ts"></code-example>
|
||||||
|
|
||||||
|
You just have to make sure, in this case, that you're supplying an array of objects because that's the type of `Item` and is what the nested component, `ItemListComponent`, expects.
|
||||||
|
|
||||||
|
In this example, `AppComponent` specifies a different `item` object
|
||||||
|
(`currentItems`) and passes it to the nested `ItemListComponent`. `ItemListComponent` was able to use `currentItems` because it matches what an `Item` object is according to `item.ts`. The `item.ts` file is where
|
||||||
|
`ItemListComponent` gets its definition of an `item`.
|
||||||
|
|
||||||
|
## Remember the brackets
|
||||||
|
|
||||||
|
The brackets, `[]`, tell Angular to evaluate the template expression.
|
||||||
|
If you omit the brackets, Angular treats the string as a constant
|
||||||
|
and *initializes the target property* with that string:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="no-evaluation" header="src/app.component.html"></code-example>
|
||||||
|
|
||||||
|
|
||||||
|
Omitting the brackets will render the string
|
||||||
|
`parentItem`, not the value of `parentItem`.
|
||||||
|
|
||||||
|
## One-time string initialization
|
||||||
|
|
||||||
|
You *should* omit the brackets when all of the following are true:
|
||||||
|
|
||||||
|
* The target property accepts a string value.
|
||||||
|
* The string is a fixed value that you can put directly into the template.
|
||||||
|
* This initial value never changes.
|
||||||
|
|
||||||
|
You routinely initialize attributes this way in standard HTML, and it works
|
||||||
|
just as well for directive and component property initialization.
|
||||||
|
The following example initializes the `prefix` property of the `StringInitComponent` to a fixed string,
|
||||||
|
not a template expression. Angular sets it and forgets about it.
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="string-init" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
The `[item]` binding, on the other hand, remains a live binding to the component's `currentItems` property.
|
||||||
|
|
||||||
|
## Property binding vs. interpolation
|
||||||
|
|
||||||
|
You often have a choice between interpolation and property binding.
|
||||||
|
The following binding pairs do the same thing:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="property-binding-interpolation" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Interpolation is a convenient alternative to property binding in
|
||||||
|
many cases. When rendering data values as strings, there is no
|
||||||
|
technical reason to prefer one form to the other, though readability
|
||||||
|
tends to favor interpolation. However, *when setting an element
|
||||||
|
property to a non-string data value, you must use property binding*.
|
||||||
|
|
||||||
|
## Content security
|
||||||
|
|
||||||
|
Imagine the following malicious content.
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.ts" region="malicious-content" header="src/app/app.component.ts"></code-example>
|
||||||
|
|
||||||
|
In the component template, the content might be used with interpolation:
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="malicious-interpolated" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Fortunately, Angular data binding is on alert for dangerous HTML. In the above case,
|
||||||
|
the HTML displays as is, and the Javascript does not execute. Angular **does not**
|
||||||
|
allow HTML with script tags to leak into the browser, neither with interpolation
|
||||||
|
nor property binding.
|
||||||
|
|
||||||
|
In the following example, however, Angular [sanitizes](guide/security#sanitization-and-security-contexts)
|
||||||
|
the values before displaying them.
|
||||||
|
|
||||||
|
<code-example path="property-binding/src/app/app.component.html" region="malicious-content" header="src/app/app.component.html"></code-example>
|
||||||
|
|
||||||
|
Interpolation handles the `<script>` tags differently than
|
||||||
|
property binding but both approaches render the
|
||||||
|
content harmlessly. The following is the browser output
|
||||||
|
of the `evilTitle` examples.
|
||||||
|
|
||||||
|
<code-example language="bash">
|
||||||
|
"Template <script>alert("evil never sleeps")</script> Syntax" is the interpolated evil title.
|
||||||
|
"Template alert("evil never sleeps")Syntax" is the property bound evil title.
|
||||||
|
</code-example>
|
@ -52,6 +52,14 @@ Any component created within a lazy loaded module’s context, such as by router
|
|||||||
|
|
||||||
Though you can provide services by lazy loading modules, not all services can be lazy loaded. For instance, some modules only work in the root module, such as the Router. The Router works with the global location object in the browser.
|
Though you can provide services by lazy loading modules, not all services can be lazy loaded. For instance, some modules only work in the root module, such as the Router. The Router works with the global location object in the browser.
|
||||||
|
|
||||||
|
As of Angular version 9, you can provide a new instance of a service with each lazy loaded module. The following code adds this functionality to `UserService`.
|
||||||
|
|
||||||
|
<code-example path="providers/src/app/user.service.2.ts" header="src/app/user.service.ts"></code-example>
|
||||||
|
|
||||||
|
With `providedIn: 'any'`, all eagerly loaded modules share a singleton instance; however, lazy loaded modules each get their own unique instance, as shown in the following diagram.
|
||||||
|
|
||||||
|
<img src="generated/images/guide/providers/any-provider.svg" alt="any-provider-scope" class="left">
|
||||||
|
|
||||||
|
|
||||||
## Limiting provider scope with components
|
## Limiting provider scope with components
|
||||||
|
|
||||||
|
@ -101,6 +101,7 @@ The following table provides the status for Angular versions under support.
|
|||||||
|
|
||||||
Version | Status | Released | Active Ends | LTS Ends
|
Version | Status | Released | Active Ends | LTS Ends
|
||||||
------- | ------ | ------------ | ------------ | ------------
|
------- | ------ | ------------ | ------------ | ------------
|
||||||
|
^10.0.0 | Active | Jun 24, 2020 | Dec 24, 2020 | Dec 24, 2021
|
||||||
^9.0.0 | Active | Feb 06, 2020 | Aug 06, 2020 | Aug 06, 2021
|
^9.0.0 | Active | Feb 06, 2020 | Aug 06, 2020 | Aug 06, 2021
|
||||||
^8.0.0 | LTS | May 28, 2019 | Nov 28, 2019 | Nov 28, 2020
|
^8.0.0 | LTS | May 28, 2019 | Nov 28, 2019 | Nov 28, 2020
|
||||||
|
|
||||||
|
@ -111,7 +111,7 @@ Let's assume that we are routing from the *Home => About*.
|
|||||||
|
|
||||||
The animation code does the following after styling the views:
|
The animation code does the following after styling the views:
|
||||||
|
|
||||||
* `query(':enter style({ left: '-100%'})` matches the view that is added and hides the newly added view by positioning it to the far left.
|
* `query(':enter', style({ left: '-100%' }))` matches the view that is added and hides the newly added view by positioning it to the far left.
|
||||||
* Calls `animateChild()` on the view that is leaving, to run its child animations.
|
* Calls `animateChild()` on the view that is leaving, to run its child animations.
|
||||||
* Uses `group()` function to make the inner animations run in parallel.
|
* Uses `group()` function to make the inner animations run in parallel.
|
||||||
* Within the `group()` function:
|
* Within the `group()` function:
|
||||||
|
2893
aio/content/guide/router-tutorial-toh.md
Normal file
2893
aio/content/guide/router-tutorial-toh.md
Normal file
File diff suppressed because it is too large
Load Diff
@ -4,7 +4,7 @@ This tutorial describes how you can build a single-page application, SPA that us
|
|||||||
|
|
||||||
|
|
||||||
In an SPA, all of your application's functions exist in a single HTML page.
|
In an SPA, all of your application's functions exist in a single HTML page.
|
||||||
As users access your application's features, the browser needs to render only the parts that matter to the user, instead of loading a new page. This pattern can significantly improve your application's user exprience.
|
As users access your application's features, the browser needs to render only the parts that matter to the user, instead of loading a new page. This pattern can significantly improve your application's user experience.
|
||||||
|
|
||||||
To define how users navigate through your application, you use routes. You can add routes to define how users navigate from one part of your application to another.
|
To define how users navigate through your application, you use routes. You can add routes to define how users navigate from one part of your application to another.
|
||||||
You can also configure routes to guard against unexpected or unauthorized behavior.
|
You can also configure routes to guard against unexpected or unauthorized behavior.
|
||||||
@ -105,7 +105,7 @@ In this section, you'll define two routes:
|
|||||||
* The route `/crisis-center` opens the `crisis-center` component.
|
* The route `/crisis-center` opens the `crisis-center` component.
|
||||||
* The route `/heroes-list` opens the `heroes-list` component.
|
* The route `/heroes-list` opens the `heroes-list` component.
|
||||||
|
|
||||||
A route definition is a JavaScript object. Each route typically has two propteries. The first property, `path`, is a string
|
A route definition is a JavaScript object. Each route typically has two properties. The first property, `path`, is a string
|
||||||
that specifies the URL path for the route. The second property, `component`, is a string that specifies
|
that specifies the URL path for the route. The second property, `component`, is a string that specifies
|
||||||
what component your application should display for that path.
|
what component your application should display for that path.
|
||||||
|
|
||||||
@ -220,7 +220,7 @@ Now when you open your application, it displays the `heroes-list` component by d
|
|||||||
## Adding a 404 page
|
## Adding a 404 page
|
||||||
|
|
||||||
It is possible for a user to try to access a route that you have not defined. To account for
|
It is possible for a user to try to access a route that you have not defined. To account for
|
||||||
this behavior, a best practice is to display a 404 page. In this section, you'll create a 404 page and
|
this behavior, the best practice is to display a 404 page. In this section, you'll create a 404 page and
|
||||||
update your route configuration to show that page for any unspecified routes.
|
update your route configuration to show that page for any unspecified routes.
|
||||||
|
|
||||||
1. From the terminal, create a new component, `PageNotFound`.
|
1. From the terminal, create a new component, `PageNotFound`.
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -1,7 +1,7 @@
|
|||||||
# Schematics for libraries
|
# Schematics for libraries
|
||||||
|
|
||||||
When you create an Angular library, you can provide and package it with schematics that integrate it with the Angular CLI.
|
When you create an Angular library, you can provide and package it with schematics that integrate it with the Angular CLI.
|
||||||
With your schematics, your users can use `ng add` to install an initial version of your library,
|
With your schematics, your users can use `ng add` to install an initial version of your library,
|
||||||
`ng generate` to create artifacts defined in your library, and `ng update` to adjust their project for a new version of your library that introduces breaking changes.
|
`ng generate` to create artifacts defined in your library, and `ng update` to adjust their project for a new version of your library that introduces breaking changes.
|
||||||
|
|
||||||
All three types of schematics can be part of a collection that you package with your library.
|
All three types of schematics can be part of a collection that you package with your library.
|
||||||
@ -57,6 +57,20 @@ The task uses the user's preferred package manager to add the library to the pro
|
|||||||
In this example, the function receives the current `Tree` and returns it without any modifications.
|
In this example, the function receives the current `Tree` and returns it without any modifications.
|
||||||
If you need to, you can do additional setup when your package is installed, such as generating files, updating configuration, or any other initial setup your library requires.
|
If you need to, you can do additional setup when your package is installed, such as generating files, updating configuration, or any other initial setup your library requires.
|
||||||
|
|
||||||
|
### Define dependency type
|
||||||
|
|
||||||
|
Use the `save` option of `ng-add` to configure if the library should be added to the `dependencies`, the `devDepedencies`, or not saved at all in the project's `package.json` configuration file.
|
||||||
|
|
||||||
|
<code-example header="projects/my-lib/package.json (ng-add Reference)" path="schematics-for-libraries/projects/my-lib/package.json" region="ng-add">
|
||||||
|
</code-example>
|
||||||
|
|
||||||
|
Possible values are:
|
||||||
|
|
||||||
|
* `false` - Don't add the package to package.json
|
||||||
|
* `true` - Add the package to the dependencies
|
||||||
|
* `"dependencies"` - Add the package to the dependencies
|
||||||
|
* `"devDependencies"` - Add the package to the devDependencies
|
||||||
|
|
||||||
## Building your schematics
|
## Building your schematics
|
||||||
|
|
||||||
To bundle your schematics together with your library, you must configure the library to build the schematics separately, then add them to the bundle.
|
To bundle your schematics together with your library, you must configure the library to build the schematics separately, then add them to the bundle.
|
||||||
@ -115,10 +129,10 @@ When you add a schematic to the collection, you have to point to it in the colle
|
|||||||
<code-example header="projects/my-lib/schematics/my-service/schema.json (Schematic JSON Schema)" path="schematics-for-libraries/projects/my-lib/schematics/my-service/schema.json">
|
<code-example header="projects/my-lib/schematics/my-service/schema.json (Schematic JSON Schema)" path="schematics-for-libraries/projects/my-lib/schematics/my-service/schema.json">
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
* *id* : A unique id for the schema in the collection.
|
* *id*: A unique id for the schema in the collection.
|
||||||
* *title* : A human-readable description of the schema.
|
* *title*: A human-readable description of the schema.
|
||||||
* *type* : A descriptor for the type provided by the properties.
|
* *type*: A descriptor for the type provided by the properties.
|
||||||
* *properties* : An object that defines the available options for the schematic.
|
* *properties*: An object that defines the available options for the schematic.
|
||||||
|
|
||||||
Each option associates key with a type, description, and optional alias.
|
Each option associates key with a type, description, and optional alias.
|
||||||
The type defines the shape of the value you expect, and the description is displayed when the user requests usage help for your schematic.
|
The type defines the shape of the value you expect, and the description is displayed when the user requests usage help for your schematic.
|
||||||
@ -130,9 +144,9 @@ When you add a schematic to the collection, you have to point to it in the colle
|
|||||||
<code-example header="projects/my-lib/schematics/my-service/schema.ts (Schematic Interface)" path="schematics-for-libraries/projects/my-lib/schematics/my-service/schema.ts">
|
<code-example header="projects/my-lib/schematics/my-service/schema.ts (Schematic Interface)" path="schematics-for-libraries/projects/my-lib/schematics/my-service/schema.ts">
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
* *name* : The name you want to provide for the created service.
|
* *name*: The name you want to provide for the created service.
|
||||||
* *path* : Overrides the path provided to the schematic. The default path value is based on the current working directory.
|
* *path*: Overrides the path provided to the schematic. The default path value is based on the current working directory.
|
||||||
* *project* : Provides a specific project to run the schematic on. In the schematic, you can provide a default if the option is not provided by the user.
|
* *project*: Provides a specific project to run the schematic on. In the schematic, you can provide a default if the option is not provided by the user.
|
||||||
|
|
||||||
### Add template files
|
### Add template files
|
||||||
|
|
||||||
@ -169,10 +183,9 @@ The Schematics framework provides a file templating system, which supports both
|
|||||||
The system operates on placeholders defined inside files or paths that loaded in the input `Tree`.
|
The system operates on placeholders defined inside files or paths that loaded in the input `Tree`.
|
||||||
It fills these in using values passed into the `Rule`.
|
It fills these in using values passed into the `Rule`.
|
||||||
|
|
||||||
For details of these data structure and syntax, see the [Schematics README](https://github.com/angular/angular-cli/blob/master/packages/angular_devkit/schematics/README.md).
|
For details of these data structures and syntax, see the [Schematics README](https://github.com/angular/angular-cli/blob/master/packages/angular_devkit/schematics/README.md).
|
||||||
|
|
||||||
|
1. Create the main file `index.ts` and add the source code for your schematic factory function.
|
||||||
1. Create the main file, `index.ts` and add the source code for your schematic factory function.
|
|
||||||
|
|
||||||
1. First, import the schematics definitions you will need. The Schematics framework offers many utility functions to create and use rules when running a schematic.
|
1. First, import the schematics definitions you will need. The Schematics framework offers many utility functions to create and use rules when running a schematic.
|
||||||
|
|
||||||
@ -271,7 +284,6 @@ For more information about rules and utility methods, see [Provided Rules](https
|
|||||||
|
|
||||||
After you build your library and schematics, you can install the schematics collection to run against your project. The steps below show you how to generate a service using the schematic you created above.
|
After you build your library and schematics, you can install the schematics collection to run against your project. The steps below show you how to generate a service using the schematic you created above.
|
||||||
|
|
||||||
|
|
||||||
### Build your library and schematics
|
### Build your library and schematics
|
||||||
|
|
||||||
From the root of your workspace, run the `ng build` command for your library.
|
From the root of your workspace, run the `ng build` command for your library.
|
||||||
|
@ -31,28 +31,27 @@ You can use these events to notify the user of a pending update or to refresh th
|
|||||||
|
|
||||||
### Checking for updates
|
### Checking for updates
|
||||||
|
|
||||||
It's possible to ask the service worker to check if any updates have been deployed to the server. You might choose to do this if you have a site that changes frequently or want updates to happen on a schedule.
|
It's possible to ask the service worker to check if any updates have been deployed to the server.
|
||||||
|
The service worker checks for updates during initialization and on each navigation request—that is, when the user navigates from a different address to your app.
|
||||||
|
However, you might choose to manually check for updates if you have a site that changes frequently or want updates to happen on a schedule.
|
||||||
|
|
||||||
Do this with the `checkForUpdate()` method:
|
Do this with the `checkForUpdate()` method:
|
||||||
|
|
||||||
<code-example path="service-worker-getting-started/src/app/check-for-update.service.ts" header="check-for-update.service.ts"></code-example>
|
<code-example path="service-worker-getting-started/src/app/check-for-update.service.ts" header="check-for-update.service.ts"></code-example>
|
||||||
|
|
||||||
|
|
||||||
This method returns a `Promise` which indicates that the update check has completed successfully, though it does not indicate whether an update was discovered as a result of the check. Even if one is found, the service worker must still successfully download the changed files, which can fail. If successful, the `available` event will indicate availability of a new version of the app.
|
This method returns a `Promise` which indicates that the update check has completed successfully, though it does not indicate whether an update was discovered as a result of the check. Even if one is found, the service worker must still successfully download the changed files, which can fail. If successful, the `available` event will indicate availability of a new version of the app.
|
||||||
|
|
||||||
<div class="alert is-important">
|
<div class="alert is-important">
|
||||||
|
|
||||||
In order to avoid negatively affecting the initial rendering, `ServiceWorkerModule` will by default
|
In order to avoid negatively affecting the initial rendering of the page, `ServiceWorkerModule` waits for up to 30 seconds by default for the app to stabilize, before registering the ServiceWorker script.
|
||||||
wait for the app to stabilize, before registering the ServiceWorker script. Constantly polling for
|
Constantly polling for updates, for example, with [setInterval()](https://developer.mozilla.org/en-US/docs/Web/API/WindowOrWorkerGlobalScope/setInterval) or RxJS' [interval()](https://rxjs.dev/api/index/function/interval), will prevent the app from stabilizing and the ServiceWorker script will not be registered with the browser until the 30 seconds upper limit is reached.
|
||||||
updates, e.g. with `interval()`, will prevent the app from stabilizing and the ServiceWorker
|
|
||||||
script will never be registered with the browser.
|
|
||||||
|
|
||||||
You can avoid that by waiting for the app to stabilize first, before starting to poll for updates
|
|
||||||
(as shown in the example above).
|
|
||||||
|
|
||||||
Note that this is true for any kind of polling done by your application.
|
Note that this is true for any kind of polling done by your application.
|
||||||
Check the {@link ApplicationRef#isStable isStable} documentation for more information.
|
Check the {@link ApplicationRef#isStable isStable} documentation for more information.
|
||||||
|
|
||||||
|
You can avoid that delay by waiting for the app to stabilize first, before starting to poll for updates, as shown in the example above.
|
||||||
|
Alternatively, you might want to define a different {@link SwRegistrationOptions#registrationStrategy registration strategy} for the ServiceWorker.
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
### Forcing update activation
|
### Forcing update activation
|
||||||
@ -61,7 +60,12 @@ If the current tab needs to be updated to the latest app version immediately, it
|
|||||||
|
|
||||||
<code-example path="service-worker-getting-started/src/app/prompt-update.service.ts" header="prompt-update.service.ts" region="sw-activate"></code-example>
|
<code-example path="service-worker-getting-started/src/app/prompt-update.service.ts" header="prompt-update.service.ts" region="sw-activate"></code-example>
|
||||||
|
|
||||||
Doing this could break lazy-loading in currently running apps, especially if the lazy-loaded chunks use filenames with hashes, which change every version.
|
<div class="alert is-important">
|
||||||
|
|
||||||
|
Calling `activateUpdate()` without reloading the page could break lazy-loading in a currently running app, especially if the lazy-loaded chunks use filenames with hashes, which change every version.
|
||||||
|
Therefore, it is recommended to reload the page once the promise returned by `activateUpdate()` is resolved.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
## More on Angular service workers
|
## More on Angular service workers
|
||||||
|
|
||||||
|
@ -193,6 +193,21 @@ The Angular service worker can use either of two caching strategies for data res
|
|||||||
|
|
||||||
* `freshness` optimizes for currency of data, preferentially fetching requested data from the network. Only if the network times out, according to `timeout`, does the request fall back to the cache. This is useful for resources that change frequently; for example, account balances.
|
* `freshness` optimizes for currency of data, preferentially fetching requested data from the network. Only if the network times out, according to `timeout`, does the request fall back to the cache. This is useful for resources that change frequently; for example, account balances.
|
||||||
|
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
You can also emulate a third strategy, [staleWhileRevalidate](https://developers.google.com/web/fundamentals/instant-and-offline/offline-cookbook/#stale-while-revalidate), which returns cached data (if available), but also fetches fresh data from the network in the background for next time.
|
||||||
|
To use this strategy set `strategy` to `freshness` and `timeout` to `0u` in `cacheConfig`.
|
||||||
|
|
||||||
|
This will essentially do the following:
|
||||||
|
|
||||||
|
1. Try to fetch from the network first.
|
||||||
|
2. If the network request does not complete after 0ms (i.e. immediately), fall back to the cache (ignoring cache age).
|
||||||
|
3. Once the network request completes, update the cache for future requests.
|
||||||
|
4. If the resource does not exist in the cache, wait for the network request anyway.
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
### `cacheQueryOptions`
|
### `cacheQueryOptions`
|
||||||
|
|
||||||
See [assetGroups](#assetgroups) for details.
|
See [assetGroups](#assetgroups) for details.
|
||||||
|
@ -107,7 +107,7 @@ Notice that all of the files the browser needs to render this application are ca
|
|||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
Pay attention to two key points:
|
Pay attention to two key points:
|
||||||
|
|
||||||
1. The generated `ngsw-config.json` includes a limited list of cachable fonts and images extentions. In some cases, you might want to modify the glob pattern to suit your needs.
|
1. The generated `ngsw-config.json` includes a limited list of cacheable fonts and images extentions. In some cases, you might want to modify the glob pattern to suit your needs.
|
||||||
|
|
||||||
1. If `resourcesOutputPath` or `assets` paths are modified after the generation of configuration file, you need to change the paths manually in `ngsw-config.json`.
|
1. If `resourcesOutputPath` or `assets` paths are modified after the generation of configuration file, you need to change the paths manually in `ngsw-config.json`.
|
||||||
</div>
|
</div>
|
||||||
|
@ -25,49 +25,48 @@ To use the Angular framework, you should be familiar with the following:
|
|||||||
|
|
||||||
Knowledge of [TypeScript](https://www.typescriptlang.org/) is helpful, but not required.
|
Knowledge of [TypeScript](https://www.typescriptlang.org/) is helpful, but not required.
|
||||||
|
|
||||||
|
To install Angular on your local system, you need the following:
|
||||||
|
|
||||||
{@a nodejs}
|
{@a nodejs}
|
||||||
### Node.js
|
|
||||||
|
|
||||||
Make sure your development environment includes `Node.js®` and an npm package manager.
|
* **Node.js**
|
||||||
|
|
||||||
|
Angular requires a [current, active LTS, or maintenance LTS](https://nodejs.org/about/releases) version of Node.js.
|
||||||
|
|
||||||
Angular requires a [current, active LTS, or maintenance LTS](https://nodejs.org/about/releases/) version of `Node.js`. See the `engines` key for the specific version requirements in our [package.json](https://unpkg.com/@angular/cli/package.json).
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
* To check your version, run `node -v` in a terminal/console window.
|
For information about specific version requirements, see the `engines` key in the [package.json](https://unpkg.com/@angular/cli/package.json) file.
|
||||||
|
|
||||||
* To get `Node.js`, go to [nodejs.org](https://nodejs.org "Nodejs.org").
|
</div>
|
||||||
|
|
||||||
|
For more information on installing Node.js, see [nodejs.org](http://nodejs.org "Nodejs.org").
|
||||||
|
If you are unsure what version of Node.js runs on your system, run `node -v` in a terminal window.
|
||||||
|
|
||||||
{@a npm}
|
{@a npm}
|
||||||
### npm package manager
|
|
||||||
|
|
||||||
Angular, the Angular CLI, and Angular apps depend on features and functionality provided by libraries that are available as [npm packages](https://docs.npmjs.com/getting-started/what-is-npm). To download and install npm packages, you must have an npm package manager.
|
* **npm package manager**
|
||||||
|
|
||||||
This setup guide uses the [npm client](https://docs.npmjs.com/cli/install) command line interface, which is installed with `Node.js` by default.
|
Angular, the Angular CLI, and Angular applications depend on [npm packages](https://docs.npmjs.com/getting-started/what-is-npm) for many features and functions.
|
||||||
|
To download and install npm packages, you need an npm package manager.
|
||||||
To check that you have the npm client installed, run `npm -v` in a terminal/console window.
|
This guide uses the [npm client](https://docs.npmjs.com/cli/install) command line interface, which is installed with `Node.js` by default.
|
||||||
|
To check that you have the npm client installed, run `npm -v` in a terminal window.
|
||||||
|
|
||||||
|
|
||||||
{@a install-cli}
|
{@a install-cli}
|
||||||
|
|
||||||
## Step 1: Install the Angular CLI
|
## Install the Angular CLI
|
||||||
|
|
||||||
You use the Angular CLI
|
You use the Angular CLI to create projects, generate application and library code, and perform a variety of ongoing development tasks such as testing, bundling, and deployment.
|
||||||
to create projects, generate application and library code, and perform a variety of ongoing development tasks such as testing, bundling, and deployment.
|
|
||||||
|
|
||||||
Install the Angular CLI globally.
|
|
||||||
|
|
||||||
To install the CLI using `npm`, open a terminal/console window and enter the following command:
|
|
||||||
|
|
||||||
|
To install the Angular CLI, open a terminal window and run the following command:
|
||||||
|
|
||||||
<code-example language="sh" class="code-shell">
|
<code-example language="sh" class="code-shell">
|
||||||
npm install -g @angular/cli
|
npm install -g @angular/cli
|
||||||
|
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
{@a create-proj}
|
{@a create-proj}
|
||||||
|
|
||||||
## Step 2: Create a workspace and initial application
|
## Create a workspace and initial application
|
||||||
|
|
||||||
You develop apps in the context of an Angular [**workspace**](guide/glossary#workspace).
|
You develop apps in the context of an Angular [**workspace**](guide/glossary#workspace).
|
||||||
|
|
||||||
@ -86,16 +85,22 @@ The Angular CLI installs the necessary Angular npm packages and other dependenci
|
|||||||
|
|
||||||
The CLI creates a new workspace and a simple Welcome app, ready to run.
|
The CLI creates a new workspace and a simple Welcome app, ready to run.
|
||||||
|
|
||||||
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
You also have the option to use Angular's strict mode, which can help you write better, more maintainable code.
|
||||||
|
For more information, see [Strict mode](/guide/strict-mode).
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
{@a serve}
|
{@a serve}
|
||||||
|
|
||||||
## Step 3: Run the application
|
## Run the application
|
||||||
|
|
||||||
The Angular CLI includes a server, so that you can easily build and serve your app locally.
|
The Angular CLI includes a server, so that you can build and serve your app locally.
|
||||||
|
|
||||||
1. Go to the workspace folder (`my-app`).
|
1. Navigate to the workspace folder, such as `my-app`.
|
||||||
|
|
||||||
1. Launch the server by using the CLI command `ng serve`, with the `--open` option.
|
1. Run the following command:
|
||||||
|
|
||||||
<code-example language="sh" class="code-shell">
|
<code-example language="sh" class="code-shell">
|
||||||
cd my-app
|
cd my-app
|
||||||
@ -108,7 +113,7 @@ and rebuilds the app as you make changes to those files.
|
|||||||
The `--open` (or just `-o`) option automatically opens your browser
|
The `--open` (or just `-o`) option automatically opens your browser
|
||||||
to `http://localhost:4200/`.
|
to `http://localhost:4200/`.
|
||||||
|
|
||||||
You will see:
|
If your installation and setup was successful, you should see a page similar to the following.
|
||||||
|
|
||||||
|
|
||||||
<div class="lightbox">
|
<div class="lightbox">
|
||||||
|
46
aio/content/guide/strict-mode.md
Normal file
46
aio/content/guide/strict-mode.md
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
# Strict mode
|
||||||
|
|
||||||
|
When you create a new workspace or a project you have an option to create them in a strict mode using the `--strict` flag.
|
||||||
|
|
||||||
|
Enabling this flag initializes your new workspace or project with a few new settings that improve maintainability, help you catch bugs ahead of time, and allow the CLI to perform advanced optimizations on your application.
|
||||||
|
Additionally, applications that use these stricter settings are easier to statically analyze, which can help the `ng update` command refactor code more safely and precisely when you are updating to future versions of Angular.
|
||||||
|
|
||||||
|
Specifically, the `strict` flag does the following:
|
||||||
|
|
||||||
|
* Enables [`strict` mode in TypeScript](https://www.staging-typescript.org/tsconfig#strict), as well as other strictness flags recommended by the TypeScript team. Specifically, `forceConsistentCasingInFileNames`, `noImplicitReturns`, `noFallthroughCasesInSwitch`.
|
||||||
|
* Turns on strict Angular compiler flags [`strictTemplates`](guide/angular-compiler-options#stricttemplates) and [`strictInjectionParameters`](guide/angular-compiler-options#strictinjectionparameters)
|
||||||
|
* [Bundle size budgets](guide/build#configuring-size-budgets) have been reduced by ~75%
|
||||||
|
* Turns on [`no-any` tslint rule](https://palantir.github.io/tslint/rules/no-any/) to prevent declarations of type `any`
|
||||||
|
* [Marks your application as side-effect free](https://webpack.js.org/guides/tree-shaking/#mark-the-file-as-side-effect-free) to enable more advanced tree-shaking
|
||||||
|
|
||||||
|
You can apply these settings at the workspace and project level.
|
||||||
|
|
||||||
|
To create a new workspace and application using the strict mode, run the following command:
|
||||||
|
|
||||||
|
<code-example language="sh" class="code-shell">
|
||||||
|
|
||||||
|
ng new [project-name] --strict
|
||||||
|
|
||||||
|
</code-example>
|
||||||
|
|
||||||
|
To create a new application in the strict mode within an existing non-strict workspace, run the following command:
|
||||||
|
|
||||||
|
<code-example language="sh" class="code-shell">
|
||||||
|
|
||||||
|
ng generate application [project-name] --strict
|
||||||
|
|
||||||
|
</code-example>
|
||||||
|
|
||||||
|
{@a side-effect}
|
||||||
|
|
||||||
|
### Non-local side effects in applications
|
||||||
|
|
||||||
|
When you create projects and workspaces using the `strict` mode, you'll notice an additional `package.json` file, located in `src/app/` directory.
|
||||||
|
This file informs tools and bundlers that the code under this directory is free of non-local side effects. Non-local side effects in the application code are not common and using them is not considered a good coding pattern.
|
||||||
|
More importantly, code with these types of side effects cannot be optimized, resulting in increased bundle sizes and applications that load more slowly.
|
||||||
|
|
||||||
|
If you need more information, the following links may be helpful.
|
||||||
|
|
||||||
|
* [Tree-shaking](https://webpack.js.org/guides/tree-shaking/)
|
||||||
|
* [Dealing with side effects and pure functions in JavaScript](https://dev.to/vonheikemen/dealing-with-side-effects-and-pure-functions-in-javascript-16mg)
|
||||||
|
* [How to deal with dirty side effects in your pure function JavaScript](https://jrsinclair.com/articles/2018/how-to-deal-with-dirty-side-effects-in-your-pure-functional-javascript/)
|
@ -39,14 +39,14 @@ No brackets. No parentheses. Just `*ngIf` set to a string.
|
|||||||
|
|
||||||
You'll learn in this guide that the [asterisk (*) is a convenience notation](guide/structural-directives#asterisk)
|
You'll learn in this guide that the [asterisk (*) is a convenience notation](guide/structural-directives#asterisk)
|
||||||
and the string is a [_microsyntax_](guide/structural-directives#microsyntax) rather than the usual
|
and the string is a [_microsyntax_](guide/structural-directives#microsyntax) rather than the usual
|
||||||
[template expression](guide/template-syntax#template-expressions).
|
[template expression](guide/interpolation#template-expressions).
|
||||||
Angular desugars this notation into a marked-up `<ng-template>` that surrounds the
|
Angular desugars this notation into a marked-up `<ng-template>` that surrounds the
|
||||||
host element and its descendents.
|
host element and its descendants.
|
||||||
Each structural directive does something different with that template.
|
Each structural directive does something different with that template.
|
||||||
|
|
||||||
Three of the common, built-in structural directives—[NgIf](guide/template-syntax#ngIf),
|
Three of the common, built-in structural directives—[NgIf](guide/built-in-directives#ngIf),
|
||||||
[NgFor](guide/template-syntax#ngFor), and [NgSwitch...](guide/template-syntax#ngSwitch)—are
|
[NgFor](guide/built-in-directives#ngFor), and [NgSwitch...](guide/built-in-directives#ngSwitch)—are
|
||||||
described in the [_Template Syntax_](guide/template-syntax) guide and seen in samples throughout the Angular documentation.
|
described in the [Built-in directives](guide/built-in-directives) guide and seen in samples throughout the Angular documentation.
|
||||||
Here's an example of them in a template:
|
Here's an example of them in a template:
|
||||||
|
|
||||||
|
|
||||||
@ -96,7 +96,7 @@ Technically it's a directive with a template.
|
|||||||
|
|
||||||
An [*attribute* directive](guide/attribute-directives) changes the appearance or behavior
|
An [*attribute* directive](guide/attribute-directives) changes the appearance or behavior
|
||||||
of an element, component, or another directive.
|
of an element, component, or another directive.
|
||||||
For example, the built-in [`NgStyle`](guide/template-syntax#ngStyle) directive
|
For example, the built-in [`NgStyle`](guide/built-in-directives#ngStyle) directive
|
||||||
changes several element styles at the same time.
|
changes several element styles at the same time.
|
||||||
|
|
||||||
You can apply many _attribute_ directives to one host element.
|
You can apply many _attribute_ directives to one host element.
|
||||||
@ -251,7 +251,7 @@ You enable these features in the string assigned to `ngFor`, which you write in
|
|||||||
|
|
||||||
Everything _outside_ the `ngFor` string stays with the host element
|
Everything _outside_ the `ngFor` string stays with the host element
|
||||||
(the `<div>`) as it moves inside the `<ng-template>`.
|
(the `<div>`) as it moves inside the `<ng-template>`.
|
||||||
In this example, the `[ngClass]="odd"` stays on the `<div>`.
|
In this example, the `[class.odd]="odd"` stays on the `<div>`.
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
@ -440,7 +440,7 @@ There are several such variables in this example: `hero`, `i`, and `odd`.
|
|||||||
All are preceded by the keyword `let`.
|
All are preceded by the keyword `let`.
|
||||||
|
|
||||||
A _template input variable_ is **_not_** the same as a
|
A _template input variable_ is **_not_** the same as a
|
||||||
[template _reference_ variable](guide/template-syntax#ref-vars),
|
[template _reference_ variable](guide/template-reference-variables),
|
||||||
neither _semantically_ nor _syntactically_.
|
neither _semantically_ nor _syntactically_.
|
||||||
|
|
||||||
You declare a template _input_ variable using the `let` keyword (`let hero`).
|
You declare a template _input_ variable using the `let` keyword (`let hero`).
|
||||||
@ -786,7 +786,7 @@ That means the directive needs an `appUnless` property, decorated with `@Input`
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
Read about `@Input` in the [_Template Syntax_](guide/template-syntax#inputs-outputs) guide.
|
Read about `@Input` in the [`@Input()` and `@Output()` properties](guide/inputs-outputs) guide.
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
@ -879,12 +879,14 @@ export type LoadingState<T> = Loaded<T> | Loading;
|
|||||||
export class IfLoadedDirective<T> {
|
export class IfLoadedDirective<T> {
|
||||||
@Input('ifLoaded') set state(state: LoadingState<T>) {}
|
@Input('ifLoaded') set state(state: LoadingState<T>) {}
|
||||||
static ngTemplateGuard_state<T>(dir: IfLoadedDirective<T>, expr: LoadingState<T>): expr is Loaded<T> { return true; };
|
static ngTemplateGuard_state<T>(dir: IfLoadedDirective<T>, expr: LoadingState<T>): expr is Loaded<T> { return true; };
|
||||||
|
}
|
||||||
|
|
||||||
export interface Person {
|
export interface Person {
|
||||||
name: string;
|
name: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
template: `<div *ifLoaded="state">{{ state.data }}</div>`,
|
template: `<div *ifLoaded="state">{{ state.data }}</div>`,
|
||||||
})
|
})
|
||||||
export class AppComponent {
|
export class AppComponent {
|
||||||
state: LoadingState<Person>;
|
state: LoadingState<Person>;
|
||||||
|
@ -3286,7 +3286,7 @@ helps instantly identify which members of the component serve which purpose.
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
**Why?** Angular allows for an [alternative syntax](guide/template-syntax#binding-syntax) `on-*`. If the event itself was prefixed with `on` this would result in an `on-onEvent` binding expression.
|
**Why?** Angular allows for an [alternative syntax](guide/binding-syntax) `on-*`. If the event itself was prefixed with `on` this would result in an `on-onEvent` binding expression.
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user