lukaswhite / meta-tags
A PHP class for managing meta tags
Requires
- lukaswhite/html-element: dev-master
Requires (Dev)
- phpunit/php-code-coverage: ^6.0
- phpunit/phpunit: 7.0
This package is auto-updated.
Last update: 2024-11-29 06:00:58 UTC
README
This library helps you manage meta tags (including Open Graph) for your website or web application.
Chances are, many of your meta tags are generated programmatically, for example from a database. Most notably the page title, but perhaps also the description and keywords.
Perhaps you make these available as data to your views, then build the tags there. What this library does is allow you to build them inside of your application (in a controller, for example), then output them in your views really easily.
So, instead of something like this (I'm using Blade syntax here):
<!DOCTYPE html> <html lang="en"> <head> <meta charset="utf-8" /> <title>{{ $title }}</title> <meta name="og:title" content="{{ $title }}" /> <meta name="description" content="{{ $description }}" /> <meta name="og:description" content="{{ $description }}" /> <meta name="keywords" content="{{ implode( ', ', $keywords ) }}" /> <meta name="robots" content="index,nofollow" /> <link rel="canonical" content="{{ $url }}" /> </head>
This library allows you to do this:
$meta = new MetaTags( ); $meta ->charSet( 'utf-8' ) ->title( $title ) ->description( $description ) ->keywords( $keywords ) ->robotsShouldIndexButNotFollow( ) ->canonical( $url );
In fact, if you're using UTF-8 then it's the default anyway, so that
->charSet( )
call is redundant.
Then in your view:
<!DOCTYPE html> <html lang="en"> <head> {!! $meta !!} </head>
If you're using Laravel, you could do something like this:
{!! app( )->make( MetaTags::class )->render( ) !!}
In the example above, we've registered the
MetaTags
class as singleton with the service container. More on that later.
So what are the benefits?
- A maximum of one variable to pass to your views
- No hand-coding the tags themselves
- No confusion as to when to use
name
vsproperty
attributes - It makes it harder to forget things like setting the character set
- Set the title and description, and the corresponding Open Graph meta tags are set for you
- It (hopefully!) helps as a reminder for important, but often overlooked things like canonicalization
- It exposes some of the lesser-known meta tags, like the one aimed directly at Googlebots
- It keeps your layouts nice and tidy
Installation
composer require lukaswhite/meta-tags
First Steps
Create an instance as follows:
use Lukaswhite\MetaTags\MetaTags; $meta = new MetaTags( );
I'd recommend creating it as a singleton, then popping it in a service container or similar.
For example, if you're using Laravel:
$this->app->singleton( MetaTags::class, function( $app ) { return new MetaTags( ); } );
Then, whenever you want to set your meta tags, for example in a controller, then you can get the instance like this:
$meta = app( )->make( MetaTags::class );
Standard Metadata
Setting the Title
$meta->title( 'My Website' );
By default, this will add the following:
<title>My Website</title> <meta property="og:title" content="My Website" />
You'll notice that it's added an Open Graph tag with the title by default; if you don't want that behavior:
$meta->includeOpenGraph( false ) ->title( 'My Website' );
The result:
<title>My Website</title>
Setting the Description
$meta->description( 'A description of the website' );
By default, this will add the following:
<meta name="description" content="A description of the website" /> <meta property="og:description" content="A description of the website" />
You'll notice that it's added an Open Graph tag with the description by default; if you don't want that behavior:
$meta->includeOpenGraph( false ) ->description( 'A description of the website' );
The result:
<meta name="description" content="A description of the website" />
Setting the Keywords
$meta->keywords( 'PHP', 'meta tags', 'SEO' );
The result:
<meta name="keywords" content="PHP, meta tags, SEO" />
Setting the URL
Set the url like this:
$meta->url( 'http://example.com' );
This will output the following:
<meta name="url" content="http://example.com" />
Setting the Character Set
The library will automatically add the character set to the meta tags, defaulting to UTF-8. So this is added automatically:
<meta charset="utf-8" />
To set the character set to something different:
$meta->charSet( 'ISO-8859-1' );
The result:
<meta charset="ISO-8859-1" />
Should, for any reason, you not wish this to be added then you can simply set it to null
:
$meta->charSet( null );
Canonical URLs
To set the canonical link meta tag:
$meta->canonical( 'http://example.com' );
This results in the following:
<link rel="canonical" content="http://example.com" />
Pagination
If you're rendering a page that lists something and it uses pagination, then it can be helpful to add links to the next and / or previous pages as links. That's to say, a link
tag with rel=prev
or rel=next
.
Suppose we're currently on page four:
$meta->previousPage( 'http://example.com/articles/page/3' ) ->nextPage( 'http://example.com/articles/page/5' );
You can also set the first and last pages:
$meta ->firstPage( 'http://example.com/articles/page/1' ) ->lastPage( 'http://example.com/articles/page/10' );
Feeds
RSS
If you have an RSS feed associated with a website or category, it's a good idea to create a link to it.
Without a title:
$meta->addRssFeed( 'http://example.com/feed.rss' );
Or with:
$meta->addRssFeed( 'http://example.com/feed.rss', 'My Website RSS Feed' );
Atom
If you have an Atom feed associated with a website or category, it's a good idea to create a link to it.
Without a title:
$meta->addAtomFeed( 'http://example.com/feed.xml' );
Or with:
$meta->addAtomFeed( 'http://example.com/feed.xml', 'My Website Atom Feed' );
Robots
By default robots will index and follow; if you want to tell them not to do one or both you can use one of the following methods:
$meta->robotsShouldIndexButNotFollow( ); // or $meta->robotsShouldFollowButNotIndex( ); // or $meta->robotsShouldNotIndexNorFollow( );
Googlebot
There are a couple of additional requests you can make from the Googlebot, in addition to the noindex
or nofollow
clauses.
To prevent a text snippet or video preview from being shown in the search results:
$meta->googleShouldNotIncludeSnippets( );
To prevent Google from showing the Cached link for a page:
$meta->googleShouldNotIncludeSnippets( );
To specify that you do not want your page to appear as the referring page for an image that appears in Google search results:
$meta->googleShouldNotShowPageAsReferringPageForImageSearchResults( );
Apologies for the ridiculously long method name, better suggestions welcome!
To request that Google stop crawling a page after a specified date:
$meta->googleShouldStopCrawlingAfter( new \DateTime( '2019-10-23' ) );
Performance-related Meta Tags
There are a number of meta tags that are performance-related; specifically things like pre-fetching and pre-rendering.
Pre-fetching
You can add a link to the meta tags to tell the browser to pre-fetch a resource:
$meta->addPrefetchLink( '/scripts.js' );
You can also add a hint as to the type of resource:
$meta->addPrefetchLink( '/scripts.js', 'script' );
Preloading
You can add a link to the meta tags to tell the browser to preload a resource:
$meta->addPreloadLink( '/scripts.js' );
You can also add a hint as to the type of resource:
$meta->addPreloadLink( '/scripts.js', 'script' );
DNS Pre-fetching
DNS pre-fetching is the process of initiating the dns resolution of each domain where we have hosted resources, before the browser makes a request for them, with the goal to save the DNS resolution time when the resource is requested. (source)
To do so:
$meta->addPreconnect( '//cdn.example.com' );
Pre-renderng
To ask the browser to pre-render content:
$meta->addPrerender( '//example.com/landing-page.html' );
http-equiv Meta Tags
There's a catgeory of meta tags that look like this:
<meta http-equiv="Pragma" content="no-cache">
They're essentially the HTML equivalent of HTTP headers.
To add one:
$meta->httpEquiv( 'Pragma', 'no-cache' );
To tell browsers that you don't want a page to be cached, you can either set the relevant tags manually, or you can do this:
$meta->tellBrowsersNotToCache( );
As per the advice on this page, since not all all client browsers and caching devices (e.g. proxy servers) are known to successfully implement all no-caching options, this method includes multiple no-caching options; specfically it adds the following meta tags:
<meta http-equiv="pragma" content="no-cache" /> <meta http-equiv="cache-control" content="max-age=0" /> <meta http-equiv="cache-control" content="no-cache" /> <meta http-equiv="cache-control" content="no-store" />
Custom Meta Tags
Custom meta tags can be useful for injecting data into a page that you can later access via JavaScript. Laravel does this for CSRF tokens, for example, or you may wish to use this approach for injecting an application ID or public key.
Here are some examples:
$meta->custom( 'google-maps', 'your-google-maps-key' ); $meta->custom( 'mixpanel', getenv( 'MIXPANEL_APP_ID' ) ); $meta->custom( 'csrf-token', csrf_token( ), 'name' );
Note that last one; Laravel outputs the CSRF meta tag like this:
<meta name="csrf-token" content="the-token" />
Whereas the Google Maps example above will use the more common property
attribute, i.e.:
<meta property="google-maps" content="your-google-maps-key" />
Open Graph Tags
The library makes adding Open Graph meta tags easy.
Enabling or Disabling
Open Graph tags are enabled by default; that includes some which get added automatically. If you wish to disable this:
$meta->includeOpenGraph( false );
Basic Tags
There are some basic — and very important — Open Graph tags that have "standard" equivalents. Notably the title, description and URL. These get automatically added as Open Graph tags in addition to "standard" tags when you specify them with the corresponding methods.
In other words, if you do this:
$meta ->title( 'My Website' ) ->description( 'A description of the website' ) ->url( 'http://example.com' );
You'll get the following:
<title>My Website</title> <meta property="og:title" content="My Website" /> <meta name="description" content="A description of the website" /> <meta property="og:description" content="A description of the website" /> <meta name="url" content="http://example.com" />
Defining the Type
By default the required og:type
meta tag is added, which by default is a website. I.e.:
<meta property="og:type" content="website" />
You can override this with the type( )
method, for example:
$meta->type( 'article' );
You can also use one of the provided constants:
$meta->type( MetaTags::OG_TYPE_ARTICLE );
You can also set any attributes at the same time, for example:
$meta->type( MetaTags::OG_TYPE_BOOK, [ 'isbn' => 'ISBN-123-1234', ] ); // or $meta->type( MetaTags::OG_TYPE_PROFILE, [ 'first_name' => 'Joe', 'last_name' => 'Bloggs', 'username' => 'joebloggs', ] );
Passing an array of attributes in this way isn't the preferred way to do this; we'll look at other ways later.
Or, call one of the following methods:
$meta->isArticle( ); $meta->isBook( ); $meta->isProfile( ); $meta->isBusiness( );
Setting the Locale
To set the locale, for example for localization with Facebook:
$meta->locale( 'en_GB' );
Setting the URL
To set the Open Graph URL — in other words, og:url
— simply do this:
$meta->ogUrl( 'https://example.com' );
Other Open Graph Tags
You can set any Open Graph tag using the openGraph( )
method:
$meta->openGraph( 'determiner', 'an' ); $meta->openGraph( 'updated_time', new \DateTime( '2018-10-12' ) ); $meta->openGraph( 'restrictions:age', 18 );
Should you be setting an Open Graph tag that can appear multiple times, pass true
as the third argument. This essentially allows you to "overwrite" Open Graph tags; which might useful if, for example, you need to override information somewhere in your application.
Images
Images are important for social networks, for example when your site is shared via Facebook, or used as the basis of a Twitter Card (more on those later).
To add an image:
$meta->addImage( ( new \Lukaswhite\MetaTags\Entities\Image( ) ) ->setUrl( 'http://example.com/image.jpeg' ) ->setSecureUrl( 'https://example.com/image.jpeg' ) ->setType( 'image/jpeg' ) ->setWidth( 500 ) ->setHeight( 300 ) ->setAlt( 'Example image' ) );
If you'd rather use your own image class, simply implement the Image
contract included in this package.
It's named
addImage( )
rather than, say,setImage( )
because the Open Graph protocol allows you to add multiple images.
Videos
To add a video:
$meta->addVideo( ( new \Lukaswhite\MetaTags\Entities\Video( ) ) ->setUrl( 'http://example.com/movie.swf' ) ->setSecureUrl( 'https://example.com/movie.swf' ) ->setType( 'application/x-shockwave-flash' ) ->setWidth( 500 ) ->setHeight( 300 ) ->setImage( 'http://example.com/movie-image.jpg' ) );
If you'd rather use your own image class, simply implement the Video
contract included in this package.
It's named
addVideo( )
rather than, say,setVideo( )
because the Open Graph protocol allows you to add multiple videos.
Audio
To add audio:
$meta->addAudio( ( new \Lukaswhite\MetaTags\Entities\Audio( ) ) ->setUrl( 'http://example.com/audio.mp3' ) ->setSecureUrl( 'https://example.com/audio.mp3' ) ->setType( 'audio/mp3' ) );
If you'd rather use your own image class, simply implement the Video
contract included in this package.
Profiles
If a web page represents a user profile, for example in a social network, then you can do this:
$meta->profile( ( new \Lukaswhite\MetaTags\Entities\Profile( ) ) ->setFirstName( 'Joe' ) ->setLastName( 'Bloggs' ) ->setUsername( 'joebloggs' ) ->setGender( \Lukaswhite\MetaTags\Contracts\Profile::MALE ) );
If you'd rather use your own profile class, simply implement the Profile
contract included in this package.
Books
If a web page represents a book:
$meta->book( ( new \Lukaswhite\MetaTags\Entities\Book( ) ) ->setReleaseDate( new \DateTime( '2018-04-03 15:20' ) ) ->setAuthor( 'Joe Bloggs' ) ->setIsbn( 'ISBN-123-1234' ) ->setTag( [ 'PHP' ] ) );
If you'd rather use your own book class, simply implement the Book
contract included in this package.
Note that since a book may have multiple authors, this is also permitted:
$book->setAuthor( [ 'Joe Bloggs', 'Harry Black' ] );
Contact Data
You can add contact data such as an address, e-mail address or telephone number to the Open Graph meta tags; this might be appropriate for a business listing site, for example.
Here's one way to do it:
$meta->contactData( ( new \Lukaswhite\MetaTags\Entities\ContactData( ) ) ->setStreetAddress( '1601 S California Ave' ) ->setLocality( 'Palo Alto' ) ->setRegion( 'CA' ) ->setPostalCode( '94304' ) ->setCountryName( 'USA' ) ->setEmail( 'me@example.com' ) ->setPhone('650-123-4567') ->setFaxNumber('+1-415-123-4567') ->setWebsite( 'http://example.com' ) );
This will add the following meta tags:
<meta name="og:street-address" content="1601 S California Ave" /> <meta name="og:locality" content="Palo Alto" /> <meta name="og:region" content="CA" /> <meta name="og:postal-code" content="94304" /> <meta name="og:country-name" content="USA" /> <meta name="og:email" content="me@example.com" /> <meta name="og:phone_number" content="650-123-4567" /> <meta name="og:fax_number" content="http://example.com" />
Alternatively, you can use your own class by implementing the ContactData
interface:
class Business implements ContactData { // getStreetAddress( ), getPostalCode( ) etc } $meta->contactData( $business );
Finally, there are also individual methods:
$meta->streetAddress( '1601 S California Ave' ) ->locality( 'Palo Alto' ) ->region( 'CA' ) ->postalCode( '94304' ) ->countryName( 'USA' ) ->email( 'me@example.com' ) ->phone('650-123-4567') ->faxNumber('+1-415-123-4567') ->website( 'http://example.com' ) );
Geographical Information
In addition to address information via contact data, Open Graph also allows you to specify the specific geographical location a page represents by setting the latitude and longitude, and optionally the altitude.
$meta->addGeopoint( new Geopoint( 37.416343, -122.153013 ) );
The result:
<meta name="og:latitude" content="37.416343" /> <meta name="og:longitude" content="-122.153013" />
Rather than pass an instance of the provided Geopoint
class, you can instead pass any class that implements the interface with the same name. Essentially this entails providing getLatitude( )
, getLatitude( )
and getAltitude( )
methods.
Business Opening Hours
To specify a business' opening hours:
$meta->businessHours( new BusinessDay( 'Monday', '09:00', '17:00' ), new BusinessDay( 'Tuesday', '09:00', '17:00' ), new BusinessDay( 'Wednesday', '09:00', '17:00' ), new BusinessDay( 'Thursday', '09:00', '17:00' ), new BusinessDay( 'Friday', '09:00', '17:00' ), new BusinessDay( 'Saturday', '08:00', '18:00' ) );
Facebook has a few Open Graph tags that are specific to the social network.
The App ID
To set the app ID of the site's app:
$meta->facebookAppId( '12345678910' );
The Profile IDs
To set the Facebook profile ID; that's to say, the Facebook ID of a user that can be followed:
$meta->facebookProfileId( '78676576576' );
Admins
To set the admins:
$meta->facebookAdmins( '12345', '678910' ); // or $meta->facebookAdmins( '12345' );
Pages
To set one or more Facebook Page IDs that are associated with a URL in order to enable link editing and instant article publishing:
$meta->facebookPages( '76236783', '67873687' ); // or $meta->facebookPages( '76236783' );
Twitter Cards
For Twitter cards, the Open Graph tags will generally suffice (previously Twitter used proprietary tags such as twitter:title
, but these are now largely redundant), although you'll also want to call this:
$meta->twitterCard( );
This will add the following:
<meta name="twitter:card" content="summary" />
For a different type, just pass it as an argument:
$meta->twitterCard( MetaTags::TWITTER_CARD_SUMMARY_LARGE_IMAGE ); $meta->twitterCard( MetaTags::TWITTER_CARD_APP ); $meta->twitterCard( MetaTags::TWITTER_CARD_PLAYER );
You can also set the site and / or creator:
$meta ->twitterSite( '@lukaswhite' ) ->twitterCreator( '@lukaswhite' );
The metadata — for example, the name and description — along with any media such as images will be fetched from the Open Graph tags.
You can, however, override these; if for any reason you want different metadata for Twitter:
$meta ->twitterTitle( 'My Twitter' ) ->twitterDescription( 'A Twitter description' );
This will set twitter:title
and twitter:description
metatags respectively.
Here's a complete example:
$meta ->title( 'My Website' ) ->description( 'A website' ) ->twitterCard( ) ->twitterSite( '@lukaswhite' ) ->twitterCreator( '@lukaswhite' ); ->addImage( ( new \Lukaswhite\MetaTags\Entities\Image( ) ) ->setUrl( 'http://example.com/image.jpeg' ) ->setSecureUrl( 'https://example.com/image.jpeg' ) ->setType( 'image/jpeg' ) ->setWidth( 500 ) ->setHeight( 300 ) ->setAlt( 'Example image' ) );
Examples (Laravel-flavored)
These simplified examples use Laravel, but you can use this library using any framework - or indeed without a framework at all.
Registering as a Singleton
One approach you could take if you're using Laravel is to register the meta tags as a singleton; that way you can also start setting your meta tags during the bootstrap phase:
$this->app->singleton( MetaTags::class, function( $app ) { $meta = new MetaTags( ); $meta->siteName( array_get( $app[ 'config' ], 'app.name' ) ); return $meta; } );
You can of course use a similar approach without Laravel, though the syntax will obviously be different.
Then, whenever you want to set your meta tags, for example in a controller, then you can get the instance like this:
$meta = app( )->make( MetaTags::class );
Because it's a singleton, you can build your meta tags from multiple parts of your application.
A Content Management System
This is a really basic example; it's pulling a page from the database, and setting the page title and page description from that.
class PagesController { public function show( Page $page ) { app( )->make( MetaTags::class ) ->title( $page->title ) ->description( $page->description ); // The rest of the method } }
A Restaurant Listing Site
In this example, the controller is pulling a restaurant record from the database in order to display it. It's using the meta tags library to match the title and description to the restaurant.
In addition, it's setting the Open Graph tags for the address of the restaurant, as well as pointing it to the geographical location.
class RestaurantController { public function show( Restaurant $restaurant ) { app( )->make( MetaTags::class ) ->isBusiness( ) ->title( $restaurant->name ) ->description( $restaurant->description ) ->addGeopoint( new Geopoint( $restaurant->lat, $restaurant->lng ) ) ->streetAddress( $restaurant->street_address ) ->locality( $restaurant->locality ) ->region( $restaurant->locality ) ->postalCode( $restaurant->postal_code ) ->country( $restaurant->country ) ->phone( $restaurant->phone ); // The rest of the method } }
Alternatively, if the Restaurant
class implements the ContactData
and Geopoint
interfaces, you can simplify the code above significantly:
class RestaurantController { public function show( Restaurant $restaurant ) { app( )->make( MetaTags::class ) ->isBusiness( ) ->title( $restaurant->name ) ->description( $restaurant->description ) ->addGeopoint( $restaurant ) ->contactData( $restaurant ); // The rest of the method } }
A Social Network
This example demonstrates a simple example whereby this particular controller display's a user's profile on a social network, setting the appropriate Open Graph tags.
class User extends Model implements Profile { public class getFirstName( ) { return $this->attributes[ 'first_name' ]; } public class getLastName( ) { return $this->attributes[ 'last_name' ]; } public class getUsername( ) { return $this->attributes[ 'username' ]; } public class getGender( ) { return $this->attributes[ 'gender' ]; } } class ProfileController { public function show( User $user ) { app( )->make( MetaTags::class ) ->profile( $user ); // The rest of the method } }
Outputting the Tags
To render the tags, just call the render( )
method, for example:
<html> <head> <?php print $meta->render( ); ?> </head>
Or using Blade:
<html>
<head>
{!! $meta->render( ) !!}
</head>
The class also implements the __toString( )
method, so you can even print
it right out. It simply calls the render( )
method.
Customizing the Output
By default, the meta tag output includes newlines. You can turn this off using the verbose( )
method.
If you want even more control over the resulting string, you can set the tag prefix and tag suffix. For example, perhaps you want pretty-printed HTML:
$meta->tagPrefix( "\t\t" )
->tagSuffix( "\n" );
The result will be along these lines:
<html>
<head>
<meta charset="utf-8" />
<title>My Website</title>
<meta name="description" content="A description here" />
</head>
Should you want even more control over the output, or want to make any modifications, then you can call the build( )
method, which will return an array of HtmlElement
instances. The documentation for that is here.
Geo Tags
You can also add geotags:
$meta->geoPosition( new Geopoint( 37.416343, -122.153013 ) ) ->geoPlaceName( 'London' ) ->geoRegion( 'GB' );
The result:
<meta name="geo.position" content="37.416343, -122.153013" /> <meta name="ICBM" content="37.416343, -122.153013" /> <meta name="geo.placename" content="London" /> <meta name="geo.region" content="GB" />
Adding Additional Meta Tags
It's worth pointing out that you don't necessarily need to add all of your meta tags to your site using this library; indeed arguably there are some that you shouldn't.
Take this example:
<!DOCTYPE html> <html lang="en"> <head> {!! $meta !!} <meta name="viewport" content="width=device-width, initial-scale=1"> <link rel="stylesheet" href="/assets/styles.min.css"> </head>
The meta tag and link tag are really theme-specific, so it might make things clearer if you add them to your layout in this way.
That's not to say you can't do it programmatically:
$meta->viewport( 'width=device-width, initial-scale=1' ) ->addLink( '/assets/styles.min.css', 'stylesheet' );