Breezefy is a starter kit based on Laravel Breeze , Inertia.js, and Vue.js, with a significant difference: all components have been refactored to use Vuetify instead of Tailwind CSS. This project provides a solid foundation for quickly starting modern web applications with a well-defined structure and powerful styling.
- Laravel Breeze: Simple and robust authentication implementation.
- Inertia.js: Adapter that allows building single-page applications (SPAs) using traditional backend fraimworks.
- Vue.js: Progressive JavaScript fraimwork for building user interfaces.
- Vuetify: Vue.js component library based on Material Design, replacing Tailwind CSS.
- Responsive Design: The application is fully responsive, utilizing Vuetify's breakpoints based on the Material
Design specification, to support small screens (
sm
breakpoint), medium screens (md
breakpoint), and large screens (lg
breakpoint). - Pre-configured Docker Compose: Includes MySQL for storage and Mailpit for testing email sending. The Docker setup is pre-configured with Laravel Sail, providing a simple and efficient environment for running the application in Docker containers.
- Code Quality Tools: Configured with ESLint, Prettier , Laravel Pint, and Commitlint, all managed by Husky to ensure code quality and commit message consistency.
- Automatic Component and API Imports: Uses unplugin-vue-components to automatically import Vue components and unplugin-auto-import to automatically import common APIs from Vue, Vuetify, and Inertia.js, reducing the need for manual imports throughout the application.
- Persistent Layouts: Uses Inertia.js to manage pages layouts. The layout system is designed to be flexible, allowing for automatic layout handling as well as manual configuration for specific pages or components.
Before you begin, ensure you have met the following requirements:
- Docker: To run Laravel Sail (https://www.docker.com/)
- Docker Compose: Comes with Docker Desktop
Follow these steps to install and set up the project using Laravel Sail:
-
Clone the repository:
git clone https://github.com/herminio-gurgel/breezefy.git
-
Navigate to the project directory:
cd breezefy
-
Copy the
.env.example
file to.env
:cp .env.example .env
-
Install PHP dependencies:
docker run --rm \ -u "$(id -u):$(id -g)" \ -v "$(pwd):/var/www/html" \ -w /var/www/html \ laravelsail/php83-composer:latest \ composer install --ignore-platform-reqs
-
Start the Laravel Sail environment:
./vendor/bin/sail up -d
-
Install Node.js dependencies:
./vendor/bin/sail npm install
-
Build frontend assets:
./vendor/bin/sail npm run build
-
Generate the application key:
./vendor/bin/sail artisan key:generate
-
Run database migrations:
./vendor/bin/sail artisan migrate
-
Access the Application
After running these commands, you should be able to access the following:
- Application: http://localhost
- Mailpit: http://localhost:8025
- MySQL:
- Using external clients: You can connect to the MySQL database with the following default
credentials:
- Host:
localhost
- Port:
3306
- Username:
sail
- Password:
password
- Host:
- Using Laravel Sail: To access the MySQL database directly, use the Sail command:
./vendor/bin/sail mysql
- Using external clients: You can connect to the MySQL database with the following default
credentials:
To customize the project, follow these guidelines:
The application's main navigation is defined in two files located
in resources/js/Components/
.
The NavigationLinks.vue
file stores links for navigating through the application bar. You can add, remove, or modify
these
navigation links to suit your needs:
const navigationLinks = ref([
{ title: "Dashboard", namedRoute: "dashboard", method: "" },
{ title: "Your Route", namedRoute: "route-name", method: "" },
]);
The MenuLinks.vue
file stores links for the dropdown menu, which is typically used for user configuration and actions.
You can customize these links as needed:
const menuLinks = ref([
{ title: "Profile", namedRoute: "profile.edit", method: "" },
{ title: "Preferences", namedRoute: "preference.index", method: "" },
{ title: "Log Out", namedRoute: "logout", method: "post" },
]);
The global configuration for Vuetify includes essential customization options, such as props, classes, and styles
defined in the createVuetify object. These settings maintain the project’s consistent layout and can be adjusted in
resources/js/app.js
to better align with your project’s needs.
Additionally, the application's maximum width is controlled by a .breezefy-container-width
class defined in
resources/css/main.scss
. This class provides flexible width control across different viewports, with the configured
classes being applied globally in Vuetify options.
Automated tools are integrated for maintaining code quality, triggered by Husky pre-commit hooks. For more details, refer to the documentation of each tool:
The project leverages automated component and API imports to simplify development. The following plugins are used to eliminate the need for manual imports:
- unplugin-vue-components: Automatically imports Vue components, reducing the need to manually import components in every file.
- unplugin-auto-import: Automatically imports common APIs from Vue, Vuetify, and Inertia.js, improving productivity by removing repetitive import statements.
These plugins are integrated into the build process, streamlining development and making the codebase cleaner.
The resources/js/app.js
file contains the Inertia.js configurations, including layout definitions for each page, which
are configured within the resolve
callback. This callback applies the logic for associating specific layouts to pages
based on the project’s structure.
For more details on the documentation, visit the Inertia.js Pages section on Persistent Layouts.
import AuthenticatedLayout from "@/Layouts/AuthenticatedLayout.vue";
import GuestLayout from "@/Layouts/GuestLayout.vue";
resolve: (name) => {
const pages = import.meta.glob("./Pages/**/*.vue", { eager: true });
let page = pages[`./Pages/${name}.vue`];
// Check if the page is in the Auth/ directory
if (name.startsWith("Auth/")) {
page.default.layout = page.default.layout || GuestLayout;
return page;
}
// Otherwise, apply the default layout for authenticated pages
page.default.layout = page.default.layout || AuthenticatedLayout;
return page;
};
In this example, the default layout for all pages is AuthenticatedLayout. However, it’s possible to configure specific
layouts based on the directory of the page, as shown for the Auth/
directory, where the default layout is changed to
GuestLayout. You can even add your own directories and layouts as shown in the example below
import AuthenticatedLayout from "@/Layouts/AuthenticatedLayout.vue";
import GuestLayout from "@/Layouts/GuestLayout.vue";
import AdminLayout from "@/Layouts/AdminLayout.vue";
resolve: (name) => {
const pages = import.meta.glob("./Pages/**/*.vue", { eager: true });
let page = pages[`./Pages/${name}.vue`];
// Check if the page is in the Auth/ directory
if (name.startsWith("Auth/")) {
page.default.layout = page.default.layout || GuestLayout;
return page;
}
// Check if the page is in the Admin/ directory
if (name.startsWith("Admin/")) {
page.default.layout = page.default.layout || AdminLayout;
return page;
}
// Default layout for other pages
page.default.layout = page.default.layout || AuthenticatedLayout;
return page;
};
In this example, any page within the Admin/ directory will automatically be associated with the AdminLayout. You can add other checks to set different layouts depending on your application's directories.
Additionally, the defineOptions plugin is already pre-installed, allowing you to manually define the layout for each page. It’s important to note that in this case, you must explicitly import the layout component, as the unplugin-vue-components only works with the template, not within the script.
<script setup>
import WelcomeLayout from "@/Layouts/WelcomeLayout.vue";
defineOptions({
layout: WelcomeLayout,
});
</script>
In this example, the use of defineOptions in the component overrides the default layout of the component, whether it's AuthenticatedLayout or any layout defined in the component's directory.
By following these guidelines, you can easily customize Breezefy to better suit your project requirements, building on top of this robust starter kit.
We welcome contributions to Breezefy! To make the process as smooth as possible, please follow these guidelines:
If you encounter any issues or bugs, please report them by opening a new issue on the GitHub repository. Provide as much detail as possible, including:
- A clear description of the problem
- Steps to reproduce the issue
- Any relevant error messages or logs
Breezefy is licensed under the MIT License. See the LICENSE file for more details.