Skip to content

Theme

Namespace: App\Infrastructure\Services\Theme

Method Description Type Parameters Return
__construct throws ContainerExceptionInterface
throws ReflectionException
throws NotFoundExceptionInterface|Exception
public
id Theme's id protected
name Theme's name protected
path The plugin's directory path
protected string
route Theme's route for submenu protected
url Theme's url protected

When creating a plugin, you can extend the abstract class Plugin and implement the two abstract methods meta and handle. The meta method returns an array of info about your plugin, and the handle method is called by the system to load assets, views, etc. The following 6 steps will walk you through creating your first plugin.

Best Practices in Plugin Development

Devflow is a framework, allowing php developers to build bespoke websites and applications for their business and their clients. But if you've built a plugin you would like to share with the community, there a few best practices you should follow.

Using Coding Standards

Devflow uses specific coding standards to help maintain consistency, quality, and clean code. The coding standard that Devflow uses and promotes is PSR-12 along with the Qubus Coding Standards.

Namespaces

Classes as well as function should be namespaced, unless there is a strong reason to not namespace a function. Namespacing functions is recommended and highly encouraged in order to keep the global namespace clean and available for native PHP functions.

Security

You must always validate and sanitize data on output. Never trust user data that was inputted. If you use the native Devflow functions that output data, that data is already sanitized or purified. Keep these things in mind when outputting data outside the native functions:

  • Sanitize data on output by using helpers Qubus\Security\Helpers\esc_html, Qubus\Security\Helpers\esc_html__, Qubus\Security\Helpers\esc_url, Qubus\Security\Helpers\esc_js, and Qubus\Security\Helpers\purify_html.
  • Check user permission before executing or saving data to prevent unauthorized operations or access.
  • Use the Devflow database helper (App\Shared\Helpers\dfdb) and prepared statements to avoid SQL injection vulnerabilities.

Step #1: Create Your Plugin File

The first step is to create a folder for your plugin in the public/plugins directory. Your directory must meet PSR-4 autoload standards. For this example, we are going to create a New Widget plugin. The name of the new folder will be NewsWidget and the name of the main class will be the same followed by the suffix Plugin.php: NewsWidgetPlugin.php.

NewsWidgetPlugin.php will extend the abstract class App\Infrastructure\Services\Plugin. We should now have a new class:

<?php

declare(strict_types=1);

namespace Plugin\NewsWidget;

use App\Infrastructure\Services\Plugin;

class NewsWidgetPlugin extends Plugin
{
    /**
     * @inheritDoc
     */
    public function meta(): array
    {
        // TODO: Implement meta() method.
    }

    /**
     * @inheritDoc
     */
    public function handle(): void
    {
        // TODO: Implement handle() method.
    }
}

We need to fill out the meta method to include our plugin's info:

  • name - The name of the plugin.
  • id - Plugin's unique identifier. This is also used for your route if you need to register a submenu.
  • author - Person or company who authored the plugin.
  • version - Current version of the plugin.
  • description - A short description of the plugin's purpose.
  • basename - Filename of the plugin.
  • path - File path of the plugin.
  • url - Plugin's directory url.
  • pluginUri - Where the plugin is hosted and updates found.
  • authorUri - Website of the plugin author.
  • className - Name of the class.

With the meta details filled out, this is now the state of our plugin class:

<?php

declare(strict_types=1);

namespace Plugin\NewsWidget;

use App\Infrastructure\Services\Plugin;
use App\Shared\Services\Registry;

use function App\Shared\Helpers\plugin_basename;
use function App\Shared\Helpers\plugin_dir_path;
use function App\Shared\Helpers\plugin_url;
use function dirname;
use function get_class;
use function Qubus\Security\Helpers\esc_html__;

class NewsWidgetPlugin extends Plugin
{
    /**
     * @inheritDoc
     */
    public function meta(): array
    {
        $newsWidgetInfo = [
            'name' => esc_html__(string: 'News Widget', domain: 'news-widget'),
            'id' => 'news-widget',
            'author' => 'Joshua Parker',
            'version' => '1.0.0',
            'description' => 'Adds a news widget to the admin dashboard.',
            'basename' => plugin_basename(dirname(__FILE__)),
            'path' => plugin_dir_path(dirname(__FILE__)),
            'url' => plugin_url('', __CLASS__),
            'pluginUri' => 'https://github.com/getdevflow/news-widget',
            'authorUri' => 'https://nomadicjosh.com/',
            'className' => get_class($this),
        ];

        Registry::getInstance()->set('news-widget', $newsWidgetInfo);

        return $newsWidgetInfo;
    }

    /**
     * @inheritDoc
     */
    public function handle(): void
    {
        // TODO: Implement handle() method.
    }
}

Now that the meta method is implemented, our plugin should now be registered on our plugins page:

image

Step #2: Create Other Methods (if needed)

For our plugin class, we will create another method called widgetNews which will give us the latest feed from TechCrunch:

