makidizajnerica/laravel-multiemail

Allow users to have more than one email address related to their account.

v1.0.0 2021-11-12 22:41 UTC

This package is auto-updated.

Last update: 2024-10-13 05:44:50 UTC


README

Laravel MultiEmail Logo

Laravel MultiEmail

Allow users to have more than one email address related to their account. Let them set their primary and recovery email addresses.

Installation

composer require makidizajnerica/laravel-multiemail

As for registering Service Provider, it is not necessary, Laravel will auto load provider using Package Discovery.

Config

Inside config/auth.php add new provider like so:

'providers' => [

    // Laravel's default provider
    'users' => [
        'driver' => 'eloquent',
        'model' => App\Models\User::class,
    ],

    'emails' => [
        'driver' => 'eloquent.email',
        'models' => [
            'user' => App\Models\User::class,
            'email' => MakiDizajnerica\MultiEmail\Models\Email::class,
        ],
    ],

],

After that you need to edit existing or create new guard:

'guards' => [
    'web' => [
        'driver' => 'session',
        'provider' => 'emails', // <- edit this line
    ],
],

Then go down under the passwords and edit it like so:

'passwords' => [
    'users' => [
        'provider' => 'emails', // <- edit this line
        'table' => 'password_resets',
        'expire' => 60,
        'throttle' => 60,
    ],
],

The last step would be to change Laravel's default Illuminate\Auth\Passwords\PasswordResetServiceProvider::class inside config/app.php like this:

'providers' => [

    /*
     * Laravel Framework Service Providers...
     */
    Illuminate\Queue\QueueServiceProvider::class,
    Illuminate\Redis\RedisServiceProvider::class,

    Illuminate\Auth\Passwords\PasswordResetServiceProvider::class, // <- remove this line
    MakiDizajnerica\MultiEmail\Providers\PasswordResetServiceProvider::class, // <- add this line

    Illuminate\Session\SessionServiceProvider::class,
    Illuminate\Translation\TranslationServiceProvider::class,
],

This part is important if you want your users to be able to reset their passwords.

Publishing config

If your User model is not in default namespace you are going to need to publish multiemail.php config file using:

php artisan vendor:publish --tag=multiemail-config

And then change the model class inside multiemail.php:

'user_model' => \Custom\Namespace\User::class,

Migrations

First you are going to need to publish migrations using command:

php artisan vendor:publish --tag=multiemail-migrations

Then run them:

php artisan migrate

After running the migrations new table emails will be created.

Please don't forget to remove email field from users table!

Usage

Go inside your App\Models\User::class and add MakiDizajnerica\MultiEmail\HasMultipleEmails::class trait and implement MakiDizajnerica\MultiEmail\Contracts\HasMultipleEmails::class:

namespace App\Models;

use MakiDizajnerica\MultiEmail\HasMultipleEmails;
use Illuminate\Foundation\Auth\User as Authenticatable;
use MakiDizajnerica\MultiEmail\Contracts\HasMultipleEmails as HasMultipleEmailsContract;

class User extends Authenticatable implements HasMultipleEmailsContract
{
    use HasMultipleEmails;
    
    //
}

Then be sure to define emails() relation method:

use MakiDizajnerica\MultiEmail\Models\Email;

public function emails() : HasMany
{
    return $this->hasMany(Email::class);
}

After that your User::class will have some methods available:

/**
 * Add new email address.
 *
 * @param  array $email
 * @param  bool $sendVerification
 * @return \MakiDizajnerica\MultiEmail\Email
 */
public function addNewEmail(array $email, $sendVerification = true);

/**
 * Find email address.
 *
 * @param  mixed $email
 * @param  string $field
 * @return \MakiDizajnerica\MultiEmail\Email|null
 */
public function findMyEmail($email, $field = 'email');

/**
 * Determine if user is the owner of the provided email address.
 *
 * @param  string $email
 * @return bool
 */
public function isMyEmail($email);

/**
 * Remove all user's email addresses.
 *
 * @return void
 */
public function removeAllEmails();

/**
 * Check if the provided email address is verified.
 *
 * @param  string $email
 * @return bool
 */
public function isVerifiedEmail($email);

/**
 * Determine if provided email address is primary.
 *
 * @param  string $email
 * @return bool
 */
public function isPrimaryEmail($email);

/**
 * Change primary email address.
 *
 * @param  string $email
 * @return void
 */
public function setEmailAsPrimary($email);

/**
 * Determine if user has recovery email address.
 *
 * @return bool
 */
public function hasRecoveryEmail();

/**
 * Determine if provided email address is recovery.
 *
 * @param  string $email
 * @return bool
 */
public function isRecoveryEmail($email);

/**
 * Change recovery email address.
 *
 * @param  string $email
 * @return void
 */
public function setEmailAsRecovery($email);

And some custom attributes:

// Get all user's verified emails
$user->verified_emails

// Get primary email
$user->email

// Get recovery email if it exists
$user->recovery_email

Adding new email address

use App\Models\User;

$user = User::first();

$email = $user->addNewEmail([
    'email' => 'test@mail.com'
]);

If user does not have primary email defined you can do something like this:

use App\Models\User;
use Illuminate\Support\Facades\Hash;

$user = User::create([
    'name' => 'Nick',
    'password' => Hash::make('password'),
]);

$email = $user->addNewEmail([
    'email' => 'test@mail.com',
    'type' => 'primary',
]);

Email verification notification will be sent every time new email is added. If you dont want to send notification you can pass second argument to the addNewEmail() method like so:

use App\Models\User;
use Illuminate\Support\Facades\Hash;

$user = User::create([
    'name' => 'Nick',
    'password' => Hash::make('password'),
]);

$email = $user->addNewEmail([
    'email' => 'test@mail.com',
    'type' => 'primary',
    'verified_at' => now(),
], false);

Email types

User may only have one primary and one recovery email address, so it is recommended to use already defined methods for changing types of email addresses:

use App\Models\User;

$user = User::first();

if ($user->isMyEmail('test@mail.com')) {

    // Set as primary
    $user->setEmailAsPrimary('test@mail.com');
    
    // Set as recovery
    $user->setEmailAsRecovery('test@mail.com');

}

Email address cannot be primary and recovery at the same time!

Password resets

Defaut email address for password resets will be user's primary email. But if there is recovery email defined, user will be able to use that email address also. Laravel's default password reset service will still be usable as normal, to learn more about password resets visit https://laravel.com/docs/8.x/passwords.

Inside multiemail.php config file you will be able to enable/disable password resets and to specify if primary email should be used for those resets.

'passwords' => [
    'allow_resets' => true,
    'reset_with_primary_email' => true,
],

Author

Nemanja Marijanovic (n.marijanovic@hotmail.com)

Licence

Copyright © 2021, Nemanja Marijanovic n.marijanovic@hotmail.com

All rights reserved.

For the full copyright and license information, please view the LICENSE file that was distributed within the source root of this package.