angular-instantsearch
Version:
Lightning-fast search for Angular apps, by Algolia.
120 lines • 15.5 kB
JavaScript
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":"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","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"]}