/**
 * @throws NotFoundExceptionInterface
 * @throws ContainerExceptionInterface
 * @throws ReflectionException
 * @throws InvalidArgumentException
 */
private function newsWidget(): array
{
    $cache = SimpleCacheObjectCacheFactory::make(namespace: 'rss');

    $rss1 = new DOMDocument();
    $rss1->load(filename: 'https://feeds.feedburner.com/TechCrunch/');
    $feed = [];
    foreach ($rss1->getElementsByTagName('item') as $node) {
        $item = [
            'title' => $node->getElementsByTagName('title')->item(0)->nodeValue,
            'link' => $node->getElementsByTagName('link')->item(0)->nodeValue,
            'date' => $node->getElementsByTagName('pubDate')->item(0)->nodeValue,
        ];
        $feed[] = $item;
    }

    if ($cache->has(key: 'techcrunch')) {
        $feed = $cache->get(key: 'techcrunch');
    } else {
        $cache->set(key: 'techcrunch', value: $feed);
    }

    $data = [];

    $limit = 4;
    for ($x = 0; $x < $limit; $x++) {
        $title = str_replace(' & ', ' &amp; ', $feed[$x]['title']);
        $link = $feed[$x]['link'];
        $date = date('l F d, Y', strtotime($feed[$x]['date']));
        $data[] = '<p><strong><a href="' . $link . '" title="' . $title . '">' . $title . '</a></strong><br />' .
        '<small><em>Posted on ' . $date . '</em></small></p>';
    }

    return $data;
}

The next method we will add will be render for our view:

/**
 * @throws ViewException
 * @throws NotFoundExceptionInterface
 * @throws InvalidTemplateNameException
 * @throws ReflectionException
 * @throws ContainerExceptionInterface
 * @throws InvalidArgumentException
 */
public function render(): void
{
    echo $this->view->render('plugin::NewsWidget/view/widget', ['data' => $this->newsWidget()]);
}

Step #3: Create Your View

Next, create a widget.phtml file in NewsWidget/view/ with the following contents:

<?php

use function Qubus\Security\Helpers\esc_html__;

$this->block('backend', function ($param) {
?>
    <div class="box">
        <div class="box-header with-border">
            <h3 class="box-title"><i class="fa fa-rss"></i> <?= esc_html__(string: 'TechCrunch Feed', domain: 'news-widget'); ?></h3>
            <div class="box-tools pull-right">
                <!-- Collapse Button -->
                <button type="button" class="btn btn-box-tool" data-widget="collapse">
                    <i class="fa fa-minus"></i>
                </button>
            </div>
            <!-- /.box-tools -->
        </div>
        <!-- /.box-header -->
        <div class="box-body">
            <?php foreach($param['data'] as $feed): echo $feed; endforeach; ?>
        </div>
        <!-- /.box-body -->
    </div>
    <!-- /.box -->

<?php }); ?>

Step #4: Implement the handle Method

Now we are ready to implement our handle method. The handle method is what the system looks for and will trigger your plugin to run. We want this widget to post to the left side of our dashboard, so we will use the widget_left_column action hook:

/**
 * @inheritDoc
 * @throws ReflectionException
 */
public function handle(): void
{
    Action::getInstance()->addAction('widget_left_column', [$this, 'render'], 5);

    Action::getInstance()->addAction('flush_cache', function () {
        SimpleCacheObjectCacheFactory::make(namespace: 'rss')->delete(key: 'techcrunch');
    }, 5);
}

We also will hook to the flush_cache action as well so that when the cache gets flushed, the TechCrunch feed cache will get purged as well. With everything in place, this is what our NewsWidgetPlugin class should look like:

<?php

declare(strict_types=1);

namespace Plugin\NewsWidget;

use App\Infrastructure\Services\Plugin;
use App\Shared\Services\Registry;
use App\Shared\Services\SimpleCacheObjectCacheFactory;
use DOMDocument;
use Psr\Container\ContainerExceptionInterface;
use Psr\Container\NotFoundExceptionInterface;
use Psr\SimpleCache\InvalidArgumentException;
use Qubus\EventDispatcher\ActionFilter\Action;
use Qubus\Exception\Exception;
use Qubus\View\Native\Exception\InvalidTemplateNameException;
use Qubus\View\Native\Exception\ViewException;
use ReflectionException;

use function App\Shared\Helpers\plugin_basename;
use function App\Shared\Helpers\plugin_dir_path;
use function App\Shared\Helpers\plugin_url;
use function date;
use function dirname;
use function get_class;
use function Qubus\Security\Helpers\esc_html__;
use function str_replace;
use function strtotime;

