- Documentation >
- Templating >
- Templates >
- Create custom view matcher
Create custom view matcher
In addition to the built-in view matchers, you can also create custom matchers to use in template configuration.
To do it, create a matcher class that implements Ibexa\Core\MVC\Symfony\Matcher\ContentBased\MatcherInterface.
Matcher class
The matcher class must implement the following methods:
- matchLocation- checks if a location object matches.
- matchContentInfo- checks if a ContentInfo object matches.
- match- checks if the View object matches.
- setMatchingConfig- receives the matcher's config from the view rule.
The following example shows how to implement an Owner matcher.
This matcher identifies content items that have the provided owner or owners.
|  1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76 | <?php declare(strict_types=1);
namespace App\View\Matcher;
use Ibexa\Contracts\Core\Repository\UserService;
use Ibexa\Contracts\Core\Repository\Values\Content\ContentInfo;
use Ibexa\Contracts\Core\Repository\Values\Content\Location;
use Ibexa\Core\MVC\Symfony\Matcher\ContentBased\MatcherInterface;
use Ibexa\Core\MVC\Symfony\View\ContentValueView;
use Ibexa\Core\MVC\Symfony\View\LocationValueView;
use Ibexa\Core\MVC\Symfony\View\View;
use Symfony\Component\DependencyInjection\Exception\InvalidArgumentException;
class Owner implements MatcherInterface
{
    /** @var string[] */
    private array $matchingUserLogins;
    public function __construct(private readonly UserService $userService)
    {
    }
    /**
     * @throws \Ibexa\Contracts\Core\Repository\Exceptions\NotFoundException
     */
    public function matchLocation(Location $location): bool
    {
        return $this->hasOwner($location->getContentInfo());
    }
    /**
     * @throws \Ibexa\Contracts\Core\Repository\Exceptions\NotFoundException
     */
    public function matchContentInfo(ContentInfo $contentInfo): bool
    {
        return $this->hasOwner($contentInfo);
    }
    /**
     * @throws \Ibexa\Contracts\Core\Repository\Exceptions\NotFoundException
     */
    public function match(View $view): ?bool
    {
        if ($view instanceof LocationValueView) {
            return $this->matchLocation($view->getLocation());
        }
        if ($view instanceof ContentValueView) {
            return $this->matchContentInfo($view->getContent()->contentInfo);
        }
        return false;
    }
    /**
     * @throws \Ibexa\Contracts\Core\Repository\Exceptions\NotFoundException
     */
    private function hasOwner(ContentInfo $contentInfo): bool
    {
        $owner = $this->userService->loadUser($contentInfo->ownerId);
        return in_array($owner->login, $this->matchingUserLogins, true);
    }
    /**
     * @param array<string> $matchingConfig
     */
    public function setMatchingConfig($matchingConfig): void
    {
        if (!is_array($matchingConfig)) {
            throw new InvalidArgumentException('App\Owner view matcher configuration has to be an array');
        }
        $this->matchingUserLogins = $matchingConfig;
    }
}
 | 
The matcher checks whether the owner of the current content (by its ContentInfo or location) matches any of the values passed in configuration.
Matcher service
You configure your matcher as a service, tag it ibexa.view.matcher, and associate it with the identifier to use in view rules:
|  | services:
    App\View\Matcher\Owner:
        autowire: true
        tags:
            - { name: ibexa.view.matcher, identifier: App\Owner }
 | 
View configuration
To apply the matcher in view configuration, indicate the matcher by its identifier.
The following configuration uses a special template to render articles owned by the users with provided logins:
|  1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15 | ibexa_design_engine:
    design_list:
        my_design: [ my_theme ]
ibexa:
    system:
        site_group:
            design: my_design
            content_view:
                full:
                    editor_articles:
                        template: '@ibexadesign/full/featured_article.html.twig'
                        match:
                            Identifier\ContentType: article
                            App\Owner: [johndoe, janedoe]
 | 
Note
If you use a matcher that is a service instead of a simple class, tag the service with ibexa.view.matcher.