artisancms / widgets
A powerful alternative to view composers. Asynchronous widgets, reloadable widgets, console generator, caching - everything you can think of.
Requires
- php: >=5.5.9
- illuminate/console: >=5.1
- illuminate/container: >=5.1
- illuminate/contracts: >=5.1
- illuminate/support: >=5.1
- illuminate/view: >=5.1
Requires (Dev)
- phpunit/phpunit: ~4.0
- dev-master
- 3.6.2
- 3.6.1
- 3.6.0
- 3.5.0
- 3.4.2
- 3.4.1
- 3.4.0
- 3.3.2
- 3.3.1
- 3.3.0
- 3.2.1
- 3.2.0
- 3.1.3
- 3.1.2
- 3.1.1
- 3.0.2
- 3.0.1
- 3.0.0
- 2.8.0
- 2.7
- 2.6
- 2.5.1
- 2.5.0
- 2.4.1
- 2.4.0
- 2.3.8
- 2.3.7
- 2.3.6
- 2.3.5
- 2.3.4
- 2.3.3
- 2.3.2
- 2.3.1
- 2.3.0
- 2.2.3
- 2.2.2
- 2.2.1
- 2.2.0
- 2.1.3
- 2.1.2
- 2.1.1
- 2.1.0
- 2.0.1
- 2.0.0
- 1.0.x-dev
- 1.0.2
- 1.0.1
- 1.0.0
- 0.2.3
- 0.2.2
- 0.2.1
- 0.2.0
- 0.1.2
- 0.1.1
- 0.1.0
- dev-scrutinizer-patch-1
This package is auto-updated.
Last update: 2024-12-19 17:07:38 UTC
README
https://img.shields.io/github/issues/artisancms/widgets.svg #Widgets for Laravel
A powerful alternative to view composers. Asynchronous widgets, reloadable widgets, console generator, caching - everything you can think of.
Installation
-
Run
composer require artisancms/widgets
-
Register a service provider in the
app.php
configuration file
<?php 'providers' => [ ... ArtisanCMS\Widgets\ServiceProvider::class, ], ?>
- Add some facades here too. If you prefer custom blade directives to facades you can skip this step.
<?php 'aliases' => [ ... 'Widget' => ArtisanCMS\Widgets\Facade::class, 'AsyncWidget' => ArtisanCMS\Widgets\AsyncFacade::class, ], ?>
Usage
Let's consider we want to make a list of recent news and reuse it in several views.
First of all we can create a Widget class using the artisan command provided by the package.
php artisan make:widget RecentNews
This command generates two files:
resources/views/widgets/recent_news.blade.php
is an empty view.
Add "--plain" option if you do not need a view.
app/Widgets/RecentNews
is a widget class.
<?php namespace App\Widgets; use ArtisanCMS\Widgets\AbstractWidget; class RecentNews extends AbstractWidget { /** * The configuration array. * * @var array */ protected $config = []; /** * Treat this method as a controller action. * Return view() or other content to display. */ public function run() { // return view("widgets.recent_news", [ 'config' => $this->config, ]); } }
Note: You can use your own stubs if you need. Publish config file to change paths.
The last step is to call the widget. You've actually got several ways to do so.
@widget('recentNews')
or
{{ Widget::run('recentNews') }}
or even
{{ Widget::recentNews() }}
There is no real difference between them. The choice is up to you.
Note: For Laravel 5.0.0 - 5.1.3 you have to use
{!! !!}
tags instead of{{ }}
Passing variables to widget
Via config array
Let's carry on with the "recent news" example.
Imagine that we usually need to show five news, but in some views we need to show ten. This can be easily achieved like that:
class RecentNews extends AbstractWidget { ... protected $config = [ 'count' => 5 ]; ... } ... @widget('recentNews') // shows 5 @widget('recentNews', ['count' => 10]) // shows 10
['count' => 10]
is a config array that can be accessed by $this->config.
Config array is available in every widget method so you can use it to configure placeholder and container too (see below)
Note: Config fields that are not specified when you call a widget aren't overridden:
class RecentNews extends AbstractWidget { ... protected $config = [ 'count' => 5, 'foo' => 'bar' ]; ... } @widget('recentNews', ['count' => 10]) // $this->config['foo'] is still 'bar'
Note 2: You may want (but you probably don't) to create your own BaseWidget and inherit from it. That's fine. The only edge case is merging config defaults from a parent and a child. In this case do the following:
-
Do not add
protected $config = [...]
line to a child. -
Add defaults like that instead:
public function __construct(array $config = []) { $this->addConfigDefaults([ 'child_key' => 'bar' ]); parent::__construct($config); }
Directly
You can also choose to pass additional parameters to run()
method directly.
@widget('recentNews', ['count' => 10], 'date', 'asc') ... public function run($sortBy, $sortOrder) { } ...
run()
method is resolved via Service Container, so method injection is also available here.
Namespaces
By default the package tries to find your widget in the App\Widgets
namespace.
You can override this by publishing package config (php artisan vendor:publish --provider="ArtisanCMS\Widgets\ServiceProvider"
) and setting default_namespace
property.
Although using the default namespace is very convenient, in some cases you may wish to have more flexibility. For example, if you've got dozens of widgets it makes sense to group them in namespaced folders.
No problem, you have several ways to call those widgets:
- Pass a full widget name from the
default_namespace
(basicallyApp\Widgets
) to therun
method.
@widget('News\RecentNews', $config)
- Use dot notation.
@widget('news.recentNews', $config)
- FQCN is also an option.
@widget('\App\Http\Some\Namespace\Widget', $config)
Asynchronous widgets
In some situations it can be very beneficial to load widget content with AJAX.
Fortunately, this can be achieved very easily!
All you need to do is to change facade or blade directive - Widget::
=> AsyncWidget::
, @widget
=> @asyncWidget
Note: Widget params are encrypted and sent via ajax call. Expect them to be json_encoded and json_decoded afterwards.
Note: Since version 3.1 you no longer need
jquery
to make ajax calls. However you can setuse_jquery_for_ajax_calls
totrue
in the config file if you want to.
By default nothing is shown until ajax call is finished.
This can be customized by adding a placeholder()
method to the widget class.
public function placeholder() { return "Loading..."; }
Side note: If you need to do smth with the routes package uses to load async widgets (e.g. you run app in a subfolder http://site.com/app/) you need to copy ArtisanCMS\Widgets\ServiceProvider to your app, modify it according to your needs and register it in Laravel instead of the former one.
Reloadable widgets
You can go even further and automatically reload widget every N seconds.
Just set the $reloadTimeout
property of the widget class and you are done.
class RecentNews extends AbstractWidget { /** * The number of seconds before each reload. * * @var int|float */ public $reloadTimeout = 10; }
Both sync and async widgets can become reloadable.
You should use this feature with care, because it can easily spam your app with ajax calls if timeouts are too low. Consider using web sockets too but they are way harder to set up.
Container
Async and Reloadable widgets both require some DOM interaction so they wrap all widget output in a html container.
This container is defined by AbstractWidget::container()
method and can be customized too.
/** * Async and reloadable widgets are wrapped in container. * You can customize it by overriding this method. * * @return array */ public function container() { return [ 'element' => 'div', 'attributes' => 'style="display:inline" class="artisancms-widget-container"', ]; }
Note: Nested async or reloadable widgets are not supported.
Caching
There is also a simple built-in way to cache entire widget output. Just set $cacheTime property in your widget class and you are done.
class RecentNews extends AbstractWidget { /** * The number of minutes before cache expires. * False means no caching at all. * * @var int|float|bool */ public $cacheTime = 60; }
No caching is turned on by default.
A cache key depends on a widget name and each widget parameter.
Override cacheKey
method if you need to adjust it.
Widget groups (extra)
In most cases Blade is a perfect tool for setting the position and order of widgets. However, sometimes you may find useful the following approach:
// add several widgets to the 'sidebar' group anywhere you want (even in controller) Widget::group('sidebar')->position(5)->addWidget('widgetName1', $config1); Widget::group('sidebar')->position(4)->addAsyncWidget('widgetName2', $config2); // display them in a view in the correct order @widgetGroup('sidebar') //or {{ Widget::group('sidebar')->display() }}
position()
can be omitted from the chain.
Widget::group('sidebar')->addWidget('files');
equals
Widget::group('sidebar')->position(100)->addWidget('files');
You can also set a separator to display between widgets in a group.
Widget::group('sidebar')->setSeparator('<hr>');
Checking the state of a widget group
Widget::group('sidebar')->isEmpty(); // bool
Widget::group('sidebar')->any(); // bool
Widget::group('sidebar')->count(); // int