brotkrueml/schema

RFC: Add option to inject virtual breadcrumb items

nostadt opened this issue · 6 comments

Currently the Breadcrumb is generated based on the simple page tree. In real projects using routeEnhancer one has virtual pages which are not represented by the page tree and those do not appear in the Breadcrumb. It would be neat to be able to add custom Breadcrumb-Items.

Looking into the code of the current implementation there's no way except to develop an own BreadcrumbSchemaManager // BreadcrumbAspect which can be enriched throughout the runtime.


POC:

1. Create a BreadcrumbListManager as Singleton.

<?php
declare(strict_types=1);

namespace Vendor\Extension\Manager;

use Brotkrueml\Schema\Model\Type\ListItem;
use TYPO3\CMS\Core\SingletonInterface;

class BreadcrumbListManager implements SingletonInterface
{
    /**
     * @var ListItem[]
     */
    protected $additionalListItems = [];

    public function addAdditionalListItem(ListItem $listItem): void
    {
        $this->additionalListItems[] = $listItem;
    }

    /**
     * @return ListItem[]
     */
    public function getAdditionalListItems(): array
    {
        return $this->additionalListItems;
    }
}

2. Create an own BreadcrumbListAspect which is basically a copy of the current implementation of ext:schema with a small addition

final class BreadcrumbListAspect implements AspectInterface
{
    // ...

    private function buildBreadCrumbList(array $rootLine): TypeInterface
    {
        // ...
        $indexCopy = 0; // <--- This is new
        foreach (\array_values($rootLine) as $index => $page) {
            // ...
        }

        /** @var BreadcrumbListManager $breadcrumbListManager */
        $breadcrumbListManager = GeneralUtility::makeInstance(BreadcrumbListManager::class);
        foreach ($breadcrumbListManager->getAdditionalListItems() as $listItem) {
            $listItem->setProperty('position', $indexCopy + 1);
            $breadcrumbList->addProperty('itemListElement', $listItem);
            $indexCopy++;
        }

        return $breadcrumbList;
    }
}

3. Replace the Aspect with your own

$GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/schema']['registerAspect']['breadcrumbList'] = \Vendor\Extension\Aspect\BreadcrumbListAspect::class;

4. Go to desired routeEnhancer aka Controller and register the additionalItem

if (ExtensionManagementUtility::isLoaded('schema')) {
    /** @var BreadcrumbListManager $breadcrumbListManager */
    $breadcrumbListManager = GeneralUtility::makeInstance(BreadcrumbListManager::class);
    $item = TypeFactory::createType('WebPage');
    $item->setId($this->uriBuilder->getRequest()->getRequestUri() . '#FOO');
    $listItem = TypeFactory::createType('ListItem')->setProperties([
        'name' => $download->getTitle(),
        'item' => $item,
    ]);
    $breadcrumbListManager->addAdditionalListItem($listItem);
}

A breadcrumb should be shown also on the page (in my opinion) to help a user to understand, where on the site she is. Also it is a nice navigational feature to get to the pages in the upper hierarchy. So, the best way is to use an own MenuProcessor to inject, e.g., a category into the breadcrumb. You can see the principle in the news extension, where the detail record is added to the breadcrumb:
https://github.com/georgringer/news/blob/master/Classes/DataProcessing/AddNewsToMenuProcessor.php

Have a look on this page for an example in frontend:
https://www.jobrouter.com/en/video/jobrpa-innovative-process-automation-with-bots/
The "Modules" category is a virtual page with an Extbase plugin. An own MenuProcessor fetches for a detail page the category and the title of the detail page and injects it into the breadcrumb

You can then use the breadcrumb view helper to inject the JSON-LD markup:
https://docs.typo3.org/p/brotkrueml/schema/1.8/en-us/Developer/Breadcrumb.html#view-helper-schema-breadcrumb

Does this fit your needs?

Thanks for your response and thoughts.

Well, the approach with a MenuProcessor leads to a duplicate query. In ProductController::showAction for example I have the desired product already.

Yet I agree that the suggestion by the issue does not consider the visible breadcrumb.

I don't like the fact that the items of the visual breadcrumb and the structured markup are decoupled from each other. I consider that as bad style. The markup should reflect the content of the page. So, to add this feature to the API is no good as it encourages developers to use it.

I understand the point of the duplicate database query, you can't even cache it on your own (perhaps in a transient cache), because the MenuProcessor is called before the Extbase controller. And the Extbase magic injects the entity always into the action method with its own query.

Back to your PoC: There is the possibility to have more than one BreadcrumbList on a page. The BreadcrumbListManager is not capable of handling multiple lists.

I thought about something like SchemaManager->clearAllBreadcrumbLists(), so you can clear the lists and build and add your own under certain circumstances. But then we are at the beginning of this comment ;-)

Yes, I indeed see your point here. I can't get rid of the thought that the routeEnhancer logic should extend the TSFE.rootline assuming TSFE is available in the future. After all the route aka rootline is IMO enhanced there. TYPO3 would fetch the object e.g. there and also pass it along to the Controller::Action.

IMO this can be closed.