Description
🚀 Feature request
Command (mark with an x
)
- new
- build
- serve
- test
- e2e
- generate
- add
- update
- lint
- extract-i18n
- run
- config
- help
- version
- doc
Description
Loading third-party scripts in a suboptimal way can cause performance regressions. A typical example is adding an SDK as script tag directly to the page head, which will block the loading of first-party scripts and delay hydration/CSR if done incorrectly.
Describe the solution you'd like
Expand scripts
the functionality already available in angular.json
, allowing developers to configure different script loading strategies. Project Aurora identified three script loading strategies that we can reuse:
afterInteractive
- this strategy would prioritize first-party scripts by executing third-party scripts after the page has been hydrated or client-side rendered.beforeInteractive
- execute third-party scripts before first-party scripts. This strategy is particularly important for loading polyfills. Since we already havepolyfill.ts
, I'm a little hesitant to include it in the Angular CLI as part of the first feature iteration. I'd suggest keeping it out of scope and collecting feedback in the meantime.lazyOnload
- lowest priority. We load these scripts and execute them inrequestIdleCallback
. Examples for libraries using this strategy for prefetching are quicklink, ngx-quicklink, and guess-js.
We'd sometimes need a hook after the script has been loaded and executed if we load it asynchronously (afterInteractive
or lazyOnload
). For example, rendering a payment dialog after we've downloaded a third-party SDK.
Using angular.json
does not provide an evident approach that would let us accomplish this. A lower-level API that would provide the necessary functionality is to allows developers to specify id
's in the script declaration. Using the id
, developers will discover the script
element in their code and hook to its onload
event.
This way, the scripts
property in angular.json
would change from an array of strings or objects to an array of strings or objects with the following properties:
input
- the source URL or path of the script. The CLI will perform different actions depending on whetherinput
is a local path or a remote URLstrategy
-afterInteractive
,beforeInteractive
, orlazyOnload
inject
- specifies if the script should be injected or not. We preserve the current semantics of theinject
property in the script objectbundleName
- applicable for local scripts only. For external scripts, we can throw an errorid
- an optional identifier that the developer can use to get a hold of the script and hook to itsonload
event
Suppose we find a string rather than an object for a given script. In that case, we can treat it as a script object with an input
the specified string, strategy
equal to beforeInteractive
for the current major, or afterInteractive
for the next major. Ideally, we'd want the default strategy to be afterInteractive
to prevent folks from causing performance regressions, but since this would be a breaking change, we don't want to do this until v13.
In v13, we can migrate all the scripts from objects to strings using the beforeInteractive
strategy.
Describe alternatives you've considered
Project Aurora considered using a script component for alternative frameworks. Their approach easier accommodates the onLoad
callback use case but seems to fit less naturally in the Angular CLI model.