divineomega / laravel-addresses
Laravel Addresses
Fund package maintenance!
DivineOmega
Installs: 7 044
Dependents: 0
Suggesters: 0
Security: 0
Stars: 9
Watchers: 5
Forks: 4
Open Issues: 2
Requires
- php: >=7.1
- divineomega/php-countries: ^2.1
- divineomega/php-distance: ^1.0
- divineomega/php-postcodes: ^4.5
- langleyfoxall/simple-google-maps: ^1.0
- laravel/framework: ^5.6||^6.0||^7.0||^8.0||^9.0
Requires (Dev)
- phpunit/phpunit: ^8.5
README
Laravel Addresses is a package that lets you associate addresses with your Laravel Eloquent models.
Features:
- Automatic geocoding of addresses on change, provided by the Google Maps API
- Validation of address details (country, postcode)
- Conversion of ISO country code to country name
- Ability to store meta data about addresses - e.g.
['type' => 'delivery', 'name' => 'home_address']
Installation
To install Laravel Addresses, just run the following Composer command.
composer require divineomega/laravel-addresses
Configuration
Run the following Artisan command to publish the configuration file.
php artisan vendor:publish --provider="DivineOmega\LaravelAddresses\ServiceProvider" --force
This will create the default configuration file at config/addresses.php
.
Note that by default, you require a Google Maps API key in order to provide address geocoding and distance calculations. If you do not wish to use geocoding, this can be disabled in the configuration.
Strict geocoding
By default, geocoding is configured as "lenient"; if, for example, the name of a real city is given but the postcode and street address refer to a nonexistent place, it will geocode as the center of that city.
Set the geocoding.strict
flag to true
in the configuration file to instead fail to geocode in
this scenario.
Usage
Assign the HasAddresses
trait to the model you wish to have associated addresses.
For example, you could give the default User
model address, as shown below.
<?php namespace App; use DivineOmega\LaravelAddresses\Traits\HasAddresses; use Illuminate\Notifications\Notifiable; use Illuminate\Foundation\Auth\User as Authenticatable; class User extends Authenticatable { use Notifiable, HasAddresses; /* ... */ }
Retrieve addresses
$addresses = $user->addresses()->get();
Create new address
$user->addresses()->create([ 'line_1' => '10 Downing Street', // 'line_2' => '', 'town_city' => 'Westminster', 'state_county' => 'London', 'postcode' => 'SW1A 2AA', 'country_code' => 'GBR', ]);
Geocoding
Geocoding is automatic when an address is created or updated. You can check if
an address was successfully geocoding using the isGeocoded
method.
$address = $user->addresses()->first(); if ($address->isGeocoded()) { dd([$address->latitude, $address->longitude]); }
You can also manually geocode the address if needed.
$address = $user->addresses()->first(); $address->geocode(); $address->save();
Note that geocoding can fail, in which case, you can detect that it failed by checking whether the address is geocoded after attempting geocoding:
$address->geocode(); if (!$address->isGeocoded()) { // Handle geocoding failure here. }
If there was an existing latitude/longitude set and geocoding fails, these are cleared.
$address->geocode(); // Succeeds // Change the address details here. $address->geocode(); // Fails // Latitude and longitude are now null.
Validation
Validation is automatic when an address is created or updated. You can expect an appropriate exception to be thrown if validation fails.
InvalidCountryException
- Providedcountry_code
is not a valid ISO 3166-1 alpha-3 country code.InvalidUKPostcodeException
- If the address is within the UK, the providedpostcode
is not a valid UK postcode.
You can also manually validate the address if needed.
$address = $user->addresses()->first(); try { $address->validate(); } catch (\DivineOmega\LaravelAddresses\Exceptions\InvalidUKPostcodeException $e) { return back()->withErrors(['Invalid UK postcode.']); }
Distance calculation
The distance between two different addresses can be calculated using the distanceTo
method.
$address1 = $user->addresses[0]; $address2 = $user->addresses[1]; $distanceKilometres = $address1->distanceTo($address2);
By default the direct distance is calculated (as the crow flies). If you want, you can specify a different type of distance calculation, such as driving distance.
use \DivineOmega\LaravelAddresses\DistanceStrategies\Driving; use \DivineOmega\LaravelAddresses\DistanceStrategies\Walking; use \DivineOmega\LaravelAddresses\DistanceStrategies\Cycling; $drivingDistanceKm = $address1->distanceTo($address2, Driving::class); $walkingDistanceKm = $address1->distanceTo($address2, Walking::class); $cyclingDistanceKm = $address1->distanceTo($address2, Cycling::class);