UNPKG

@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
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"]}