angular-instantsearch
Version:
Lightning-fast search for Angular apps, by Algolia.
184 lines (181 loc) • 18.9 kB
JavaScript
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"]}