butschster / meta-tags
The most powerful and extendable tools for managing SEO Meta Tags in your Laravel project
Fund package maintenance!
Patreon
Open Collective
Installs: 484 889
Dependents: 5
Suggesters: 0
Security: 0
Stars: 552
Watchers: 14
Forks: 52
Open Issues: 5
Requires
- php: >=8.0
- ext-json: *
- illuminate/config: ^9.0|^10.0|^11.0
- illuminate/console: ^9.0|^10.0|^11.0
- illuminate/contracts: ^9.0|^10.0|^11.0
- illuminate/support: ^9.0|^10.0|^11.0
Requires (Dev)
- mockery/mockery: ^1.0
- phpunit/phpunit: ^9.3|^10.5
- vimeo/psalm: ^5.15
- 3.x-dev
- v3.1.1
- v3.1.0
- v3.0.3
- v3.0.2
- v3.0.1
- v3.0.0
- v3.0-beta1
- 2.x-dev
- v2.3.2
- v2.3.1
- v2.3.0
- v2.2.1
- v2.2
- v2.1
- v2.0
- v1.15
- v1.14
- v1.13
- v1.12
- 1.9
- v1.8
- v1.7
- v1.6.8
- v1.6.7
- v1.6.6
- v1.6.5
- v1.6.4
- v1.6.3
- v1.5.3
- v1.4.2
- v1.4.1
- v1.4
- v1.3
- v1.2
- v1.1
- v1.0
- v0.16
- v0.8
- v0.6.3-beta
- v0.5-beta
- v0.3.8-beta
- v0.3.6-beta
- v0.2-beta
- dev-feature/google-tagmanager
- dev-feature/google-analytics
- dev-feature/yandex-metrika-builder
This package is auto-updated.
Last update: 2024-12-12 22:40:40 UTC
README
The most powerful and extendable tools for managing SEO Meta Tags in your Laravel project
Laravel SEO Meta Tags offers a sophisticated solution for Laravel applications, allowing developers to seamlessly manage header meta tags, CSS, JavaScript, and other relevant tags. Its primary objective is to simplify the process of managing search engine optimization (SEO) tags within your application.
Moreover, its versatile API makes it compatible with frameworks like Inertiajs, VueJS and other JavaScript frameworks.
For any questions or further assistance, please join our official telegram group
Contributing
We enthusiastically invite you to contribute to the package! Whether you've uncovered a bug, have innovative feature suggestions, or wish to contribute in any other capacity, we warmly welcome your participation. Simply open an issue or submit a pull request on our GitHub repository to get started.
Remember, every great developer was once a beginner. Contributing to open source projects is a step in your journey to becoming a better developer. So, don't hesitate to jump in and start contributing!
We appreciate any contributions to help make the package better!
Supercharge Your Development with Buggregator
Pair it with our tool for a more robust development environment. For more information visit buggregator.dev
Features
- Meta Management: Effortlessly set titles, charset, pagination links, and more.
- Styles & Scripts: Organize and place styles and scripts anywhere in your HTML.
- Custom Tags: Make your own tags to suit specific needs.
- Rich Media Integration: Supports both Open Graph & Twitter Cards.
- Analytics Ready: Comes with Google Analytics and Yandex Metrika tracking code support, including a code builder for the latter.
- Site Verification: Supports webmaster tools site verifier tags.
- Package System: Group tags, styles, and scripts into named packages for easy inclusion anywhere.
- Robust Documentation: Clear instructions and guidelines for a seamless setup.
- Thoroughly Tested: Built to ensure reliability and stability.
Requirements
- Laravel version: 9.x to 10.x
- PHP version: 8.0 or higher
Installation and Configuration
- Install the package: Use the following command to install the Meta Tags package in your awesome application.
composer require butschster/meta-tags
- Register the Service Provider: After installing the package, you must register its service provider.
You can do it using the following artisan command:
php artisan meta-tags:install
This command will activate the App\Providers\MetaTagsServiceProvider
and publish the configuration
at config/meta_tags.php
. Within this configuration file, you can set default titles, keywords, descriptions, and other
meta tags which will be automatically inserted into your HTML.
- Verification: Ensure that
App\Providers\MetaTagsServiceProvider
has been added to the providers array in yourconfig/app.php
configuration file. If it hasn't, you'll need to add it manually. Remember, if your application isn't using theApp
namespace, update the provider class name accordingly.
And that's all! Your Laravel project is now equipped to handle SEO meta tags with ease.
Usage
Controller
You can use either Facade \Butschster\Head\Facades\Meta
or \Butschster\Head\Contracts\MetaTags\MetaInterface
in your
controller
use Butschster\Head\MetaTags\MetaInterface; class HomeController extends Controller { public function __contruct( protected MetaInterface $meta ) { } public function index() { $news = News::paginate(); // Prepend title part to the default title $this->meta // Will render "Home page - Default Title" ->prependTitle('Home page') // Will include next, prev, canonical links ->setPaginationLinks($news) // Will change default favicon ->setFavicon('/favicon-index.ico') } } // Or you can use the facade use Butschster\Head\Facades\Meta; class HomeController extends Controller { public function index() { $news = News::paginate(); Meta::prependTitle('Home page') ->setPaginationLinks($news) ->setFavicon('favicon-index.ico'); } }
View
To integrate meta tags into your HTML, simply insert {!! Meta::toHtml() !!}
wherever required.
Note You have two options to insert meta tags:
{!! Meta::toHtml() !!}
or the Blade directive@meta_tags
.
Here's an example of how you can use it in your view:
<!DOCTYPE html> <html lang="en"> <head> {!! Meta::toHtml() !!} </head> ... </html>
Placements
The package provides flexibility to insert meta tags beyond the header. You can define specific placements in your templates.
- To place meta tags in the footer, use:
Meta::footer()->toHtml()
- For custom placements, use:
Meta::placement('placement_name')->toHtml()
- Alternatively, the Blade directive can also be used:
@meta_tags('placement_name')
<body> ... {!! Meta::placement('middle_of_the_page')->toHtml() !!} ... {!! Meta::footer()->toHtml() !!} </body>
Packages
To avoid code repetition and improve code organization, you can group tags, assets, etc., into named packages. This allows for streamlined inclusion of sets of meta tags or assets where needed.
To create a new package:
- Navigate to the Service provider.
- Add your package within
\App\Providers\MetaTagsServiceProvider
.
To define packages, you can use the PackageManager::create
method:
namespace App\Providers; use Butschster\Head\Facades\PackageManager; use Butschster\Head\Packages\Package; use Illuminate\Support\ServiceProvider; class MetaTagsServiceProvider extends ServiceProvider { ... protected function packages() { PackageManager::create('jquery', function(Package $package) { $package->addScript( 'jquery.js', 'https://code.jquery.com/jquery-3.3.1.min.js', ['defer'] ); }); PackageManager::create('calendar', function(Package $package) { $package->requires('jquery'); $package->addScript( 'fullcalendar.js', 'https://cdn.jsdelivr.net/npm/@fullcalendar/core@4.0.1/main.min.js', ['defer'] )->addScript( 'fullcalendar.locales.js', 'https://cdn.jsdelivr.net/npm/@fullcalendar/core@4.0.1/locales-all.min.js', ['defer'] )->addStyle( 'fullcalendar.css', 'https://cdn.jsdelivr.net/npm/@fullcalendar/core@4.0.1/main.min.css' ); }); } ... }
Using Packages in Controllers
When specific packages are required within a controller, you can include them by referencing the package name:
use Butschster\Head\Facades\Meta; class EventsController extends Controller { public function show(Event $event) { // Will include all tags from calendar package Meta::includePackages(['calendar', ...]); } }
Setting Global Packages
For packages that should be included on every page, you can define them in the config/meta_tags.php
:
... 'packages' => [ 'jquery', 'calendar', ... ], ...
And there you have it! With these steps, handling and organizing meta tags and assets becomes a breeze.
Note All methods available in the
Meta
class can also be utilized alongside these package functions.
API
Meta
\Butschster\Head\MetaTags\Meta
- This class implements
Illuminate\Contracts\Support\Htmlable
interface
Methods
Set the main part of meta title
Meta::setTitle('Laravel'); // <title>Laravel</title> // You can specify max length. (By default it gets from config.) Meta::setTitle('Laravel', 4); // <title>Lara...</title>
Prepend title part to main title
Meta::setTitle('Laravel') ->prependTitle('Home page'); // <title>Home page - Laravel</title>
Set the title separator
By default it gets from config
Meta::setTitleSeparator('->') ->setTitle('Laravel') ->prependTitle('Home page'); // <title>Home page -> Laravel</title>
Set the description
Meta::setDescription('Awesome page'); // <meta name="description" content="Awesome page"> // You can specify max length. (By default it gets from config.) Meta::setDescription('Awesome page', 7); // <meta name="description" content="Awesome...">
Set the keywords
Meta::setKeywords('Awesome keywords'); // <meta name="keywords" content="Awesome keywords"> Meta::setKeywords(['Awesome keyword', 'keyword2']); // <meta name="keywords" content="Awesome keyword, keyword2"> // You can specify max length. (By default it gets from config.) Meta::setKeywords(['keyword', 'keyword2'], 10); // <meta name="keywords" content="keyword, key...">
Set the robots
Meta::setRobots('nofollow,noindex'); // <meta name="robots" content="nofollow,noindex">
Set the content type
Meta::setContentType('text/html'); // <meta http-equiv="Content-Type" content="text/html; charset=utf-8"> Meta::setContentType('text/html', 'ISO-8859-1'); // <meta http-equiv="Content-Type" content="text/html; ISO-8859-1">
Set the viewport
Meta::setViewport('width=device-width, initial-scale=1'); // <meta name="viewport" content="width=device-width, initial-scale=1">
Add webmaster tags You can add multiple tags
use Butschster\Head\MetaTags\Entities\Webmaster; // Supported services [google, yandex, pinterest, alexa, bing] Meta::addWebmaster(Webmaster::GOOGLE, 'f+e-Ww4=[Pp4wyEPLdVx4LxTsQ'); // <meta name="google-site-verification" content="f+e-Ww4=[Pp4wyEPLdVx4LxTsQ"> Meta::addWebmaster('yandex', 'f+e-Ww4=[Pp4wyEPLdVx4LxTsQ'); // <meta name="yandex-verification" content="f+e-Ww4=[Pp4wyEPLdVx4LxTsQ"> Meta::addWebmaster('bing', 'f+e-Ww4=[Pp4wyEPLdVx4LxTsQ'); // <meta name="msvalidate.01" content="f+e-Ww4=[Pp4wyEPLdVx4LxTsQ"> Meta::addWebmaster('alexa', 'f+e-Ww4=[Pp4wyEPLdVx4LxTsQ'); // <meta name="alexaVerifyID" content="f+e-Ww4=[Pp4wyEPLdVx4LxTsQ"> Meta::addWebmaster(Webmaster::PINTEREST, 'f+e-Ww4=[Pp4wyEPLdVx4LxTsQ'); // <meta name="p:domain_verify" content="f+e-Ww4=[Pp4wyEPLdVx4LxTsQ">
Set the prev href
Meta::setPrevHref('http://site.com/prev'); // <link rel="prev" href="http://site.com/prev" />
Set the next href
Meta::setNextHref('http://site.com/next'); // <link rel="next" href="http://site.com/next" />
Set the canonical link
Meta::setCanonical('http://site.com'); // <link rel="canonical" href="http://site.com" />
Set canonical link, prev and next from paginator object
$news = \App\News::paginate(); Meta::setPaginationLinks($news); // <link rel="prev" href="http://site.com/prev" /> // <link rel="next" href="http://site.com/next" /> // <link rel="canonical" href="http://site.com" />
Add a hreflang link
Meta::setHrefLang('en', http://site.com/en'); Meta::setHrefLang('ru', http://site.com/ru'); // <link rel="alternate" hreflang="en" href="http://site.com/en" /> // <link rel="alternate" hreflang="ru" href="http://site.com/ru" />
Set the character encoding for the HTML document
Meta::setCharset(); // <meta charset="utf-8"> Meta::setCharset('ISO-8859-1'); // <meta charset="ISO-8859-1">
Set the favicon
Meta::setFavicon('http://site.com/favicon.ico'); // <link rel="icon" type="image/x-icon" href="http://site.com/favicon.ico" /> Meta::setFavicon('http://site.com/favicon.png'); // <link rel="icon" type="image/png" href="http://site.com/favicon.png" /> Meta::setFavicon('http://site.com/favicon.gif'); // <link rel="icon" type="image/gif" href="http://site.com/favicon.gif" /> Meta::setFavicon('http://site.com/favicon.svg'); // <link rel="icon" type="image/svg+xml" href="http://site.com/favicon.svg" /> //You can set additional attributes Meta::setFavicon('http://site.com/favicon.svg', ['sizes' => '16x16', 'type' => 'custom_type']); // <link rel="icon" type="custom_type" href="http://site.com/favicon.svg" sizes="16x16" />
Add a custom link tag
Meta::addLink('apple-touch-icon-precomposed', [ 'href' => 'http://site.com', 'id' => 'id:213' ]); // <link rel="apple-touch-icon-precomposed" href="http://site.com" id="id:213" />
Add a link to a css file
Meta::addStyle('style.css', 'http://site.com/style.css'); // <link media="all" type="text/css" rel="stylesheet" href="http://site.com/style.css" /> // You can override or add attributes Meta::addStyle('style.css', 'http://site.com/style.css', [ 'media' => 'custom', 'defer', 'async' ]); // <link media="custom" type="text/css" rel="stylesheet" href="http://site.com/style.css" defer async />
Add a link to a script file
Meta::addScript('script.js', 'http://site.com/script.js'); // <script src="http://site.com/script.js"></script> // You can override or add attributes Meta::addScript('script.js', 'http://site.com/script.js', ['async', 'defer', 'id' => 'hj2b3424iu2-dfsfsd']); // <script src="http://site.com/script.js" async defer id="hj2b3424iu2-dfsfsd"></script> // You can specify placement. By default, for scripts it's footer Meta::addScript('script.js', 'http://site.com/script.js', [], 'custom_placement'); // <script src="http://site.com/script.js" async defer id="hj2b3424iu2-dfsfsd"></script>
Register a custom tag Our package has a lot of ways of extending. One of them is creating new tags. You are able to create a new class and share it with friends or with the laravel community. You can also create a new pull request if you think that your awesome tag is really useful.
class FacebookPixelTag implements \Butschster\Head\Contracts\MetaTags\Entities\TagInterface { private $pixel_id; public function __construct(string $id) { $this->pixel_id = $id } public function getPlacement(): string { return 'footer' } public function toArray() { return [ 'type' => 'facebook_pixel_tag', 'pixel_id' => $this->pixel_id ]; } public function toHtml() { return '<script type="text/javascript">...</script>' } } Meta::addTag('facebook.pixel', new FacebookPixelTag('42b3h23-34234')); // <script type="text/javascript">...</script>
Register tags from TagsCollection
$tags = new \Butschster\Head\MetaTags\TagsCollection([ ... ]); Meta::registerTags($tags); // You can specify the placement Meta::registerTags($tags, 'footer');
Get a tag by name
Meta::getTag('author');
Remove a tag by name
Meta::removeTag('author');
Add a meta tag
Meta::addMeta('author', [ 'content' => 'butschster', ]); // <meta name="author" content="butschster">
Add the CSRF-token tag
Meta::addCsrfToken(); // <meta name="csrf-token" content="....">
Remove all tags
Meta::reset();
Include required packages
Meta::includePackages('jquery', 'vuejs'); Meta::includePackages(['jquery', 'vuejs']); Will load registered packages with names jquery and vuejs and append tags from there to Meta
Register a new package and register all tags from this package
Meta::setTitle('Current title'); $package = new \Butschster\Head\Packages\Package('custom_package'); $package->setTitle('New title'); // Will replace "Current title" to "New title" after package registration Meta::registerPackage($package);
Replace package with the same name When you replace package with a new one, old tags will be removed
$package = new \Butschster\Head\Packages\Package('custom_package'); $package->setTitle('Custom title'); $newPackage = new \Butschster\Head\Packages\Package('custom_package'); $newPackage->setTitle('New title'); // Will replace "Current title" to "New title" after package registration Meta::registerPackage($package); Meta::replacePackage($newPackage);
Remove package by name
$package = new \Butschster\Head\Packages\Package('custom_package'); $package->setTitle('Custom title'); // Will replace "Current title" to "New title" after package registration Meta::registerPackage($package); Meta::removePackage('custom_package');
Get package by name
$package = new \Butschster\Head\Packages\Package('custom_package'); $package->setTitle('Custom title'); // Will replace "Current title" to "New title" after package registration Meta::registerPackage($package); $package = Meta::getPackage('custom_package');
Using meta interfaces
A package has different interfaces which help you set meta tags from your objects
Seo tags
namespace App; use Butschster\Head\Contracts\MetaTags\SeoMetaTagsInterface; use Butschster\Head\Contracts\MetaTags\RobotsTagsInterface; class Page extends Model implements SeoMetaTagsInterface, RobotsTagsInterface { public function getTitle(): ?string { return $this->title; } public function getDescription(): ?string { return $this->description; } public function getKeywords() { return $this->keywords; } public function getRobots(): ?string { return 'noindex, nofollow'; } } // Controller use Butschster\Head\Facades\Meta; class PageController extends Controller { public function show(\App\Page $page) { Meta::setMetaFrom($page); } }
Meta extending
Meta object contains Macroable
trait and you can extend it!
For example
//Service Provider Meta::macro('registerSeoMetaTagsForPage', function (\App\Page $page) { $this ->prependTitle($page->title) ->setKeywords($page->meta_keywords) ->setDescription($page->meta_description); }); // Controller use Butschster\Head\Facades\Meta; class PageController extends Controller { public function show(\App\Page $page) { Meta::registerSeoMetaTagsForPage($page); } }
A little bit infirmation about macroable trait https://unnikked.ga/understanding-the-laravel-macroable-trait-dab051f09172
Meta tags placements
By default, tags place to head placement. You can specify your own placement and use their all available methods.
Meta::placement('twitter.meta') ->addMeta('twitter:card', [ 'content' => 'summary_large_image', ]) ->includePackages('twitter') // There is the method for footer placement Meta::footer()->... // View <body> ... @meta_tags('twitter.meta') ... @meta_tags('footer') </body>
Package
A package object has the same methods as Meta object. You can use it for extending and creating custom tags sets.
\Butschster\Head\Packages\Package
- This class extend
Butschster\Head\MetaTags\Meta
class - This class implements
Illuminate\Contracts\Support\Htmlable
interface
Create a new package and register it in PackageManager
$package = new \Butschster\Head\Packages\Package('jquery'); PackageManager::register($package); // or PackageManager::create('jquery', function($package) { ... });
Get the name of the package
$package = new \Butschster\Head\Packages\Package('jquery'); $package->getName(); // jquery
Add a link to a css file
$package = new \Butschster\Head\Packages\Package('jquery'); $package->addStyle('style.css', 'http://site.com/style.css'); // <link media="all" type="text/css" rel="stylesheet" href="http://site.com/style.css" /> // You can change or add attributes $package->addStyle('style.css', 'http://site.com/style.css', [ 'media' => 'custom', 'defer', 'async' ]); // <link media="custom" type="text/css" rel="stylesheet" href="http://site.com/style.css" defer async />
Add a link to a script file
$package = new \Butschster\Head\Packages\Package('jquery'); $package->addScript('script.js', 'http://site.com/script.js'); // <script src="http://site.com/script.js"></script> // You can change or add attributes $package->addScript('script.js', 'http://site.com/script.js', ['async', 'defer', 'id' => 'hj2b3424iu2-dfsfsd']); // <script src="http://site.com/script.js" async defer id="hj2b3424iu2-dfsfsd"></script> // You can placement. By default it's footer $package->addScript('script.js', 'http://site.com/script.js', [], 'custom_placement'); // <script src="http://site.com/script.js" async defer id="hj2b3424iu2-dfsfsd"></script> Meta::includePackages('jquery')->placement('custom_placement')->toHtml();
Available packages
OpenGraphPackage
Butschster\Head\Packages\Entities\OpenGraphPackage
You can use this package for managing OpenGraph meta tags
$og = new Butschster\Head\Packages\Entities\OpenGraphPackage('some_name'); $og->setType('website') ->setSiteName('My awesome site') ->setTitle('Post title'); // You can render itself $og->toHtml(); // <meta name="og:type" content="website"> // <meta name="og:site_name" content="My awesome site"> // <meta name="og:title" content="Post title"> // Or just register this package in Meta class and it will be rendered automatically Meta::registerPackage($og);
setType Set the type of your object, e.g., "video.movie".
$og->setType('website'); // <meta name="og:type" content="website">
setTitle Set the title of your object as it should appear within the graph, e.g., "The Rock".
$og->setTitle('Post title'); // <meta name="og:title" content="Post title">
setDescription Set the description
$og->setDescription('View the album on Flickr.'); // <meta name="og:description" content="View the album on Flickr.">
setSiteName Set the site name
$og->setSiteName('My awesome site'); // <meta name="og:site_name" content="My awesome site">
setUrl Set the canonical URL of your object that will be used as its permanent ID in the graph.
$og->setUrl('https://site.com'); // <meta name="og:url" content="https://site.com">
setLocale Set the locale these tags are marked up in. Of the format language_TERRITORY
$og->setLocale('en_US'); // <meta name="og:locale" content="en_US">
addAlternateLocale
$og->addAlternateLocale('en_US', 'ru_RU'); // <meta name="og:locale:alternate" content="en_US"> // <meta name="og:locale:alternate" content="ru_RU">
addImage Add an image URL which should represent your object within the graph.
$og->addImage('http://site.com'); // <meta name="og:image" content="http://site.com"> // You can pass properties $og->addImage('http://site.com', [ 'secure_url' => 'https://site.com', 'type' => 'image/png' ]); // <meta name="og:image" content="http://site.com"> // <meta name="og:image:secure_url" content="https://site.com"> // <meta name="og:image:type" content="image/png">
addVideo Add an image URL which should represent your object within the graph.
$og->addVideo('http://site.com'); // <meta name="og:video" content="http://site.com"> // You can pass properties $og->addVideo('http://site.com', [ 'secure_url' => 'https://site.com', 'type' => 'application/x-shockwave-flash' ]); // <meta name="og:video" content="http://site.com"> // <meta name="og:video:secure_url" content="https://site.com"> // <meta name="og:video:type" content="application/x-shockwave-flash">
TwitterCardPackage
Butschster\Head\Packages\Entities\TwitterCardPackage
You can use this package for managing Twitter card meta tags
$card = new Butschster\Head\Packages\Entities\TwitterCardPackage('some_name'); $card->setType('summary') ->setSite('@username') ->setTitle('Post title'); // You can render itself $card->toHtml(); // <meta name="twitter:card" content="summary"> // <meta name="twitter:site" content="@username"> // <meta name="twitter:title" content="Post title"> // Or just register this package in Meta class and it will be rendered automatically Meta::registerPackage($card);
setType Set the type of the card: summary, summary_large_image, player, app
$card->setType('summary'); // <meta name="twitter:card" content="summary">
setSite Set the @username for the website used in the card footer.
$card->setSite('@username'); // <meta name="twitter:site" content="@username">
setCreator Set the @username for the content creator / author.
$card->setCreator('@username'); // <meta name="twitter:creator" content="@username">
setTitle Set the title
$card->setTitle('Post title'); // <meta name="twitter:title" content="Post title">
setDescription Set the description
$card->setDescription('View the album on Flickr.'); // <meta name="twitter:title" content="View the album on Flickr.">
setImage Set an image for cards that are of type summary
or summary_large_image
$card->setImage('https://site.com'); // <meta name="twitter:image" content="https://site.com">
setVideo Set a video to cards that are of type player
$card->setVideo('https://site.com/video.mp4', ['width' => 1920, 'height' => 1280]); // <meta name="twitter:player" content="https://site.com/video.mp4"> // <meta name="twitter:player:width" content="1920"> // <meta name="twitter:player:height" content="1280">
addMeta Set a custom meta tag
$card->addMeta('image:alt', 'Picture of Pavel Buchnev'); // <meta name="twitter:image:alt" content="Picture of Pavel Buchnev">
PackageManager API
Package manager provide a store for your packages or presets. You can get them by name.
Create a new package
PackageManager::create('jquery', function($package) { ... });
Register a new package
$package = new \Butschster\Head\Packages\Package('jquery'); PackageManager::register($package);
Get all registered packages
PackageManager::getPackages(): array;
Get registered package by name
PackageManager::create('jquery', function($package) { ... }); PackageManager::getPackage('jquery'); // Will return the registered pacakge or null;
Dependencies
A package can have dependencies.
PackageManager::create('jquery', function($package) { $package->addScript('jquery.js', 'http://site.com/jquery.min.js'); }); PackageManager::create('bootstrap4', function($package) { $package->requires('jquery'); $package->addScript('bootstrap4.js', 'http://site.com/bootstrap4.min.js'); $package->addStyle('bootstrap4.css', 'http://site.com/bootstrap4.min.css'); }); Meta::includePackages('bootstrap4'); // Will load jquery package also // Head // <link media="all" type="text/css" rel="stylesheet" href="http://site.com/bootstrap4.min.css" /> // Footer // <script src="http://site.com/jquery.min.js"></script> // <script src="http://site.com/bootstrap4.min.js"></script>
Helper classes
Tag
\Butschster\Head\MetaTags\Entities\Tag
Create a new tag
$tag = new \Butschster\Head\MetaTags\Entities\Tag('meta', [ 'name' => 'author', 'content' => 'butschster' ]); // or $tag = \Butschster\Head\MetaTags\Entities\Tag::meta([ 'name' => 'author', 'content' => 'butschster' ]); $tag->toHtml(); // <meta name="author" content="butschster"> // Closed tag $tag = new \Butschster\Head\MetaTags\Entities\Tag('link', [ 'rel' => 'favicon', 'href' => 'http://site.com' ], true); // or $tag = \Butschster\Head\MetaTags\Entities\Tag::link([ 'rel' => 'favicon', 'href' => 'http://site.com' ]); $tag->toHtml(); // <link rel="favicon" href="http://site.com" /> // Tag with anonymous function $tag = new \Butschster\Head\MetaTags\Entities\Tag('meta', [ 'name' => 'csrf-token', 'content' => function () { return Session::token(); } ]); $tag->toHtml(); // <meta name="csrf-token" content="8760b1d530d60d2cba6fe81cb12d67c0">
Set the placement
$tag = new \Butschster\Head\MetaTags\Entities\Tag(...); $tag->setPlacement('footer');
Get the placement
$tag = new \Butschster\Head\MetaTags\Entities\Tag(...); $tag->getPlacement() // Will return specified placement;
Set visibility condition
$tag = new \Butschster\Head\MetaTags\Entities\Tag(...); $tag->visibleWhen(function () { return Request::ip() === '127.0.0.1'; });
Title
\Butschster\Head\MetaTags\Entities\Title
This class is responsible for title generation
Set the default part of the title
$title = new \Butschster\Head\MetaTags\Entities\Title(); $title->setTitle('Laravel'); $title->toHtml(); // <title>Laravel</title>
Prepend a new part of title
$title = new \Butschster\Head\MetaTags\Entities\Title(); $title->setTitle('Laravel'); $title->prepend('Index page'); $title->toHtml(); // <title>Index page | Laravel</title>
Change default title parts separator
$title = new \Butschster\Head\MetaTags\Entities\Title(); $title->setTitle('Laravel'); $title->prepend('Index page'); $title->setSeparator('-'); $title->toHtml(); // <title>Index page - Laravel</title>
Specify max length (default length is 255)
$title = new \Butschster\Head\MetaTags\Entities\Title('Lorem Ipsum is simply dummy text of the printing and typesetting'); $title->setMaxLength(20); $title->toHtml(); // <title>Lorem Ipsum is simpl...</title>
Description
\Butschster\Head\MetaTags\Entities\Description
Script
\Butschster\Head\MetaTags\Entities\Script
This class is responsible for script links generation
use Butschster\Head\MetaTags\Entities\Script; $script = new Script('jquery.js', 'http://site.com/script.js', ['defer', 'async']); $script->toHtml(); // <script src="http://site.com/script.js" defer async></script> Meta::addTag($script);
Style
\Butschster\Head\MetaTags\Entities\Style
This class is responsible for css links generation
use Butschster\Head\MetaTags\Entities\Style; $style = new Style('style.css', 'http://site.com/style.css'); $style->toHtml(); // <link media="all" type="text/css" rel="stylesheet" href="http://site.com/style.css" /> Meta::addTag($style);
Favicon
\Butschster\Head\MetaTags\Entities\Favicon
This class is responsible for favicon link generation
Comment
\Butschster\Head\MetaTags\Entities\Comment
This class is a wrapper for tags, that allows to add comments to your tags
use Butschster\Head\MetaTags\Entities\Comment; use Butschster\Head\MetaTags\Entities\Favicon; $favicon = new Favicon('http://site.com/favicon.ico'); $comment = new Comment($favicon, 'Favicon'); Meta::addTag('favicon', $comment); // Will render <!-- Favicon --> <link rel="icon" type="image/x-icon" href="http://site.com/favicon.ico" /> <!-- /Favicon -->
Conditional comment
\Butschster\Head\MetaTags\Entities\ConditionalComment
This class is a wrapper for tags, that allows to add conditional comments to your tags
use Butschster\Head\MetaTags\Entities\ConditionalComment; use Butschster\Head\MetaTags\Entities\Favicon; $favicon = new Favicon('http://site.com/favicon.ico'); $comment = new ConditionalComment($favicon, 'IE 6'); Meta::addTag('favicon', $comment); <!--[if IE 6]> <link rel="icon" type="image/x-icon" href="http://site.com/favicon.ico" /> <![endif]-->
Google Analytics
Has header placement!
use Butschster\Head\MetaTags\Entities\GoogleAnalytics; $script = new GoogleAnalytics('UA-12345678-1'); Meta::addTag('google.analytics', $script);
Will return
<script> (function (i, s, o, g, r, a, m) { i['GoogleAnalyticsObject'] = r; i[r] = i[r] || function () { (i[r].q = i[r].q || []).push(arguments) }, i[r].l = 1 * new Date(); a = s.createElement(o), m = s.getElementsByTagName(o)[0]; a.async = 1; a.src = g; m.parentNode.insertBefore(a, m) })(window, document, 'script', 'https://www.google-analytics.com/analytics.js', 'ga'); ga('create', 'UA-12345678-1', 'auto'); ga('send', 'pageview'); </script>
Google TagManager
Has header placement!
use Butschster\Head\MetaTags\Entities\GoogleTagManager; $script = new GoogleTagManager('UA-12345678-1'); Meta::addTag('google.tagmanager', $script);
Will return
<script async src="https://www.googletagmanager.com/gtag/js?id=UA-12345678-1"></script> <script> window.dataLayer = window.dataLayer || []; function gtag() { dataLayer.push(arguments); } gtag('js', new Date()); gtag('config', 'UA-12345678-1'); </script>
Yandex Metrika
Has footer placement!
use \Butschster\Head\MetaTags\Entities\YandexMetrika; $script = new YandexMetrika('20925319'); Meta::addTag('yandex.metrika', $script);
Will return
<script type="text/javascript"> (function (m, e, t, r, i, k, a) { m[i] = m[i] || function () { (m[i].a = m[i].a || []).push(arguments) }; m[i].l = 1 * new Date(); k = e.createElement(t), a = e.getElementsByTagName(t)[0], k.async = 1, k.src = r, a.parentNode.insertBefore(k, a) }) (window, document, "script", "https://mc.yandex.ru/metrika/tag.js", "ym"); ym(20925319, "init", {clickmap: true, trackLinks: true, accurateTrackBounce: true, webvisor: true}); </script> <noscript> <div><img src="https://mc.yandex.ru/watch/20925319" style="position:absolute; left:-9999px;" alt=""/></div> </noscript>
Change clickmap setting
$script = new YandexMetrika('20925319'); $script->clickmap(bool);
Change webvisor setting
$script = new YandexMetrika('20925319'); $script->webvisor(bool);
Change trackLinks setting
$script = new YandexMetrika('20925319'); $script->trackLinks(bool);
Change accurateTrackBounce setting
$script = new YandexMetrika('20925319'); $script->accurateTrackBounce(bool);
Change trackHash setting
$script = new YandexMetrika('20925319'); $script->trackHash(bool);
Change eCommerce setting
$script = new YandexMetrika('20925319'); $script->eCommerce(string $containerName);
Use alternate CDN
$script = new YandexMetrika('20925319'); $script->useCDN();
Disable noscript tag
$script = new YandexMetrika('20925319'); $script->disableNoScript();
Javascript variables
use \Butschster\Head\MetaTags\Entities\JavascriptVariables; $variables = new JavascriptVariables([ 'string' => 'Hello world', 'number' => 4815162342, 'bool' => true, 'nullable' => null ]); // you can put new variable $variables->put('array', ['jquery', 'vuejs']); Meta::addTag('variables', $variables);
Will return
<script> window.array = ["jquery", "vuejs"]; window.string = 'Hello world'; window.number = 4815162342; window.bool = true; window.nullable = null; </script>
You can change namespace
use \Butschster\Head\MetaTags\Entities\JavascriptVariables; $variables = new JavascriptVariables([ 'string' => 'Hello world', 'number' => 4815162342, ], 'custom');
Will return
<script> window.custom = window.custom || {}; custom.string = 'Hello world'; custom.number = 4815162342; </script>
Use cases
Multiple favicons
You can use your own package for that.
At first create your package in the MetaTagsServiceProvider App\Providers\MetaTagsServiceProvider
namespace App\Providers; use Butschster\Head\MetaTags\Entities\Favicon; use Butschster\Head\MetaTags\Entities\ConditionalComment; use Butschster\Head\Facades\PackageManager; use Butschster\Head\Packages\Package; class MetaTagsServiceProvider extends ServiceProvider { ... protected function packages() { PackageManager::create('favicons', function(Package $package) { $sizes = ['16x16', '32x32', '64x64']; foreach ($sizes as $size) { $package->addTag( 'favicon.'.$size, new Favicon('http://site.com/favicon-'.$size.'.png', [ 'sizes' => $size ]) ); } $package->addTag('favicon.ie', new ConditionalComment( new Favicon('http://site.com/favicon-ie.png'), 'IE gt 6' )); }); } ... }
And then append this package into packages
section in config/meta_tags.php
:
... 'packages' => [ 'favicons' ], ...
And the every page you will see in the head seaction something like that:
<head> ... <title>...</title> <link rel="icon" type="image/png" href="http://site.com/favicon-16x16.png" sizes="16x16"/> <link rel="icon" type="image/png" href="http://site.com/favicon-32x32.png" sizes="32x32"/> <link rel="icon" type="image/png" href="http://site.com/favicon-64x64.png" sizes="64x64"/> <!--[if IE gt 6]> <link rel="icon" type="image/png" href="http://site.com/favicon-ie.png" /> <![endif]--> ... </head>
Extending
If you want to extend Meta class you can do it in the App\Providers\MetaTagsServiceProvider
. Just
override registerMeta
method.
namespace App\Providers; use Butschster\Head\MetaTags\Meta; use Butschster\Head\Contracts\MetaTags\MetaInterface; use Butschster\Head\Contracts\Packages\ManagerInterface; class MetaTagsServiceProvider { ... protected function registerMeta(): void { $this->app->singleton(MetaInterface::class, function () { $meta = new Meta( $this->app[ManagerInterface::class], $this->app['config'] ); // It just an imagination, you can automatically // add favicon if it exists if (file_exists(public_path('favicon.ico'))) { $meta->setFavicon('/favicon.ico'); } $meta->includePackages('fonts', 'assets'); // This method gets default values from config and creates tags // If you don't want to use default values just remove it. $meta->initialize(); return $meta; }); } ... }
Using with inertiajs or vue-meta
You can easily convert Meta object to array ant the use values in your Js project
$meta = Meta::setTitle('Laravel') ->setDescription('Awesome page') ->setKeywords('php, laravel, ...'); dd($meta->toArray()); [ 'head' => [ [ 'tag' => 'title', 'content' => 'Laravel', ], [ 'name' => 'description', 'content' => 'Awesome page', 'type' => 'tag', 'tag' => 'meta', ], [ 'name' => 'keywords', 'content' => 'php, laravel, ...', 'type' => 'tag', 'tag' => 'meta', ], ] ]
Example for inertiaJs
use Inertia\Inertia; use Butschster\Head\MetaTags\MetaInterface; use Butschster\Head\Hydrator\VueMetaHydrator; class EventsController extends Controller { protected $meta; public function __contruct(MetaInterface $meta) { $this->meta = $meta; } public function show(Event $event, VueMetaHydrator $hydrator) { $this->meta->setTitle('Laravel') ->setDescription('Awesome page') ->setKeywords('php, laravel, ...'); return Inertia::render('Event/Show', [ 'event' => $event->only('id', 'title', 'start_date', 'description'), // this.$page.props.meta... 'meta' => $hydrator->hydrate($this->meta) ]); } }