UNPKG

angular-instantsearch

Version:

Lightning-fast search for Angular apps, by Algolia.

133 lines (132 loc) 15.7 kB
import { Component, Input, Inject, forwardRef, NgZone, ContentChild, Optional, } from '@angular/core'; import { connectVoiceSearch } 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 NgAisVoiceSearch extends TypedBaseWidget { constructor(parentIndex, instantSearchInstance, zone) { super('VoiceSearch'); this.parentIndex = parentIndex; this.instantSearchInstance = instantSearchInstance; this.zone = zone; // rendering options this.buttonTitle = 'Search by voice'; this.disabledButtonTitle = 'Search by voice (not supported on this browser)'; this.state = { isBrowserSupported: undefined, isListening: undefined, toggleListening: noop, voiceListeningState: { status: 'initial', transcript: '', isSpeechFinal: false, errorCode: undefined, }, }; this.templateContext = { status: 'initial', errorCode: undefined, transcript: '', isSpeechFinal: false, isListening: false, isBrowserSupported: false, }; this.handleClick = (event) => { event.currentTarget.blur(); this.state.toggleListening(); }; this.isNotAllowedError = () => this.state.voiceListeningState.status === 'error' && this.state.voiceListeningState.errorCode === 'not-allowed'; this.updateState = (state) => { this.zone.run(() => { this.templateContext = { status: state.voiceListeningState.status, errorCode: state.voiceListeningState.errorCode, transcript: state.voiceListeningState.transcript, isSpeechFinal: state.voiceListeningState.isSpeechFinal, isListening: state.isListening, isBrowserSupported: state.isBrowserSupported, }; this.state = state; }); }; } ngOnInit() { this.createWidget(connectVoiceSearch, { searchAsYouSpeak: this.searchAsYouSpeak, }, { $$widgetType: 'ais.voiceSearch', }); super.ngOnInit(); } } NgAisVoiceSearch.decorators = [ { type: Component, args: [{ selector: 'ais-voice-search', template: ` <div [class]="cx()"> <button type="button" [class]="cx('button')" [title]="state.isBrowserSupported ? buttonTitle : disabledButtonTitle" [disabled]="!state.isBrowserSupported" (click)="handleClick($event)" > <ng-container *ngTemplateOutlet="button ? button : defaultButton; context: templateContext"></ng-container> </button> <div [class]="cx('status')"> <ng-container *ngTemplateOutlet="status ? status : defaultStatus; context: templateContext"></ng-container> </div> </div> <ng-template #defaultButton let-status="status" let-errorCode="errorCode" let-isListening="isListening"> <svg xmlns='http://www.w3.org/2000/svg' width='16' height='16' viewBox='0 0 24 24' fill='none' stroke='currentColor' strokeWidth='2' strokeLinecap='round' strokeLinejoin='round' > <ng-container *ngIf="isNotAllowedError(); then errorSvgContent else normalSvgContent"> </ng-container> <ng-template #errorSvgContent> <line x1="1" y1="1" x2="23" y2="23"></line> <path d="M9 9v3a3 3 0 0 0 5.12 2.12M15 9.34V4a3 3 0 0 0-5.94-.6"></path> <path d="M17 16.95A7 7 0 0 1 5 12v-2m14 0v2a7 7 0 0 1-.11 1.23"></path> <line x1="12" y1="19" x2="12" y2="23"></line> <line x1="8" y1="23" x2="16" y2="23"></line> </ng-template> <ng-template #normalSvgContent> <path d="M12 1a3 3 0 0 0-3 3v8a3 3 0 0 0 6 0V4a3 3 0 0 0-3-3z" [attr.fill]="isListening ? 'currentColor' : 'none'" ></path> <path d="M19 10v2a7 7 0 0 1-14 0v-2"></path> <line x1="12" y1="19" x2="12" y2="23"></line> <line x1="8" y1="23" x2="16" y2="23"></line> </ng-template> </svg> </ng-template> <ng-template #defaultStatus let-transcript="transcript"> <p>{{transcript}}</p> </ng-template> ` },] } ]; NgAisVoiceSearch.ctorParameters = () => [ { type: NgAisIndex, decorators: [{ type: Inject, args: [forwardRef(() => NgAisIndex),] }, { type: Optional }] }, { type: NgAisInstantSearch, decorators: [{ type: Inject, args: [forwardRef(() => NgAisInstantSearch),] }] }, { type: NgZone } ]; NgAisVoiceSearch.propDecorators = { button: [{ type: ContentChild, args: ['button', { static: false },] }], status: [{ type: ContentChild, args: ['status', { static: false },] }], buttonTitle: [{ type: Input }], disabledButtonTitle: [{ type: Input }], searchAsYouSpeak: [{ type: Input }] }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-search.js","sourceRoot":"","sources":["../../../src/voice-search/voice-search.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Inject,\n  forwardRef,\n  NgZone,\n  ContentChild,\n  ElementRef,\n  TemplateRef,\n  OnInit,\n  Optional,\n} from '@angular/core';\n\nimport { connectVoiceSearch } 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  VoiceSearchConnectorParams,\n  VoiceSearchWidgetDescription,\n  VoiceSearchRenderState,\n} from 'instantsearch.js/es/connectors/voice-search/connectVoiceSearch';\n\n@Component({\n  selector: 'ais-voice-search',\n  template: `\n    <div [class]=\"cx()\">\n      <button\n        type=\"button\"\n        [class]=\"cx('button')\"\n        [title]=\"state.isBrowserSupported ? buttonTitle : disabledButtonTitle\"\n        [disabled]=\"!state.isBrowserSupported\"\n        (click)=\"handleClick($event)\"\n      >\n        <ng-container *ngTemplateOutlet=\"button ? button : defaultButton; context: templateContext\"></ng-container>\n      </button>\n      <div [class]=\"cx('status')\">\n        <ng-container *ngTemplateOutlet=\"status ? status : defaultStatus; context: templateContext\"></ng-container>\n      </div>\n    </div>\n\n    <ng-template #defaultButton let-status=\"status\" let-errorCode=\"errorCode\" let-isListening=\"isListening\">\n      <svg\n        xmlns='http://www.w3.org/2000/svg'\n        width='16'\n        height='16'\n        viewBox='0 0 24 24'\n        fill='none'\n        stroke='currentColor'\n        strokeWidth='2'\n        strokeLinecap='round'\n        strokeLinejoin='round'\n      >\n        <ng-container *ngIf=\"isNotAllowedError(); then errorSvgContent else normalSvgContent\">\n        </ng-container>\n        <ng-template #errorSvgContent>\n          <line x1=\"1\" y1=\"1\" x2=\"23\" y2=\"23\"></line>\n          <path d=\"M9 9v3a3 3 0 0 0 5.12 2.12M15 9.34V4a3 3 0 0 0-5.94-.6\"></path>\n          <path d=\"M17 16.95A7 7 0 0 1 5 12v-2m14 0v2a7 7 0 0 1-.11 1.23\"></path>\n          <line x1=\"12\" y1=\"19\" x2=\"12\" y2=\"23\"></line>\n          <line x1=\"8\" y1=\"23\" x2=\"16\" y2=\"23\"></line>\n        </ng-template>\n        <ng-template #normalSvgContent>\n          <path\n            d=\"M12 1a3 3 0 0 0-3 3v8a3 3 0 0 0 6 0V4a3 3 0 0 0-3-3z\"\n            [attr.fill]=\"isListening ? 'currentColor' : 'none'\"\n          ></path>\n          <path d=\"M19 10v2a7 7 0 0 1-14 0v-2\"></path>\n          <line x1=\"12\" y1=\"19\" x2=\"12\" y2=\"23\"></line>\n          <line x1=\"8\" y1=\"23\" x2=\"16\" y2=\"23\"></line>\n        </ng-template>\n      </svg>\n    </ng-template>\n    <ng-template #defaultStatus let-transcript=\"transcript\">\n      <p>{{transcript}}</p>\n    </ng-template>\n  `,\n})\nexport class NgAisVoiceSearch\n  extends TypedBaseWidget<\n    VoiceSearchWidgetDescription,\n    VoiceSearchConnectorParams\n  >\n  implements OnInit {\n  @ContentChild('button', { static: false })\n  button: TemplateRef<ElementRef>;\n  @ContentChild('status', { static: false })\n  status: TemplateRef<ElementRef>;\n\n  // rendering options\n  @Input() public buttonTitle: string = 'Search by voice';\n  @Input()\n  public disabledButtonTitle: string =\n    'Search by voice (not supported on this browser)';\n\n  // instance option\n  @Input()\n  public searchAsYouSpeak?: VoiceSearchConnectorParams['searchAsYouSpeak'];\n\n  public state: VoiceSearchRenderState = {\n    isBrowserSupported: undefined,\n    isListening: undefined,\n    toggleListening: noop,\n    voiceListeningState: {\n      status: 'initial',\n      transcript: '',\n      isSpeechFinal: false,\n      errorCode: undefined,\n    },\n  };\n\n  public templateContext: VoiceSearchRenderState['voiceListeningState'] & {\n    isListening: boolean;\n    isBrowserSupported: boolean;\n  } = {\n    status: 'initial',\n    errorCode: undefined,\n    transcript: '',\n    isSpeechFinal: false,\n    isListening: false,\n    isBrowserSupported: false,\n  };\n\n  constructor(\n    @Inject(forwardRef(() => NgAisIndex))\n    @Optional()\n    public parentIndex: NgAisIndex,\n    @Inject(forwardRef(() => NgAisInstantSearch))\n    public instantSearchInstance: NgAisInstantSearch,\n    private zone: NgZone\n  ) {\n    super('VoiceSearch');\n  }\n\n  ngOnInit() {\n    this.createWidget(\n      connectVoiceSearch,\n      {\n        searchAsYouSpeak: this.searchAsYouSpeak,\n      },\n      {\n        $$widgetType: 'ais.voiceSearch',\n      }\n    );\n    super.ngOnInit();\n  }\n\n  public handleClick = (event: MouseEvent): void => {\n    (event.currentTarget as HTMLElement).blur();\n    this.state.toggleListening();\n  };\n\n  public isNotAllowedError = (): boolean =>\n    this.state.voiceListeningState.status === 'error' &&\n    this.state.voiceListeningState.errorCode === 'not-allowed';\n\n  public updateState = (state: VoiceSearchRenderState): void => {\n    this.zone.run(() => {\n      this.templateContext = {\n        status: state.voiceListeningState.status,\n        errorCode: state.voiceListeningState.errorCode,\n        transcript: state.voiceListeningState.transcript,\n        isSpeechFinal: state.voiceListeningState.isSpeechFinal,\n        isListening: state.isListening,\n        isBrowserSupported: state.isBrowserSupported,\n      };\n      this.state = state;\n    });\n  };\n}\n"]}