search ESC

Searching…

No results for "".

Type at least 2 characters to search.

Docs
You are viewing an older version (1.0.0-alpha.10). Go to the latest.

Directory Structure

Introduction

The default Magic application structure is intended to provide a great starting point for both large and small applications. But you are free to organize your application however you like. Magic imposes almost no restrictions on where any given class is located—as long as the classes can be imported.

The Root Directory

A fresh Magic project contains the following directories:

my_app/
├── lib/
│   ├── app/                 # Application logic
│   ├── config/              # Configuration files
│   ├── database/            # Migrations, seeders, factories
│   ├── resources/           # Views and assets
│   ├── routes/              # Route definitions
│   └── main.dart            # Application entry point
├── assets/
│   └── lang/                # Localization JSON files
├── .env                     # Environment variables
└── pubspec.yaml             # Dependencies

The App Directory

The app directory contains the core code of your application. Almost all of the classes in your application will be in this directory.

lib/app/
├── controllers/             # Request handlers
├── middleware/              # Route middleware
├── models/                  # Eloquent models
├── policies/                # Authorization policies
├── providers/               # Service providers
└── kernel.dart              # Application kernel

The Controllers Directory

The controllers directory contains all of your application's controller classes. Controllers are responsible for handling incoming requests and returning responses.

// lib/app/controllers/user_controller.dart
class UserController extends MagicController with MagicStateMixin> {
  static UserController get instance => Magic.findOrPut(UserController.new);
  
  Widget index() => UserListView();
  Widget show(String id) => UserShowView(id: id);
}

The Middleware Directory

The middleware directory contains your application's route middleware. Middleware provide a convenient mechanism for filtering requests entering your application.

// lib/app/middleware/auth_middleware.dart
class AuthMiddleware extends Middleware {
  @override
  Future handle() async {
    if (!Auth.check()) {
      MagicRoute.to('/login');
      return false;
    }
    return true;
  }
}

The Models Directory

The models directory contains all of your Eloquent model classes. Each database table has a corresponding "Model" which is used to interact with that table.

// lib/app/models/user.dart
class User extends Model with HasTimestamps, InteractsWithPersistence {
  @override String get table => 'users';
  @override String get resource => 'users';
  
  String? get name => getAttribute('name') as String?;
  String? get email => getAttribute('email') as String?;
}

The Policies Directory

The policies directory contains the authorization policy classes for your application. Policies are used to determine if a user can perform a given action against a resource.

The Providers Directory

The providers directory contains all of the service providers for your application. Service providers bootstrap your application by binding services in the service container and registering events.

// lib/app/providers/route_service_provider.dart
class RouteServiceProvider extends ServiceProvider {
  @override
  void boot() {
    registerRoutes();
  }
}

The Config Directory

The config directory contains all of your application's configuration files. Each file returns a Map that is merged into Magic's configuration.

lib/config/
├── app.dart                 # App name, env, providers
├── auth.dart                # Guards, session config
├── database.dart            # Database connections
├── network.dart             # API endpoints, timeouts
├── cache.dart               # Cache drivers
├── logging.dart             # Log channels
└── localization.dart        # Supported locales

The Resources Directory

The resources directory contains your views and other UI resources.

lib/resources/
└── views/
    ├── auth/                # Authentication views
    │   ├── login_view.dart
    │   └── register_view.dart
    ├── layouts/             # Layout components
    │   ├── app_layout.dart
    │   └── guest_layout.dart
    ├── components/          # Reusable UI components
    │   ├── app_sidebar.dart
    │   └── app_header.dart
    └── dashboard_view.dart

Views in Magic are Dart classes that extend MagicView or MagicStatefulView:

// lib/resources/views/dashboard_view.dart
class DashboardView extends MagicView {
  @override
  Widget build(BuildContext context) {
    return WDiv(
      className: 'p-4 flex flex-col gap-4',
      children: [
        WText('Dashboard', className: 'text-2xl font-bold'),
      ],
    );
  }
}

The Routes Directory

The routes directory contains all of the route definitions for your application.

lib/routes/
├── web.dart                 # Main application routes
└── api.dart                 # API routes (optional)

Routes are defined using the MagicRoute facade:

// lib/routes/web.dart
void registerRoutes() {
  MagicRoute.group(
    prefix: '/',
    middleware: ['auth'],
    layout: (child) => AppLayout(child: child),
    routes: () {
      MagicRoute.page('/', () => DashboardView());
      MagicRoute.page('/users', () => UserController.instance.index());
      MagicRoute.page('/users/:id', (id) => UserController.instance.show(id));
    },
  );
}

The Database Directory

The database directory contains your database migrations, model factories, and seeders.

lib/database/
├── migrations/              # Schema migrations
│   └── 2024_01_01_000000_create_users_table.dart
├── seeders/                 # Data seeders
│   └── user_seeder.dart
└── factories/               # Model factories
    └── user_factory.dart

Migrations

Migrations are like version control for your database, allowing you to modify your database schema:

class CreateUsersTable extends Migration {
  @override
  Future up(Schema schema) async {
    await schema.create('users', (table) {
      table.id();
      table.string('name');
      table.string('email').unique();
      table.timestamps();
    });
  }
  
  @override
  Future down(Schema schema) async {
    await schema.drop('users');
  }
}

Seeders

Seeders populate your database with test data:

class UserSeeder extends Seeder {
  @override
  Future run() async {
    await User.factory().count(10).create();
  }
}

Factories

Factories define how to generate fake model instances:

class UserFactory extends Factory {
  @override
  Map definition() => {
    'name': faker.person.name(),
    'email': faker.internet.email(),
  };
}