J4.x:Developing an MVC Component/Adding a Model to the Site Part
From Joomla! Documentation
< J4.x:Developing an MVC Component
This article is part of the tutorial "Developing an MVC Component for Joomla 4.x". It is intended to be a follow-along programming tutorial, so if you have not read the previous parts of the tutorial you are encouraged to do so.
Now that we have the "Controller" and "View" portions of the Model-View-Controller design pattern in place, in this article we will add a model to our Hello World component. The model will hold some sample data for our template to render. To create a model we extend one of the abstract models provided by Joomla! Several such models exist, and can be found in the Joomla\CMS\MVC\Model namespace:
1 | Joomla\CMS\MVC\Model\AdminModel | Despite its name, this model is not actually specific to the admin panel section of Joomla! It is actually for administering records of a particular type, allowing for batch update/delete operations, etc. |
2 | Joomla\CMS\MVC\Model\BaseModel | The basic model from which other models inherit. If you don't require any additional functionality you can extend this model, but you likely don't want to. |
3 | Joomla\CMS\MVC\Model\FormModel | Used in forms, allows for rows to be checked in or out for editing, and has methods for validation of form inputs. |
4 | Joomla\CMS\MVC\Model\ItemModel | Represents a single database record, allows for basic database operations. |
5 | Joomla\CMS\MVC\Model\ListModel | Used to display paginated lists of records, with optional filtering. |
Later in the tutorial we will back this model with a database query, so we're going to use ItemModel as the base for our new model class. Let's go ahead and create the new files we will need to add our model.
1 | Update: site/language/en-GB/en-GB.com_helloworld.ini | We'll add a new message to pull from the language system, to verify that our model is working |
2 | Create: site/src/Model/MessageModel.php | Our new model, which will hold a message for display in the public web page |
3 | Update: site/src/Controller/DisplayController.php | We'll tell the default display controller to fetch the new model |
4 | Update: site/tmpl/hello/default.php | We'll add an output in the template to show the message from the model |
5 | Update: helloworld.xml | A simple version bump, for our new code |
File Details
site/language/en-GB/en-GB.com_helloworld.ini
We'll (temporarily) put a new message in here that we can use to demo the model.
; Hello World Admin Strings
; Copyright (C) 2020 John Smith. All rights reserved.
COM_HELLOWORLD_MSG_HELLO_WORLD="Hello World!"
COM_HELLOWORLD_MSG_GREETING="This message is coming from the item model!"
site/src/Model/MessageModel.php
The new single-item data model. The getItem() method is the only one required when extending ItemModel. It returns an object that represents the model's item. This "item" could be anything, but is usually a database record. Until we hook this model up to the database in a future part of the tutorial, we will return a basic object with our new message in it, straight from the language system.
<?php
namespace JohnSmith\Component\HelloWorld\Site\Model;
defined('_JEXEC') or die;
use Joomla\CMS\MVC\Model\ItemModel;
use Joomla\CMS\Language\Text;
/**
* @package Joomla.Site
* @subpackage com_helloworld
*
* @copyright Copyright (C) 2020 John Smith. All rights reserved.
* @license GNU General Public License version 3; see LICENSE
*/
/**
* Hello World Message Model
* @since 0.0.5
*/
class MessageModel extends ItemModel {
/**
* Returns a message for display
* @param integer $pk Primary key of the "message item", currently unused
* @return object Message object
*/
public function getItem($pk= null): object {
$item = new \stdClass();
$item->message = Text::_('COM_HELLOWORLD_MSG_GREETING');
return $item;
}
}
site/src/Controller/DisplayController.php
We're going to tell the DisplayController to add our new model to the view as it is being loaded. As we currently have only one view, this will do for now. As our component becomes more complex, our views will gain their own controllers, and can load their own models independently.
<?php
namespace JohnSmith\Component\HelloWorld\Site\Controller;
defined('_JEXEC') or die;
use Joomla\CMS\MVC\Controller\BaseController;
use Joomla\CMS\Factory;
/**
* @package Joomla.Site
* @subpackage com_helloworld
*
* @copyright Copyright (C) 2020 John Smith. All rights reserved.
* @license GNU General Public License version 3; see LICENSE
*/
/**
* HelloWorld Component Controller
* @since 0.0.2
*/
class DisplayController extends BaseController {
public function display($cachable = false, $urlparams = array()) {
$document = Factory::getDocument();
$viewName = $this->input->getCmd('view', 'login');
$viewFormat = $document->getType();
$view = $this->getView($viewName, $viewFormat);
$view->setModel($this->getModel('Message'), true);
$view->document = $document;
$view->display();
}
}
site/tmpl/hello/default.php
The last functional change to make is to add the message output into our site template. In the context of a component template, the $this variable refers to the view object. Remember setting the model on the view in the previous file? We retrieve it here, and display the message.
<?php
use Joomla\CMS\Language\Text;
/**
* @package Joomla.Administrator
* @subpackage com_helloworld
*
* @copyright Copyright (C) 2020 John Smith. All rights reserved.
* @license GNU General Public License version 3; see LICENSE
*/
// No direct access to this file
defined('_JEXEC') or die('Restricted Access');
?>
<h2><?= Text::_('COM_HELLOWORLD_MSG_HELLO_WORLD') ?></h2>
<p><?= $this->getModel()->getItem()->message; ?></p>
helloworld.xml
There's nothing new needed in the manifest file for these changes, so this update isn't technically required. However, for consistency, we will bump the component's version in the manifest file.
<?xml version="1.0" encoding="utf-8"?>
<extension type="component" version="4.0" method="upgrade">
<name>Hello World</name>
<!-- The following elements are optional and free of formatting constraints -->
<creationDate>December 2020</creationDate>
<!-- Dummy author, feel free to replace anywhere you see it-->
<author>John Smith</author>
<authorUrl>https://smith.ca</authorUrl>
<copyright>John Smith</copyright>
<license>GPL v3</license>
<!-- The version string is recorded in the components table -->
<version>0.0.5</version>
<!-- The description is optional and defaults to the name -->
<description>
A hello world component!
</description>
<!-- This is the PHP namespace under which the extension's
code is organised. It should follow this format:
Vendor\Component\ComponentName
"Vendor" can be your company or your own name
The "ComponentName" section MUST match the name used
everywhere else for your component. Whatever the name of
this XML file is, the namespace must match (ignoring CamelCase).
-->
<namespace path="src/">JohnSmith\Component\HelloWorld</namespace>
<files folder="site/">
<folder>language</folder>
<folder>src</folder>
<folder>tmpl</folder>
</files>
<languages>
<language tag="en-GB">site/language/en-GB/en-GB.com_helloworld.ini</language>
</languages>
<administration>
<!-- The link that will appear in the Admin panel's "Components" menu -->
<menu link="index.php?option=com_helloworld">Hello World</menu>
<!-- List of files and folders to copy, and where to copy them -->
<files folder="admin/">
<folder>language</folder>
<folder>services</folder>
<folder>src</folder>
<folder>tmpl</folder>
</files>
<languages>
<language tag="en-GB">admin/language/en-GB/en-GB.com_helloworld.ini</language>
<language tag="en-GB">admin/language/en-GB/en-GB.com_helloworld.sys.ini</language>
</languages>
</administration>
</extension>
Testing the Component
Just as before, zip up your new component version and upload it into your Joomla! install's admin panel. Once you have done so, go to the front-end of your Joomla site. Click the "Hello World" link we created in the menu. You should see the new message in the page template.
Next, we'll learn how to add parameters to menu items, and how to read them within the code. After that, we'll start wiring our component up to the database.