Ajax form submission in Laravel
In this tutorial I will show you how to create an ajax file upload field (with image preview) when other form fields present also.
Assuming that you have a fresh install of Laravel, with database and .env
file is setup correctly. If you need further guidance on how to install and set up Laravel, you may find it in the first part of my previous tutorial: Laravel 8 login authentication. So let’s get started!
Create Model
Head to the terminal with your project folder set as the base path, and enter the following command to create a model. The -m
at the end means that it will create a migration file too:
php artisan make:model ImageForm -m
Next open our newly created model App/Models/ImageForm.php
file, and replace with this:
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
class ImageForm extends Model
{
use HasFactory;
protected $fillable = ['image', 'name', 'email'];
}
If you want to have more inputs, you need to add the name
attributes in this model file like this.
Create Migration
Go to database/migrations/create_image_forms_table.php
which is the new migration file created with the Model, and edit the function like this:
public function up()
{
Schema::create('image_forms', function (Blueprint $table) {
$table->id();
$table->string('image')->nullable();
$table->string('name');
$table->string('email');
$table->timestamps();
});
}
Next, you need to migrate it to the database (meaning that the rows added in this file will be columns in the new image_forms
table) using this command:
php artisan migrate
Now if you log in to your phpMyAdmin, you will see the empty table with columns, we just created:
Create Controller
Now we need to create a controller, what will handle our form inputs. Type this in the Terminal:
php artisan make:controller ImageFormController
A new file was created App/Http/Controllers/ImageFormController.php
, so add this:
<?php
namespace App\Http\Controllers;
use App\Models\ImageForm;
use Illuminate\Http\Request;
class ImageFormController extends Controller
{
/**
* Display the view file of our form
*/
public function index()
{
return view('imageform');
}
/**
* Upload files and fill form fields
*/
public function create(Request $request)
{
// Get the text fields
$form = ImageForm::updateOrCreate([
'id' => $request->id,
'name' => $request->name,
'email' => $request->email,
]);
// Get the uploaded image
if ($request->file('image')) {
$file = $request->file('image');
$filename = time().'_'.$file->getClientOriginalName();
$extension = $file->getClientOriginalExtension();
$location = storage_path('app/public/images');
$file->move($location, $filename);
$filepath = url('storage/images/'.$filename);
$form->image = $filename;
$form->save();
}
return $form;
}
}
Create Routes
Now we need to create the routes to get the newly created controller functions work as expected. Open routes/web.php
file, and add the following content:
<?php
use Illuminate\Support\Facades\Route;
use App\Http\Controllers\ImageFormController;
// Image form
Route::get('/imageform', [ImageFormController::class, 'index']);
Route::post('/uploadfile', [ImageFormController::class, 'uploadFile']);
Create Link
If you have a fresh laravel install, you have to create a link between storage and the public folders:
php artisan storage:link
Create View
Let’s create a new file resources/views/imageform.blade.php
which will be the front-end:
<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="csrf-token" content="">
<title>Image Form Tutorial</title>
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/css/bootstrap.min.css" rel="stylesheet">
</head>
<body class="bg-dark">
<div class="container">
<div class="row d-flex justify-content-center mt-5">
<div class="col-md-6">
<div class="card p-5">
<!-- Image & Preview -->
<div class="form-group mb-3">
<div class="text-center">
<img class="image-preview rounded-circle" src="" width="150" height="150">
<input class="image-input form-control d-none" type="file" id="image" name="image">
</div>
</div>
<!-- Other fields -->
<div class="form-group mb-3">
<label class="form-label" for="name">Name</label>
<input class="form-control form-control-lg fw-bold" type="text" name="name" id="name">
</div>
<div class="form-group mb-3">
<label class="form-label" for="email">Email</label>
<input class="form-control form-control-lg fw-bold" type="email" name="email" id="email">
</div>
</div>
</div>
</div>
</div>
</body>
</html>
Add the following AJAX script to the bottom of the document, just above the closing </body>
tag:
<script src="https://code.jquery.com/jquery-3.4.1.min.js"></script>
<script>
$(document).ready(function(){
/**
* Ajax csrf initialize
*/
$.ajaxSetup({
headers: {
'X-CSRF-TOKEN': $('meta[name="csrf-token"]').attr('content')
}
});
/**
* Create new item function
*/
$('#submit').click(function() {
// define variables
var file = $('#image')[0].files;
var name = $('#name').val();
var email = $('#email').val();
// formdata object
var fd = new FormData();
// append data
fd.append('image', file[0]);
fd.append('name', name);
fd.append('email', email);
// ajax request
$.ajax({
url: "/uploadfile",
method: 'post',
data: fd,
contentType: false,
processData: false,
dataType: 'json',
success: function(res){
alert('Item saved successfully!')
},
error: function(res){
console.log("error : " + JSON.stringify(res) );
}
});
});
/**
* Image preview function
*/
$(function(){
// Load the preview image when browsed
$('.image-input').change(function(){
let reader = new FileReader();
reader.onload = (e) => {
$('.image-preview').attr('src', e.target.result);
}
reader.readAsDataURL(this.files[0]);
});
// Opens the upload browser on click
$('.image-preview').click(function(){
$('.image-input').trigger('click');
});
});
});
</script>
Run Server
Now if you go to run the following command and go to /imageform
you will see what we created:
php artisan serve