UNPKG

@firestitch/address

Version:
178 lines 26.4 kB
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, Input, Optional, Output, ViewChild, inject } from '@angular/core'; import { ControlContainer, NgForm, NgModel, FormsModule } from '@angular/forms'; import { FsAutocompleteComponent, FsAutocompleteModule } from '@firestitch/autocomplete'; import { guid } from '@firestitch/common'; import { controlContainerFactory } from '@firestitch/core'; import { Subject, of } from 'rxjs'; import { map, takeUntil } from 'rxjs/operators'; import { Countries } from '../../consts'; import { Country } from '../../enums/country.enum'; import { FsFormModule } from '@firestitch/form'; import * as i0 from "@angular/core"; import * as i1 from "@firestitch/autocomplete"; import * as i2 from "@angular/forms"; import * as i3 from "@firestitch/form"; export class FsAddressRegionComponent { _cdRef = inject(ChangeDetectorRef); autocompleteModel; set region(regionCode) { const region = this.addressCountries .reduce((accum, addressCountry) => { return [ ...accum, ...(addressCountry.regions || []) .filter((addressRegion) => (addressRegion.code === regionCode && (!this.country || this.country === addressCountry.code))), ]; }, [])[0]; this.regionModel = (region ? region : (regionCode ? { name: regionCode } : null)); } get region() { return this.regionModel?.code; } disabled = false; country; label; required = false; regionCountryOrder = [Country.Canada, Country.UnitedStates]; set countries(countryCodes) { countryCodes = countryCodes || [Country.Canada, Country.UnitedStates]; this._countries = countryCodes .map((countryCode) => { return Countries.find((country) => country.code === countryCode); }); this.updateCountryRegionLabels(); } get addressCountries() { return this._countries; } regionChange = new EventEmitter(); regionModel; controlName = `region${guid('xxxxxx')}`; regionLabel; countryEnum = Country; _countries = []; _destroy$ = new Subject(); constructor() { this.countries = [Country.Canada, Country.UnitedStates]; } ngOnInit() { this.updateCountryRegionLabels(); this._listenControlStateChanges(); } clear() { this.regionModel = null; } ngOnDestroy() { this._destroy$.next(null); this._destroy$.complete(); } fetch = (keyword) => { keyword = keyword.toLowerCase(); return of(null) .pipe(map(() => { const regions = this._countries .reduce((accum, country) => { const countryRegions = (country.regions || []) .filter((region) => { const regionName = region.name.toLowerCase().trim(); return regionName.indexOf(keyword) !== -1; }); if (countryRegions.length) { console.log(country, keyword, countryRegions); } return [ ...accum, ...countryRegions .map((countryRegion) => { return { ...countryRegion, country: country.name, }; }), ]; }, []); console.log(regions, keyword); return regions; })); }; displayWith = (data) => { return data?.name; }; selectUserOption(keyword) { this.regionModel = { code: keyword, name: keyword, }; this.autocompleteModel.control.markAsDirty(); this.regionChange.emit(keyword); } regionChanged() { this.regionChange.emit(this.regionModel?.code); } justUseShow = (keyword) => { return !!keyword; }; updateCountryRegionLabels() { this.regionLabel = this.label ? this.label : Object.keys(this._countries .reduce((accum, country) => { return { ...accum, [country.regionLabel || 'Province']: true, }; }, {})) .join('/'); } // we need this to get updated ng-(invalid/dirty) classes _listenControlStateChanges() { this.autocompleteModel .control .statusChanges .pipe(takeUntil(this._destroy$)) .subscribe(() => { this._cdRef.markForCheck(); }); } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: FsAddressRegionComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.7", type: FsAddressRegionComponent, isStandalone: true, selector: "fs-address-region", inputs: { region: "region", disabled: "disabled", country: "country", label: "label", required: "required", regionCountryOrder: "regionCountryOrder", countries: "countries" }, outputs: { regionChange: "regionChange" }, viewQueries: [{ propertyName: "autocompleteModel", first: true, predicate: FsAutocompleteComponent, descendants: true, read: NgModel, static: true }], ngImport: i0, template: "<fs-autocomplete\n [fetch]=\"fetch\"\n [displayWith]=\"displayWith\"\n [fetchOnFocus]=\"true\"\n [(ngModel)]=\"regionModel\"\n (ngModelChange)=\"regionChanged()\"\n [placeholder]=\"regionLabel\"\n [disabled]=\"disabled\"\n [fsFormRequired]=\"required\"\n [name]=\"controlName\">\n <ng-template\n fsAutocompleteTemplate\n let-data=\"data\">\n <span class=\"country-region\">\n <span>\n {{ data.name }}\n </span>\n @if (!regionModel && addressCountries.length > 1) {\n <span>\n {{ data.country }}\n </span>\n }\n </span>\n </ng-template>\n <ng-template\n fsAutocompleteStatic\n let-keyword\n (selected)=\"selectUserOption($event)\"\n [show]=\"justUseShow\">\n Just Use \"{{ keyword }}\"\n </ng-template>\n <ng-template fsAutocompleteNoResults></ng-template>\n</fs-autocomplete>", styles: ["@charset \"UTF-8\";.country-region{display:inline-flex}.country-region span:not(:last-child):after{content:\",\\a0\"}\n"], dependencies: [{ kind: "ngmodule", type: FsAutocompleteModule }, { kind: "component", type: i1.FsAutocompleteComponent, selector: "fs-autocomplete", inputs: ["fetch", "displayWith", "placeholder", "fetchOnFocus", "readonly", "required", "disabled", "formFieldClass", "appearance", "hint", "panelWidth", "panelClass", "showClear"], outputs: ["cleared", "opened", "closed"] }, { kind: "directive", type: i1.FsAutocompleteTemplateDirective, selector: "[fsAutocompleteTemplate]" }, { kind: "directive", type: i1.FsAutocompleteStaticDirective, selector: "[fsAutocompleteStatic],[fsAutocompleteStaticTemplate]", inputs: ["show", "disable"], outputs: ["selected"] }, { kind: "directive", type: i1.FsAutocompleteNoResultsDirective, selector: "[fsAutocompleteNoResults]", inputs: ["show"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: FsFormModule }, { kind: "directive", type: i3.FsFormRequiredDirective, selector: "[fsFormRequired],[ngModel][required]", inputs: ["fsFormRequired", "required", "fsFormRequiredMessage"] }], viewProviders: [ { provide: ControlContainer, useFactory: controlContainerFactory, deps: [[new Optional(), NgForm]], }, ], changeDetection: i0.ChangeDetectionStrategy.OnPush }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: FsAddressRegionComponent, decorators: [{ type: Component, args: [{ selector: 'fs-address-region', changeDetection: ChangeDetectionStrategy.OnPush, viewProviders: [ { provide: ControlContainer, useFactory: controlContainerFactory, deps: [[new Optional(), NgForm]], }, ], standalone: true, imports: [ FsAutocompleteModule, FormsModule, FsFormModule, ], template: "<fs-autocomplete\n [fetch]=\"fetch\"\n [displayWith]=\"displayWith\"\n [fetchOnFocus]=\"true\"\n [(ngModel)]=\"regionModel\"\n (ngModelChange)=\"regionChanged()\"\n [placeholder]=\"regionLabel\"\n [disabled]=\"disabled\"\n [fsFormRequired]=\"required\"\n [name]=\"controlName\">\n <ng-template\n fsAutocompleteTemplate\n let-data=\"data\">\n <span class=\"country-region\">\n <span>\n {{ data.name }}\n </span>\n @if (!regionModel && addressCountries.length > 1) {\n <span>\n {{ data.country }}\n </span>\n }\n </span>\n </ng-template>\n <ng-template\n fsAutocompleteStatic\n let-keyword\n (selected)=\"selectUserOption($event)\"\n [show]=\"justUseShow\">\n Just Use \"{{ keyword }}\"\n </ng-template>\n <ng-template fsAutocompleteNoResults></ng-template>\n</fs-autocomplete>", styles: ["@charset \"UTF-8\";.country-region{display:inline-flex}.country-region span:not(:last-child):after{content:\",\\a0\"}\n"] }] }], ctorParameters: () => [], propDecorators: { autocompleteModel: [{ type: ViewChild, args: [FsAutocompleteComponent, { read: NgModel, static: true }] }], region: [{ type: Input }], disabled: [{ type: Input }], country: [{ type: Input }], label: [{ type: Input }], required: [{ type: Input }], regionCountryOrder: [{ type: Input }], countries: [{ type: Input }], regionChange: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"address-region.component.js","sourceRoot":"","sources":["../../../../../src/app/components/address-region/address-region.component.ts","../../../../../src/app/components/address-region/address-region.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, Input, OnDestroy, OnInit, Optional, Output, ViewChild, inject } from '@angular/core';\nimport { ControlContainer, NgForm, NgModel, FormsModule } from '@angular/forms';\n\n\nimport { FsAutocompleteComponent, FsAutocompleteModule } from '@firestitch/autocomplete';\nimport { guid } from '@firestitch/common';\nimport { controlContainerFactory } from '@firestitch/core';\n\nimport { Subject, of } from 'rxjs';\nimport { map, takeUntil } from 'rxjs/operators';\n\nimport { Countries } from '../../consts';\nimport { Country } from '../../enums/country.enum';\nimport { IAddressCountry } from '../../interfaces/address-country.interface';\nimport { IAddressRegion } from '../../interfaces/address-region.interface';\nimport { FsFormModule } from '@firestitch/form';\n\n\n@Component({\n    selector: 'fs-address-region',\n    templateUrl: './address-region.component.html',\n    styleUrls: ['./address-region.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    viewProviders: [\n        {\n            provide: ControlContainer,\n            useFactory: controlContainerFactory,\n            deps: [[new Optional(), NgForm]],\n        },\n    ],\n    standalone: true,\n    imports: [\n        FsAutocompleteModule,\n        FormsModule,\n        FsFormModule,\n    ],\n})\nexport class FsAddressRegionComponent implements OnInit, OnDestroy {\n  private _cdRef = inject(ChangeDetectorRef);\n\n\n  @ViewChild(FsAutocompleteComponent, { read: NgModel, static: true })\n  public autocompleteModel: NgModel;\n\n  @Input() public set region(regionCode: string) {\n    const region = this.addressCountries\n      .reduce((accum, addressCountry) => {\n        return [\n          ...accum,\n          ...(addressCountry.regions || [])\n            .filter((addressRegion) => (\n              addressRegion.code === regionCode &&\n              (!this.country || this.country === addressCountry.code)),\n            ),\n        ];\n      }, [])[0];\n\n    this.regionModel = (region ? region : (regionCode ? { name: regionCode } : null));\n  }\n\n  public get region() {\n    return this.regionModel?.code;\n  }\n\n  @Input() public disabled = false;\n  @Input() public country: Country | string;\n  @Input() public label;\n  @Input() public required = false;\n  @Input() public regionCountryOrder = [Country.Canada, Country.UnitedStates];\n  @Input() public set countries(countryCodes: (string | Country)[]) {\n    countryCodes = countryCodes || [Country.Canada, Country.UnitedStates];\n    this._countries = countryCodes\n      .map((countryCode: string) => {\n        return Countries.find((country) => country.code === countryCode);\n      });\n\n    this.updateCountryRegionLabels();\n  }\n\n  public get addressCountries() {\n    return this._countries;\n  }\n\n  @Output() public regionChange = new EventEmitter<string>();\n\n  public regionModel: IAddressRegion;\n  public controlName = `region${guid('xxxxxx')}`;\n  public regionLabel: string;\n  public countryEnum = Country;\n\n  private _countries: IAddressCountry[] = [];\n  private _destroy$ = new Subject<void>();\n\n  constructor() {\n    this.countries = [Country.Canada, Country.UnitedStates];\n  }\n\n  public ngOnInit() {\n    this.updateCountryRegionLabels();\n    this._listenControlStateChanges();\n  }\n\n  public clear() {\n    this.regionModel = null;\n  }\n\n  public ngOnDestroy(): void {\n    this._destroy$.next(null);\n    this._destroy$.complete();\n  }\n\n  public fetch = (keyword: string) => {\n    keyword = keyword.toLowerCase();\n\n    return of(null)\n      .pipe(\n        map(() => {\n          const regions: IAddressRegion[] = this._countries\n            .reduce((accum, country) => {\n              const countryRegions = (country.regions || [])\n                .filter((region) => {\n                  const regionName = region.name.toLowerCase().trim();\n\n                  return regionName.indexOf(keyword) !== -1;\n                });\n\n              if (countryRegions.length ) {\n                console.log(country, keyword, countryRegions);\n              }\n\n              return [\n                ...accum,\n                ...countryRegions\n                  .map((countryRegion) => {\n                    return {\n                      ...countryRegion,\n                      country: country.name,\n                    };\n                  }),\n              ];\n            }, []);\n          console.log(regions, keyword);\n\n          return regions;\n        }),\n      );\n  };\n\n  public displayWith = (data) => {\n    return data?.name;\n  };\n\n  public selectUserOption(keyword) {\n    this.regionModel = {\n      code: keyword,\n      name: keyword,\n    };\n\n    this.autocompleteModel.control.markAsDirty();\n\n    this.regionChange.emit(keyword);\n  }\n\n  public regionChanged() {\n    this.regionChange.emit(this.regionModel?.code);\n  }\n\n  public justUseShow = (keyword) => {\n    return !!keyword;\n  };\n\n  public updateCountryRegionLabels() {\n    this.regionLabel = this.label ? this.label : Object.keys(\n      this._countries\n        .reduce((accum, country) => {\n          return {\n            ...accum,\n            [country.regionLabel || 'Province']: true,\n          };\n        }, {}),\n    )\n      .join('/');\n  }\n\n  // we need this to get updated ng-(invalid/dirty) classes\n  private _listenControlStateChanges(): void {\n    this.autocompleteModel\n      .control\n      .statusChanges\n      .pipe(\n        takeUntil(this._destroy$),\n      )\n      .subscribe(() => {\n        this._cdRef.markForCheck();\n      });\n  }\n\n}\n","<fs-autocomplete\n  [fetch]=\"fetch\"\n  [displayWith]=\"displayWith\"\n  [fetchOnFocus]=\"true\"\n  [(ngModel)]=\"regionModel\"\n  (ngModelChange)=\"regionChanged()\"\n  [placeholder]=\"regionLabel\"\n  [disabled]=\"disabled\"\n  [fsFormRequired]=\"required\"\n  [name]=\"controlName\">\n  <ng-template\n    fsAutocompleteTemplate\n    let-data=\"data\">\n    <span class=\"country-region\">\n      <span>\n        {{ data.name }}\n      </span>\n      @if (!regionModel && addressCountries.length > 1) {\n        <span>\n          {{ data.country }}\n        </span>\n      }\n    </span>\n  </ng-template>\n  <ng-template\n    fsAutocompleteStatic\n    let-keyword\n    (selected)=\"selectUserOption($event)\"\n    [show]=\"justUseShow\">\n    Just Use \"{{ keyword }}\"\n  </ng-template>\n  <ng-template fsAutocompleteNoResults></ng-template>\n</fs-autocomplete>"]}