prometa / sleek
Bootstrap UI components with aggressive defaults
Requires
- php: ^8.0
- laravel/framework: >=7.0.0
- twbs/bootstrap-icons: ^1.10.5
Requires (Dev)
- orchestra/testbench: ^8.21
- phpunit/phpunit: ^10.1
- dev-master
- v0.0.125
- v0.0.124
- v0.0.123
- v0.0.122
- v0.0.121
- v0.0.120
- v0.0.119
- v0.0.118
- v0.0.117
- v0.0.116
- v0.0.115
- v0.0.114
- v0.0.113
- v0.0.112
- v0.0.111
- v0.0.110
- v0.0.109
- v0.0.108
- v0.0.107
- v0.0.106
- v0.0.105
- v0.0.104
- v0.0.103
- v0.0.102
- v0.0.101
- v0.0.100
- v0.0.99
- v0.0.98
- v0.0.97
- v0.0.96
- v0.0.95
- v0.0.94
- v0.0.93
- v0.0.92
- v0.0.91
- v0.0.90
- v0.0.89
- v0.0.88
- v0.0.87
- v0.0.86
- v0.0.85
- v0.0.84
- v0.0.83
- v0.0.82
- v0.0.81
- v0.0.80
- v0.0.79
- v0.0.78
- v0.0.77
- v0.0.76
- v0.0.75
- v0.0.74
- v0.0.73
- v0.0.72
- v0.0.71
- v0.0.70
- v0.0.69
- v0.0.68
- v0.0.67
- v0.0.66
- v0.0.65
- v0.0.64
- v0.0.63
- v0.0.62
- v0.0.61
- v0.0.60
- v0.0.59
- v0.0.58
- v0.0.57
- v0.0.56
- v0.0.55
- v0.0.54
- v0.0.53
- v0.0.52
- v0.0.51
- v0.0.50
- v0.0.49
- v0.0.48
- v0.0.47
- v0.0.46
- v0.0.45
- v0.0.44
- v0.0.43
- v0.0.42
- v0.0.41
- v0.0.40
- v0.0.39
- v0.0.38
- v0.0.37
- v0.0.36
- v0.0.35
- v0.0.34
- v0.0.33
- v0.0.32
- v0.0.31
- v0.0.30
- v0.0.29
- v0.0.28
- v0.0.27
- v0.0.26
- v0.0.25
- v0.0.24
- v0.0.23
- v0.0.22
- v0.0.21
- v0.0.20
- v0.0.19
- v0.0.18
- v0.0.17
- v0.0.16
- v0.0.15
- v0.0.14
- v0.0.13
- v0.0.12
- v0.0.11
- v0.0.10
- v0.0.9
- v0.0.8
- v0.0.7
- v0.0.6
- v0.0.5
- v0.0.4
- v0.0.3
- v0.0.2
- v0.0.1
This package is auto-updated.
Last update: 2024-12-20 11:33:36 UTC
README
Sleek is a Laravel package that provides a variety of useful features for your Laravel application. All components are styled with Bootstrap for a cohesive and attractive design. The package comes with Bootstrap UI components featuring aggressive defaults.
Table of Contents
Installation
To get started, install Sleek via the Composer package manager:
composer require prometa/sleek
By default, the service provider is automatically registered via Laravel's package auto-discovery. No additional steps are required.
However, if auto-discovery is disabled or does not work as expected, you can register the service provider manually. To do this, add the following line to the providers
array in your config/app.php
file:
\Prometa\Sleek\Providers\SleekServiceProvider::class,
Sleek offers a setup command to automatically install and set up the necessary dependencies. The sleek:setup
command
will check your bootstrap and bootstrap-icons installations and inject an import to sleek's sass into your app.scss:
php artisan sleek:setup
Page Layout
Sleek features a dynamic page layout system, offering a strong default while staying configurable. Simply use the
sleek::view
component to get a complete html-page, assets, menu and scaffolding included!
<x-sleek::view> <div>Your page goes here</div> </x-sleek::view>
Defining Assets
Assets are defined via the Sleek-Facade in your ServiceProvider:
Sleek::assets([ 'vite' => [ /* Your vite bundles go here */ ], /* additional dependencies go here */ ]);
Defining the Menu Structure
Menu items can be defined on 2 levels. On the one hand, you can define menu items via the Sleek-Facade:
Sleek::menu([ [ 'route' => route('index'), 'label' => __('navbar.index'), ], [ /* Enables flexible icon usage via the `icon` attribute: * * - Simple Bootstrap icon (e.g., 'people'): 'people' will use the 'bi-people' class on the icon tag. * - Explicit Bootstrap icon with 'bi:' prefix (e.g., 'bi:people'): 'bi-people' class on the icon tag. * - Blade component with 'component:' prefix (e.g., 'component:my-icon'): will use the component 'my-icon'. * It's also possible to use namespaced components (e.g., 'my-icons::my-icon') */ 'icon' => 'people', 'route' => 'routes('customers'), 'label' => __('navbar.customers'), ], [ 'label' => __('navbar.settings'), /* Navigation items can be nested. Items have the same structure as top-level items. * As of now, only one level of nesting is supported. */ 'items' => [ 'route' => route('settings.general'), 'label' => __('navbar.general'), ] ], ])
Since the service provider is executed outside of a request-context, authentication information an the like are not available. To circumvent this problem, all methods on the Sleek-Facade also accept a callback. Sleek will use the DI-Container for the callback's parameters and will execute it when the view is rendered.
Sleek::menu(fn () => [ [ 'route' => route('index'), 'label' => __('navbar.index'), ], ])
For one-off changes to the navigation structure, items can also be defined on the sleek::view
component:
<x-sleek::view :page:nav:items="[ /* The same structure as above is valid here */ ]"> </x-sleek::view>
This is especially powerful if you need to define sub-sections of your applications which need a different
navigation structure. You can do this by defining your own view component, based on sleek::view
:
{{-- resources/components/custom-view.blade.php --}} <x-sleek::view {{ $attributes->merge([ 'page:nav:items' => [ /* Menu structure goes here */ ] ], false) }}> {{ $slot }} </x-sleek::view> {{-- resources/components/nested/page.blade.php --}} <x-custom-view> Neseted page comes here </x-custom-view>
Authentication
By default, the navbar shows Login/Logout actions for the user, using the standard login
and logout
route names.
You can customize these routes using the Sleek-Facade:
Sleek::authentication(['login' => '/custom-login', 'logout' => '/custom-logout']);
If you do not need authentication routes, you can deactivate it as follows.
Sleek::authentication(false);
Language Switcher
The package has a built-in language switcher. You only need to define the available languages in the AppServiceProvider. For example:
Sleek::language(['de' => 'Deutsch', 'en' => 'Englisch']);
Forms
Sleek attempts to make writing forms as easy as possible. Beyond components for defining and managing forms, sleek also provides natural shortcuts for defining forms upon model instances.
Defining a Form
Sleek provides the sleek::form
component to define a standard HTML-Form. Beyond being a normal form element, this
form will automatically handling the methods for 'PUT', 'PATCH' and 'DELETE' and automatically insert the CSRF-Token
when applicable:
{{-- This form uses method="POST" by default, as it's way more useful than a method="GET" default --}} <x-sleek::form :action="route('some.route')"></x-sleek::form> {{-- This form will automatically transform to use method="POST" and inject @method('PUT') and @csrf into it's body. --}} <x-sleek::form method="PUT" :action="route('some.put.route')"></x-sleek::form> {{-- If you still need a method="GET" form, you just need to explicitly say so --}} <x-sleek::form method="GET" :action="route('some.get.route')"></x-sleek::form>
Defining Form Fields
To define a form field, sleek provides the sleek::form-field
component. This component wraps the standard HTML-Input,
-Select and -Textarea fields (an does a whole lot more magic behind the scenes), allowing you to define form fields
with one unified component:
{{-- Will use the standard <input type="text"> --}} <x-sleek::form-field name="first-field" /> {{-- Will use the standard <textarea></textarea> --}} <x-sleek::form-field type="textarea" name="textarea-fields" /> {{-- All of them will forward their type to <input>-Element --}} <x-sleek::form-field type="checkbox" /> <x-sleek::form-field type="number" /> <x-sleek::form-field type="date" /> <x-sleek::form-field type="datetime" /> <x-sleek::form-field type="hidden" /> {{-- Will render a <select>-Element --}} <x-sleek::form-field type="select"> <option value="1">One</option> <option value="2">Two</option> <option value="3">Three</option> </select> <x-sleek::form-field type="select" :options="['1' => 'One', '2', => 'Two', 3 => 'Three']" /> {{-- Will render a list of radio buttons --}} <x-sleek::form-field type="radio-group" :options="['1' => 'One', '2' => 'Two', 3 => 'Three']" />
Field Names
Like normal form fields, every field needs a name. Beyond just setting a name, sleek::form-field
will automagically
convert dot-notation to nested field names and automatically append []
for multi-selects.
<x-sleek::form-field name="the-name" /> {{-- name will be "nested[name]" --}} <x-sleek::form-field name="nested.name" /> {{-- name will be "multi-select[]" --}} <x-sleek::form-field type="select" multiple name="multi-select">{{-- ... --}}</x-sleek::form-field>
Field Labels
Form fields include labels in their rendered output. By default, it will try to be smart and guess an appropriate translation key for the label, based on the current route's name.
For example, if the current route is named 'users.show' (standard naming convention when using Resource-Controllers), the translation key will be set to 'users.fields.'.
Generally, the form field takes the route, strips the last segment from it and uses the rest as a prefix for the translation key: '.fields.'.
As long as you adhere to standard naming and framework conventions, you should generally have no need to set the label explicitly. However, if you need to, there are several ways override the standard guessing logic, depending on the needed granularity.
Setting an Explicit Label
Of course you can just set an explicit label. sleek::form-field
both accepts the label as an attribute and slot:
<x-sleek::form-field label="Custom Label" /> <x-sleek::form-field> <x-slot:label> Custom Label </x-slot:label> </x-sleek::form-field>
Setting a Custom Prefix
If you only need a custom prefix to find the correct translation file, you can use the i18nPrefix
property:
{{-- will resolve the label from 'custom.prefix.fields.the-field' --}} <x-sleek::form-field i18nPrefix="custom.prefix" name="the-field" />
This property is also checked on any parent component, making it especially useful to define on form components, meaning all containing fields will use the custom prefix:
<x-sleek::form :action="route('some.route')" i18nPrefix="custom.prefix"> {{-- translation key: 'custom.prefix.first-field' --}} <x-sleek::form-field name="first-field" /> {{-- translation key: 'custom.prefix.second-field' --}} <x-sleek::form-field name="second-field" /> </x-sleek::form>
Field values
Beyond setting a value directly, sleek::form-field
will automatically resolve the old input value from the session
upon re-render (for example when validation fails).
The value will be used appropriately depending on the input type, so you don't have to manage "checked" or "selected" attributes manually.
<x-sleek::form-field name="field-name" value="the value" /> {{-- Renders as: --}} <input name="field-name" value="the value"> <x-sleek::form-field name="checkbox-field" type="checkbox" :value="true" /> {{-- Renders as: --}} <input type="checkbox" name="checkbox-field" checked> <x-sleek::form-field name="select-field" type="select" value="1" :options="['1' => 'One', '2' => 'Two']" /> {{-- Renders as: --}} <select name="select-field"> <option value="1" selected>One</option> <option value="2">One</option> </select>
Grouping Form Fields
Sometimes, you want to group fields together, so they form a nested attribute in the submit payload. For example, the following form would group otherwise redundant names into a nested attribute:
<x-sleek::form> <x-sleek::form-field name="name" /> <x-sleek::form-field name="address" /> <x-sleek::form-field name="contact.name" /> <x-sleek::form-field name="contact.address" /> </x-sleek::form>
When laravel parses the payload from submitting the above form, it would look like this:
[ 'name' => '...', 'address' => '...', 'contact' => [ 'name' => '...', 'address' => '...' ], ]
However, having to specify the group key each time can be redundant and increase the complexity of composite forms. To
combat this problem, you can simply wrap fields in a sleek::form-group
:
<x-sleek::form-group name="contact"> {{-- Will have name="contact[name]" --}} <x-sleek::form-field name="name" /> {{-- Will have name="contact[address]" --}} <x-sleek::form-field name="address" /> </x-sleek::form-group>
The form group is also a nice place to set a custom i18nPrefix
if you need to change the lookup for a subset of
form fields:
<x-sleek::form-group i18nPrefix="contacts"> {{-- Will use the translation key "contacts.fields.name" --}} <x-sleek::form-field name="name" /> {{-- Will use the translation key "contacts.fields.address" --}} <x-sleek::form-field name="address" /> </x-sleek::form-group>
Nesting Form Groups
Form groups nest properly! So you can do crazy stuff like the following when you find the need to:
<x-sleek::form-group name="contact"> <x-sleek::form-group name="address"> {{-- Will have name="contact[address][street]" --}} <x-sleek::form-field name="street" /> </x-sleek::form-group> </x-sleek::form-group>
While the above example is pretty contrived, this behavior allows you to compose forms from multiple specialized components without thinking about it too much. Imagine the following component:
{{-- contacts/form.blade.php --}} @props(['name' => null]) <x-sleek::form-group :name="$name"> <x-sleek::form-field name="name" /> <x-sleek::form-field name="address" /> </x-sleek::form-group>
We can now use this component in other forms and ensure the data nests properly, allowing for easy reuse:
{{-- users/create.blade.php --}} <x-sleek::form> <x-sleek::form-field name="name" /> <x-contacts.form name="contact" /> </x-sleek::form>
A Note on From Group Markup
Currently, the Form Group simply renders it's slot, not adding any Markup in the process. We're keeping the option open
to add sensible markup to actually group form elements visibly, but if you want to be absolutely sure that
sleek::form-group
only groups logically and does not add markup, add the passthrough
property to the component:
<x-sleek::form-group passthrough> I will <strong>never</strong> be wrapped in markup! </x-sleek::form-group>
Entity Forms
While the form and field components are already useful by themselves, Sleek also provides a sleek::entity-form
component, which can automatically build a form from a model!
<x-sleek::entity-form :model="$user" :fields="[ /* Assumed type="text" */ 'name', /* Explicit name => type combination */ 'is_admin' => 'checkbox', /* Verbose definition */ 'roles' => [ 'type' => 'select', /* additional properties */ 'multiple', ], ]" />
The above declaration autmagically sets up a few things:
- Since we pass in a model the form assumes an update operation and sets the method to 'PUT'
- The form automatically assumes an "action" property based on the current route.
- The form creates form fields from the "fields" attribute.
- The form pulls out the field values from the given model.
Lets go through them one by one:
Form Method Guessing
Depending on if you provide a model or not, the method of the form will be set to 'POST' or 'PUT' respectively. This aligns with standard assumptions about the routes for creating and updating resources in RESTful APIs.
{{-- Will set method="POST" --}} <x-sleek::entity-form /> {{-- Will set method="PUT" --}} <x-sleek::entity-form :model="$user" /> {{-- Explicitly setting the method overrides automatic guessing, so method="PATCH" --}} <x-sleek::entity-form :model="$user" method="PATCH" />
Form Action Guessing
Depending on the resolved form method, an appropriate form action will be set. Sleek assumes standard form names for ResourceControllers and tries to resolve the route based on those route names.
As with translation keys for form fields, the current route name is used as a basis for constructing the route name.
For example, if the current route is named users.edit
and the method="PUT", the action will be resolved to the route
named users.update
.
Alternatively, if the current route is named users.create
and the method="POST", the action will be resolved to the
route named users.store
.
Generally, as before, the form takes the route, strips the last segment from it and uses the rest as a prefix for the route name: '.update' or '.store'.
{{-- Will use route named 'users.store' --}} <x-sleek::entity-form method="POST" /> {{-- Will use route named 'users.update' --}} <x-sleek::entity-form method="PUT" /> {{-- Explicitly setting an action overridese automatic guessing --}} <x-sleek::entity-form :action="route('custom.action')" />
Form Field Generation
Under the hood, sleek::entity-form
will use sleek::form-field
s under the hood to convert the fields
property
into form elements, so the same magic for it's properties apply here. The array accepts a few forms, depending on
your required level of detail:
<x-sleek::entity-form :fields="[ '<name>', '<name>' => '<type>', '<name>' => [ 'type' => '<type>', /* additional properties for the form-field go here */ ], ]" />
The form fields generated through these methods are simply rendered in a straight down without any complex styling, so
this method is ideal for quick and simple forms. For more complex layouts you can still use sleek::form-field
inside
of sleek::entity-form
without any problem:
<x-sleek::entity-form :model="$user"> <div style="display: grid; grid-template-columns: repeat(2, 1fr);"> <x-sleek::form-field name="first-name" /> <x-sleek::form-field name="last-name" /> </div> </x-sleek::entity-form>
These form fields will automagically be aware of the model passed to sleek::entity-form
, which brings us to the last
part:
Value extraction
Not strictly a feature of sleek::entity-form
(the implementation actually lives in sleek::form-field
), when a model
is passed to the form component, the current value for each form field will be extracted from the model and set on the
form control.
<x-sleek::entity-form :model="new User(['name' => 'James Bond', 'active' => true, 'role' => 'special-agent'])" > {{-- will have the property value="James Bond" --}} <x-sleek::form-field name="name" /> {{-- will have the property "checked" --}} <x-sleek::form-field name="active" type="checkbox" /> {{-- option with value "special-agent" will have property "selected" --}} <x-sleek::form-field name="role" type="select" :options="[/* ... */]" /> </x-sleek::entity-form>
When the data on your model is nested for any reason (relations, json columns, complex casts that create nested fields), you can use dot-notation to access these nested values:
<x-sleek::entity-form :model="new User(['tenant' => ['name' => 'FBI']])"> {{-- will have the properties name="tenant[name]" and value="FBI" --}} <x-sleek::form-field name="tenant.name" /> </x-sleek::entity-form>
As you can see, the name property is used both for setting the input's name and fetching the value from the model. If
you need a different way to access the value, you can also set the accessor
property on sleek::form-field
resolve
the value. The same dot-notation syntax is supported here:
<x-sleek::entity-form :model="new User(['tenant' => ['name' => 'FBI']])"> {{-- will have the properties name="tenant-name" and value="FBI" --}} <x-sleek::form-field name="tenant-name" accessor="tenant.name" /> </x-sleek::entity-form>
Using Form Groups in Entity-Forms
Form groups are especially powerful in Entity-Forms! As discussed previously, sleek::form-group
can set common
prefixes for multiple field names and as discussed just above, field names are by default used to extract current
values from models.
This synergy makes form groups very convenient when you need form fields for nested data:
<x-sleek::entity-form :model="new User(['tenant' => 'name' => 'FBI'])"> <x-sleek::form-group name="tenant"> {{-- will have the properties name="tenant[name]" and value="FBI" --}} <x-sleek::form-field name="name" /> </x-sleek::form-group> </x-sleek::entity-form>
This means that it's often a good idea to let your form structure mirror your model's data structure, which also helps when filling the submitted form values back into the model on update.
Entity Tables
Just like sleek::entity-form
, sleek::entity-table
is a data-driven table component with much the same magic imbued.
Entity-Tables allow for very efficient creation of tables for a collection of models - sorting and pagination included!
Here's a simple example:
<x-sleek::entity-table :entities="new Paginator( collect([ new User(['name' => 'Jon Doe', 'role' => 'user']), new User(['name' => 'Jon Bovi', 'role' => 'artist']), ]), pageSize: 1 )" :columns="['name', 'role']" :sortable="true" />
(paginators are usually proveded by eloquent's query builder, we instantiate one directly in this example for clarity)
With this simple definition we get for free:
- Translated table headers
- Controls for sorting for each column
- Pagination controls
- A row for each model in the
entities
property, extracting the value for each column
There's a lot more to cover, but lets go through the above points for now:
Table Header Generation
sleek::entity-table
will use the provided columns
array to generate the column headers for the table. The field
names will be used to generate translation keys, just like when using sleek::form-field
. For example, if our current
route name is users.show
, the name
-column will look for a translation with the key users.fields.name
.
As before, if you need a custom prefix for key generation, you can set i18nPrefix
to use a custom prefix:
<x-sleek::entity-table :entities="[...]" i18nPrefix="custom-prefix" :columns="['name']" />
Here, the columns will use a prefix of custom-prefix
, so the generated translation key will be
custom-prefix.fields.name
.
If you need to set a custom label for a single column, you can switch to the long form for column-definitions:
<x-sleek::entity-table :entities="[...]" :columns="['name' => ['label' => __('my.custom.label')]" />
Sorting Controls
Setting the sortable
property will append sorting controls to column headers. These controls are really just links to
the current page with additional parameters, so you still need to handle them on the server side.
Tip
Check out our auto sort helpers to automate handling of sorting parameters!
The sortable
property either accepts a boolean, where true
indicates all columns are sortable, or an array of field
names specifying the columns that are sortable.
There are 2 attributes that will be set:
sort-by
will identify the column to sort by, by namesort-direction
will indicate the direction, eitherasc
ordesc
As a simple example, here's how you could use those in your Controller:
User::query() ->when(request('sort-by')) ->orderBy( request('sort-by'), requres('sort-direction'), ) ->get();
Pagination Controls
Whenever the entities
property holds an instance of either a Paginator
or CursorPaginator
, sleek::entity-table
will automatically render the pagination links from that paginator. Additionally to Laravel's default pagination
controls, we also added page size links.
By default, the pagination links are rendered both above and under the table. You can specify the location of the
pagination links by specifying the navigation
property:
navigation="top"
will only render above the tablenavigation="bottom"
will only render below the table:navigation="false"
will disable pagination links
As with sorting links, pagination links are just links to the same page with additional parameters, so you need to handle those on the server yourself.
Tip
Check out our auto paginate helpers to automate handling of pagination parameters!
Again, there are 2 relevant attributes:
page
indicates the page that needs to be renderedpage-size
indicates the number of elements the page should hold
For example, here's how you could use those in your Controller:
User::query() ->paginate(request('page-size'));
Note
We do not supply the "page" since Laravel takes care of that internally.
Scoping Parameter Names
Sometimes you might want to render multiple tables on the same page. While this can get complicated pretty quickly, with
multiple sets of sorting and pagination parameters, sleek::entity-table
tries to simplify this as much as possible.
You can provide a scoped
property to the component which will prefix all parameters this component appends to links:
<x-slot::entity-table :entities="[...]" :sortable="true" scoped="users" />
In this example, the above discussed parameters will instead be named:
users[sort-by]
users[sort-direction]
users[page]
users[page-size]
Value Extraction
Just like sleek::form-field
, sleek::entity-table
will use the column name to automagically extract values from each
model instance to generate columns.
As usual, you can use dot-notation to access nested data. However, there is a slight caveat with guessing translation keys. In this case, only the first section of the name is used for guessing the translation key. For example:
<x-sleek::entity-table :entities="[ new User(['tenant' => ['name' => 'FBI']]), ]" :columns="['tenant.name']" />
Here, the key for translating the column header will only be users.fields.tenant
, while the value for the row will
correctly resolve to 'FBI'. This logic is more intuitive when dotting into nested data, as usually, you're looking for
a displayable value inside a nested structure that describes this structure. In this specific example, we want to
display the 'tenant' by using it's name, so only using the first section of a dotted name feels more correct.
As usual, if this doesn't work for you, you can explicitly specify an accessor. Since a custom accessor is a common occurance, there's a semi-shortcut in addition to the explicit syntax:
<x-sleek::entity-table :entities="[ new User(['tenant' => ['name' => 'FBI']]), ]" :columns="['tenant' => 'tenant.name', 'also-tenant' => ['accessor' => 'tenant.name']]" />
In this example, both column definitions are equivalent.
Custom Columns
Value extraction always needs to resolve to a "stringable" value, but sometimes you need markup in a column - for bades, buttons, etc. To facilitate this, we patched the blade compiler to support our own flavour of "slot properties":
<x-sleek::entity-table :entities="[ new User(['name' => 'James Bond']), ]" > <x-slot:column-name bind="$name"> <strong>{{ $name }}</strong> </x-slot:column-name> </x-sleek::entity-table>
Let's unpack this:
Here we have a slot called column-name
with a special attribute bind
. We patched the blade
compiler to transform any slot with the attribute bind
into a "callable slot" by turning the slot into a callback.
This allows us to render the slot multiple times and pass parameters to it, the parameters to the callback are named
via the value to the bind
property - so in our example, the first parameter to the callback will be called name
.
You can then use the parameter in your slot as it will be in scope when the slot is rendered.
For each column of each row, sleek::entity-table
will look for a slot called column-<name>
, where name is the name
of the column to be rendered. If found, it will then call this callable slot, passing the extracted value of that column
and the full row value as the first and second parameter respectively.
In effect, this allows us to define custom markup for each column or even handle virtual columns where we ignore the extracted value and render custom html instead. For example, you might define an "actions" column that renders navigational links for each record:
<x-sleek::entity-table :entitites="[ new User(['name' => 'James Bond']) ]" :columns="['name', 'actions']" > <x-sleek:column-actions bind="$_, $user"> <a href="{{ route('users.show', $user) }}">Show</a> </x-sleek:column-actions> </x-sleek::entity-table>
Styling Columns
In addition to the bind attribute, you can pass any additional attributes to the slot and they will be passed to the
underlying <td>
-element:
<x-sleek::entity-table :entitites="[ new User(['name' => 'James Bond']) ]" :columns="['name']" > <x-sleek:column-name bind="$name" class="bg-primary"> {{ $name }} </x-sleek:column-name> </x-sleek::entity-table>
Important
Passing attributes is currently limited and cannot be evaluated conditionally based on the current row or column values.
Styling Rows
If you want to append attributes to the <tr>
-element, you can do that via the special row
slot:
<x-sleek::entity-table :entitites="[ new User(['name' => 'James Bond']) ]" :columns="['name']" > <x-slot:row class="bg-success" /> </x-sleek::entity-table>
Important
Passing attributes is currently limited and cannot be evaluated conditionally based on the current row value.
Note
The content of this slot is always ignored and can only be used to pass attributes.
Eloquent Extensions
Auto Sort
As described in the Sorting Controls section, sorting can require repetitive boilerplate. The auto sort helper automates this process based on request parameters.
When autoSort()
is called, it checks for the following request parameters:
sort-by
: The column name to sort by.sort-direction
: The sort direction (asc
ordesc
, defaults todesc
).
If these parameters are present, autoSort()
automatically applies the necessary orderBy
clause.
$request->all(); /* [ * 'sort-by' => 'name', * 'sort-direction' => 'asc' * ] */ User::query() // Will apply ->orderBy('name', 'asc') ->autoSort() ->get()
Auto Paginate
Building on the behavior described in the Pagination Controls, the auto paginate helper automatically applies pagination parameters from the request to your query. When you call autoPaginate()
, it looks for the following parameters:
page-size
: The number of items per page (defaults to a given value if it's not provided in the request)page
: The page that needs to be rendered
If a request includes a page-size
parameter, as shown below:
$request->all(); /* [ * 'page-size' => '100', * 'page' => '1' * ] */
Calling autoPaginate()
like this:
User::query() ->autoPaginate(50) // Default items per page 50 // Will append ->paginate(100)
Automatically adjusts the query to display 100 items per page by appending ->paginate(100)
. This ensures that the page-size
parameter from the request takes precedence over the default value.
If the request does not include a page-size
parameter, autoPaginate()
will use the default size specified in the method call, such as 50 in this case.
Auto Filter
The auto filter helper simplifies the process of applying query filters based on request parameters. When autoFilter()
is called, it dynamically checks the request for filter parameters and applies them to the query using a configured filter pipeline.
Filter Pipeline
There are several built-in filters for the most common use cases:
equals
: Filters by exact matchlike
: Filters by partial match using LIKEcontains
: Filters JSON fields to check if they contain a specific valuegt
: Filters by greater thanlt
: Filters by less thangte
: Filters by greater than or equal tolte
: Filters by less than or equal tofor_each
: Processes each value in a comma-sperated list with a given operator from the list above (e.g.for_each|equals
)boolean
: This filter ensures that a value is properly cast to a boolean before applying a comparison operator (e.g.,boolean|equals
). It is especially helpful when working with values that might otherwise be compared as strings, such as'true'
or'false'
.
Example Usage
To use the filtering functionality, you need to create a form on your page with the corresponding input fields. For the example provided, the form should contain three fields: name
, age
and role
. These fields allow users to define their filter criteria, which will be processed by autoFilter()
. The autoFilter()
helper automatically retrieves and applies the parameters from the request to filter the query.
If the request includes the following parameters:
/* [ * 'name' => 'James', * 'age' => 25, * 'role' => 'admin,user' * ] */
You can configure autoFilter()
for example as follows:
User::query() ->autoFilter([ 'name' => 'like', // ->where('name', 'like', '%James%') 'age' => 'gte', // `->where('age', '>=', 25)` 'role' => 'for_each|equals' // ->where('role', 'admin')->where('role', 'user') ]) ->get()
This configuration tells autoFilter()
to dynamically append the suitable conditions to the query based on the request parameters:
name
: appends->where('name', 'like', '%James%')
age
: appends->where('age', '>=', 25)
role
: appends->where('role', 'admin')->where('role', 'user')
UI Components
Alert
Usage
If you use the Sleek layout, the alert is automatically included and ready to use. Otherwise you can add it to your own layout or to an individual page.
<x-sleek::alert />
After that you can use and trigger the alert in your controller.
use Prometa\Sleek\Facades\Sleek; Sleek::raise('Your message goes here', 'danger');
The first parameter is the message and the second is the type. The icons displayed are dependent on the type.
The following types are supported:
danger
warning
info
success
primary
secondary
light
dark
You can set the position of the alert in the AppServiceProvider
.
Sleek::alert([ 'position' => 'top-right' ]);
Available positions: center
, top-left
, top-right
, bottom
, bottom-left
, bottom-right
Entity-Table
You can easily create a table with data using the entity-table
component. The table also supports pagination out-of-the-box.
Here's a basic example:
<x-sleek::entity-table :entities="$users" :columns="['name', 'email', 'actions']" / >
Parameters
entities
: Specifies the collection that should be displayed in the table.columns
: Defines which fields from the collection should be displayed in the table. Note that you can include fields that are not in the entities collection, such asactions
, to customize your table further.
In the above example, the entities
parameter is set to $users
, which means the table will display data from the $users
collection. The columns
parameter specifies that the fields 'name', 'email', 'actions'
will be shown.
Customize the Table
You can further customise each field of the table, e.g. by adjusting the formatting in each table cell using slots, or you can display other fields.
Customizing Cell Format
It is important that the name of the column is used in the slot.
<x-slot:column-<columnName> />
To customize the format of a specific column, you can use the slot with the column name. This is very useful for date fields to display in the format of your region. Here's an example that customizes the name
column:
<x-slot:column-name bind="$name"> User: {$name} </x-slot:column-name>
This changes the way names are displayed in the name
column
Accessing Full Model Data
You can also pass the entire model as an argument to the slot to have access to all of its fields. This can be especially useful for incorporating model IDs into routes for actions like deleting an entry. Here's how you can do it:
<x-slot:column-actions bind="$_,$user"> {$user->name} {$user->id} <a href="{{route('users.show',$user->id)}}">Details</a> </x-slot:column-actions>
As you can see, this allows you to access fields that may not even be displayed in the table, such as the model's ID.
Entity-Form
The Entity Form feature allows you to easily create forms for editing existing entities and create forms. Instead of manually specifying each field and its value, you can simply pass the entity and the list of fields to display. The CSRF Token will be set automatically. You can also set the Method to PUT
, POST
, DELETE
, GET
<x-sleek::entity-form :fields="['name' , 'email']"> <button type="submit" class="btn btn-primary">Submit</button> </x-sleek::entity-form>
This example would create a form with two field name and email.
Customize Route
You can easily change the route of the form by defining the action.
<x-sleek::entity-form action="{{route('profil.update',$user)}}" :fields="['name','email']" > <button type="submit" class="btn btn-primary">Submit</button> </x-sleek::entity-form>
Use a model
If you give the model attribute a user model, for example, then the fields are automatically filled with the existing data. This way you can easily develop e.g. a user edit page.
<x-sleek::entity-form action="{{route('profil.update',$user)}}" :model="$user" :fields="['name','email']" > <button type="submit" class="btn btn-primary">Submit</button> </x-sleek::entity-form>
Modal Form
This component can be used to create a form in a dialogue. The form also uses Alpine.js to deactivate the button after the submit and display a loading spinner.
Usage
<button data-bs-target="#add-user-modal" data-bs-toggle="modal">User Create</button> <x-sleek::modal-form title="User" :action="route('users.store')" method="POST" id="add-user-modal" > <x-sleek::form-field type="text" name="name" label="Name"/> </x-sleek::modal-form>
The attributes in the modal form are required. The button to open the dialogue must be linked to the ID of the dialogue. To define fields, it is best to use the form-field component.
Customization
If you want to change the text of the buttons in the dialogue, you can do this as follows.
<button data-bs-target="#add-user-modal" data-bs-toggle="modal">User Create</button> <x-sleek::modal-form title="User" :action="route('users.store')" method="POST" id="add-user-modal" > <x-slot:submit label="Submit" ></x-slot:submit> <x-slot:cancel label="Cancel" ></x-slot:cancel> </x-sleek::modal-form>
Example
Here you can find a full CRUD example Show Code