Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/alwaysblank/schemer
Help with schema.org generation.
https://github.com/alwaysblank/schemer
helper microdata php php7 schema
Last synced: 3 months ago
JSON representation
Help with schema.org generation.
- Host: GitHub
- URL: https://github.com/alwaysblank/schemer
- Owner: alwaysblank
- Created: 2019-07-29T23:25:41.000Z (over 5 years ago)
- Default Branch: master
- Last Pushed: 2020-04-27T20:27:56.000Z (over 4 years ago)
- Last Synced: 2024-09-30T14:09:01.612Z (4 months ago)
- Topics: helper, microdata, php, php7, schema
- Language: PHP
- Size: 102 KB
- Stars: 9
- Watchers: 4
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# Schemer 🦹
Help with [schema.org](https://schema.org/) markup generation.
[![Build Status](https://travis-ci.org/alwaysblank/schemer.svg?branch=master)](https://travis-ci.org/alwaysblank/schemer)
## Usage
> **Note:** Currently, Schemer generates only the "microdata" format.
Schemer is designed to be pretty simple to use: Just pass a structured array to the `::build()` static method on the Scheme you want:
```php
echo AlwaysBlank\Schemer\Scheme\PostalAddress::build([
['street', '123 Oak St'],
['state', 'OR'],
['city', 'Portland'],
['zip', '97123'],
['pobox', 'P.O. 1234'],
['country', 'USA'],
]);// 123 Oak StORPortland97123P.O. 1234USA
```You can optionally pass a second argument, which is a array of arguments that will override the defaults. This allows you to do things like change the wrapping element tag, add arbitrary attributes, etc.
```php
echo AlwaysBlank\Schemer\Scheme\PostalAddress::build([
['street', '123 Oak St'],
['state', 'OR'],
['city', 'Portland'],
['zip', '97123'],
['pobox', 'P.O. 1234'],
['country', 'USA'],
], [
'tag' => 'a',
'attributes' => [
'href' => 'https://www.alwaysblank.org',
'hidden' => true,
],
]);
// 123 Oak StORPortland97123P.O. 1234USA
```> The actual HTML output is slightly different: To allow for simple breaks but maximum styling options, each inline element is followed by `​`, a zero-width space. This allows the browser to break at this point, but contributes no formatting apart from that.
### Schemes
Currently Schemer supports the following Schemes in some fashion. More will be added in the future! Please feel free to file an issue with schemes you'd like to see, or—even better!—a pull request adding them. See "How It Works" below for more information on how to create new Schemes and Properties.
- [`PostalAddress`](https://schema.org/PostalAddress) can understand the following:
- `street`
- `state`
- `city`
- `zip` (zip code)
- `pobox` (P.O. box)
- `country`
- [`LocalBusiness`](https://schema.org/LocalBusiness) can understand the following:
- `name`
- `phone` (phone segments will be rendered as links with `tel:`)
- `url` (url can take a string, or an array with two items keyed `url` and `content`)
- `address` (this is an array containing keys that will be understood by `PostalAddress`)
## How It WorksThere are three basic parts to Schemer:
- `Node`
- `Property`
- `Scheme`### Node
The `Node` is the simplest element, and also the only element that actually behaves like a class (`Property`s and `Scheme`s, though classes, are really just wrappers for static methods). It provides a sort of wrapper for individual segments of a `Propery` or `Scheme`, and is what is used to generate the actual HTML.
In general, you probably won't be interacting with `Node`s very much outside of creating them by passing a set of arguments:
```php
$Node = Node::add([
'itemscope' => true,
'itemtype' => 'http://schema.org/LocalBusiness',
'itemprop' => 'description',
'tag' => 'div',
'content' => "Always Blank",
'attributes' => [
'href' => 'https://www.alwaysblank.org',
'hidden' => true,
]
]);
```Understanding this syntax is helpful, because it will be necessary for creating new `Property`s, or to modify existing properties or `Scheme`s.
### Property
A `Property`'s one purpose in life is to return a Node with the appropriate arguments for the the schema.org property it represents. It is also implemented as a **Trait** so that it can be composed into a `Scheme` later on.
A `Property` must meet the following requirements:
- It implements a public static method that has the same name as whatever you want to use as the key/name for the field in question—i.e. `address` or `phone`. In general, these names should be *simple* rather than directly matching the schema.org property they represent. For instance, `telephone` is represented here as just `phone`.
- This method takes only one argument.
- This method must return a `Node`.
- The Class and file names of the `Property` must be, essentially, CamelCase versions of the method name. i.e. `pobox` => `POBox`, `name` => `Name`, etc.A `Property` can take input of any kind (although most take strings), but you should be aware that there isn't a obvious mechanism (apart from reading the source code) to know what specific format a `Property` might be expecting its argument to be in, so keep user experience in mind when building these things.
### Scheme
A `Scheme` is the top dog; the final thing Schemer exists to do. They are surprisingly simple. A `Scheme` has to do the following:
- `extend` the class `AlwaysBlank\Schemer\Scheme\Scheme`. This actually probably most of the functionality.
- Implement a public static method called `wrap`. It takes two arguments: `content`, which is a string, and `args` which is an optional array. It returns a `Node`. Use this method to set the properties (i.e. `itemprop`) that the element that contains this `Scheme` will need.
- `use` any `Property` Traits that you want this `Scheme` to have access to. This defines the arguments it will accept.Here's an example `Scheme`:
```php
class Example extends AlwaysBlank\Schemer\Scheme\Scheme
{
use City;
use Name;
public static function wrap(string $content, array $args = []): Node
{
return Node::add(array_merge([
'itemscope' => true,
'itemtype' => 'http://schema.org/Example',
'content' => $content,
'tag' => 'section',
'attributes' => [
'data-section' => 'example',
],
], $args));
}
```#### Aliases
When defining Nodes, you can use abbreviations for the properties. This may be useful if you find yourself in a situation where you need to type up a lot of Nodes.
Supported abbreviations are:
| Property | Alias(es) |
|------------ |--------------- |
| itemscope | scope, iscope |
| itemprop | prop, iprop |
| itemtype | type, itype |
| attributes | attr, attrs |This will create a `Scheme` called `Example` that will understand `city` and `name`, and will use a `` with an additional attribute of `data-section="example"`.
### Going Further
If you have more questions about how these things work, I encourage you to dig into the source code. I've made an effort to document the code clearly inline, and keep everything simple and clear.
## Debugging
To debug Schemer, set the PHP constant `ALWAYSBLANK_SCHEMER_DEBUG` to `true`. As with any debug flag, I don't recommend you set this in production.
Currently the debug flag has the following effects (this list may expand with time):
- Stop suppressing exceptions when calling `Property`s with bad arguments.
## Limitations
### Nesting
The output of a `Scheme` is generally pretty "flat" in the sense that it's just a list of elements that aren't nested within on another. (The exception being if the `Scheme` in question includes other `Scheme`s as `Property`s, as `LocalBusiness` does with `Address`.) This is more or less by design: So far as I can tell, building a system that would allow a user to dynamically modify the nesting of arbitrary elements would create an extreme amount of complexity in a library that is supposed to be very simple. In general, `Scheme`-level content isn't usually heavily nested anyway (apart from the aforementioned exceptions).
If you find yourself needing a more complex structure, Schemer is design to be flexible enough that you can build `Scheme`s and `Property`s "on the fly"—or even just use `Property`s directly without the need for a `Scheme`.
### Options
This library is currently limited in terms of what it "understands". While it can be easily expanded, I have no intention of ever making it comprehensive with respect to the full schema.org specifications. Polite feature requests are always welcome (and good PRs will likely be merged without question), but please keep in mind that the primary purpose of this library is to make a small slice of repetitive tasks less repetitive, not to solve all microdata-related problems.