@spartacus/storefront
Version:
Spartacus Storefront is a package that you can include in your application, which allows you to add default storefront features.
236 lines • 39.4 kB
JavaScript
import { ChangeDetectionStrategy, Component, Input, Optional, } from '@angular/core';
import { PageType, } from '@spartacus/core';
import { of } from 'rxjs';
import { filter, map, switchMap, tap } from 'rxjs/operators';
import { ICON_TYPE } from '../../../cms-components/misc/icon/index';
import * as i0 from "@angular/core";
import * as i1 from "./search-box-component.service";
import * as i2 from "../../../cms-structure/page/model/cms-component-data";
import * as i3 from "@spartacus/core";
import * as i4 from "../../misc/icon/icon.component";
import * as i5 from "../../../shared/components/media/media.component";
import * as i6 from "@angular/common";
import * as i7 from "@angular/router";
import * as i8 from "./highlight.pipe";
const DEFAULT_SEARCH_BOX_CONFIG = {
minCharactersBeforeRequest: 1,
displayProducts: true,
displaySuggestions: true,
maxProducts: 5,
maxSuggestions: 5,
displayProductImages: true,
};
export class SearchBoxComponent {
constructor(searchBoxComponentService, componentData, winRef, routingService) {
var _a;
this.searchBoxComponentService = searchBoxComponentService;
this.componentData = componentData;
this.winRef = winRef;
this.routingService = routingService;
this.iconTypes = ICON_TYPE;
/**
* In some occasions we need to ignore the close event,
* for example when we click inside the search result section.
*/
this.ignoreCloseEvent = false;
this.chosenWord = '';
/**
* Returns the SearchBox configuration. The configuration is driven by multiple
* layers: default configuration, (optional) backend configuration and (optional)
* input configuration.
*/
this.config$ = (((_a = this.componentData) === null || _a === void 0 ? void 0 : _a.data$) || of({})).pipe(map((config) => {
const isBool = (obj, prop) => (obj === null || obj === void 0 ? void 0 : obj[prop]) !== 'false' && (obj === null || obj === void 0 ? void 0 : obj[prop]) !== false;
return Object.assign(Object.assign(Object.assign(Object.assign({}, DEFAULT_SEARCH_BOX_CONFIG), config), { displayProducts: isBool(config, 'displayProducts'), displayProductImages: isBool(config, 'displayProductImages'), displaySuggestions: isBool(config, 'displaySuggestions') }), this.config);
}), tap((config) => (this.config = config)));
this.results$ = this.config$.pipe(switchMap((config) => this.searchBoxComponentService.getResults(config)));
}
/**
* Sets the search box input field
*/
set queryText(value) {
if (value) {
this.search(value);
}
}
ngOnInit() {
this.subscription = this.routingService
.getRouterState()
.pipe(filter((data) => !data.nextState))
.subscribe((data) => {
var _a, _b;
if (!(((_a = data.state.context) === null || _a === void 0 ? void 0 : _a.id) === 'search' &&
((_b = data.state.context) === null || _b === void 0 ? void 0 : _b.type) === PageType.CONTENT_PAGE))
this.chosenWord = '';
});
}
/**
* Closes the searchBox and opens the search result page.
*/
search(query) {
this.searchBoxComponentService.search(query, this.config);
// force the searchBox to open
this.open();
}
/**
* Opens the type-ahead searchBox
*/
open() {
this.searchBoxComponentService.toggleBodyClass('searchbox-is-active', true);
}
/**
* Dispatch UI events for Suggestion selected
*
* @param eventData the data for the event
*/
dispatchSuggestionEvent(eventData) {
this.searchBoxComponentService.dispatchSuggestionSelectedEvent(eventData);
}
/**
* Dispatch UI events for Product selected
*
* @param eventData the data for the event
*/
dispatchProductEvent(eventData) {
this.searchBoxComponentService.dispatchProductSelectedEvent(eventData);
}
/**
* Closes the type-ahead searchBox.
*/
close(event, force) {
// Use timeout to detect changes
setTimeout(() => {
if ((!this.ignoreCloseEvent && !this.isSearchBoxFocused()) || force) {
this.blurSearchBox(event);
}
});
}
blurSearchBox(event) {
this.searchBoxComponentService.toggleBodyClass('searchbox-is-active', false);
if (event && event.target) {
event.target.blur();
}
}
// Check if focus is on searchbox or result list elements
isSearchBoxFocused() {
return (this.getResultElements().includes(this.getFocusedElement()) ||
this.winRef.document.querySelector('input[aria-label="Search"]') ===
this.getFocusedElement());
}
/**
* Especially in mobile we do not want the search icon
* to focus the input again when it's already open.
* */
avoidReopen(event) {
if (this.searchBoxComponentService.hasBodyClass('searchbox-is-active')) {
this.close(event);
event.preventDefault();
}
}
// Return result list as HTMLElement array
getResultElements() {
return Array.from(this.winRef.document.querySelectorAll('.products > li a, .suggestions > li a'));
}
// Return focused element as HTMLElement
getFocusedElement() {
return this.winRef.document.activeElement;
}
updateChosenWord(chosenWord) {
this.chosenWord = chosenWord;
}
getFocusedIndex() {
return this.getResultElements().indexOf(this.getFocusedElement());
}
// Focus on previous item in results list
focusPreviousChild(event) {
event.preventDefault(); // Negate normal keyscroll
const [results, focusedIndex] = [
this.getResultElements(),
this.getFocusedIndex(),
];
// Focus on last index moving to first
if (results.length) {
if (focusedIndex < 1) {
results[results.length - 1].focus();
}
else {
results[focusedIndex - 1].focus();
}
}
}
// Focus on next item in results list
focusNextChild(event) {
this.open();
event.preventDefault(); // Negate normal keyscroll
const [results, focusedIndex] = [
this.getResultElements(),
this.getFocusedIndex(),
];
// Focus on first index moving to last
if (results.length) {
if (focusedIndex >= results.length - 1) {
results[0].focus();
}
else {
results[focusedIndex + 1].focus();
}
}
}
/**
* Opens the PLP with the given query.
*
* TODO: if there's a single product match, we could open the PDP.
*/
launchSearchResult(event, query) {
if (!query || query.trim().length === 0) {
return;
}
this.close(event);
this.searchBoxComponentService.launchSearchPage(query);
}
/**
* Disables closing the search result list.
*/
disableClose() {
this.ignoreCloseEvent = true;
}
preventDefault(ev) {
ev.preventDefault();
}
/**
* Clears the search box input field
*/
clear(el) {
this.disableClose();
el.value = '';
this.searchBoxComponentService.clearResults();
// Use Timeout to run after blur event to prevent the searchbox from closing on mobile
setTimeout(() => {
// Retain focus on input lost by clicking on icon
el.focus();
this.ignoreCloseEvent = false;
});
}
ngOnDestroy() {
var _a;
(_a = this.subscription) === null || _a === void 0 ? void 0 : _a.unsubscribe();
}
}
SearchBoxComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: SearchBoxComponent, deps: [{ token: i1.SearchBoxComponentService }, { token: i2.CmsComponentData, optional: true }, { token: i3.WindowRef }, { token: i3.RoutingService }], target: i0.ɵɵFactoryTarget.Component });
SearchBoxComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.0.5", type: SearchBoxComponent, selector: "cx-searchbox", inputs: { config: "config", queryText: "queryText" }, ngImport: i0, template: "<label class=\"searchbox\" [class.dirty]=\"!!searchInput.value\">\n <input\n #searchInput\n [placeholder]=\"'searchBox.placeholder' | cxTranslate\"\n autocomplete=\"off\"\n aria-describedby=\"initialDescription\"\n aria-controls=\"results\"\n [attr.aria-label]=\"'common.search' | cxTranslate\"\n (focus)=\"open()\"\n (click)=\"open()\"\n (input)=\"search(searchInput.value)\"\n (blur)=\"close($event)\"\n (keydown.escape)=\"close($event)\"\n (keydown.enter)=\"\n close($event, true);\n launchSearchResult($event, searchInput.value);\n updateChosenWord(searchInput.value)\n \"\n (keydown.arrowup)=\"focusPreviousChild($event)\"\n (keydown.arrowdown)=\"focusNextChild($event)\"\n value=\"{{ chosenWord }}\"\n />\n\n <button\n [attr.aria-label]=\"'common.reset' | cxTranslate\"\n (mousedown)=\"clear(searchInput)\"\n (keydown.enter)=\"clear(searchInput)\"\n class=\"reset\"\n >\n <cx-icon [type]=\"iconTypes.RESET\"></cx-icon>\n </button>\n\n <div role=\"presentation\" class=\"search-icon\">\n <cx-icon [type]=\"iconTypes.SEARCH\"></cx-icon>\n </div>\n\n <button\n [attr.aria-label]=\"'common.search' | cxTranslate\"\n class=\"search\"\n (click)=\"open()\"\n >\n <cx-icon [type]=\"iconTypes.SEARCH\"></cx-icon>\n </button>\n</label>\n\n<div\n *ngIf=\"results$ | async as result\"\n class=\"results\"\n id=\"results\"\n (click)=\"close($event, true)\"\n role=\"listbox\"\n>\n <div\n *ngIf=\"result.message\"\n class=\"message\"\n [innerHTML]=\"result.message\"\n ></div>\n\n <ul\n class=\"suggestions\"\n attr.aria-label=\"{{ 'searchBox.ariaLabelSuggestions' | cxTranslate }}\"\n tabindex=\"0\"\n >\n <li *ngFor=\"let suggestion of result.suggestions\">\n <a\n [innerHTML]=\"suggestion | cxHighlight: searchInput.value\"\n [routerLink]=\"\n {\n cxRoute: 'search',\n params: { query: suggestion }\n } | cxUrl\n \"\n (keydown.arrowup)=\"focusPreviousChild($event)\"\n (keydown.arrowdown)=\"focusNextChild($event)\"\n (keydown.enter)=\"close($event, true)\"\n (keydown.escape)=\"close($event, true)\"\n (blur)=\"close($event)\"\n (mousedown)=\"preventDefault($event)\"\n (click)=\"\n dispatchSuggestionEvent({\n freeText: searchInput.value,\n selectedSuggestion: suggestion,\n searchSuggestions: result.suggestions\n });\n updateChosenWord(suggestion)\n \"\n >\n </a>\n </li>\n </ul>\n\n <ul\n class=\"products\"\n *ngIf=\"result.products\"\n attr.aria-label=\"{{ 'searchBox.ariaLabelProducts' | cxTranslate }}\"\n tabindex=\"0\"\n >\n <li *ngFor=\"let product of result.products\">\n <a\n [routerLink]=\"\n {\n cxRoute: 'product',\n params: product\n } | cxUrl\n \"\n [class.has-media]=\"config.displayProductImages\"\n (keydown.arrowup)=\"focusPreviousChild($event)\"\n (keydown.arrowdown)=\"focusNextChild($event)\"\n (keydown.enter)=\"close($event, true)\"\n (keydown.escape)=\"close($event, true)\"\n (blur)=\"close($event)\"\n (mousedown)=\"preventDefault($event)\"\n (click)=\"\n dispatchProductEvent({\n freeText: searchInput.value,\n productCode: product.code\n })\n \"\n >\n <cx-media\n *ngIf=\"config.displayProductImages\"\n [container]=\"product.images?.PRIMARY\"\n format=\"thumbnail\"\n role=\"presentation\"\n ></cx-media>\n <div class=\"name\" [innerHTML]=\"product.nameHtml\"></div>\n <span class=\"price\">{{ product.price?.formattedValue }}</span>\n </a>\n </li>\n </ul>\n <span id=\"initialDescription\" class=\"cx-visually-hidden\">\n {{ 'searchBox.initialDescription' | cxTranslate }}\n </span>\n <div\n *ngIf=\"result.suggestions?.length || result.products?.length\"\n aria-live=\"assertive\"\n class=\"cx-visually-hidden\"\n >\n {{\n 'searchBox.suggestionsResult'\n | cxTranslate: { count: result.suggestions?.length }\n }}\n {{\n 'searchBox.productsResult'\n | cxTranslate: { count: result.products?.length }\n }}\n {{ 'searchBox.initialDescription' | cxTranslate }}\n </div>\n</div>\n", components: [{ type: i4.IconComponent, selector: "cx-icon,[cxIcon]", inputs: ["cxIcon", "type"] }, { type: i5.MediaComponent, selector: "cx-media", inputs: ["container", "format", "alt", "role", "loading"], outputs: ["loaded"] }], directives: [{ type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i7.RouterLinkWithHref, selector: "a[routerLink],area[routerLink]", inputs: ["routerLink", "target", "queryParams", "fragment", "queryParamsHandling", "preserveFragment", "skipLocationChange", "replaceUrl", "state", "relativeTo"] }], pipes: { "cxTranslate": i3.TranslatePipe, "async": i6.AsyncPipe, "cxHighlight": i8.HighlightPipe, "cxUrl": i3.UrlPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: SearchBoxComponent, decorators: [{
type: Component,
args: [{
selector: 'cx-searchbox',
templateUrl: './search-box.component.html',
changeDetection: ChangeDetectionStrategy.OnPush,
}]
}], ctorParameters: function () { return [{ type: i1.SearchBoxComponentService }, { type: i2.CmsComponentData, decorators: [{
type: Optional
}] }, { type: i3.WindowRef }, { type: i3.RoutingService }]; }, propDecorators: { config: [{
type: Input
}], queryText: [{
type: Input,
args: ['queryText']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search-box.component.js","sourceRoot":"","sources":["../../../../../../projects/storefrontlib/cms-components/navigation/search-box/search-box.component.ts","../../../../../../projects/storefrontlib/cms-components/navigation/search-box/search-box.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  OnDestroy,\n  OnInit,\n  Optional,\n} from '@angular/core';\nimport {\n  CmsSearchBoxComponent,\n  PageType,\n  RoutingService,\n  WindowRef,\n} from '@spartacus/core';\nimport { Observable, of, Subscription } from 'rxjs';\nimport { filter, map, switchMap, tap } from 'rxjs/operators';\nimport { ICON_TYPE } from '../../../cms-components/misc/icon/index';\nimport { CmsComponentData } from '../../../cms-structure/page/model/cms-component-data';\nimport { SearchBoxComponentService } from './search-box-component.service';\nimport {\n  SearchBoxProductSelectedEvent,\n  SearchBoxSuggestionSelectedEvent,\n} from './search-box.events';\nimport { SearchBoxConfig, SearchResults } from './search-box.model';\n\nconst DEFAULT_SEARCH_BOX_CONFIG: SearchBoxConfig = {\n  minCharactersBeforeRequest: 1,\n  displayProducts: true,\n  displaySuggestions: true,\n  maxProducts: 5,\n  maxSuggestions: 5,\n  displayProductImages: true,\n};\n\n@Component({\n  selector: 'cx-searchbox',\n  templateUrl: './search-box.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SearchBoxComponent implements OnInit, OnDestroy {\n  @Input() config: SearchBoxConfig;\n\n  /**\n   * Sets the search box input field\n   */\n  @Input('queryText')\n  set queryText(value: string) {\n    if (value) {\n      this.search(value);\n    }\n  }\n\n  iconTypes = ICON_TYPE;\n\n  /**\n   * In some occasions we need to ignore the close event,\n   * for example when we click inside the search result section.\n   */\n  private ignoreCloseEvent = false;\n  chosenWord = '';\n  public subscription: Subscription;\n\n  constructor(\n    protected searchBoxComponentService: SearchBoxComponentService,\n    @Optional()\n    protected componentData: CmsComponentData<CmsSearchBoxComponent>,\n    protected winRef: WindowRef,\n    protected routingService: RoutingService\n  ) {}\n\n  /**\n   * Returns the SearchBox configuration. The configuration is driven by multiple\n   * layers: default configuration, (optional) backend configuration and (optional)\n   * input configuration.\n   */\n  protected config$: Observable<SearchBoxConfig> = (\n    this.componentData?.data$ || of({} as any)\n  ).pipe(\n    map((config) => {\n      const isBool = (obj: SearchBoxConfig, prop: string): boolean =>\n        obj?.[prop] !== 'false' && obj?.[prop] !== false;\n\n      return {\n        ...DEFAULT_SEARCH_BOX_CONFIG,\n        ...config,\n        displayProducts: isBool(config, 'displayProducts'),\n        displayProductImages: isBool(config, 'displayProductImages'),\n        displaySuggestions: isBool(config, 'displaySuggestions'),\n        // we're merging the (optional) input of this component, but write the merged\n        // result back to the input property, as the view logic depends on it.\n        ...this.config,\n      };\n    }),\n    tap((config) => (this.config = config))\n  );\n\n  results$: Observable<SearchResults> = this.config$.pipe(\n    switchMap((config) => this.searchBoxComponentService.getResults(config))\n  );\n\n  ngOnInit(): void {\n    this.subscription = this.routingService\n      .getRouterState()\n      .pipe(filter((data) => !data.nextState))\n      .subscribe((data) => {\n        if (\n          !(\n            data.state.context?.id === 'search' &&\n            data.state.context?.type === PageType.CONTENT_PAGE\n          )\n        )\n          this.chosenWord = '';\n      });\n  }\n\n  /**\n   * Closes the searchBox and opens the search result page.\n   */\n  search(query: string): void {\n    this.searchBoxComponentService.search(query, this.config);\n    // force the searchBox to open\n    this.open();\n  }\n\n  /**\n   * Opens the type-ahead searchBox\n   */\n  open(): void {\n    this.searchBoxComponentService.toggleBodyClass('searchbox-is-active', true);\n  }\n\n  /**\n   * Dispatch UI events for Suggestion selected\n   *\n   * @param eventData the data for the event\n   */\n  dispatchSuggestionEvent(eventData: SearchBoxSuggestionSelectedEvent): void {\n    this.searchBoxComponentService.dispatchSuggestionSelectedEvent(eventData);\n  }\n\n  /**\n   * Dispatch UI events for Product selected\n   *\n   * @param eventData the data for the event\n   */\n  dispatchProductEvent(eventData: SearchBoxProductSelectedEvent): void {\n    this.searchBoxComponentService.dispatchProductSelectedEvent(eventData);\n  }\n\n  /**\n   * Closes the type-ahead searchBox.\n   */\n  close(event: UIEvent, force?: boolean): void {\n    // Use timeout to detect changes\n    setTimeout(() => {\n      if ((!this.ignoreCloseEvent && !this.isSearchBoxFocused()) || force) {\n        this.blurSearchBox(event);\n      }\n    });\n  }\n\n  protected blurSearchBox(event: UIEvent): void {\n    this.searchBoxComponentService.toggleBodyClass(\n      'searchbox-is-active',\n      false\n    );\n    if (event && event.target) {\n      (<HTMLElement>event.target).blur();\n    }\n  }\n\n  // Check if focus is on searchbox or result list elements\n  private isSearchBoxFocused(): boolean {\n    return (\n      this.getResultElements().includes(this.getFocusedElement()) ||\n      this.winRef.document.querySelector('input[aria-label=\"Search\"]') ===\n        this.getFocusedElement()\n    );\n  }\n\n  /**\n   * Especially in mobile we do not want the search icon\n   * to focus the input again when it's already open.\n   * */\n  avoidReopen(event: UIEvent): void {\n    if (this.searchBoxComponentService.hasBodyClass('searchbox-is-active')) {\n      this.close(event);\n      event.preventDefault();\n    }\n  }\n\n  // Return result list as HTMLElement array\n  private getResultElements(): HTMLElement[] {\n    return Array.from(\n      this.winRef.document.querySelectorAll(\n        '.products > li a, .suggestions > li a'\n      )\n    );\n  }\n\n  // Return focused element as HTMLElement\n  private getFocusedElement(): HTMLElement {\n    return <HTMLElement>this.winRef.document.activeElement;\n  }\n\n  updateChosenWord(chosenWord: string): void {\n    this.chosenWord = chosenWord;\n  }\n\n  private getFocusedIndex(): number {\n    return this.getResultElements().indexOf(this.getFocusedElement());\n  }\n\n  // Focus on previous item in results list\n  focusPreviousChild(event: UIEvent) {\n    event.preventDefault(); // Negate normal keyscroll\n    const [results, focusedIndex] = [\n      this.getResultElements(),\n      this.getFocusedIndex(),\n    ];\n    // Focus on last index moving to first\n    if (results.length) {\n      if (focusedIndex < 1) {\n        results[results.length - 1].focus();\n      } else {\n        results[focusedIndex - 1].focus();\n      }\n    }\n  }\n\n  // Focus on next item in results list\n  focusNextChild(event: UIEvent) {\n    this.open();\n    event.preventDefault(); // Negate normal keyscroll\n    const [results, focusedIndex] = [\n      this.getResultElements(),\n      this.getFocusedIndex(),\n    ];\n    // Focus on first index moving to last\n    if (results.length) {\n      if (focusedIndex >= results.length - 1) {\n        results[0].focus();\n      } else {\n        results[focusedIndex + 1].focus();\n      }\n    }\n  }\n\n  /**\n   * Opens the PLP with the given query.\n   *\n   * TODO: if there's a single product match, we could open the PDP.\n   */\n  launchSearchResult(event: UIEvent, query: string): void {\n    if (!query || query.trim().length === 0) {\n      return;\n    }\n    this.close(event);\n    this.searchBoxComponentService.launchSearchPage(query);\n  }\n\n  /**\n   * Disables closing the search result list.\n   */\n  disableClose(): void {\n    this.ignoreCloseEvent = true;\n  }\n\n  preventDefault(ev: UIEvent): void {\n    ev.preventDefault();\n  }\n\n  /**\n   * Clears the search box input field\n   */\n  clear(el: HTMLInputElement): void {\n    this.disableClose();\n    el.value = '';\n    this.searchBoxComponentService.clearResults();\n\n    // Use Timeout to run after blur event to prevent the searchbox from closing on mobile\n    setTimeout(() => {\n      // Retain focus on input lost by clicking on icon\n      el.focus();\n      this.ignoreCloseEvent = false;\n    });\n  }\n\n  ngOnDestroy(): void {\n    this.subscription?.unsubscribe();\n  }\n}\n","<label class=\"searchbox\" [class.dirty]=\"!!searchInput.value\">\n  <input\n    #searchInput\n    [placeholder]=\"'searchBox.placeholder' | cxTranslate\"\n    autocomplete=\"off\"\n    aria-describedby=\"initialDescription\"\n    aria-controls=\"results\"\n    [attr.aria-label]=\"'common.search' | cxTranslate\"\n    (focus)=\"open()\"\n    (click)=\"open()\"\n    (input)=\"search(searchInput.value)\"\n    (blur)=\"close($event)\"\n    (keydown.escape)=\"close($event)\"\n    (keydown.enter)=\"\n      close($event, true);\n      launchSearchResult($event, searchInput.value);\n      updateChosenWord(searchInput.value)\n    \"\n    (keydown.arrowup)=\"focusPreviousChild($event)\"\n    (keydown.arrowdown)=\"focusNextChild($event)\"\n    value=\"{{ chosenWord }}\"\n  />\n\n  <button\n    [attr.aria-label]=\"'common.reset' | cxTranslate\"\n    (mousedown)=\"clear(searchInput)\"\n    (keydown.enter)=\"clear(searchInput)\"\n    class=\"reset\"\n  >\n    <cx-icon [type]=\"iconTypes.RESET\"></cx-icon>\n  </button>\n\n  <div role=\"presentation\" class=\"search-icon\">\n    <cx-icon [type]=\"iconTypes.SEARCH\"></cx-icon>\n  </div>\n\n  <button\n    [attr.aria-label]=\"'common.search' | cxTranslate\"\n    class=\"search\"\n    (click)=\"open()\"\n  >\n    <cx-icon [type]=\"iconTypes.SEARCH\"></cx-icon>\n  </button>\n</label>\n\n<div\n  *ngIf=\"results$ | async as result\"\n  class=\"results\"\n  id=\"results\"\n  (click)=\"close($event, true)\"\n  role=\"listbox\"\n>\n  <div\n    *ngIf=\"result.message\"\n    class=\"message\"\n    [innerHTML]=\"result.message\"\n  ></div>\n\n  <ul\n    class=\"suggestions\"\n    attr.aria-label=\"{{ 'searchBox.ariaLabelSuggestions' | cxTranslate }}\"\n    tabindex=\"0\"\n  >\n    <li *ngFor=\"let suggestion of result.suggestions\">\n      <a\n        [innerHTML]=\"suggestion | cxHighlight: searchInput.value\"\n        [routerLink]=\"\n          {\n            cxRoute: 'search',\n            params: { query: suggestion }\n          } | cxUrl\n        \"\n        (keydown.arrowup)=\"focusPreviousChild($event)\"\n        (keydown.arrowdown)=\"focusNextChild($event)\"\n        (keydown.enter)=\"close($event, true)\"\n        (keydown.escape)=\"close($event, true)\"\n        (blur)=\"close($event)\"\n        (mousedown)=\"preventDefault($event)\"\n        (click)=\"\n          dispatchSuggestionEvent({\n            freeText: searchInput.value,\n            selectedSuggestion: suggestion,\n            searchSuggestions: result.suggestions\n          });\n          updateChosenWord(suggestion)\n        \"\n      >\n      </a>\n    </li>\n  </ul>\n\n  <ul\n    class=\"products\"\n    *ngIf=\"result.products\"\n    attr.aria-label=\"{{ 'searchBox.ariaLabelProducts' | cxTranslate }}\"\n    tabindex=\"0\"\n  >\n    <li *ngFor=\"let product of result.products\">\n      <a\n        [routerLink]=\"\n          {\n            cxRoute: 'product',\n            params: product\n          } | cxUrl\n        \"\n        [class.has-media]=\"config.displayProductImages\"\n        (keydown.arrowup)=\"focusPreviousChild($event)\"\n        (keydown.arrowdown)=\"focusNextChild($event)\"\n        (keydown.enter)=\"close($event, true)\"\n        (keydown.escape)=\"close($event, true)\"\n        (blur)=\"close($event)\"\n        (mousedown)=\"preventDefault($event)\"\n        (click)=\"\n          dispatchProductEvent({\n            freeText: searchInput.value,\n            productCode: product.code\n          })\n        \"\n      >\n        <cx-media\n          *ngIf=\"config.displayProductImages\"\n          [container]=\"product.images?.PRIMARY\"\n          format=\"thumbnail\"\n          role=\"presentation\"\n        ></cx-media>\n        <div class=\"name\" [innerHTML]=\"product.nameHtml\"></div>\n        <span class=\"price\">{{ product.price?.formattedValue }}</span>\n      </a>\n    </li>\n  </ul>\n  <span id=\"initialDescription\" class=\"cx-visually-hidden\">\n    {{ 'searchBox.initialDescription' | cxTranslate }}\n  </span>\n  <div\n    *ngIf=\"result.suggestions?.length || result.products?.length\"\n    aria-live=\"assertive\"\n    class=\"cx-visually-hidden\"\n  >\n    {{\n      'searchBox.suggestionsResult'\n        | cxTranslate: { count: result.suggestions?.length }\n    }}\n    {{\n      'searchBox.productsResult'\n        | cxTranslate: { count: result.products?.length }\n    }}\n    {{ 'searchBox.initialDescription' | cxTranslate }}\n  </div>\n</div>\n"]}