Components
As web development has matured, our applications have become more complex. We have moved from simple HTML pages to dynamic web applications that use JavaScript, CSS, and HTML to create rich user experiences. This complexity has led to the development of a new way to organize our code: components.
Components are self-contained pieces of code that can be reused across our application. They can be as simple as a button or as complex as a form. Components are a way to encapsulate the logic and presentation of a piece of the user interface. They can be used to create reusable elements that can be shared across different parts of your application.
There are two types of components: simple components and controlled components. Simple components are often stateless and are used to render UI elements that do not have any logic requirements. They are often used to help with styling the components using CSS frameworks like TailwindCSS. Controlled components work much like controlled routes, where the component is responsible for managing its own state and rendering the UI based on that state.
Creating Components
All components should be located within the app/Components
directory. This directory is auto-loaded by the framework, so you can reference components by their class name without needing to import them. You can add additional directories to search for components in by adding it to the componentPaths
array in the config/app.php
file.
return [
'componentPaths' => [
'x' => 'app/Components',
'x' => 'app/CustomComponents',
],
];
Each component's tag in the HTML will have a prefix of x-
to differentiate it from standard HTML tags. This is a convention that Monarch uses to make it clear that a tag is a custom component. You can change the prefix by updating the key of the componentPaths
array in the config/app.php
file.
return [
'componentPaths' => [
'my' => 'app/Components',
],
];
Simple Components
Simple components consist of a single PHP file that contains the HTML for the component. The file should be located in the app/Components
directory and have a name that matches the desired tag name. Note that all custom tags must contain a hyphen (-
) to be valid HTML, and Monarch standardizes this by prefixing all custom tags with x-
.
For example, to create a simple component that renders a button, you would create a file named app/Components/button.php
with content similar to this:
<button type="button">
<x-slot></x-slot>
</button>
Once this file is created, you can use the component in your views like this:
<x-button>Click me</x-button>
The slot
tag is a placeholder that will be replaced with the content inside the component tag.
Nested Component Folders
You can create nested folders within the app/Components
directory to organize your components. For example, if you wanted to create a form
component, you could create a folder named form
and place the component file inside it. The name of the component would replace the directory separator with a period (.). So form/fieldset
would be named x-form.fieldset
. The component would then be used like this:
<x-form.fieldset></x-form.fieldset>
Attributes
You can pass attributes to a component by adding them to the component tag. The attributes will be available in the component as an object called $attributes
. If echoed out directly, it will be a string of HTML attributes that can be added to the component tag.
// app/components/button.php
<button type="button" class="btn btn-primary" <?= $attributes ?>>
<x-slot></x-slot>
</button>
You can then use the component like this:
<x-button id="submit" name="submit" value="Submit">Click me</x-button>
Attributes can also be accessed individually by using the $attributes->get()
method.
// app/components/button.php
<button
type="button"
class="btn btn-primary"
id="<?= $attributes->get('id') ?>"
<?= $attributes->except('id', 'class') ?>
>
<x-slot></x-slot>
</button>
You can also restrict the attributes that are output by using the only
method.
// app/components/button.php
<button
type="button"
class="btn btn-primary"
<?= $attributes->only('id', 'name') ?>
>
<x-slot></x-slot>
</button>
Default / Merged Attributes
Sometimes you may want to provide default attributes for a component that can be overridden by the user. You can do this by merging the default attributes with the user-provided attributes.
// app/components/button.php
<button
type="button"
<?= $attributes->merge(['class' => 'btn btn-primary']) ?>
>
<x-slot></x-slot>
</button>
You can then use the component like this:
<x-button class="btn-lg">Click me</x-button>
The resulting HTML will be:
<button type="button" class="btn btn-primary btn-lg">Click me</button>
Dynamic Attributes
Since the component file is just a PHP file, you can use PHP to generate dynamic attributes.
// app/components/button.php
<?php
if ($attributes->has('disabled')) {
$class = 'btn btn-primary btn-disabled';
} else {
$class = 'btn btn-primary';
}
?>
<button
type="button"
<?= $attributes->merge(['class' => $class]) ?>
>
<x-slot></x-slot>
</button>
Slots
As we've seen, slots are a way to pass content into a component. They are placeholders that can be replaced with content when the component is used. Slots are defined using the x-slot
tag in the component file and are replaced with the content inside the component tag.
<!-- app/components/alert.php -->
<div class="alert alert-<?= $type ?>">
<x-slot></x-slot>
</div>
You can then use the component like this:
<x-alert type="success">
<p>Success! Your changes have been saved.</p>
</x-alert>
Default Slot Content
You can provide default content for a slot by using the x-slot
tag with a default value.
<!-- app/components/alert.php -->
<div class="alert alert-<?= $type ?>">
<x-slot>
<p>Default content goes here.</p>
</x-slot>
</div>
You can then use the component like this:
<x-alert type="success"></x-alert>
The default content will be used if no content is provided.
Named Slots
You can define multiple slots in a component by using the name
attribute on the x-slot
tag.
<!-- app/components/card.php -->
<div class="card">
<div class="card-header">
<x-slot name="header"></x-slot>
</div>
<div class="card-body">
<x-slot name="body"></x-slot>
</div>
</div>
You can then use the component like this:
<x-card>
<x-slot name="header">
<h2>Card Title</h2>
</x-slot>
<x-slot name="body">
<p>Card content goes here.</p>
</x-slot>
</x-card>
Controlled Components
Controlled components are used when you need more logic, or access to more of the framework, than a simple component provides. They are similar to controllers in that they are responsible for managing their own state and rendering the UI based on that state.
Creating a Controlled Component
Controlled components are PHP classes that extend the Component
class. They should be located in the app/components
directory and have a name that matches the desired tag name, and have the .control.php
extension. You will typically have a corresponding view file with the same name as the component, though it is not required.
For example, to create a controlled component that renders a form, you would create a file named app/components/form.control.php
with content similar to this:
<?php
use Monarch\Components\Component;
return new class() extends Component
{
public $name;
public function render(): string
{
return $this->view('form', ['name' => $this->name]);
}
}
You are required to implement the render()
method which must return a string. Other than that, you are free to add any additional methods or properties to the class as needed.
The class provides a view()
method that returns a simple component view like was described earlier in this document. It automatically makes the $attributes
object available to the view, and handles parsing any slots. In this case, the form
view file would be located at app/Components/form.php
.
Using a Controlled Component
Controlled Components are called exactly like simple components, but the control file's render
method is called instead of the component file's content being returned.
<x-form name="contact">
<x-slot></x-slot>
</x-form>
Within the component's control file, you can access the attributes and slots using the $this->attributes
instance.
<?php
use Monarch\Components\Component;
return new class() extends Component
{
public function render(): string
{
return $this->view('form', ['name' => $this->attributes->get('name')]);
}
}