UNPKG

angular-instantsearch

Version:

Lightning-fast search for Angular apps, by Algolia.

120 lines 15.5 kB
import { Component, Input, ViewChild, Inject, forwardRef, Optional, } from '@angular/core'; import { connectRange } from 'instantsearch.js/es/connectors'; import * as noUiSlider from 'nouislider'; import { TypedBaseWidget } from '../typed-base-widget'; import { NgAisInstantSearch } from '../instantsearch/instantsearch'; import { NgAisIndex } from '../index-widget/index-widget'; import { parseNumberInput, noop } from '../utils'; export class NgAisRangeSlider extends TypedBaseWidget { constructor(parentIndex, instantSearchInstance) { super('RangeSlider'); this.parentIndex = parentIndex; this.instantSearchInstance = instantSearchInstance; // rendering options this.pips = true; this.tooltips = true; this.state = { canRefine: false, format: { from: () => '', to: () => '', }, range: { min: 0, max: 1 }, refine: noop, start: [0, 1], sendEvent: noop, }; this.updateState = (state, isFirstRendering) => { if (isFirstRendering) { // create slider const config = { animate: false, behaviour: 'snap', connect: true, range: { min: 0, max: 1 }, start: [0, 1], step: this.step, tooltips: this.tooltips && [ { to: this.formatTooltip }, { to: this.formatTooltip }, ], }; // tslint:disable-next-line: no-boolean-literal-compare (pips is @Input, so could be not a boolean) if (this.pips === true || typeof this.pips === 'undefined') { Object.assign(config, { pips: { density: 3, mode: 'positions', stepped: true, values: [0, 50, 100], }, }); } else if (this.pips !== undefined) { Object.assign(config, { pips: this.pips }); } this.slider = noUiSlider.create(this.sliderContainer.nativeElement, config); // register listen events this.sliderContainer.nativeElement.noUiSlider.on('change', this.handleChange); } // update component inner state this.state = state; // update the slider state const { range: { min, max }, start, } = state; const disabled = min === max; const range = disabled ? { min, max: max + 0.0001 } : { min, max }; // TODO: test this as we're nolonger passing disable // it seems the API has changed: slider.setAttribute('disabled', true) / slider.removeAttribute('disabled'); // see: https://refreshless.com/nouislider/more/#section-disable this.slider.updateOptions({ range, start }); }; this.handleChange = (values) => { this.state.refine(values); }; this.formatTooltip = (value) => { return value.toFixed(parseNumberInput(this.precision)); }; } get step() { // compute step from the precision value const precision = parseNumberInput(this.precision) || 2; return 1 / Math.pow(10, precision); } ngOnInit() { this.createWidget(connectRange, { attribute: this.attribute, max: parseNumberInput(this.max), min: parseNumberInput(this.min), precision: parseNumberInput(this.precision), }, { $$widgetType: 'ais.rangeSlider', }); super.ngOnInit(); } } NgAisRangeSlider.decorators = [ { type: Component, args: [{ selector: 'ais-range-slider', template: ` <div [class]="cx()"> <div [class]="cx('body')"> <div #sliderContainer></div> </div> </div> ` },] } ]; NgAisRangeSlider.ctorParameters = () => [ { type: NgAisIndex, decorators: [{ type: Inject, args: [forwardRef(() => NgAisIndex),] }, { type: Optional }] }, { type: NgAisInstantSearch, decorators: [{ type: Inject, args: [forwardRef(() => NgAisInstantSearch),] }] } ]; NgAisRangeSlider.propDecorators = { sliderContainer: [{ type: ViewChild, args: ['sliderContainer', { static: false },] }], pips: [{ type: Input }], tooltips: [{ type: Input }], attribute: [{ type: Input }], min: [{ type: Input }], max: [{ type: Input }], precision: [{ type: Input }] }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"range-slider.js","sourceRoot":"","sources":["../../../src/range-slider/range-slider.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EACT,KAAK,EACL,SAAS,EACT,MAAM,EACN,UAAU,EACV,QAAQ,GACT,MAAM,eAAe,CAAC;AAEvB,OAAO,EAAE,YAAY,EAAE,MAAM,gCAAgC,CAAC;AAC9D,OAAO,KAAK,UAAU,MAAM,YAAY,CAAC;AAEzC,OAAO,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAC;AACvD,OAAO,EAAE,kBAAkB,EAAE,MAAM,gCAAgC,CAAC;AACpE,OAAO,EAAE,UAAU,EAAE,MAAM,8BAA8B,CAAC;AAC1D,OAAO,EAAE,gBAAgB,EAAE,IAAI,EAAE,MAAM,UAAU,CAAC;AAkBlD,MAAM,OAAO,gBAAiB,SAAQ,eAGrC;IAkCC,YAGS,WAAuB,EAEvB,qBAAyC;QAEhD,KAAK,CAAC,aAAa,CAAC,CAAC;QAJd,gBAAW,GAAX,WAAW,CAAY;QAEvB,0BAAqB,GAArB,qBAAqB,CAAoB;QAnClD,oBAAoB;QACJ,SAAI,GAAY,IAAI,CAAC;QACrB,aAAQ,GAAY,IAAI,CAAC;QAQlC,UAAK,GAAqB;YAC/B,SAAS,EAAE,KAAK;YAChB,MAAM,EAAE;gBACN,IAAI,EAAE,GAAG,EAAE,CAAC,EAAE;gBACd,EAAE,EAAE,GAAG,EAAE,CAAC,EAAE;aACb;YACD,KAAK,EAAE,EAAE,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAE;YACzB,MAAM,EAAE,IAAI;YACZ,KAAK,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC;YACb,SAAS,EAAE,IAAI;SAChB,CAAC;QAqCK,gBAAW,GAAG,CAAC,KAAuB,EAAE,gBAAyB,EAAE,EAAE;YAC1E,IAAI,gBAAgB,EAAE;gBACpB,gBAAgB;gBAChB,MAAM,MAAM,GAAG;oBACb,OAAO,EAAE,KAAK;oBACd,SAAS,EAAE,MAAM;oBACjB,OAAO,EAAE,IAAI;oBACb,KAAK,EAAE,EAAE,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAE;oBACzB,KAAK,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC;oBACb,IAAI,EAAE,IAAI,CAAC,IAAI;oBACf,QAAQ,EAAE,IAAI,CAAC,QAAQ,IAAI;wBACzB,EAAE,EAAE,EAAE,IAAI,CAAC,aAAa,EAAE;wBAC1B,EAAE,EAAE,EAAE,IAAI,CAAC,aAAa,EAAE;qBAC3B;iBACF,CAAC;gBAEF,mGAAmG;gBACnG,IAAI,IAAI,CAAC,IAAI,KAAK,IAAI,IAAI,OAAO,IAAI,CAAC,IAAI,KAAK,WAAW,EAAE;oBAC1D,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE;wBACpB,IAAI,EAAE;4BACJ,OAAO,EAAE,CAAC;4BACV,IAAI,EAAE,WAAW;4BACjB,OAAO,EAAE,IAAI;4BACb,MAAM,EAAE,CAAC,CAAC,EAAE,EAAE,EAAE,GAAG,CAAC;yBACrB;qBACF,CAAC,CAAC;iBACJ;qBAAM,IAAI,IAAI,CAAC,IAAI,KAAK,SAAS,EAAE;oBAClC,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;iBAC5C;gBAED,IAAI,CAAC,MAAM,GAAG,UAAU,CAAC,MAAM,CAC7B,IAAI,CAAC,eAAe,CAAC,aAAa,EAClC,MAAM,CACP,CAAC;gBAEF,yBAAyB;gBACzB,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,UAAU,CAAC,EAAE,CAC9C,QAAQ,EACR,IAAI,CAAC,YAAY,CAClB,CAAC;aACH;YAED,+BAA+B;YAC/B,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;YAEnB,0BAA0B;YAC1B,MAAM,EACJ,KAAK,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,EACnB,KAAK,GACN,GAAG,KAAK,CAAC;YAEV,MAAM,QAAQ,GAAG,GAAG,KAAK,GAAG,CAAC;YAC7B,MAAM,KAAK,GAAG,QAAQ,CAAC,CAAC,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,GAAG,MAAM,EAAE,CAAC,CAAC,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC;YAEnE,oDAAoD;YACpD,4GAA4G;YAC5G,gEAAgE;YAChE,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC;QAC9C,CAAC,CAAC;QAEK,iBAAY,GAAG,CAAC,MAAuB,EAAE,EAAE;YAChD,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QAC5B,CAAC,CAAC;QAEK,kBAAa,GAAG,CAAC,KAAa,EAAE,EAAE;YACvC,OAAO,KAAK,CAAC,OAAO,CAAC,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;QACzD,CAAC,CAAC;IArFF,CAAC;IAdD,IAAI,IAAI;QACN,wCAAwC;QACxC,MAAM,SAAS,GAAG,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QACxD,OAAO,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC;IACrC,CAAC;IAYM,QAAQ;QACb,IAAI,CAAC,YAAY,CACf,YAAY,EACZ;YACE,SAAS,EAAE,IAAI,CAAC,SAAS;YACzB,GAAG,EAAE,gBAAgB,CAAC,IAAI,CAAC,GAAG,CAAC;YAC/B,GAAG,EAAE,gBAAgB,CAAC,IAAI,CAAC,GAAG,CAAC;YAC/B,SAAS,EAAE,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC;SAC5C,EACD;YACE,YAAY,EAAE,iBAAiB;SAChC,CACF,CAAC;QAEF,KAAK,CAAC,QAAQ,EAAE,CAAC;IACnB,CAAC;;;YAxEF,SAAS,SAAC;gBACT,QAAQ,EAAE,kBAAkB;gBAC5B,QAAQ,EAAE;;;;;;GAMT;aACF;;;YAlBQ,UAAU,uBAyDd,MAAM,SAAC,UAAU,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,cACnC,QAAQ;YA3DJ,kBAAkB,uBA6DtB,MAAM,SAAC,UAAU,CAAC,GAAG,EAAE,CAAC,kBAAkB,CAAC;;;8BArC7C,SAAS,SAAC,iBAAiB,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE;mBAI9C,KAAK;uBACL,KAAK;wBAGL,KAAK;kBACL,KAAK;kBACL,KAAK;wBACL,KAAK","sourcesContent":["import {\n  Component,\n  Input,\n  ViewChild,\n  Inject,\n  forwardRef,\n  Optional,\n} from '@angular/core';\n\nimport { connectRange } from 'instantsearch.js/es/connectors';\nimport * as noUiSlider from 'nouislider';\n\nimport { TypedBaseWidget } from '../typed-base-widget';\nimport { NgAisInstantSearch } from '../instantsearch/instantsearch';\nimport { NgAisIndex } from '../index-widget/index-widget';\nimport { parseNumberInput, noop } from '../utils';\nimport {\n  RangeBoundaries,\n  RangeConnectorParams,\n  RangeWidgetDescription,\n  RangeRenderState,\n} from 'instantsearch.js/es/connectors/range/connectRange';\n\n@Component({\n  selector: 'ais-range-slider',\n  template: `\n    <div [class]=\"cx()\">\n      <div [class]=\"cx('body')\">\n        <div #sliderContainer></div>\n      </div>\n    </div>\n  `,\n})\nexport class NgAisRangeSlider extends TypedBaseWidget<\n  RangeWidgetDescription,\n  RangeConnectorParams\n> {\n  @ViewChild('sliderContainer', { static: false })\n  public sliderContainer: any;\n\n  // rendering options\n  @Input() public pips: boolean = true;\n  @Input() public tooltips: boolean = true;\n\n  // instance options\n  @Input() public attribute: RangeConnectorParams['attribute'];\n  @Input() public min?: RangeConnectorParams['min'];\n  @Input() public max?: RangeConnectorParams['max'];\n  @Input() public precision?: RangeConnectorParams['precision'];\n\n  public state: RangeRenderState = {\n    canRefine: false,\n    format: {\n      from: () => '',\n      to: () => '',\n    },\n    range: { min: 0, max: 1 },\n    refine: noop,\n    start: [0, 1],\n    sendEvent: noop,\n  };\n\n  private slider: any;\n\n  get step() {\n    // compute step from the precision value\n    const precision = parseNumberInput(this.precision) || 2;\n    return 1 / Math.pow(10, precision);\n  }\n\n  constructor(\n    @Inject(forwardRef(() => NgAisIndex))\n    @Optional()\n    public parentIndex: NgAisIndex,\n    @Inject(forwardRef(() => NgAisInstantSearch))\n    public instantSearchInstance: NgAisInstantSearch\n  ) {\n    super('RangeSlider');\n  }\n\n  public ngOnInit() {\n    this.createWidget(\n      connectRange,\n      {\n        attribute: this.attribute,\n        max: parseNumberInput(this.max),\n        min: parseNumberInput(this.min),\n        precision: parseNumberInput(this.precision),\n      },\n      {\n        $$widgetType: 'ais.rangeSlider',\n      }\n    );\n\n    super.ngOnInit();\n  }\n\n  public updateState = (state: RangeRenderState, isFirstRendering: boolean) => {\n    if (isFirstRendering) {\n      // create slider\n      const config = {\n        animate: false,\n        behaviour: 'snap',\n        connect: true,\n        range: { min: 0, max: 1 },\n        start: [0, 1],\n        step: this.step,\n        tooltips: this.tooltips && [\n          { to: this.formatTooltip },\n          { to: this.formatTooltip },\n        ],\n      };\n\n      // tslint:disable-next-line: no-boolean-literal-compare (pips is @Input, so could be not a boolean)\n      if (this.pips === true || typeof this.pips === 'undefined') {\n        Object.assign(config, {\n          pips: {\n            density: 3,\n            mode: 'positions',\n            stepped: true,\n            values: [0, 50, 100],\n          },\n        });\n      } else if (this.pips !== undefined) {\n        Object.assign(config, { pips: this.pips });\n      }\n\n      this.slider = noUiSlider.create(\n        this.sliderContainer.nativeElement,\n        config\n      );\n\n      // register listen events\n      this.sliderContainer.nativeElement.noUiSlider.on(\n        'change',\n        this.handleChange\n      );\n    }\n\n    // update component inner state\n    this.state = state;\n\n    // update the slider state\n    const {\n      range: { min, max },\n      start,\n    } = state;\n\n    const disabled = min === max;\n    const range = disabled ? { min, max: max + 0.0001 } : { min, max };\n\n    // TODO: test this as we're nolonger passing disable\n    // it seems the API has changed: slider.setAttribute('disabled', true) / slider.removeAttribute('disabled');\n    // see: https://refreshless.com/nouislider/more/#section-disable\n    this.slider.updateOptions({ range, start });\n  };\n\n  public handleChange = (values: RangeBoundaries) => {\n    this.state.refine(values);\n  };\n\n  public formatTooltip = (value: number) => {\n    return value.toFixed(parseNumberInput(this.precision));\n  };\n}\n"]}