UNPKG

angular-instantsearch

Version:

Lightning-fast search for Angular apps, by Algolia.

184 lines (181 loc) 18.9 kB
import { Component, Input, Output, EventEmitter, Inject, forwardRef, ViewChild, Optional, } from '@angular/core'; import { DOCUMENT } from '@angular/common'; import { connectSearchBox } from 'instantsearch.js/es/connectors'; import { TypedBaseWidget } from '../typed-base-widget'; import { NgAisInstantSearch } from '../instantsearch/instantsearch'; import { NgAisIndex } from '../index-widget/index-widget'; import { noop } from '../utils'; export class NgAisSearchBox extends TypedBaseWidget { constructor(parentIndex, instantSearchInstance, injectedDocument) { super('SearchBox'); this.parentIndex = parentIndex; this.instantSearchInstance = instantSearchInstance; this.injectedDocument = injectedDocument; this.placeholder = 'Search'; this.submitTitle = 'Submit'; this.resetTitle = 'Reset'; this.searchAsYouType = true; this.autofocus = false; this.showLoadingIndicator = true; // Output events // form this.submit = new EventEmitter(); this.reset = new EventEmitter(); // input this.change = new EventEmitter(); this.focus = new EventEmitter(); this.blur = new EventEmitter(); this.query = ''; this.state = { query: '', refine: noop, clear: noop, isSearchStalled: false, }; this.createWidget(connectSearchBox, {}, { $$widgetType: 'ais.searchBox', }); this.document = injectedDocument; } ngAfterViewInit() { if (this.autofocus) { this.searchBox.nativeElement.focus(); } } ngDoCheck() { // We bypass the state update if the input is focused to avoid concurrent // updates when typing. if (this.query !== this.state.query && this.searchBox && this.searchBox.nativeElement && this.document.activeElement !== this.searchBox.nativeElement) { this.query = this.state.query; } } handleChange(query) { this.change.emit(query); if (this.searchAsYouType) { this.state.refine(query); } } handleSubmit(event) { // send submit event to parent component this.submit.emit(event); event.preventDefault(); if (!this.searchAsYouType) { this.state.refine(this.searchBox.nativeElement.value); } } handleReset(event) { // send reset event to parent component this.reset.emit(event); // reset search this.state.refine(''); } } NgAisSearchBox.decorators = [ { type: Component, args: [{ selector: 'ais-search-box', template: ` <div [class]="cx()"> <form [class]="cx('form')" novalidate (submit)="handleSubmit($event)" > <input [class]="cx('input')" autocapitalize="off" autocorrect="off" placeholder="{{placeholder}}" role="textbox" spellcheck="false" type="text" [value]="query" (input)="handleChange($event.target.value)" (focus)="focus.emit($event)" (blur)="blur.emit($event)" #searchBox /> <button [class]="cx('submit')" type="submit" title="{{submitTitle}}" > <svg [ngClass]="cx('submitIcon')" viewBox="0 0 40 40" width="40" height="40" > <path d="M26.804 29.01c-2.832 2.34-6.465 3.746-10.426 3.746C7.333 32.756 0 25.424 0 16.378 0 7.333 7.333 0 16.378 0c9.046 0 16.378 7.333 16.378 16.378 0 3.96-1.406 7.594-3.746 10.426l10.534 10.534c.607.607.61 1.59-.004 2.202-.61.61-1.597.61-2.202.004L26.804 29.01zm-10.426.627c7.323 0 13.26-5.936 13.26-13.26 0-7.32-5.937-13.257-13.26-13.257C9.056 3.12 3.12 9.056 3.12 16.378c0 7.323 5.936 13.26 13.258 13.26z"></path> </svg> </button> <button [class]="cx('reset')" type="reset" title="{{resetTitle}}" (click)="handleReset($event)" [hidden]="!state.query || (state.query && !state.query.trim()) || (state.isSearchStalled && showLoadingIndicator)"> <svg [ngClass]="cx('resetIcon')" viewBox="0 0 20 20" width="20" height="20" > <path d="M8.114 10L.944 2.83 0 1.885 1.886 0l.943.943L10 8.113l7.17-7.17.944-.943L20 1.886l-.943.943-7.17 7.17 7.17 7.17.943.944L18.114 20l-.943-.943-7.17-7.17-7.17 7.17-.944.943L0 18.114l.943-.943L8.113 10z"></path> </svg> </button> <span [class]="cx('loadingIndicator')" [hidden]="!showLoadingIndicator || !state.isSearchStalled" > <svg width="16" height="16" viewBox="0 0 38 38" stroke="#444" [ngClass]="cx('loadingIcon')" > <g fill="none" fillRule="evenodd"> <g transform="translate(1 1)" strokeWidth="2"> <circle strokeOpacity=".5" cx="18" cy="18" r="18" /> <path d="M36 18c0-9.94-8.06-18-18-18"> <animateTransform attributeName="transform" type="rotate" from="0 18 18" to="360 18 18" dur="1s" repeatCount="indefinite" /> </path> </g> </g> </svg> </span> </form> </div> ` },] } ]; NgAisSearchBox.ctorParameters = () => [ { type: NgAisIndex, decorators: [{ type: Inject, args: [forwardRef(() => NgAisIndex),] }, { type: Optional }] }, { type: NgAisInstantSearch, decorators: [{ type: Inject, args: [forwardRef(() => NgAisInstantSearch),] }] }, { type: Document, decorators: [{ type: Inject, args: [DOCUMENT,] }] } ]; NgAisSearchBox.propDecorators = { searchBox: [{ type: ViewChild, args: ['searchBox', { static: false },] }], placeholder: [{ type: Input }], submitTitle: [{ type: Input }], resetTitle: [{ type: Input }], searchAsYouType: [{ type: Input }], autofocus: [{ type: Input }], showLoadingIndicator: [{ type: Input }], submit: [{ type: Output }], reset: [{ type: Output }], change: [{ type: Output }], focus: [{ type: Output }], blur: [{ type: Output }] }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search-box.js","sourceRoot":"","sources":["../../../src/search-box/search-box.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  Inject,\n  forwardRef,\n  ViewChild,\n  AfterViewInit,\n  ElementRef,\n  Optional,\n  DoCheck,\n} from '@angular/core';\nimport { DOCUMENT } from '@angular/common';\n\nimport { connectSearchBox } from 'instantsearch.js/es/connectors';\nimport { TypedBaseWidget } from '../typed-base-widget';\nimport { NgAisInstantSearch } from '../instantsearch/instantsearch';\nimport { NgAisIndex } from '../index-widget/index-widget';\nimport { noop } from '../utils';\nimport {\n  SearchBoxConnectorParams,\n  SearchBoxWidgetDescription,\n  SearchBoxRenderState,\n} from 'instantsearch.js/es/connectors/search-box/connectSearchBox';\n\n@Component({\n  selector: 'ais-search-box',\n  template: `\n    <div [class]=\"cx()\">\n      <form\n        [class]=\"cx('form')\"\n        novalidate\n        (submit)=\"handleSubmit($event)\"\n      >\n        <input\n          [class]=\"cx('input')\"\n          autocapitalize=\"off\"\n          autocorrect=\"off\"\n          placeholder=\"{{placeholder}}\"\n          role=\"textbox\"\n          spellcheck=\"false\"\n          type=\"text\"\n          [value]=\"query\"\n          (input)=\"handleChange($event.target.value)\"\n          (focus)=\"focus.emit($event)\"\n          (blur)=\"blur.emit($event)\"\n          #searchBox\n        />\n\n        <button\n          [class]=\"cx('submit')\"\n          type=\"submit\"\n          title=\"{{submitTitle}}\"\n        >\n          <svg\n            [ngClass]=\"cx('submitIcon')\"\n            viewBox=\"0 0 40 40\"\n            width=\"40\"\n            height=\"40\"\n          >\n            <path d=\"M26.804 29.01c-2.832 2.34-6.465 3.746-10.426 3.746C7.333 32.756 0 25.424 0 16.378 0 7.333 7.333 0 16.378 0c9.046 0 16.378 7.333 16.378 16.378 0 3.96-1.406 7.594-3.746 10.426l10.534 10.534c.607.607.61 1.59-.004 2.202-.61.61-1.597.61-2.202.004L26.804 29.01zm-10.426.627c7.323 0 13.26-5.936 13.26-13.26 0-7.32-5.937-13.257-13.26-13.257C9.056 3.12 3.12 9.056 3.12 16.378c0 7.323 5.936 13.26 13.258 13.26z\"></path>\n          </svg>\n        </button>\n\n        <button\n          [class]=\"cx('reset')\"\n          type=\"reset\"\n          title=\"{{resetTitle}}\"\n          (click)=\"handleReset($event)\"\n          [hidden]=\"!state.query || (state.query && !state.query.trim()) || (state.isSearchStalled && showLoadingIndicator)\">\n          <svg\n            [ngClass]=\"cx('resetIcon')\"\n            viewBox=\"0 0 20 20\"\n            width=\"20\"\n            height=\"20\"\n          >\n            <path d=\"M8.114 10L.944 2.83 0 1.885 1.886 0l.943.943L10 8.113l7.17-7.17.944-.943L20 1.886l-.943.943-7.17 7.17 7.17 7.17.943.944L18.114 20l-.943-.943-7.17-7.17-7.17 7.17-.944.943L0 18.114l.943-.943L8.113 10z\"></path>\n          </svg>\n        </button>\n\n        <span\n          [class]=\"cx('loadingIndicator')\"\n          [hidden]=\"!showLoadingIndicator || !state.isSearchStalled\"\n        >\n          <svg\n            width=\"16\"\n            height=\"16\"\n            viewBox=\"0 0 38 38\"\n            stroke=\"#444\"\n            [ngClass]=\"cx('loadingIcon')\"\n          >\n            <g fill=\"none\" fillRule=\"evenodd\">\n              <g transform=\"translate(1 1)\" strokeWidth=\"2\">\n                <circle strokeOpacity=\".5\" cx=\"18\" cy=\"18\" r=\"18\" />\n                <path d=\"M36 18c0-9.94-8.06-18-18-18\">\n                  <animateTransform\n                    attributeName=\"transform\"\n                    type=\"rotate\"\n                    from=\"0 18 18\"\n                    to=\"360 18 18\"\n                    dur=\"1s\"\n                    repeatCount=\"indefinite\"\n                  />\n                </path>\n              </g>\n            </g>\n          </svg>\n        </span>\n      </form>\n    </div>\n  `,\n})\nexport class NgAisSearchBox\n  extends TypedBaseWidget<SearchBoxWidgetDescription, SearchBoxConnectorParams>\n  implements AfterViewInit, DoCheck {\n  @ViewChild('searchBox', { static: false })\n  searchBox: ElementRef;\n  @Input() public placeholder: string = 'Search';\n  @Input() public submitTitle: string = 'Submit';\n  @Input() public resetTitle: string = 'Reset';\n  @Input() public searchAsYouType: boolean = true;\n  @Input() public autofocus: boolean = false;\n  @Input() public showLoadingIndicator: boolean = true;\n\n  // Output events\n  // form\n  @Output() submit = new EventEmitter();\n  @Output() reset = new EventEmitter();\n\n  // input\n  @Output() change = new EventEmitter();\n  @Output() focus = new EventEmitter();\n  @Output() blur = new EventEmitter();\n\n  public query = '';\n\n  public state: SearchBoxRenderState = {\n    query: '',\n    refine: noop,\n    clear: noop,\n    isSearchStalled: false,\n  };\n\n  private document: Document;\n\n  constructor(\n    @Inject(forwardRef(() => NgAisIndex))\n    @Optional()\n    public parentIndex: NgAisIndex,\n    @Inject(forwardRef(() => NgAisInstantSearch))\n    public instantSearchInstance: NgAisInstantSearch,\n    @Inject(DOCUMENT) private injectedDocument: Document\n  ) {\n    super('SearchBox');\n    this.createWidget(\n      connectSearchBox,\n      {},\n      {\n        $$widgetType: 'ais.searchBox',\n      }\n    );\n    this.document = injectedDocument;\n  }\n\n  public ngAfterViewInit() {\n    if (this.autofocus) {\n      this.searchBox.nativeElement.focus();\n    }\n  }\n\n  public ngDoCheck() {\n    // We bypass the state update if the input is focused to avoid concurrent\n    // updates when typing.\n    if (\n      this.query !== this.state.query &&\n      this.searchBox &&\n      this.searchBox.nativeElement &&\n      this.document.activeElement !== this.searchBox.nativeElement\n    ) {\n      this.query = this.state.query;\n    }\n  }\n\n  public handleChange(query: string) {\n    this.change.emit(query);\n    if (this.searchAsYouType) {\n      this.state.refine(query);\n    }\n  }\n\n  public handleSubmit(event: Event) {\n    // send submit event to parent component\n    this.submit.emit(event);\n\n    event.preventDefault();\n\n    if (!this.searchAsYouType) {\n      this.state.refine(this.searchBox.nativeElement.value);\n    }\n  }\n\n  public handleReset(event: MouseEvent) {\n    // send reset event to parent component\n    this.reset.emit(event);\n\n    // reset search\n    this.state.refine('');\n  }\n}\n"]}