Creating the page module

Now that we are ready with the initial setup of Laravel and Twill we can start building our CMS.

In Twill we use Modules. A module is a single "content type" and exists out of a few files:

  • A model class (in app/Models)
    • This is your regular Laravel eloquent model. It tells Laravel how your content type is defined and is used to make the link to the database.
  • A module controller class (in app/Http/Controllers/Twill)
    • The module controls controls how your module interacts with the cms, you can see it as a bridge between your model and the CMS front-end ui.
  • A repository class (in app/Repositories)
    • The repository is used to handle saving, reading and more of your model. This is also the place where we would add custom pre/post save actions and much more!
  • A requests class (in app/Http/Requests/Twill)
    • The request class is in between form requests. It is mainly used to provide validation!
  • (Optionally) a form blade file
    • In Twill 3 there are 2 ways to build the form that you will see in the CMS, we will go more in depth about these later in the guide.

So, we now know a bit about what files there are (specific to Twill) and what their purpose is. The next step would be to create them.

Creating these files manually would be, painful. Their naming conventions would need to match, the base content would need to be set, that is a lot of work!

Luckily, Twill comes with a few useful artisan commands. (Artisan is the command line toolkit of Laravel. You used this before to install Twill itself.)

While there are many commands that you can run, we will focus on just one at the moment.

Exploring php artisan twill:make:module

The php artisan twill:make:module is the main way of creating a module to be added in your CMS.

If you want to see what options this command takes, you can add --help like this:

php artisan twill:make:module --help

This will show you all the available options:

1$ php artisan twill:make:module --help
2 
3Description:
4 Create a new Twill Module
5 
6Usage:
7 twill:make:module [options] [--] <moduleName>
8 
9Arguments:
10 moduleName
11 
12Options:
13 -B, --hasBlocks
14 -T, --hasTranslation
15 -S, --hasSlug
16 -M, --hasMedias
17 -F, --hasFiles
18 -P, --hasPosition
19 -R, --hasRevisions
20 -N, --hasNesting
21 --bladeForm
22 -E, --generatePreview
23 --parentModel[=PARENTMODEL]
24 --all

As you can see, we can pass a few options, but there is only one thing mandatory, and that is the name of our module.

If Twill needs to know more about your intentions, the command will ask you.

The module name should always be in a plural form, this is important for the code generator. After all, we are talking to a computer, it does not always correctly guess if a word is singular or plural.

Creating the pages module

Now that we know what a module is and how we can generate one. We can go ahead and actually do it!

php artisan twill:make:module pages

You will be prompted to answer some questions and while we could have used the options to configure these, it is good to see what the options are with a bit more explanation.

For this module we will answer:

  • Do you need to use the block editor on this module: yes
  • Do you need to translate content on this module: yes
  • Do you need to generate slugs on this module: yes
  • Do you need to attach images on this module: yes
  • Do you need to attach files on this module: no
  • Do you need to manage the position of records on this module: no
  • Do you need to enable revisions on this module: yes
  • Do you need to enable nesting on this module: no
  • Do you also want to generate the preview file: yes

Great! That was quite a bit, but our module should now be as we want it.

As a quick note, you may not need translatable content, but usually it is good to prepare for the future when it comes to translations. When we build the front-end, as long as you do not enable more that one language it is basically a non translated website.

Once the command is complete you will be prompted with some instructions:

1The following snippet has been added to routes/twill.php:
2┌──────────────────────────────────┐
3use A17\Twill\Facades\TwillRoutes│
4TwillRoutes::module('pages');
5└──────────────────────────────────┘
6To add a navigation entry add the following to your AppServiceProvider BOOT method.
7**************************
8use A17\Twill\Facades\TwillNavigation;
9use A17\Twill\View\Components\Navigation\NavigationLink;
10 
11TwillNavigation::addLink(
12 NavigationLink::make()->forModule('pages')
13);
14**************************
15Do not forget to migrate your database after modifying the migrations.