class NewsWidgetPlugin extends Plugin
{
    /**
     * @inheritDoc
     * @throws ReflectionException
     * @throws Exception
     */
    public function meta(): array
    {
        $newsWidgetInfo = [
            'name' => esc_html__(string: 'News Widget', domain: 'news-widget'),
            'id' => 'news-widget',
            'author' => 'Joshua Parker',
            'version' => '1.0.0',
            'description' => 'Adds a news widget to the admin dashboard.',
            'basename' => plugin_basename(dirname(__FILE__)),
            'path' => plugin_dir_path(dirname(__FILE__)),
            'url' => plugin_url('', __CLASS__),
            'pluginUri' => 'https://github.com/getdevflow/news-widget',
            'authorUri' => 'https://nomadicjosh.com/',
            'className' => get_class($this),
        ];

        Registry::getInstance()->set('news-widget', $newsWidgetInfo);

        return $newsWidgetInfo;
    }

    /**
     * @inheritDoc
     * @throws ReflectionException
     */
    public function handle(): void
    {
        Action::getInstance()->addAction('widget_left_column', [$this, 'render'], 5);

        Action::getInstance()->addAction('flush_cache', function () {
            SimpleCacheObjectCacheFactory::make(namespace: 'rss')->delete(key: 'techcrunch');
        }, 5);
    }

    /**
     * @throws NotFoundExceptionInterface
     * @throws ContainerExceptionInterface
     * @throws ReflectionException
     * @throws InvalidArgumentException
     */
    private function newsWidget(): array
    {
        $cache = SimpleCacheObjectCacheFactory::make(namespace: 'rss');

        $rss1 = new DOMDocument();
        $rss1->load(filename: 'https://feeds.feedburner.com/TechCrunch/');
        $feed = [];
        foreach ($rss1->getElementsByTagName('item') as $node) {
            $item = [
                'title' => $node->getElementsByTagName('title')->item(0)->nodeValue,
                'link' => $node->getElementsByTagName('link')->item(0)->nodeValue,
                'date' => $node->getElementsByTagName('pubDate')->item(0)->nodeValue,
            ];
            $feed[] = $item;
        }

        if ($cache->has(key: 'techcrunch')) {
            $feed = $cache->get(key: 'techcrunch');
        } else {
            $cache->set(key: 'techcrunch', value: $feed);
        }

        $data = [];

        $limit = 4;
        for ($x = 0; $x < $limit; $x++) {
            $title = str_replace(' & ', ' &amp; ', $feed[$x]['title']);
            $link = $feed[$x]['link'];
            $date = date('l F d, Y', strtotime($feed[$x]['date']));
            $data[] = '<p><strong><a href="' . $link . '" title="' . $title . '">' . $title . '</a></strong><br />' .
            '<small><em>Posted on ' . $date . '</em></small></p>';
        }

        return $data;
    }

    /**
     * @throws ViewException
     * @throws NotFoundExceptionInterface
     * @throws InvalidTemplateNameException
     * @throws ReflectionException
     * @throws ContainerExceptionInterface
     * @throws InvalidArgumentException
     */
    public function render(): void
    {
        echo $this->view->render('plugin::NewsWidget/view/widget', ['data' => $this->newsWidget()]);
    }
}

And plugin's tree should look similar to this:

NewsWidget
├── NewsWidgetPlugin.php
└── view
    └── widget.phtml

Step #5: Activate the Plugin

Once we activate our plugin, the new widget should appear on our dashboard:

image

Step #6: Share Your Plugin

If we wanted to share this plugin to allow others in the community to install it via composer, we would add a composer.json file to the root of our plugin's directory:

{
  "name": "getdevflow/techcrunch-news-widget",
  "description": "Adds a TechCrunch widget to the admin dashboard.",
  "type": "devflow-plugin",
  "keywords": ["techcrunch","devflow-plugin","plugins","widget"],
  "license": "GPL-2.0-only",
  "authors": [
    {
      "name": "Joshua Parker",
      "email": "joshua@joshuaparker.dev"
    }
  ],
  "require": {
    "php": ">=8.3",
    "oomphinc/composer-installers-extender": "^2.0"
  },
  "extra": {
    "installer-name": "NewsWidget",
    "installer-types": ["devflow-plugin"]
  },
  "minimum-stability": "stable",
  "prefer-stable": true,
  "config": {
    "allow-plugins": {
      "composer/installers": true,
      "oomphinc/composer-installers-extender": true
    }
  }
}

There are several important key points to point out in the json data above:

  1. Make sure to change the vendor name
  2. Make sure to add the type devflow-plugin
  3. Make sure to include installer-name and make it PSR-4 compatible. It will install the plugin as NewsWidget instead of as techcrunch-news-widget.
  4. Make sure the installer-types includes devflow-plugin

When someone runs composer require getdevflow/techcrunch-news-widget, the plugin will be installed with the correct folder name public/plugins/NewsWidget/.

There you go. These are the steps you can take to create a plugin as well as how to share your new plugin with the Devflow community. In the near future, there will be a GitHub repository available where developers will be able to open a pull request to add their plugin to the list of plugins available for install.