benjaminhaeberli / kirby-seo
🔎 Minimal SEO plugin for Kirby CMS – for PHP lovers.
Installs: 917
Dependents: 0
Suggesters: 0
Security: 0
Stars: 4
Watchers: 2
Forks: 0
Open Issues: 0
Type:kirby-plugin
pkg:composer/benjaminhaeberli/kirby-seo
Requires
- php: ^8.2
- funkjedi/composer-include-files: ^1.1
- getkirby/cms: ^4.0.1 || ^5.0
- getkirby/composer-installer: ^1.2.1
Requires (Dev)
- laravel/pint: ^1.17
- pestphp/pest: ^4.0
- phpstan/phpstan: ^2.0
- rector/rector: ^2.0
- dev-main
- v1.1.0
- v1.0.0
- v0.3.0
- v0.2.1
- v0.2.0
- v0.1.1
- v0.1.0
- dev-claude/docs-and-deps-upgrade-n8yi9
- dev-claude/update-kirby-seo-plugin-PA75V
- dev-dependabot/github_actions/dependabot/fetch-metadata-2.5.0
- dev-dependabot/github_actions/actions/checkout-6
- dev-dependabot/composer/pestphp/pest-3.7.1
- dev-dependabot/composer/rector/rector-1.2.10
This package is auto-updated.
Last update: 2026-02-02 00:36:35 UTC
README
Minimal SEO plugin for Kirby CMS
Features
- Meta tags (title, description, keywords, robots, canonical URL)
- Open Graph tags (title, type, site name, URL, image, description, locale)
- Twitter Card tags (title, card, site, creator, image, description)
- Automatic image cropping to 1200x630 for social sharing
- Panel blueprints for page-level and site-level SEO fields
- Multi-language support (locale detection)
- Conditional rendering (empty fields produce no HTML output)
Requirements
- PHP ^8.2
- Kirby ^4.0.1 or ^5.0
Installation
composer require benjaminhaeberli/kirby-seo
Usage
1. Add the site-level blueprint
In your site.yml blueprint, include the SEO fields for global settings (site title, OG image, Twitter handles):
# site/blueprints/site.yml tabs: seo: label: SEO fields: seo: fields/seo/site
2. Add the page-level blueprint
In any page blueprint, include the SEO fields (meta title, description, keywords, canonical URL):
# site/blueprints/pages/default.yml tabs: seo: label: SEO fields: seo: fields/seo/meta
3. Render the meta tags
In your header.php snippet (or equivalent), add the SEO snippet inside <head>:
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<?php snippet('seo/meta') ?>
</head>
Output example
The snippet generates the following HTML (tags with empty values are omitted):
<base href="https://example.com"> <title>Page title</title> <meta content="Page description." name="description"> <meta content="keyword1, keyword2" name="keywords"> <meta content="index, follow, noodp" name="robots"> <link href="https://example.com/canonical" rel="canonical"> <meta content="Page title" property="og:title"> <meta content="website" property="og:type"> <meta content="My Site" property="og:site_name"> <meta content="https://example.com/page" property="og:url"> <meta content="https://example.com/image.jpg" property="og:image"> <meta content="Page description." property="og:description"> <meta content="fr_FR" property="og:locale"> <meta content="Page title" name="twitter:title"> <meta content="summary_large_image" name="twitter:card"> <meta content="@site" name="twitter:site"> <meta content="@creator" name="twitter:creator"> <meta content="https://example.com/image.jpg" name="twitter:image"> <meta content="Page description." name="twitter:description">
Panel fields
Site-level (fields/seo/site)
| Field | Type | Description |
|---|---|---|
| Site title | text |
Used as og:site_name |
| Image | files |
OG/Twitter image (cropped to 1200x630) |
| Twitter (Creator) | text |
@handle of the content author |
| Twitter (Site) | text |
@handle of the website |
Page-level (fields/seo/meta)
| Field | Type | Description |
|---|---|---|
| Meta title | text |
<title> and og:title |
| Meta description | textarea |
<meta name="description"> and og:description |
| Keywords | tags |
<meta name="keywords"> |
| Canonical URL | url |
<link rel="canonical"> (defaults to site URL) |
Panel translations
Blueprint labels and help texts use Kirby's plugin translations system. English and French are included.
To add a new language, create a file in translations/ (e.g. de.php) following the same format as en.php, and register it in index.php. Contributions are welcome via pull request.
Development
composer install composer test # Run all checks (rector, pint, phpstan, pest) composer test:unit # Run Pest tests only composer test:types # Run PHPStan only composer lint # Run Pint code formatter composer refacto # Run Rector refactoring