mirror of
https://github.com/angular/angular
synced 2026-05-24 09:28:37 +00:00
415 lines
24 KiB
Markdown
415 lines
24 KiB
Markdown
# Getting started with NgOptimizedImage
|
||
|
||
The `NgOptimizedImage` directive makes it easy to adopt performance best practices for loading images.
|
||
|
||
The directive ensures that the loading of the [Largest Contentful Paint (LCP)](http://web.dev/lcp) image is prioritized by:
|
||
|
||
* Automatically setting the `fetchpriority` attribute on the `<img>` tag
|
||
* Lazy loading other images by default
|
||
* Automatically generating a preconnect link tag in the document head
|
||
* Automatically generating a `srcset` attribute
|
||
* Generating a [preload hint](https://developer.mozilla.org/en-US/docs/Web/HTML/Link_types/preload) if app is using SSR
|
||
|
||
In addition to optimizing the loading of the LCP image, `NgOptimizedImage` enforces a number of image best practices, such as:
|
||
|
||
* Using [image CDN URLs to apply image optimizations](https://web.dev/image-cdns/#how-image-cdns-use-urls-to-indicate-optimization-options)
|
||
* Preventing layout shift by requiring `width` and `height`
|
||
* Warning if `width` or `height` have been set incorrectly
|
||
* Warning if the image will be visually distorted when rendered
|
||
|
||
**Note: Although the `NgOptimizedImage` directive was made a stable feature in Angular version 15, it has been backported and is available as a stable feature in versions 13.4.0 and 14.3.0 as well. **
|
||
|
||
## Getting Started
|
||
|
||
#### Step 1: Import NgOptimizedImage
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
|
||
import { NgOptimizedImage } from '@angular/common'
|
||
|
||
</code-example>
|
||
|
||
The directive is defined as a [standalone directive](/guide/standalone-components), so components should import it directly.
|
||
|
||
#### Step 2: (Optional) Set up a Loader
|
||
|
||
An image loader is not **required** in order to use NgOptimizedImage, but using one with an image CDN enables powerful performance features, including automatic `srcset`s for your images.
|
||
|
||
A brief guide for setting up a loader can be found in the [Configuring an Image Loader](#configuring-an-image-loader-for-ngoptimizedimage) section at the end of this page.
|
||
|
||
#### Step 3: Enable the directive
|
||
|
||
To activate the `NgOptimizedImage` directive, replace your image's `src` attribute with `ngSrc`.
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
|
||
<img ngSrc="cat.jpg">
|
||
|
||
</code-example>
|
||
|
||
If you're using a [built-in third-party loader](#built-in-loaders), make sure to omit the base URL path from `src`, as that will be prepended automatically by the loader.
|
||
|
||
#### Step 4: Mark images as `priority`
|
||
|
||
Always mark the [LCP image](https://web.dev/lcp/#what-elements-are-considered) on your page as `priority` to prioritize its loading.
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
|
||
<img ngSrc="cat.jpg" width="400" height="200" priority>
|
||
|
||
</code-example>
|
||
|
||
Marking an image as `priority` applies the following optimizations:
|
||
|
||
* Sets `fetchpriority=high` (read more about priority hints [here](https://web.dev/priority-hints))
|
||
* Sets `loading=eager` (read more about native lazy loading [here](https://web.dev/browser-level-image-lazy-loading))
|
||
* Automatically generates a [preload link element](https://developer.mozilla.org/en-US/docs/Web/HTML/Link_types/preload) if [rendering on the server](/guide/ssr).
|
||
|
||
Angular logs an error during development if the LCP element is an image that does not have the `priority` attribute, as this can hurt loading performance significantly. A page’s LCP element can vary based on a number of factors - such as the dimensions of a user's screen, so a page may have multiple images that should be marked `priority`. See [CSS for Web Vitals](https://web.dev/css-web-vitals/#images-and-largest-contentful-paint-lcp) for more details.
|
||
|
||
#### Step 5: Include Height and Width
|
||
|
||
In order to prevent [image-related layout shifts](https://web.dev/css-web-vitals/#images-and-layout-shifts), NgOptimizedImage requires that you specify a height and width for your image, as follows:
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
|
||
<img ngSrc="cat.jpg" width="400" height="200">
|
||
|
||
</code-example>
|
||
|
||
For **responsive images** (images which you've styled to grow and shrink relative to the viewport), the `width` and `height` attributes should be the intrinsic size of the image file. For responsive images it's also important to [set a value for `sizes`.](#responsive-images)
|
||
|
||
For **fixed size images**, the `width` and `height` attributes should reflect the desired rendered size of the image. The aspect ratio of these attributes should always match the intrinsic aspect ratio of the image.
|
||
|
||
Note: If you don't know the size of your images, consider using "fill mode" to inherit the size of the parent container, as described below:
|
||
|
||
### Using `fill` mode
|
||
|
||
In cases where you want to have an image fill a containing element, you can use the `fill` attribute. This is often useful when you want to achieve a "background image" behavior. It can also be helpful when you don't know the exact width and height of your image, but you do have a parent container with a known size that you'd like to fit your image into (see "object-fit" below).
|
||
|
||
When you add the `fill` attribute to your image, you do not need and should not include a `width` and `height`, as in this example:
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
|
||
<img ngSrc="cat.jpg" fill>
|
||
|
||
</code-example>
|
||
|
||
You can use the [object-fit](https://developer.mozilla.org/en-US/docs/Web/CSS/object-fit) CSS property to change how the image will fill its container. If you style your image with `object-fit: "contain"`, the image will maintain its aspect ratio and be "letterboxed" to fit the element. If you set `object-fit: "cover"`, the element will retain its aspect ratio, fully fill the element, and some content may be "cropped" off.
|
||
|
||
See visual examples of the above at the [MDN object-fit documentation.](https://developer.mozilla.org/en-US/docs/Web/CSS/object-fit)
|
||
|
||
You can also style your image with the [object-position property](https://developer.mozilla.org/en-US/docs/Web/CSS/object-position) to adjust its position within its containing element.
|
||
|
||
**Important note:** For the "fill" image to render properly, its parent element **must** be styled with `position: "relative"`, `position: "fixed"`, or `position: "absolute"`.
|
||
|
||
## Using placeholders
|
||
|
||
### Automatic placeholders
|
||
|
||
NgOptimizedImage can provide an automatic low-resolution placeholder for your image, as long as you're using a CDN or image host which provides automatic image resizing. To take advantage of this feature, just add the `placeholder` attribute to your image, as so:
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
|
||
<img ngSrc="cat.jpg" width="400" height="200" placeholder>
|
||
|
||
</code-example>
|
||
|
||
Adding this attribute will automatically request a second, smaller version of the image, using your specified image loader. This small image will be applied as a `background-image` style with a CSS blur while your image loads. If no image loader is provided, no placeholder image can be generated and an error will be thrown.
|
||
|
||
The default size for generated placeholders is 30px wide, but you can make it larger or smaller by specifying a value in the `IMAGE_CONFIG` provider, as seen below:
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
providers: [
|
||
{
|
||
provide: IMAGE_CONFIG,
|
||
useValue: {
|
||
placeholderResolution: 40
|
||
}
|
||
},
|
||
],
|
||
</code-example>
|
||
|
||
If you want sharp edges around your blurred placeholder, you can wrap your image in a containing `<div>` with the `overflow: hidden` style. As long as the `<div>` is the same size as the image (such as by using the `width: fit-content` style), the "fuzzy edges" of the placeholder will be hidden.
|
||
|
||
### Data URL placeholders
|
||
|
||
You can also specify a placeholder using a base64 [data URL](https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/Data_URLs) without an image loader. The data url format is `data:image/[imagetype];[data]`, where `[imagetype]` is the image format, just as `png`, and `[data]` is a base64 encoding of the image. That encoding can be done using the command line or in JavaScript. For specific commands, see [the MDN documentation](https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/Data_URLs#encoding_data_into_base64_format). An example of a data URL placeholder with truncated data is shown below:
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
|
||
<img ngSrc="cat.jpg" width="400" height="200" placeholder="data:image/png;base64,iVBORw0K...">
|
||
|
||
</code-example>
|
||
|
||
However, large data URLs increase the size of your Angular bundles and slow down page load. If you cannot use an image loader, the Angular team recommends keeping base64 placeholder images smaller than 4KB and using them exclusively on critical images. In addition to decreasing placeholder dimensions, consider changing image formats or parameters used when saving images. At very low resolutions, these parameters can have a large effect on file size.
|
||
|
||
### Non-blurred placeholders
|
||
|
||
By default, NgOptimizedImage applies a CSS blur effect to image placeholders. To render a placeholder without blur, provide a `placeholderConfig` argument with an object that includes the `blur` property, set to false. For example:
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
|
||
<img ngSrc="cat.jpg" width="400" height="200" placeholder [placeholderConfig]="{blur: false}">
|
||
|
||
</code-example>
|
||
|
||
## Adjusting image styling
|
||
|
||
Depending on the image's styling, adding `width` and `height` attributes may cause the image to render differently. `NgOptimizedImage` warns you if your image styling renders the image at a distorted aspect ratio.
|
||
|
||
You can typically fix this by adding `height: auto` or `width: auto` to your image styles. For more information, see the [web.dev article on the `<img>` tag](https://web.dev/patterns/web-vitals-patterns/images/img-tag).
|
||
|
||
If the `height` and `width` attribute on the image are preventing you from sizing the image the way you want with CSS, consider using "fill" mode instead, and styling the image's parent element.
|
||
|
||
## Performance Features
|
||
|
||
NgOptimizedImage includes a number of features designed to improve loading performance in your app. These features are described in this section.
|
||
|
||
### Resource hints
|
||
|
||
A [`preconnect` resource hint](https://web.dev/preconnect-and-dns-prefetch) for your image origin ensures that the LCP image loads as quickly as possible.
|
||
|
||
Preconnect links are automatically generated for domains provided as an argument to a [loader](#configuring-an-image-loader-for-ngoptimizedimage). If an image origin cannot be automatically identified, and no preconnect link is detected for the LCP image, `NgOptimizedImage` will warn during development. In that case, you should manually add a resource hint to `index.html`. Within the `<head>` of the document, add a `link` tag with `rel="preload"`, as shown below:
|
||
|
||
<code-example format="html" language="html">
|
||
|
||
<link rel="preconnect" href="https://my.cdn.origin" />
|
||
|
||
</code-example>
|
||
|
||
To disable preconnect warnings, inject the `PRECONNECT_CHECK_BLOCKLIST` token:
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
|
||
providers: [
|
||
{provide: PRECONNECT_CHECK_BLOCKLIST, useValue: 'https://your-domain.com'}
|
||
],
|
||
|
||
See more information on automatic preconnect generation [here](#why-is-a-preconnect-element-not-being-generated-for-my-image-domain).
|
||
|
||
</code-example>
|
||
|
||
### Request images at the correct size with automatic `srcset`
|
||
|
||
Defining a [`srcset` attribute](https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/srcset) ensures that the browser requests an image at the right size for your user's viewport, so it doesn't waste time downloading an image that's too large. `NgOptimizedImage` generates an appropriate `srcset` for the image, based on the presence and value of the [`sizes` attribute](https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/sizes) on the image tag.
|
||
|
||
#### Fixed-size images
|
||
|
||
If your image should be "fixed" in size (i.e. the same size across devices, except for [pixel density](https://web.dev/codelab-density-descriptors/)), there is no need to set a `sizes` attribute. A `srcset` can be generated automatically from the image's width and height attributes with no further input required.
|
||
|
||
Example srcset generated: `<img ... srcset="image-400w.jpg 1x, image-800w.jpg 2x">`
|
||
|
||
#### Responsive images
|
||
|
||
If your image should be responsive (i.e. grow and shrink according to viewport size), then you will need to define a [`sizes` attribute](https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/sizes) to generate the `srcset`.
|
||
|
||
If you haven't used `sizes` before, a good place to start is to set it based on viewport width. For example, if your CSS causes the image to fill 100% of viewport width, set `sizes` to `100vw` and the browser will select the image in the `srcset` that is closest to the viewport width (after accounting for pixel density). If your image is only likely to take up half the screen (ex: in a sidebar), set `sizes` to `50vw` to ensure the browser selects a smaller image. And so on.
|
||
|
||
If you find that the above does not cover your desired image behavior, see the documentation on [advanced sizes values](#advanced-sizes-values).
|
||
|
||
By default, the responsive breakpoints are:
|
||
|
||
`[16, 32, 48, 64, 96, 128, 256, 384, 640, 750, 828, 1080, 1200, 1920, 2048, 3840]`
|
||
|
||
If you would like to customize these breakpoints, you can do so using the `IMAGE_CONFIG` provider:
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
providers: [
|
||
{
|
||
provide: IMAGE_CONFIG,
|
||
useValue: {
|
||
breakpoints: [16, 48, 96, 128, 384, 640, 750, 828, 1080, 1200, 1920]
|
||
}
|
||
},
|
||
],
|
||
</code-example>
|
||
|
||
If you would like to manually define a `srcset` attribute, you can provide your own using the `ngSrcset` attribute:
|
||
|
||
<code-example format="html" language="html">
|
||
|
||
<img ngSrc="hero.jpg" ngSrcset="100w, 200w, 300w">
|
||
|
||
</code-example>
|
||
|
||
If the `ngSrcset` attribute is present, `NgOptimizedImage` generates and sets the `srcset` based on the sizes included. Do not include image file names in `ngSrcset` - the directive infers this information from `ngSrc`. The directive supports both width descriptors (e.g. `100w`) and density descriptors (e.g. `1x`).
|
||
|
||
<code-example format="html" language="html">
|
||
|
||
<img ngSrc="hero.jpg" ngSrcset="100w, 200w, 300w" sizes="50vw">
|
||
|
||
</code-example>
|
||
|
||
### Disabling automatic srcset generation
|
||
|
||
To disable srcset generation for a single image, you can add the `disableOptimizedSrcset` attribute on the image:
|
||
|
||
<code-example format="html" language="html">
|
||
|
||
<img ngSrc="about.jpg" disableOptimizedSrcset>
|
||
|
||
</code-example>
|
||
|
||
### Disabling image lazy loading
|
||
|
||
By default, `NgOptimizedImage` sets `loading=lazy` for all images that are not marked `priority`. You can disable this behavior for non-priority images by setting the `loading` attribute. This attribute accepts values: `eager`, `auto`, and `lazy`. [See the documentation for the standard image `loading` attribute for details](https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/loading#value).
|
||
|
||
<code-example format="html" language="html">
|
||
|
||
<img ngSrc="cat.jpg" width="400" height="200" loading="eager">
|
||
|
||
</code-example>
|
||
|
||
### Advanced 'sizes' values
|
||
|
||
You may want to have images displayed at varying widths on differently-sized screens. A common example of this pattern is a grid- or column-based layout that renders a single column on mobile devices, and two columns on larger devices. You can capture this behavior in the `sizes` attribute, using a "media query" syntax, such as the following:
|
||
|
||
<code-example format="html" language="html">
|
||
|
||
<img ngSrc="cat.jpg" width="400" height="200" sizes="(max-width: 768px) 100vw, 50vw">
|
||
|
||
</code-example>
|
||
|
||
The `sizes` attribute in the above example says "I expect this image to be 100 percent of the screen width on devices under 768px wide. Otherwise, I expect it to be 50 percent of the screen width.
|
||
|
||
For additional information about the `sizes` attribute, see [web.dev](https://web.dev/learn/design/responsive-images/#sizes) or [MDN](https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/sizes).
|
||
|
||
## Configuring an image loader for `NgOptimizedImage`
|
||
|
||
A "loader" is a function that generates an [image transformation URL](https://web.dev/image-cdns/#how-image-cdns-use-urls-to-indicate-optimization-options) for a given image file. When appropriate, `NgOptimizedImage` sets the size, format, and image quality transformations for an image.
|
||
|
||
`NgOptimizedImage` provides both a generic loader that applies no transformations, as well as loaders for various third-party image services. It also supports writing your own custom loader.
|
||
|
||
| Loader type| Behavior |
|
||
|:--- |:--- |
|
||
| Generic loader | The URL returned by the generic loader will always match the value of `src`. In other words, this loader applies no transformations. Sites that use Angular to serve images are the primary intended use case for this loader.|
|
||
| Loaders for third-party image services | The URL returned by the loaders for third-party image services will follow API conventions used by that particular image service. |
|
||
| Custom loaders | A custom loader's behavior is defined by its developer. You should use a custom loader if your image service isn't supported by the loaders that come preconfigured with `NgOptimizedImage`.|
|
||
|
||
Based on the image services commonly used with Angular applications, `NgOptimizedImage` provides loaders preconfigured to work with the following image services:
|
||
|
||
| Image Service | Angular API | Documentation |
|
||
|:--- |:--- |:--- |
|
||
| Cloudflare Image Resizing | `provideCloudflareLoader` | [Documentation](https://developers.cloudflare.com/images/image-resizing/) |
|
||
| Cloudinary | `provideCloudinaryLoader` | [Documentation](https://cloudinary.com/documentation/resizing_and_cropping) |
|
||
| ImageKit | `provideImageKitLoader` | [Documentation](https://docs.imagekit.io/) |
|
||
| Imgix | `provideImgixLoader` | [Documentation](https://docs.imgix.com/) |
|
||
| Netlify | `provideNetlifyLoader` | [Documentation](https://docs.netlify.com/image-cdn/overview/) |
|
||
|
||
To use the **generic loader** no additional code changes are necessary. This is the default behavior.
|
||
|
||
### Built-in Loaders
|
||
|
||
To use an existing loader for a **third-party image service**, add the provider factory for your chosen service to the `providers` array. In the example below, the Imgix loader is used:
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
providers: [
|
||
provideImgixLoader('https://my.base.url/'),
|
||
],
|
||
</code-example>
|
||
|
||
The base URL for your image assets should be passed to the provider factory as an argument. For most sites, this base URL should match one of the following patterns:
|
||
|
||
* https://yoursite.yourcdn.com
|
||
* https://subdomain.yoursite.com
|
||
* https://subdomain.yourcdn.com/yoursite
|
||
|
||
You can learn more about the base URL structure in the docs of a corresponding CDN provider.
|
||
|
||
### Custom Loaders
|
||
|
||
To use a **custom loader**, provide your loader function as a value for the `IMAGE_LOADER` DI token. In the example below, the custom loader function returns a URL starting with `https://example.com` that includes `src` and `width` as URL parameters.
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
providers: [
|
||
{
|
||
provide: IMAGE_LOADER,
|
||
useValue: (config: ImageLoaderConfig) => {
|
||
return `https://example.com/images?src=${config.src}&width=${config.width}`;
|
||
},
|
||
},
|
||
],
|
||
</code-example>
|
||
|
||
A loader function for the `NgOptimizedImage` directive takes an object with the `ImageLoaderConfig` type (from `@angular/common`) as its argument and returns the absolute URL of the image asset. The `ImageLoaderConfig` object contains the `src` property, and optional `width` and `loaderParams` properties.
|
||
|
||
Note: even though the `width` property may not always be present, a custom loader must use it to support requesting images at various widths in order for `ngSrcset` to work properly.
|
||
|
||
### The `loaderParams` Property
|
||
|
||
There is an additional attribute supported by the `NgOptimizedImage` directive, called `loaderParams`, which is specifically designed to support the use of custom loaders. The `loaderParams` attribute takes an object with any properties as a value, and does not do anything on its own. The data in `loaderParams` is added to the `ImageLoaderConfig` object passed to your custom loader, and can be used to control the behavior of the loader.
|
||
|
||
A common use for `loaderParams` is controlling advanced image CDN features.
|
||
|
||
### Example custom loader
|
||
|
||
The following shows an example of a custom loader function. This example function concatenates `src` and `width`, and uses `loaderParams` to control a custom CDN feature for rounded corners:
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
const myCustomLoader = (config: ImageLoaderConfig) => {
|
||
let url = `https://example.com/images/${config.src}?`;
|
||
let queryParams = [];
|
||
if (config.width) {
|
||
queryParams.push(`w=${config.width}`);
|
||
}
|
||
if (config.loaderParams?.roundedCorners) {
|
||
queryParams.push('mask=corners&corner-radius=5');
|
||
}
|
||
return url + queryParams.join('&');
|
||
};
|
||
</code-example>
|
||
|
||
Note that in the above example, we've invented the 'roundedCorners' property name to control a feature of our custom loader. We could then use this feature when creating an image, as follows:
|
||
|
||
<code-example format="html" language="html">
|
||
|
||
<img ngSrc="profile.jpg" width="300" height="300" [loaderParams]="{roundedCorners: true}">
|
||
|
||
</code-example>
|
||
|
||
## Frequently Asked Questions
|
||
|
||
### Does NgOptimizedImage support the `background-image` css property?
|
||
The NgOptimizedImage does not directly support the `background-image` css property, but it is designed to easily accommodate the use case of having an image as the background of another element.
|
||
|
||
Here's a simple step-by-step process for migrating from `background-image` to `NgOptimizedImage`. For these steps, we'll refer to the element that has an image background as the "containing element":
|
||
|
||
1) Remove the `background-image` style from the containing element.
|
||
2) Ensure that the containing element has `position: "relative"`, `position: "fixed"`, or `position: "absolute"`.
|
||
3) Create a new image element as a child of the containing element, using `ngSrc` to enable the `NgOptimizedImage` directive.
|
||
4) Give that element the `fill` attribute. Do not include a `height` and `width`.
|
||
5) If you believe this image might be your [LCP element](https://web.dev/lcp/), add the `priority` attribute to the image element.
|
||
|
||
You can adjust how the background image fills the container as described in the [Using fill mode](#using-fill-mode) section.
|
||
|
||
### Why can't I use `src` with `NgOptimizedImage`?
|
||
The `ngSrc` attribute was chosen as the trigger for NgOptimizedImage due to technical considerations around how images are loaded by the browser. NgOptimizedImage makes programmatic changes to the `loading` attribute--if the browser sees the `src` attribute before those changes are made, it will begin eagerly downloading the image file, and the loading changes will be ignored.
|
||
|
||
### Can I use two different image domains in the same page?
|
||
The [image loaders](#configuring-an-image-loader-for-ngoptimizedimage) provider pattern is designed to be as simple as possible for the common use case of having only a single image CDN used within a component. However, it's still very possible to manage multiple image CDNs using a single provider.
|
||
|
||
To do this, we recommend writing a [custom image loader](#custom-loaders) which uses the [`loaderParams` property](#the-loaderparams-property) to pass a flag that specifies which image CDN should be used, and then invokes the appropriate loader based on that flag.
|
||
|
||
### Why is a preconnect element not being generated for my image domain?
|
||
Preconnect generation is performed based on static analysis of your application. That means that the image domain must be directly included in the loader parameter, as in the following example:
|
||
|
||
<code-example format="typescript" language="typescript">
|
||
providers: [
|
||
provideImgixLoader('https://my.base.url/'),
|
||
],
|
||
</code-example>
|
||
|
||
If you use a variable to pass the domain string to the loader, or you're not using a loader, the static analysis will not be able to identify the domain, and no preconnect link will be generated. In this case you should manually add a preconnect link to the document head, as [described above.](#resource-hints).
|
||
|
||
### Can you add a new built-in loader for my preferred CDN?
|
||
|
||
For maintenance reasons, we don't currently plan to support additional built-in loaders in the Angular repository. Instead, we encourage developers to publish any additional image loaders as third-party packages.
|
||
|
||
<!-- links -->
|
||
|
||
<!-- external links -->
|
||
|
||
<!--end links -->
|
||
|
||
@reviewed 2023-07-18
|