Let's have a look at them:

Module routes

We can see that Route::module('pages'); has been added to routes/twill.php.

This is automatic, because it is simple enough to do.

The routes/twill.php file is a Twill specific list of routes. These routes are protected and loaded specifically for the CMS.

In standard Laravel there is no module method on a Route object, this is something Twill provides and it will build many routes for your module, these are then used by the cms, controllers and requests to make everything work together.

Navigation link

The second step, we have to do ourself. So let's open app/Providers/AppServiceProvider.php.

We will not go into detail about what a service provider is, for that you can check the official documentation.

In our boot method we will add the suggested snippet:

1<?php
2 
3namespace App\Providers;
4 
5use Illuminate\Support\ServiceProvider;
6use A17\Twill\Facades\TwillNavigation;
7use A17\Twill\View\Components\Navigation\NavigationLink;
8 
9class AppServiceProvider extends ServiceProvider
10{
11 /**
12 * Register any application services.
13 *
14 * @return void
15 */
16 public function register()...
17 {
18 //
19 }
20 
21 /**
22 * Bootstrap any application services.
23 *
24 * @return void
25 */
26 public function boot()
27 {
28 TwillNavigation::addLink(
29 NavigationLink::make()->forModule('pages')
30 );
31 }
32}

TwillNavigation is a Facade that Twill provides to manage the administration menu.

NavigationLink is a blade component, but with some additional methods. This is used to render the actual link.

In the future, you might define more links. For this, order is important as it is this order that is also used for displaying.

Migration

We ran database migrations before, but every time you add a new migration to database/migrations you need to run php artisan migrate.

But before that, have a look at database/migrations/..._create_pages_table.php:

1<?php
2 
3use Illuminate\Database\Migrations\Migration;
4use Illuminate\Database\Schema\Blueprint;
5use Illuminate\Support\Facades\Schema;
6 
7return new class extends Migration
8{
9 public function up()
10 {
11 Schema::create('pages', function (Blueprint $table) {
12 // this will create an id, a "published" column, and soft delete and timestamps columns
13 createDefaultTableFields($table);
14 
15 // add those 2 columns to enable publication timeframe fields
16 // (you can use publish_start_date only if you don't need to provide the ability to specify an end date)
17 // $table->timestamp('publish_start_date')->nullable();
18 // $table->timestamp('publish_end_date')->nullable();
19 });
20 
21 Schema::create('page_translations', function (Blueprint $table) {
22 createDefaultTranslationsTableFields($table, 'page');
23 $table->string('title', 200)->nullable();
24 $table->text('description')->nullable();
25 });
26 
27 Schema::create('page_slugs', function (Blueprint $table) {
28 createDefaultSlugsTableFields($table, 'page');
29 });
30 
31 Schema::create('page_revisions', function (Blueprint $table) {
32 createDefaultRevisionsTableFields($table, 'page');
33 });
34 }
35 
36 public function down()
37 {
38 Schema::dropIfExists('page_revisions');
39 Schema::dropIfExists('page_translations');
40 Schema::dropIfExists('page_slugs');
41 Schema::dropIfExists('pages');
42 }
43};

This file will create the minimum required tables and columns that Twill uses to provide the CMS functionality. Later in the guide we may add some more fields to the database, but will will do that in a new migration.

Once you are more experienced with Twill, you may want to add fields at this moment, before you run the migrate command. That way, you do not have to immediately add a new migration file.

Alright, now lets run:

php artisan migrate

Creating our first page

Great, we looked at all the files and did our migration. Let's create some content now!

Head back over to the browser and visit your CMS. If you followed the steps correctly, the navigation bar should now show our Pages link!

Twill navigation link

Open it up, and you will be presented with an empty page list, almost there!

On the top right press the button "Add new", enter a title and press "Create". You now have your first page!

Awesome, in the next steps we will improve a bit the default controller so that our page fit's our needs.