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