@angular-material-extensions/google-maps-autocomplete
Version:
Autocomplete input component and directive for google-maps built with angular and material design
228 lines • 28.9 kB
JavaScript
import { Directive, EventEmitter, forwardRef, HostListener, Inject, Input, Output, PLATFORM_ID, ViewChild } from '@angular/core';
import { NG_VALIDATORS, Validators } from '@angular/forms';
import { isPlatformBrowser } from '@angular/common';
import { ApiKeyToken } from "../tokens";
import * as i0 from "@angular/core";
import * as i1 from "../services/script-loader.service";
export class MatGoogleMapsAutocompleteDirective {
platformId;
apiKey;
elemRef;
cf;
loaderService;
ngZone;
inputField;
autocomplete;
address;
country;
placeIdOnly;
strictBounds;
types;
type;
autoCompleteOptions = {};
onChange = new EventEmitter();
onAutocompleteSelected = new EventEmitter();
onGermanAddressMapped = new EventEmitter();
onLocationSelected = new EventEmitter();
disabled;
_value;
get value() {
return this._value;
}
set value(value) {
this._value = value;
this.propagateChange(this.value);
this.cf.markForCheck();
}
onNewPlaceResult = new EventEmitter();
propagateChange = (_) => {
};
constructor(platformId, apiKey, elemRef, cf, loaderService, ngZone) {
this.platformId = platformId;
this.apiKey = apiKey;
this.elemRef = elemRef;
this.cf = cf;
this.loaderService = loaderService;
this.ngZone = ngZone;
}
ngOnDestroy() {
if (this.autocomplete) {
google.maps.event.clearInstanceListeners(this.autocomplete);
}
}
ngAfterViewInit() {
this.loadMap();
}
ngOnInit() {
}
validate(fc) {
return fc.hasValidator(Validators.required) ? !!fc?.value : true;
}
onChangeInputValue() {
const value = this.elemRef.nativeElement?.value;
this.value = value;
}
initGoogleMapsAutocomplete() {
const autocomplete = new google.maps.places.Autocomplete(this.elemRef.nativeElement, this.autoCompleteOptions);
autocomplete.addListener('place_changed', () => {
this.ngZone.run(() => {
// get the place result
const place = autocomplete.getPlace();
const germanAddress = {
gmID: place.id,
icon: place.icon,
url: place.url,
placeID: place.place_id,
displayAddress: place.formatted_address,
name: place.name,
vicinity: place.vicinity,
locality: {},
state: {},
country: {},
geoLocation: { latitude: -1, longitude: -1 },
};
if (place.geometry && place.geometry.location) {
germanAddress.geoLocation.latitude = place.geometry.location.lat();
germanAddress.geoLocation.longitude = place.geometry.location.lng();
}
// console.log("place.address_components --> ", place.address_components);
place.address_components.forEach(value => {
if (value.types.indexOf('street_number') > -1) {
germanAddress.streetNumber = value.short_name;
}
if (value.types.indexOf('route') > -1) {
germanAddress.streetName = value.long_name;
}
if (value.types.indexOf('postal_code') > -1) {
germanAddress.postalCode = value.short_name;
}
if (value.types.indexOf('sublocality') > -1) {
germanAddress.sublocality = value.long_name;
}
if (value.types.indexOf('locality') > -1) {
germanAddress.locality.long = value.long_name;
germanAddress.locality.short = value.short_name;
}
if (value.types.indexOf('administrative_area_level_1') > -1) {
germanAddress.state.long = value.long_name;
germanAddress.state.short = value.short_name;
}
if (value.types.indexOf('country') > -1) {
germanAddress.country.long = value.long_name;
germanAddress.country.short = value.short_name;
}
if (value.types.indexOf('administrative_area_level_3') > -1) {
germanAddress.locality.short = value.short_name;
}
});
this.onGermanAddressMapped.emit(germanAddress);
this.value = place.formatted_address;
this.address = place.formatted_address;
this.onAutocompleteSelected.emit(place);
this.onLocationSelected.emit({
latitude: place.geometry.location.lat(),
longitude: place.geometry.location.lng()
});
});
});
}
registerOnChange(fn) {
this.propagateChange = fn;
}
registerOnTouched(fn) {
}
setDisabledState(isDisabled) {
this.disabled = isDisabled;
}
writeValue(obj) {
if (obj) {
this.value = obj;
}
}
loadMap() {
this.loaderService.loadScript(`https://maps.googleapis.com/maps/api/js?key=${this.apiKey}&libraries=places`)
.then(() => {
this.initMap();
})
.catch(error => console.error('Google Maps loading failed: ', error));
}
initMap() {
if (isPlatformBrowser(this.platformId)) {
console.log("on after view init --> ", this.elemRef.nativeElement);
this.autocomplete = new google.maps.places.Autocomplete(this.elemRef.nativeElement);
const options = {
// types: ['address'],
// componentRestrictions: {country: this.country},
placeIdOnly: this.placeIdOnly,
strictBounds: this.strictBounds,
// types: this.types,
type: this.type
};
// tslint:disable-next-line:no-unused-expression
this.country ? options.componentRestrictions = { country: this.country } : null;
// tslint:disable-next-line:no-unused-expression
this.country ? options.types = this.types : null;
this.autoCompleteOptions = Object.assign(this.autoCompleteOptions, options);
this.initGoogleMapsAutocomplete();
}
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.2", ngImport: i0, type: MatGoogleMapsAutocompleteDirective, deps: [{ token: PLATFORM_ID }, { token: ApiKeyToken }, { token: i0.ElementRef }, { token: i0.ChangeDetectorRef }, { token: i1.ScriptLoaderService }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Directive });
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.2", type: MatGoogleMapsAutocompleteDirective, selector: "[matGoogleMapsAutocomplete]", inputs: { address: "address", country: "country", placeIdOnly: "placeIdOnly", strictBounds: "strictBounds", types: "types", type: "type", autoCompleteOptions: "autoCompleteOptions", value: "value" }, outputs: { onChange: "onChange", onAutocompleteSelected: "onAutocompleteSelected", onGermanAddressMapped: "onGermanAddressMapped", onLocationSelected: "onLocationSelected" }, host: { listeners: { "change": "onChangeInputValue()" } }, providers: [
{
provide: NG_VALIDATORS,
useExisting: forwardRef(() => MatGoogleMapsAutocompleteDirective),
multi: true
}
], viewQueries: [{ propertyName: "inputField", first: true, predicate: ["inputField"], descendants: true }], exportAs: ["matGoogleMapsAutocomplete"], ngImport: i0 });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.2", ngImport: i0, type: MatGoogleMapsAutocompleteDirective, decorators: [{
type: Directive,
args: [{
selector: '[matGoogleMapsAutocomplete]',
exportAs: 'matGoogleMapsAutocomplete',
providers: [
{
provide: NG_VALIDATORS,
useExisting: forwardRef(() => MatGoogleMapsAutocompleteDirective),
multi: true
}
]
}]
}], ctorParameters: function () { return [{ type: undefined, decorators: [{
type: Inject,
args: [PLATFORM_ID]
}] }, { type: undefined, decorators: [{
type: Inject,
args: [ApiKeyToken]
}] }, { type: i0.ElementRef }, { type: i0.ChangeDetectorRef }, { type: i1.ScriptLoaderService }, { type: i0.NgZone }]; }, propDecorators: { inputField: [{
type: ViewChild,
args: ['inputField']
}], address: [{
type: Input
}], country: [{
type: Input
}], placeIdOnly: [{
type: Input
}], strictBounds: [{
type: Input
}], types: [{
type: Input
}], type: [{
type: Input
}], autoCompleteOptions: [{
type: Input
}], onChange: [{
type: Output
}], onAutocompleteSelected: [{
type: Output
}], onGermanAddressMapped: [{
type: Output
}], onLocationSelected: [{
type: Output
}], value: [{
type: Input
}], onChangeInputValue: [{
type: HostListener,
args: ['change']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mat-google-maps-autocomplete.directive.js","sourceRoot":"","sources":["../../../../../../projects/angular-material-extensions/google-maps-autocomplete/src/lib/directives/mat-google-maps-autocomplete.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  HostListener,\n  Inject,\n  Input,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Output,\n  PLATFORM_ID,\n  ViewChild\n} from '@angular/core';\nimport {ControlValueAccessor, FormControl, NG_VALIDATORS, Validators} from '@angular/forms';\nimport {GermanAddress, Location} from '../interfaces';\nimport {isPlatformBrowser} from '@angular/common';\nimport {ScriptLoaderService} from \"../services/script-loader.service\";\nimport {ApiKeyToken} from \"../tokens\";\nimport PlaceResult = google.maps.places.PlaceResult;\nimport AutocompleteOptions = google.maps.places.AutocompleteOptions;\n\n@Directive({\n  selector: '[matGoogleMapsAutocomplete]',\n  exportAs: 'matGoogleMapsAutocomplete',\n  providers: [\n    {\n      provide: NG_VALIDATORS,\n      useExisting: forwardRef(() => MatGoogleMapsAutocompleteDirective),\n      multi: true\n    }\n  ]\n})\nexport class MatGoogleMapsAutocompleteDirective implements OnInit, AfterViewInit, OnDestroy, ControlValueAccessor {\n\n  @ViewChild('inputField')\n  inputField!: ElementRef;\n\n  autocomplete: google.maps.places.Autocomplete | undefined;\n\n  @Input()\n  address: PlaceResult | string;\n\n  @Input()\n  country: string | string[];\n\n  @Input()\n  placeIdOnly?: boolean;\n\n  @Input()\n  strictBounds?: boolean;\n\n  @Input()\n  types?: string[];\n\n  @Input()\n  type?: string;\n\n  @Input()\n  autoCompleteOptions: AutocompleteOptions = {};\n\n  @Output()\n  onChange: EventEmitter<PlaceResult | string | null> = new EventEmitter<PlaceResult | string | null>();\n\n  @Output()\n  onAutocompleteSelected: EventEmitter<PlaceResult> = new EventEmitter<PlaceResult>();\n\n  @Output()\n  onGermanAddressMapped: EventEmitter<GermanAddress> = new EventEmitter<GermanAddress>();\n\n  @Output()\n  onLocationSelected: EventEmitter<Location> = new EventEmitter<Location>();\n\n  disabled: boolean\n\n  _value: string;\n\n  get value(): string {\n    return this._value;\n  }\n\n  @Input()\n  set value(value: string) {\n    this._value = value;\n    this.propagateChange(this.value);\n    this.cf.markForCheck();\n  }\n\n  private onNewPlaceResult: EventEmitter<any> = new EventEmitter();\n\n  propagateChange = (_: any) => {\n  };\n\n  constructor(@Inject(PLATFORM_ID) public platformId: string,\n              @Inject(ApiKeyToken)\n              public apiKey: string,\n              public elemRef: ElementRef,\n              private cf: ChangeDetectorRef,\n              private loaderService: ScriptLoaderService,\n              private ngZone: NgZone) {\n  }\n\n  ngOnDestroy(): void {\n    if (this.autocomplete) {\n      google.maps.event.clearInstanceListeners(this.autocomplete);\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.loadMap();\n  }\n\n  ngOnInit(): void {\n  }\n\n  validate(fc: FormControl) {\n    return fc.hasValidator(Validators.required) ? !!fc?.value : true;\n  }\n\n  @HostListener('change')\n  onChangeInputValue(): void {\n    const value = (this.elemRef.nativeElement as HTMLInputElement)?.value;\n    this.value = value;\n  }\n\n  public initGoogleMapsAutocomplete() {\n    const autocomplete = new google.maps.places.Autocomplete(this.elemRef.nativeElement, this.autoCompleteOptions);\n    autocomplete.addListener('place_changed', () => {\n      this.ngZone.run(() => {\n        // get the place result\n        const place: PlaceResult = autocomplete.getPlace();\n\n        const germanAddress: GermanAddress = {\n          gmID: place.id,\n          icon: place.icon,\n          url: place.url,\n          placeID: place.place_id,\n          displayAddress: place.formatted_address,\n          name: place.name,\n          vicinity: place.vicinity,\n          locality: {},\n          state: {},\n          country: {},\n          geoLocation: {latitude: -1, longitude: -1},\n        };\n\n        if (place.geometry && place.geometry.location) {\n          germanAddress.geoLocation.latitude = place.geometry.location.lat();\n          germanAddress.geoLocation.longitude = place.geometry.location.lng();\n        }\n\n        // console.log(\"place.address_components --> \", place.address_components);\n        place.address_components.forEach(value => {\n          if (value.types.indexOf('street_number') > -1) {\n            germanAddress.streetNumber = value.short_name;\n          }\n          if (value.types.indexOf('route') > -1) {\n            germanAddress.streetName = value.long_name;\n          }\n          if (value.types.indexOf('postal_code') > -1) {\n            germanAddress.postalCode = value.short_name;\n          }\n          if (value.types.indexOf('sublocality') > -1) {\n            germanAddress.sublocality = value.long_name;\n          }\n          if (value.types.indexOf('locality') > -1) {\n            germanAddress.locality.long = value.long_name;\n            germanAddress.locality.short = value.short_name;\n          }\n          if (value.types.indexOf('administrative_area_level_1') > -1) {\n            germanAddress.state.long = value.long_name;\n            germanAddress.state.short = value.short_name;\n          }\n          if (value.types.indexOf('country') > -1) {\n            germanAddress.country.long = value.long_name;\n            germanAddress.country.short = value.short_name;\n          }\n          if (value.types.indexOf('administrative_area_level_3') > -1) {\n            germanAddress.locality.short = value.short_name;\n          }\n        });\n\n        this.onGermanAddressMapped.emit(germanAddress);\n\n        this.value = place.formatted_address;\n        this.address = place.formatted_address;\n        this.onAutocompleteSelected.emit(place);\n        this.onLocationSelected.emit(\n          {\n            latitude: place.geometry.location.lat(),\n            longitude: place.geometry.location.lng()\n          });\n      });\n    });\n  }\n\n  registerOnChange(fn: any): void {\n    this.propagateChange = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n  }\n\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n\n  writeValue(obj: any): void {\n    if (obj) {\n      this.value = obj;\n    }\n  }\n\n  loadMap(): void {\n    this.loaderService.loadScript(`https://maps.googleapis.com/maps/api/js?key=${this.apiKey}&libraries=places`)\n      .then(() => {\n        this.initMap();\n      })\n      .catch(error => console.error('Google Maps loading failed: ', error));\n  }\n\n  initMap() {\n    if (isPlatformBrowser(this.platformId)) {\n\n      console.log(\"on after view init --> \", this.elemRef.nativeElement)\n\n      this.autocomplete = new google.maps.places.Autocomplete(\n        this.elemRef.nativeElement\n      );\n\n      const options: AutocompleteOptions = {\n        // types: ['address'],\n        // componentRestrictions: {country: this.country},\n        placeIdOnly: this.placeIdOnly,\n        strictBounds: this.strictBounds,\n        // types: this.types,\n        type: this.type\n      };\n\n      // tslint:disable-next-line:no-unused-expression\n      this.country ? options.componentRestrictions = {country: this.country} : null;\n      // tslint:disable-next-line:no-unused-expression\n      this.country ? options.types = this.types : null;\n\n      this.autoCompleteOptions = Object.assign(this.autoCompleteOptions, options);\n      this.initGoogleMapsAutocomplete();\n    }\n  }\n\n}\n"]}