places-autocomplete-svelte 1.0.0 → 2.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +181 -1
- package/dist/PlaceAutocomplete.svelte +231 -150
- package/dist/PlaceAutocomplete.svelte.d.ts +38 -34
- package/package.json +26 -19
package/README.md
CHANGED
|
@@ -1 +1,181 @@
|
|
|
1
|
-
#
|
|
1
|
+
# Places Autocomplete Svelte
|
|
2
|
+
|
|
3
|
+
This Svelte component leverages the [Google Maps Places Autocomplete API](https://developers.google.com/maps/documentation/javascript/place-autocomplete-overview) to provide a user-friendly way to search for and retrieve detailed address information within your [SvelteKit](https://kit.svelte.dev) applications.
|
|
4
|
+
|
|
5
|
+
## Features:
|
|
6
|
+
|
|
7
|
+
- **Seamless Integration:** Easily integrate the component into your SvelteKit projects.
|
|
8
|
+
- **Autocomplete Suggestions:** Provide users with real-time suggestions as they type, enhancing the search experience.
|
|
9
|
+
- **Detailed Address Retrieval:** Retrieve comprehensive address information, including street address, city, region, postal code, and country.
|
|
10
|
+
- **Formatted and Unformatted Responses:** Access both formatted address strings and raw address component data for flexible use cases.
|
|
11
|
+
- **Country/Region Selection:** Allow users to specify a region for more targeted results.
|
|
12
|
+
- **Customizable:** Tailor the component's appearance and behavior using language settings and placeholder text.
|
|
13
|
+
|
|
14
|
+

|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
## Requirements
|
|
19
|
+
|
|
20
|
+
- **Google Maps API Key:** Create an API key with the Places API (New) enabled. Refer to [Use API Keys](https://developers.google.com/maps/documentation/javascript/get-api-key) for detailed instructions.
|
|
21
|
+
|
|
22
|
+
## Installation Svelte 5
|
|
23
|
+
|
|
24
|
+
```bash
|
|
25
|
+
npm i places-autocomplete-svelte
|
|
26
|
+
```
|
|
27
|
+
|
|
28
|
+
## Installation Svelte 4
|
|
29
|
+
|
|
30
|
+
```bash
|
|
31
|
+
npm i places-autocomplete-svelte@1.0.1
|
|
32
|
+
```
|
|
33
|
+
|
|
34
|
+
## Import Component
|
|
35
|
+
|
|
36
|
+
```javascript
|
|
37
|
+
import PlaceAutocomplete from 'places-autocomplete-svelte';
|
|
38
|
+
```
|
|
39
|
+
|
|
40
|
+
## Basic Usage
|
|
41
|
+
|
|
42
|
+
1. **Provide your Google Maps API Key:** Replace `'--YOUR_API_KEY--'` with your actual Google Maps API key.
|
|
43
|
+
2. **Bind to Address Variables:** Use `bind:formattedAddress` to capture the selected address as string.
|
|
44
|
+
|
|
45
|
+
```svelte
|
|
46
|
+
<script>
|
|
47
|
+
import { PlaceAutocomplete } from 'places-autocomplete-svelte';
|
|
48
|
+
|
|
49
|
+
let formattedAddress = '';
|
|
50
|
+
let PUBLIC_GOOGLE_MAPS_API_KEY = '--YOUR_API_KEY--';
|
|
51
|
+
</script>
|
|
52
|
+
|
|
53
|
+
<PlaceAutocomplete
|
|
54
|
+
bind:PUBLIC_GOOGLE_MAPS_API_KEY
|
|
55
|
+
bind:formattedAddress />
|
|
56
|
+
|
|
57
|
+
<p>Formatted Address: {formattedAddress}</p>
|
|
58
|
+
```
|
|
59
|
+
|
|
60
|
+
|
|
61
|
+
## Specifying Countries/Regions
|
|
62
|
+
|
|
63
|
+
Use the `countries` property to refine search by region:
|
|
64
|
+
|
|
65
|
+
```svelte
|
|
66
|
+
<script>
|
|
67
|
+
// ... other imports
|
|
68
|
+
|
|
69
|
+
let countries = [
|
|
70
|
+
{ name: 'United Kingdom', region: 'GB' , language:'en-GB'},
|
|
71
|
+
{ name: 'United States', region: 'US',language:'en-US' },
|
|
72
|
+
// ... more countries
|
|
73
|
+
];
|
|
74
|
+
</script>
|
|
75
|
+
|
|
76
|
+
<PlaceAutocomplete
|
|
77
|
+
bind:PUBLIC_GOOGLE_MAPS_API_KEY
|
|
78
|
+
bind:formattedAddress
|
|
79
|
+
bind:countries />
|
|
80
|
+
```
|
|
81
|
+
|
|
82
|
+
- The `region` code follows the [CLDR two-character format](https://developers.google.com/maps/documentation/javascript/reference/autocomplete-data#AutocompleteRequest).
|
|
83
|
+
- The `language` in which to return results. [See details](https://developers.google.com/maps/documentation/javascript/reference/autocomplete-data#AutocompleteRequest.language)
|
|
84
|
+
|
|
85
|
+
## Accessing Full Response Data
|
|
86
|
+
|
|
87
|
+
For maximum flexibility, access the complete unformatted response from the Google Maps API:
|
|
88
|
+
|
|
89
|
+
```svelte
|
|
90
|
+
<script>
|
|
91
|
+
// ... other imports
|
|
92
|
+
let fullResponse = {};
|
|
93
|
+
</script>
|
|
94
|
+
|
|
95
|
+
<PlaceAutocomplete bind:PUBLIC_GOOGLE_MAPS_API_KEY bind:fullResponse />
|
|
96
|
+
|
|
97
|
+
<pre>{JSON.stringify(fullResponse, null, 2)}</pre>
|
|
98
|
+
```
|
|
99
|
+
|
|
100
|
+
## Example
|
|
101
|
+
```svelte
|
|
102
|
+
<script>
|
|
103
|
+
import { PlaceAutocomplete } from 'places-autocomplete-svelte';
|
|
104
|
+
|
|
105
|
+
let PUBLIC_GOOGLE_MAPS_API_KEY = '--YOUR_API_KEY--';
|
|
106
|
+
let formattedAddress = '';
|
|
107
|
+
let fullResponse = {};
|
|
108
|
+
let formattedAddressObj = {};
|
|
109
|
+
let countries = [
|
|
110
|
+
{ name: 'United Kingdom', region: 'GB' , language:'en-GB'},
|
|
111
|
+
{ name: 'United States', region: 'US',language:'en-US' },
|
|
112
|
+
// ... more countries
|
|
113
|
+
];
|
|
114
|
+
</script>
|
|
115
|
+
|
|
116
|
+
<PlaceAutocomplete
|
|
117
|
+
bind:PUBLIC_GOOGLE_MAPS_API_KEY
|
|
118
|
+
bind:formattedAddress
|
|
119
|
+
bind:fullResponse
|
|
120
|
+
bind:formattedAddressObj
|
|
121
|
+
bind:countries
|
|
122
|
+
placeholder="Enter your address...">
|
|
123
|
+
|
|
124
|
+
```
|
|
125
|
+
- The `formattedAddress` - selected address as string.
|
|
126
|
+
- The `fullResponse` - the complete unformatted response from the Google Maps API.
|
|
127
|
+
- The `formattedAddressObj` - parsed address components, containing individual elements like street number, town, and postcode.
|
|
128
|
+
|
|
129
|
+
The `formattedAddressObj` mapping corresponds to the following component types:
|
|
130
|
+
|
|
131
|
+
- `formattedAddressObj.street_number`: longText property of the street_number
|
|
132
|
+
- `formattedAddressObj.street`: longText property of the route
|
|
133
|
+
- `formattedAddressObj.town`: longText property of the postal_town
|
|
134
|
+
- `formattedAddressObj.county`: longText property of the administrative_area_level_2
|
|
135
|
+
- `formattedAddressObj.country_iso2`: shortText property of the country
|
|
136
|
+
- `formattedAddressObj.postcode`: longText property of the postal_code
|
|
137
|
+
|
|
138
|
+
|
|
139
|
+
|
|
140
|
+
## Error Handling
|
|
141
|
+
|
|
142
|
+
The component will throw an error if:
|
|
143
|
+
- The Google Maps API key is invalid or missing.
|
|
144
|
+
- There are network issues connecting to the Google Maps service.
|
|
145
|
+
|
|
146
|
+
Handle these errors gracefully in your application:
|
|
147
|
+
|
|
148
|
+
```svelte
|
|
149
|
+
<script>
|
|
150
|
+
// ... other imports
|
|
151
|
+
|
|
152
|
+
// Error handler
|
|
153
|
+
let pacError = '';
|
|
154
|
+
let onError = (error:string) => {
|
|
155
|
+
console.error(error);
|
|
156
|
+
pacError = error;
|
|
157
|
+
};
|
|
158
|
+
</script>
|
|
159
|
+
|
|
160
|
+
<PlaceAutocomplete bind:PUBLIC_GOOGLE_MAPS_API_KEY {onError} />
|
|
161
|
+
|
|
162
|
+
{#if pacError}
|
|
163
|
+
<p class="error">{pacError}</p>
|
|
164
|
+
{/if}
|
|
165
|
+
```
|
|
166
|
+
|
|
167
|
+
|
|
168
|
+
|
|
169
|
+
|
|
170
|
+
|
|
171
|
+
|
|
172
|
+
|
|
173
|
+
## Contributing
|
|
174
|
+
|
|
175
|
+
Contributions are welcome! Please open an issue or submit a pull request on the [GitHub repository](https://github.com/alexpechkarev/places-autocomplete-svelte/).
|
|
176
|
+
|
|
177
|
+
## License
|
|
178
|
+
|
|
179
|
+
[MIT](LICENSE)
|
|
180
|
+
|
|
181
|
+
|
|
@@ -1,149 +1,233 @@
|
|
|
1
|
-
<script
|
|
2
|
-
import
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
export let
|
|
9
|
-
export let
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
};
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import { onMount } from 'svelte';
|
|
3
|
+
import * as GMaps from '@googlemaps/js-api-loader';
|
|
4
|
+
const { Loader } = GMaps;
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
export let PUBLIC_GOOGLE_MAPS_API_KEY: string = '';
|
|
9
|
+
export let fetchFields: string[] = ['displayName', 'formattedAddress', 'addressComponents'];
|
|
10
|
+
export let countries: { name: string; region: string }[] = [];
|
|
11
|
+
let hasCountries = countries.length > 0;
|
|
12
|
+
|
|
13
|
+
export let formattedAddress: string = '';
|
|
14
|
+
export let fullResponse: { longText: string; shortText: string; types: Array<string> }[] = [];
|
|
15
|
+
export let formattedAddressObj: {
|
|
16
|
+
street_number: string;
|
|
17
|
+
street: string;
|
|
18
|
+
town: string;
|
|
19
|
+
county: string;
|
|
20
|
+
country_iso2: string;
|
|
21
|
+
postcode: string;
|
|
22
|
+
} = {
|
|
23
|
+
street_number: '',
|
|
24
|
+
street: '',
|
|
25
|
+
town: '',
|
|
26
|
+
county: '',
|
|
27
|
+
country_iso2: '',
|
|
28
|
+
postcode: ''
|
|
29
|
+
};
|
|
30
|
+
export let onError: (error: string) => void = (error: string) => {};
|
|
31
|
+
|
|
32
|
+
let inputRef: HTMLInputElement;
|
|
33
|
+
let currentSuggestion = -1;
|
|
34
|
+
let title: string = '';
|
|
35
|
+
let results: any[] = [];
|
|
36
|
+
let token;
|
|
37
|
+
let loader: GMaps.Loader;
|
|
38
|
+
let placesApi: { [key: string]: any } = {};
|
|
39
|
+
//https://developers.google.com/maps/documentation/javascript/reference/autocomplete-data#AutocompleteRequest.includedPrimaryTypes
|
|
40
|
+
let request = {
|
|
41
|
+
input: '',
|
|
42
|
+
language: 'en-GB',
|
|
43
|
+
region: 'GB',
|
|
44
|
+
sessionToken: ''
|
|
45
|
+
};
|
|
46
|
+
|
|
47
|
+
$: {
|
|
48
|
+
if (request.input == '') {
|
|
49
|
+
results = [];
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Reset search input and results.
|
|
55
|
+
*/
|
|
56
|
+
const reset = () => {
|
|
57
|
+
currentSuggestion = -1;
|
|
58
|
+
request.input = '';
|
|
59
|
+
results = [];
|
|
60
|
+
refreshToken(request);
|
|
61
|
+
};
|
|
62
|
+
|
|
63
|
+
/**
|
|
64
|
+
* Make request and get autocomplete suggestions.
|
|
65
|
+
* @param event
|
|
66
|
+
*/
|
|
67
|
+
const makeAcRequest = async (
|
|
68
|
+
event: Event & { currentTarget: HTMLInputElement }
|
|
69
|
+
): Promise<void> => {
|
|
70
|
+
const target = event.currentTarget as HTMLInputElement;
|
|
71
|
+
if (target?.value == '') {
|
|
72
|
+
title = '';
|
|
73
|
+
request.input = '';
|
|
74
|
+
results = [];
|
|
75
|
+
return;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
request.input = target.value;
|
|
79
|
+
try {
|
|
80
|
+
const { suggestions } =
|
|
81
|
+
await placesApi.AutocompleteSuggestion.fetchAutocompleteSuggestions(request);
|
|
82
|
+
results = [];
|
|
83
|
+
// iterate suggestions and add results to an array
|
|
84
|
+
for (const suggestion of suggestions) {
|
|
85
|
+
// add suggexstions to results
|
|
86
|
+
results.push({
|
|
87
|
+
to_pace: suggestion.placePrediction.toPlace(),
|
|
88
|
+
text: suggestion.placePrediction.text.toString()
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
} catch (e) {
|
|
92
|
+
onError((e.name || 'An error occurred') + ' - ' + (e.message || 'see console for details.'));
|
|
93
|
+
}
|
|
94
|
+
};
|
|
95
|
+
/**
|
|
96
|
+
* Event handler for clicking on a suggested place.
|
|
97
|
+
//https://developers-dot-devsite-v2-prod.appspot.com/maps/documentation/javascript/reference/autocomplete-data#AutocompleteSuggestion
|
|
98
|
+
* @param place
|
|
99
|
+
*/
|
|
100
|
+
const onPlaceSelected = async (place: {
|
|
101
|
+
[x: string]: any;
|
|
102
|
+
fetchFields: (arg0: { fields: string[] }) => any;
|
|
103
|
+
addressComponents: any;
|
|
104
|
+
formattedAddress: string;
|
|
105
|
+
}): Promise<void> => {
|
|
106
|
+
try {
|
|
107
|
+
|
|
108
|
+
await place.fetchFields({
|
|
109
|
+
fields: fetchFields
|
|
110
|
+
});
|
|
111
|
+
let placeData = place.toJSON();
|
|
112
|
+
formattedAddressObj = {
|
|
113
|
+
street_number: '',
|
|
114
|
+
street: '',
|
|
115
|
+
town: '',
|
|
116
|
+
county: '',
|
|
117
|
+
country_iso2: '',
|
|
118
|
+
postcode: ''
|
|
119
|
+
};
|
|
120
|
+
|
|
121
|
+
title = 'Selected address: ' + placeData.formattedAddress;
|
|
122
|
+
formattedAddress = placeData.formattedAddress;
|
|
123
|
+
fullResponse = placeData.addressComponents;
|
|
124
|
+
|
|
125
|
+
for (const component of placeData.addressComponents) {
|
|
126
|
+
switch (component.types[0]) {
|
|
127
|
+
case 'street_number':
|
|
128
|
+
case 'point_of_interest':
|
|
129
|
+
case 'establishment':
|
|
130
|
+
formattedAddressObj.street_number = component.longText;
|
|
131
|
+
break;
|
|
132
|
+
case 'route':
|
|
133
|
+
case 'premise':
|
|
134
|
+
formattedAddressObj.street = component.longText;
|
|
135
|
+
break;
|
|
136
|
+
case 'postal_town':
|
|
137
|
+
formattedAddressObj.town = component.longText;
|
|
138
|
+
break;
|
|
139
|
+
case 'administrative_area_level_2':
|
|
140
|
+
formattedAddressObj.county = component.longText;
|
|
141
|
+
break;
|
|
142
|
+
case 'country':
|
|
143
|
+
formattedAddressObj.country_iso2 = component.shortText;
|
|
144
|
+
break;
|
|
145
|
+
case 'postal_code':
|
|
146
|
+
formattedAddressObj.postcode = component.longText;
|
|
147
|
+
break;
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
} catch (e) {
|
|
151
|
+
onError((e.name || 'An error occurred') + ' - ' + (e.message || 'error fetching place details'));
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
// reset srarch input and results
|
|
155
|
+
reset();
|
|
156
|
+
};
|
|
157
|
+
|
|
158
|
+
/**
|
|
159
|
+
* Helper function to refresh the session token.
|
|
160
|
+
* @param request
|
|
161
|
+
*/
|
|
162
|
+
const refreshToken = async (request: {
|
|
163
|
+
input?: string;
|
|
164
|
+
language?: string;
|
|
165
|
+
region?: string;
|
|
166
|
+
sessionToken?: any;
|
|
167
|
+
}): Promise<{ input?: string; language?: string; region?: string; sessionToken?: any }> => {
|
|
168
|
+
try{
|
|
169
|
+
token = new placesApi.AutocompleteSessionToken();
|
|
170
|
+
request.sessionToken = token;
|
|
171
|
+
return request;
|
|
172
|
+
|
|
173
|
+
}catch(e){
|
|
174
|
+
onError((e.name || 'An error occurred') + ' - ' + (e.message || 'error fetch token'));
|
|
175
|
+
return request;
|
|
176
|
+
}
|
|
177
|
+
};
|
|
178
|
+
/**
|
|
179
|
+
* Initialize the Google Maps JavaScript API Loader.
|
|
180
|
+
*/
|
|
181
|
+
onMount(async (): Promise<void> => {
|
|
182
|
+
inputRef.focus();
|
|
183
|
+
|
|
184
|
+
try {
|
|
185
|
+
loader = new Loader({
|
|
186
|
+
apiKey: PUBLIC_GOOGLE_MAPS_API_KEY,
|
|
187
|
+
version: 'weekly',
|
|
188
|
+
libraries: ['places']
|
|
189
|
+
});
|
|
190
|
+
|
|
191
|
+
const { AutocompleteSessionToken, AutocompleteSuggestion } =
|
|
192
|
+
await loader.importLibrary('places');
|
|
193
|
+
placesApi.AutocompleteSessionToken = AutocompleteSessionToken;
|
|
194
|
+
placesApi.AutocompleteSuggestion = AutocompleteSuggestion;
|
|
195
|
+
token = new placesApi.AutocompleteSessionToken();
|
|
196
|
+
request.sessionToken = token;
|
|
197
|
+
} catch (e) {
|
|
198
|
+
onError(
|
|
199
|
+
(e.name || 'An error occurred') + ' - ' + (e.message || 'Error loading Google Maps API')
|
|
200
|
+
);
|
|
201
|
+
}
|
|
202
|
+
});
|
|
203
|
+
/**
|
|
204
|
+
* Handles keyboard events for navigating the suggestions.
|
|
205
|
+
*/
|
|
206
|
+
function onKeyDown(e: KeyboardEvent) {
|
|
207
|
+
if (e.key === 'ArrowDown') {
|
|
208
|
+
currentSuggestion = Math.min(currentSuggestion + 1, results.length - 1);
|
|
209
|
+
} else if (e.key === 'ArrowUp') {
|
|
210
|
+
currentSuggestion = Math.max(currentSuggestion - 1, 0);
|
|
211
|
+
} else if (e.key === 'Enter') {
|
|
212
|
+
e.preventDefault();
|
|
213
|
+
if (currentSuggestion >= 0) {
|
|
214
|
+
onPlaceSelected(results[currentSuggestion].to_pace);
|
|
215
|
+
}
|
|
216
|
+
} else if (e.key === 'Escape') {
|
|
217
|
+
// reset srarch input and results
|
|
218
|
+
reset();
|
|
219
|
+
}
|
|
220
|
+
}
|
|
128
221
|
</script>
|
|
129
222
|
|
|
130
223
|
<svelte:window on:keydown={onKeyDown} />
|
|
131
224
|
|
|
132
225
|
<section class="my-10">
|
|
133
|
-
{#if placesError}
|
|
134
|
-
<div
|
|
135
|
-
class="bg-red-100 border border-red-400 text-red-700 px-4 py-3 rounded relative"
|
|
136
|
-
role="alert"
|
|
137
|
-
>
|
|
138
|
-
<strong class="font-bold">Error!</strong>
|
|
139
|
-
<span class="block">Check your PUBLIC_GOOGLE_MAPS_API_KEY and API restrictions.</span>
|
|
140
|
-
<span class="block sm:inline">{placesError}</span>
|
|
141
|
-
</div>
|
|
142
|
-
{/if}
|
|
143
|
-
|
|
144
226
|
<div class="grid grid-cols-1 lg:grid-cols-6 gap-x-4">
|
|
145
227
|
<div class:lg:col-span-4={hasCountries} class:lg:col-span-6={!hasCountries}>
|
|
146
|
-
<label class="text-sm text-gray-
|
|
228
|
+
<label class="mt-1 text-sm leading-6 text-gray-600" for="search"
|
|
229
|
+
>Start typing your address</label
|
|
230
|
+
>
|
|
147
231
|
<div class="relative z-50 transform rounded-xl mt-4">
|
|
148
232
|
<div class="pointer-events-none absolute inset-y-0 left-0 flex items-center pl-3">
|
|
149
233
|
<svg
|
|
@@ -157,11 +241,12 @@ function onKeyDown(e) {
|
|
|
157
241
|
stroke-linejoin="round"><circle cx="11" cy="11" r="8" /><path d="m21 21-4.3-4.3" /></svg
|
|
158
242
|
>
|
|
159
243
|
</div>
|
|
244
|
+
|
|
160
245
|
<input
|
|
161
246
|
type="text"
|
|
162
247
|
name="search"
|
|
163
248
|
bind:this={inputRef}
|
|
164
|
-
class="border-1 w-full rounded-md border-
|
|
249
|
+
class="border-1 w-full rounded-md border-0 shadow-sm bg-gray-100 px-4 py-2.5 pl-10 pr-20 text-gray-900 ring-1 ring-inset ring-gray-300 focus:ring-2 sm:text-sm"
|
|
165
250
|
placeholder="Search..."
|
|
166
251
|
aria-controls="options"
|
|
167
252
|
bind:value={request.input}
|
|
@@ -171,15 +256,15 @@ function onKeyDown(e) {
|
|
|
171
256
|
{#if results.length > 0}
|
|
172
257
|
<div class="absolute inset-y-0 right-0 flex py-1.5 pr-1.5">
|
|
173
258
|
<kbd
|
|
174
|
-
class="inline-flex items-center rounded border border-gray-
|
|
259
|
+
class="inline-flex items-center rounded border border-gray-300 px-1 font-sans text-xs text-gray-500 w-8 mr-1"
|
|
175
260
|
>Esc</kbd
|
|
176
261
|
>
|
|
177
262
|
<kbd
|
|
178
|
-
class="inline-flex items-center justify-center rounded border border-gray-
|
|
263
|
+
class="inline-flex items-center justify-center rounded border border-gray-300 px-1 font-sans text-xs text-gray-500 w-6"
|
|
179
264
|
>⇑</kbd
|
|
180
265
|
>
|
|
181
266
|
<kbd
|
|
182
|
-
class="inline-flex items-center rounded border border-gray-
|
|
267
|
+
class="inline-flex items-center rounded border border-gray-400 px-1 font-sans text-xs text-gray-500 justify-center w-6"
|
|
183
268
|
>⇓</kbd
|
|
184
269
|
>
|
|
185
270
|
</div>
|
|
@@ -209,13 +294,13 @@ function onKeyDown(e) {
|
|
|
209
294
|
</div>
|
|
210
295
|
|
|
211
296
|
<div class:lg:col-span-2={hasCountries} class:hidden={!hasCountries}>
|
|
212
|
-
<label class="text-sm text-gray-
|
|
297
|
+
<label class="mt-1 text-sm leading-6 text-gray-600" for="search">Address country</label>
|
|
213
298
|
<div class="flex items-center mt-4">
|
|
214
299
|
<label for="country" class="sr-only">Country</label>
|
|
215
300
|
<select
|
|
216
301
|
id="country"
|
|
217
302
|
name="country"
|
|
218
|
-
class="h-10 w-full rounded-md border-
|
|
303
|
+
class="h-10 w-full rounded-md border-0 bg-transparent py-0 pl-2 pr-7 text-gray-500 focus:ring-2 ring-1 ring-inset ring-gray-300 focus:ring-inset focus:ring-indigo-600 sm:text-sm"
|
|
219
304
|
bind:value={request.region}
|
|
220
305
|
>
|
|
221
306
|
{#each countries as country}
|
|
@@ -229,8 +314,4 @@ function onKeyDown(e) {
|
|
|
229
314
|
<div class="text-sm font-medium leading-6 text-gray-500">{title}</div>
|
|
230
315
|
</div>
|
|
231
316
|
</div>
|
|
232
|
-
</section>
|
|
233
|
-
|
|
234
|
-
<style>
|
|
235
|
-
input,select,ul{margin:0;padding:0}.absolute,.sr-only{position:absolute}.block,svg{display:block}*,.border-gray-200,.border-gray-300{--tw-border-opacity:1}.text-gray-400,.text-gray-500,.text-gray-700,.text-gray-900,.text-red-700,.text-white{--tw-text-opacity:1}*,:after,:before{box-sizing:border-box;border:0 solid #e5e7eb;--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:rgb(59 130 246 / 0.5);--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }:after,:before{--tw-content:''}:host,section{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:system-ui,sans-serif,apple color emoji,segoe ui emoji,Segoe UI Symbol,noto color emoji;font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}strong{font-weight:bolder}kbd{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}input,select{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit}input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:transparent;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}ul{list-style:none}.cursor-default,:disabled{cursor:default}svg{vertical-align:middle}[type=text],input:where(:not([type])),select{-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:#fff;border-color:#6b7280;border-width:1px;border-radius:0;padding:.5rem .75rem;font-size:1rem;line-height:1.5rem;--tw-shadow:0 0 #0000}[type=text]:focus,input:where(:not([type])):focus,select:focus{outline:transparent solid 2px;outline-offset:2px;--tw-ring-inset:var(--tw-empty,);--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:#2563eb;--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow);border-color:#2563eb}input::-moz-placeholder{color:#6b7280;opacity:1}input::placeholder{color:#6b7280;opacity:1}select{text-transform:none;background-image:url();background-position:right .5rem center;background-repeat:no-repeat;background-size:1.5em 1.5em;padding-right:2.5rem;-webkit-print-color-adjust:exact;print-color-adjust:exact}:root{--background:0 0% 100%;--foreground:222.2 84% 4.9%;--muted:210 40% 96.1%;--muted-foreground:215.4 16.3% 46.9%;--popover:0 0% 100%;--popover-foreground:222.2 84% 4.9%;--card:0 0% 100%;--card-foreground:222.2 84% 4.9%;--border:214.3 31.8% 91.4%;--input:214.3 31.8% 91.4%;--primary:222.2 47.4% 11.2%;--primary-foreground:210 40% 98%;--secondary:210 40% 96.1%;--secondary-foreground:222.2 47.4% 11.2%;--accent:210 40% 96.1%;--accent-foreground:222.2 47.4% 11.2%;--destructive:0 72.2% 50.6%;--destructive-foreground:210 40% 98%;--ring:222.2 84% 4.9%;--radius:0.5rem}*{border-color:hsl(var(--border) / var(--tw-border-opacity))}::backdrop{--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:rgb(59 130 246 / 0.5);--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }.sr-only{width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border-width:0}.pointer-events-none{pointer-events:none}.relative{position:relative}.inset-y-0{top:0;bottom:0}.left-0{left:0}.right-0{right:0}.z-50{z-index:50}.-mb-2{margin-bottom:-.5rem}.mr-1{margin-right:.25rem}.mt-4{margin-top:1rem}.flex{display:flex}.inline-flex{display:inline-flex}.hidden{display:none}.h-10{height:2.5rem}.h-5{height:1.25rem}.max-h-60{max-height:15rem}.w-5{width:1.25rem}.w-6{width:1.5rem}.w-8{width:2rem}.w-full{width:100%}.transform{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.select-none{-webkit-user-select:none;-moz-user-select:none;user-select:none}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr));grid-template-columns:1fr}.items-center{align-items:center}.gap-x-4{-moz-column-gap:1rem;column-gap:1rem;row-gap:1rem}.rounded-md{border-radius:calc(var(--radius) - 2px)}.rounded-xl{border-radius:.75rem}.border{border-width:1px}.border-gray-200{border-color:rgb(229 231 235 / var(--tw-border-opacity))}.border-gray-300{border-color:rgb(209 213 219 / var(--tw-border-opacity))}.bg-gray-100{--tw-bg-opacity:1;background-color:rgb(243 244 246 / var(--tw-bg-opacity))}.bg-indigo-500,.hover\:bg-indigo-500:hover{--tw-bg-opacity:1;background-color:rgb(99 102 241 / var(--tw-bg-opacity))}.bg-transparent{background-color:transparent}.bg-white{--tw-bg-opacity:1;background-color:rgb(255 255 255 / var(--tw-bg-opacity))}.px-1{padding-left:.25rem;padding-right:.25rem}.px-4{padding-left:1rem;padding-right:1rem}.py-0{padding-top:0;padding-bottom:0}.py-1{padding-top:.25rem;padding-bottom:.25rem}.py-1\.5{padding-top:.375rem;padding-bottom:.375rem}.py-2{padding-top:.5rem;padding-bottom:.5rem}.py-2\.5{padding-top:.625rem;padding-bottom:.625rem}.py-3{padding-top:.75rem;padding-bottom:.75rem}.pl-10{padding-left:2.5rem}.pl-2{padding-left:.5rem}.pl-3{padding-left:.75rem}.pl-4{padding-left:1rem}.pr-1\.5{padding-right:.375rem}.pr-20{padding-right:5rem}.pr-7{padding-right:1.75rem}.text-base{font-size:1rem;line-height:1.5rem}.text-sm{font-size:.875rem;line-height:1.25rem}.text-xs{font-size:.75rem;line-height:1rem}.text-gray-400{color:rgb(156 163 175 / var(--tw-text-opacity))}.text-gray-500{color:rgb(107 114 128 / var(--tw-text-opacity))}.text-gray-700{color:rgb(55 65 81 / var(--tw-text-opacity))}.text-gray-900{color:rgb(17 24 39 / var(--tw-text-opacity))}.text-red-700{color:rgb(185 28 28 / var(--tw-text-opacity))}.text-white{color:rgb(255 255 255 / var(--tw-text-opacity))}.shadow-lg{--tw-shadow:0 10px 15px -3px rgb(0 0 0 / 0.1),0 4px 6px -4px rgb(0 0 0 / 0.1);--tw-shadow-colored:0 10px 15px -3px var(--tw-shadow-color),0 4px 6px -4px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.focus\:ring-1:focus,.focus\:ring-2:focus,.ring-1{--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow,0 0 #0000)}.focus\:ring-1:focus,.ring-1{--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color)}.ring-black{--tw-ring-opacity:1;--tw-ring-color:rgb(0 0 0 / var(--tw-ring-opacity))}.focus\:outline-none:focus{outline:transparent solid 2px;outline-offset:2px}.focus\:ring-2:focus{--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color)}.focus\:ring-inset:focus{--tw-ring-inset:inset}@media (min-width:640px){.sm\:text-sm{font-size:.875rem;line-height:1.25rem}}@media (min-width:1024px){.lg\:col-span-2{grid-column:span 2/span 2}}@keyframes swipe-out{0%{transform:translateY(calc(var(--lift) * var(--offset) + var(--swipe-amount)));opacity:1}to{transform:translateY(calc(var(--lift) * var(--offset) + var(--swipe-amount) + var(--lift) * -100%));opacity:0}}.grid{display:grid}@media (min-width:768px){.lg\:grid-cols-6{grid-template-columns:repeat(6,1fr)}.lg\:col-span-6{grid-column-start:span 6}.lg\:col-span-4{grid-column-start:span 4}.lg\:col-span-2{grid-column-start:span 2}}.my-10{margin-top:2.5rem;margin-bottom:2.5rem}
|
|
236
|
-
</style>
|
|
317
|
+
</section>
|
|
@@ -1,36 +1,40 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
longText: string;
|
|
12
|
-
shortText: string;
|
|
13
|
-
types: Array<string>;
|
|
14
|
-
}[];
|
|
15
|
-
address?: {
|
|
16
|
-
building_name_number: string;
|
|
17
|
-
street: string;
|
|
18
|
-
town: string;
|
|
19
|
-
county: string;
|
|
20
|
-
country_iso2: string;
|
|
21
|
-
postcode: string;
|
|
22
|
-
};
|
|
1
|
+
interface $$__sveltets_2_IsomorphicComponent<Props extends Record<string, any> = any, Events extends Record<string, any> = any, Slots extends Record<string, any> = any, Exports = {}, Bindings = string> {
|
|
2
|
+
new (options: import('svelte').ComponentConstructorOptions<Props>): import('svelte').SvelteComponent<Props, Events, Slots> & {
|
|
3
|
+
$$bindings?: Bindings;
|
|
4
|
+
} & Exports;
|
|
5
|
+
(internal: unknown, props: Props & {
|
|
6
|
+
$$events?: Events;
|
|
7
|
+
$$slots?: Slots;
|
|
8
|
+
}): Exports & {
|
|
9
|
+
$set?: any;
|
|
10
|
+
$on?: any;
|
|
23
11
|
};
|
|
24
|
-
|
|
25
|
-
[evt: string]: CustomEvent<any>;
|
|
26
|
-
};
|
|
27
|
-
slots: {};
|
|
28
|
-
exports?: {} | undefined;
|
|
29
|
-
bindings?: string | undefined;
|
|
30
|
-
};
|
|
31
|
-
export type PlaceAutocompleteProps = typeof __propDef.props;
|
|
32
|
-
export type PlaceAutocompleteEvents = typeof __propDef.events;
|
|
33
|
-
export type PlaceAutocompleteSlots = typeof __propDef.slots;
|
|
34
|
-
export default class PlaceAutocomplete extends SvelteComponent<PlaceAutocompleteProps, PlaceAutocompleteEvents, PlaceAutocompleteSlots> {
|
|
12
|
+
z_$$bindings?: Bindings;
|
|
35
13
|
}
|
|
36
|
-
|
|
14
|
+
declare const PlaceAutocomplete: $$__sveltets_2_IsomorphicComponent<{
|
|
15
|
+
PUBLIC_GOOGLE_MAPS_API_KEY?: string;
|
|
16
|
+
fetchFields?: string[];
|
|
17
|
+
countries?: {
|
|
18
|
+
name: string;
|
|
19
|
+
region: string;
|
|
20
|
+
}[];
|
|
21
|
+
formattedAddress?: string;
|
|
22
|
+
fullResponse?: {
|
|
23
|
+
longText: string;
|
|
24
|
+
shortText: string;
|
|
25
|
+
types: Array<string>;
|
|
26
|
+
}[];
|
|
27
|
+
formattedAddressObj?: {
|
|
28
|
+
street_number: string;
|
|
29
|
+
street: string;
|
|
30
|
+
town: string;
|
|
31
|
+
county: string;
|
|
32
|
+
country_iso2: string;
|
|
33
|
+
postcode: string;
|
|
34
|
+
};
|
|
35
|
+
onError?: (error: string) => void;
|
|
36
|
+
}, {
|
|
37
|
+
[evt: string]: CustomEvent<any>;
|
|
38
|
+
}, {}, {}, string>;
|
|
39
|
+
type PlaceAutocomplete = InstanceType<typeof PlaceAutocomplete>;
|
|
40
|
+
export default PlaceAutocomplete;
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "places-autocomplete-svelte",
|
|
3
3
|
"license": "MIT",
|
|
4
|
-
"version": "
|
|
4
|
+
"version": "2.0.0",
|
|
5
5
|
"description": "A Svelte component that leverages the power of Google Places New API to provide a user-friendly autocomplete experience for location searches.",
|
|
6
6
|
"keywords": [
|
|
7
7
|
"svelte",
|
|
@@ -55,33 +55,40 @@
|
|
|
55
55
|
"!dist/**/*.test.*",
|
|
56
56
|
"!dist/**/*.spec.*"
|
|
57
57
|
],
|
|
58
|
+
"publishConfig": {
|
|
59
|
+
"@alexpechkarev:registry": "https://npm.pkg.github.com"
|
|
60
|
+
},
|
|
58
61
|
"peerDependencies": {
|
|
59
62
|
"svelte": "^4.0.0"
|
|
60
63
|
},
|
|
61
64
|
"devDependencies": {
|
|
62
|
-
"@sveltejs/adapter-auto": "^3.
|
|
63
|
-
"@sveltejs/
|
|
64
|
-
"@sveltejs/
|
|
65
|
-
"@sveltejs/
|
|
66
|
-
"@
|
|
67
|
-
"eslint": "^9.
|
|
65
|
+
"@sveltejs/adapter-auto": "^3.3.1",
|
|
66
|
+
"@sveltejs/adapter-cloudflare": "^4.7.4",
|
|
67
|
+
"@sveltejs/kit": "^2.7.3",
|
|
68
|
+
"@sveltejs/package": "^2.3.7",
|
|
69
|
+
"@sveltejs/vite-plugin-svelte": "^4.0.0",
|
|
70
|
+
"@types/eslint": "^9.6.1",
|
|
71
|
+
"autoprefixer": "^10.4.20",
|
|
72
|
+
"eslint": "^9.13.0",
|
|
68
73
|
"eslint-config-prettier": "^9.1.0",
|
|
69
|
-
"eslint-plugin-svelte": "^2.
|
|
70
|
-
"globals": "^15.
|
|
74
|
+
"eslint-plugin-svelte": "^2.46.0",
|
|
75
|
+
"globals": "^15.11.0",
|
|
76
|
+
"postcss": "^8.4.47",
|
|
71
77
|
"prettier": "^3.3.3",
|
|
72
|
-
"prettier-plugin-svelte": "^3.2.
|
|
73
|
-
"publint": "^0.2.
|
|
74
|
-
"svelte": "^
|
|
75
|
-
"svelte-check": "^
|
|
76
|
-
"
|
|
77
|
-
"
|
|
78
|
-
"typescript
|
|
79
|
-
"
|
|
78
|
+
"prettier-plugin-svelte": "^3.2.7",
|
|
79
|
+
"publint": "^0.2.12",
|
|
80
|
+
"svelte": "^5.1.3",
|
|
81
|
+
"svelte-check": "^4.0.5",
|
|
82
|
+
"tailwindcss": "^3.4.14",
|
|
83
|
+
"tslib": "^2.8.0",
|
|
84
|
+
"typescript": "^5.6.3",
|
|
85
|
+
"typescript-eslint": "^8.11.0",
|
|
86
|
+
"vite": "^5.4.10"
|
|
80
87
|
},
|
|
81
88
|
"svelte": "./dist/index.js",
|
|
82
89
|
"types": "./dist/PlaceAutocomplete.svelte.d.ts",
|
|
83
90
|
"type": "module",
|
|
84
91
|
"dependencies": {
|
|
85
|
-
"@googlemaps/js-api-loader": "^1.16.
|
|
92
|
+
"@googlemaps/js-api-loader": "^1.16.8"
|
|
86
93
|
}
|
|
87
|
-
}
|
|
94
|
+
}
|