Blocchi (Gutenberg)
Ecco come recuperare i dati dei blocchi Gutenberg.
Lo schema GraphQL include i seguenti campi aggiunti a tutti i tipi CustomPost (come Post e Page):
blocksblockDataItemsblockFlattenedDataItems
Questi campi non sono abilitati se il plugin Classic Editor è attivo.
blocks
Il campo CustomPost.blocks: [BlockUnion!] recupera l'elenco di tutti i blocchi contenuti nel custom post.
blocks restituisce un elenco dei tipi Block che sono stati mappati nello schema GraphQL. Questi tipi Block fanno tutti parte del tipo BlockUnion e implementano l'interfaccia Block.
Il plugin implementa un tipo Block, GenericBlock, già sufficiente per recuperare i dati di qualsiasi blocco (tramite il campo attributes: JSONObject).
Questa query:
{
post(by: { id: 1 }) {
blocks {
...on Block {
name
attributes
innerBlocks {
...on Block {
name
attributes
innerBlocks {
...on Block {
name
attributes
}
}
}
}
}
}
}
}
}...produrrà questa risposta:
{
"data": {
"post": {
"blocks": [
{
"name": "core/gallery",
"attributes": {
"linkTo": "none",
"className": "alignnone",
"images": [
{
"url": "https://d.pr/i/zd7Ehu+",
"alt": "",
"id": "1706"
},
{
"url": "https://d.pr/i/jXLtzZ+",
"alt": "",
"id": "1705"
}
],
"ids": [],
"shortCodeTransforms": [],
"imageCrop": true,
"fixedHeight": true,
"sizeSlug": "large",
"allowResize": false
},
"innerBlocks": null
},
{
"name": "core/heading",
"attributes": {
"content": "List Block",
"level": 2
},
"innerBlocks": null
},
{
"name": "core/list",
"attributes": {
"ordered": false,
"values": "<li>List item 1</li><li>List item 2</li><li>List item 3</li><li>List item 4</li>"
},
"innerBlocks": null
},
{
"name": "core/heading",
"attributes": {
"className": "has-top-margin",
"content": "Columns Block",
"level": 2
},
"innerBlocks": null
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"innerBlocks": [
{
"name": "core/column",
"attributes": {},
"innerBlocks": [
{
"name": "core/image",
"attributes": {
"id": 1701,
"className": "layout-column-1",
"url": "https://d.pr/i/fW6V3V+",
"alt": ""
},
"innerBlocks": null
}
]
},
{
"name": "core/column",
"attributes": {},
"innerBlocks": [
{
"name": "core/paragraph",
"attributes": {
"className": "layout-column-2",
"content": "Phosfluorescently morph intuitive relationships rather than customer directed human capital.",
"dropCap": false
},
"innerBlocks": null
}
]
}
]
},
{
"name": "core/heading",
"attributes": {
"content": "Columns inside Columns (nested inner blocks)",
"level": 2
},
"innerBlocks": null
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"innerBlocks": [
{
"name": "core/column",
"attributes": {},
"innerBlocks": [
{
"name": "core/image",
"attributes": {
"id": 1701,
"className": "layout-column-1",
"url": "https://d.pr/i/fW6V3V+",
"alt": ""
},
"innerBlocks": null
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"innerBlocks": [
{
"name": "core/column",
"attributes": {
"width": "33.33%"
},
"innerBlocks": [
{
"name": "core/heading",
"attributes": {
"fontSize": "large",
"content": "Life is so rich",
"level": 2
},
"innerBlocks": null
},
{
"name": "core/heading",
"attributes": {
"level": 3,
"content": "Life is so dynamic"
},
"innerBlocks": null
}
]
},
{
"name": "core/column",
"attributes": {
"width": "66.66%"
},
"innerBlocks": [
{
"name": "core/paragraph",
"attributes": {
"content": "This rhyming poem is the spark that can reignite the fires within you. It challenges you to go out and live your life in the present moment as a “hero” and leave your mark on this world.",
"dropCap": false
},
"innerBlocks": null
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"innerBlocks": [
{
"name": "core/column",
"attributes": {},
"innerBlocks": [
{
"name": "core/image",
"attributes": {
"id": 361,
"sizeSlug": "large",
"linkDestination": "none",
"url": "https://gato-graphql.lndo.site/wp-content/uploads/2022/05/graphql-voyager-public-1024x622.jpg",
"alt": ""
},
"innerBlocks": null
}
]
},
{
"name": "core/column",
"attributes": {},
"innerBlocks": null
},
{
"name": "core/column",
"attributes": {},
"innerBlocks": [
{
"name": "core/image",
"attributes": {
"id": 362,
"sizeSlug": "large",
"linkDestination": "none",
"url": "https://gato-graphql.lndo.site/wp-content/uploads/2022/05/namespaced-interactive-schema-1024x598.webp",
"alt": ""
},
"innerBlocks": null
}
]
}
]
}
]
}
]
}
]
}
]
}
]
}
}
}Lo schema GraphQL per i tipi Block si presenta così:
interface Block {
name: String!
attributes: JSONObject
innerBlocks: [BlockUnion!]
contentSource: HTML!
}
type GenericBlock implements Block {
name: String!
attributes: JSONObject
innerBlocks: [BlockUnion!]
contentSource: HTML!
}
union BlockUnion = GenericBlockCampi di Block
L'interfaccia Block (e, di conseguenza, il tipo GeneralBlock) contiene i seguenti campi:
namerecupera il nome del blocco:"core/paragraph","core/heading","core/image", ecc.attributesrecupera un oggetto JSON contenente tutti gli attributi del blocco.innerBlocksrecupera[BlockUnion!], permettendoci di interrogarlo per navigare nella gerarchia dei blocchi contenenti blocchi interni e recuperare i dati per tutti loro, per quanti livelli di profondità esistano nel contenuto.contentSourcerecupera il codice sorgente HTML (Gutenberg) del blocco, inclusi i delimitatori di commento che contengono gli attributi. Tuttavia, questo campo non recupera esattamente gli stessi dati di come vengono memorizzati nel DB (vedi #2346), quindi usare questo campo con cautela.
Recuperare GeneralBlock direttamente (invece di BlockUnion)
Poiché attualmente esiste un solo tipo Block che mappa i blocchi — GeneralBlock — ha senso che CustomPost.blocks (e anche Block.innerBlocks) recuperi questo tipo direttamente, invece del tipo BlockUnion.
È possibile farlo nella pagina Settings nella scheda Blocks, spuntando l'opzione Use single type instead of union type?:

A questo punto, la query GraphQL si semplifica:
{
post(by: { id: 1 }) {
blocks {
name
attributes
innerBlocks {
name
attributes
}
}
}
}Si noti che mantenere il tipo di risposta come BlockUnion è utile per la compatibilità futura: se in futuro si decidesse di aggiungere tipi specifici per i blocchi nello schema (vedi la sezione seguente), non ci saranno modifiche incompatibili.
Mappare tipi specifici per blocchi
Il tipo JSONObject (come recuperato da Block.attributes) non è tipizzato in modo rigido: le sue proprietà possono avere qualsiasi tipo e cardinalità (String, Int, [Boolean!], ecc.), quindi è necessario conoscere queste informazioni per ogni blocco e gestire ciascun caso lato client.
Se si necessita di una tipizzazione rigida, occorre estendere lo schema GraphQL tramite codice PHP, aggiungendo tipi specifici per i blocchi che mappano gli attributi specifici di un blocco come campi, e includerli in BlockUnion.
Ad esempio, è possibile aggiungere il tipo CoreParagraphBlock che mappa il blocco core/paragraph, con il campo content di tipo String.
Consultare la documentazione in GatoGraphQL/GatoGraphQL per scoprire come estendere lo schema GraphQL (attualmente in corso di sviluppo).
Filtrare i blocchi
Il campo CustomPost.blocks contiene l'argomento filterBy con due proprietà: include ed exclude. È possibile usarle per filtrare i blocchi recuperati, in base al nome del blocco:
{
post(by: { id: 1 }) {
id
blocks(
filterBy: {
include: [
"core/heading",
"core/gallery"
]
}
) {
name
attributes
}
}
}Questo produrrà:
{
"data": {
"post": {
"blocks": [
{
"name": "core/gallery",
"attributes": {
"linkTo": "none",
"className": "alignnone",
"images": [
{
"url": "https://d.pr/i/zd7Ehu+",
"alt": "",
"id": "1706"
},
{
"url": "https://d.pr/i/jXLtzZ+",
"alt": "",
"id": "1705"
}
],
"ids": [],
"shortCodeTransforms": [],
"imageCrop": true,
"fixedHeight": true,
"sizeSlug": "large",
"allowResize": false
},
"innerBlocks": null
},
{
"name": "core/heading",
"attributes": {
"content": "List Block",
"level": 2
},
"innerBlocks": null
},
{
"name": "core/heading",
"attributes": {
"className": "has-top-margin",
"content": "Columns Block",
"level": 2
},
"innerBlocks": null
},
{
"name": "core/heading",
"attributes": {
"content": "Columns inside Columns (nested inner blocks)",
"level": 2
},
"innerBlocks": null
}
]
}
}
}Si noti che non tutti i blocchi di tipo core/heading sono stati inclusi: quelli annidati sotto core/column sono stati esclusi, poiché non c'è modo di raggiungerli (dato che i blocchi core/columns e core/column sono essi stessi esclusi).
Inconvenienti del campo blocks
Il campo blocks presenta l'inconveniente che, per recuperare tutti i dati dei blocchi contenuti nel custom post, inclusi i dati dei blocchi interni, e i loro blocchi interni, e così via, è necessario conoscere quanti livelli di blocchi annidati esistono nel contenuto, e riflettere questa informazione nella query GraphQL.
Oppure, se non lo si sa, occorre comporre la query con abbastanza livelli da essere sicuri che tutti i dati vengano recuperati.
Ad esempio, questa query recupera fino a 7 livelli di annidamento di blocchi interni:
{
post(by: { id: 1 }) {
blocks {
...BlockData
}
}
}
fragment BlockData on Block {
name
attributes
innerBlocks {
...on Block {
name
attributes
innerBlocks {
...on Block {
name
attributes
innerBlocks {
...on Block {
name
attributes
innerBlocks {
...on Block {
name
attributes
innerBlocks {
...on Block {
name
attributes
innerBlocks {
...on Block {
name
attributes
innerBlocks {
...on Block {
name
attributes
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}blockDataItems
Per evitare l'inconveniente legato al modo in cui il campo blocks recupera tutti i dati (inclusi quelli dei blocchi interni, e dei loro blocchi interni, e così via), esiste il campo CustomPost.blockDataItems.
Questo campo, invece di restituire [BlockUnion], restituisce [JSONObject!]:
type CustomPost {
blockDataItems: [JSONObject!]
}In altre parole, invece di seguire il tipico approccio GraphQL in cui le entità si relazionano ad altre entità e si naviga tra di esse, ogni entità Block al livello superiore produce già tutti i dati del blocco per sé stessa e per tutti i suoi figli, all'interno di un singolo risultato JSONObject.
L'oggetto JSON contiene le proprietà del blocco (nelle voci name e attributes) e quelle dei suoi blocchi interni (nella voce innerBlocks), in modo ricorsivo.
Ad esempio, la seguente query:
{
post(by: { id: 1 }) {
blockDataItems
}
}...produrrà:
{
"data": {
"post": {
"blockDataItems": [
{
"name": "core/gallery",
"attributes": {
"linkTo": "none",
"className": "alignnone",
"images": [
{
"url": "https://d.pr/i/zd7Ehu+",
"alt": "",
"id": "1706"
},
{
"url": "https://d.pr/i/jXLtzZ+",
"alt": "",
"id": "1705"
}
],
"ids": [],
"shortCodeTransforms": [],
"imageCrop": true,
"fixedHeight": true,
"sizeSlug": "large",
"allowResize": false
}
},
{
"name": "core/heading",
"attributes": {
"content": "List Block",
"level": 2
}
},
{
"name": "core/list",
"attributes": {
"ordered": false,
"values": "<li>List item 1</li><li>List item 2</li><li>List item 3</li><li>List item 4</li>"
}
},
{
"name": "core/heading",
"attributes": {
"className": "has-top-margin",
"content": "Columns Block",
"level": 2
}
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"innerBlocks": [
{
"name": "core/column",
"attributes": {},
"innerBlocks": [
{
"name": "core/image",
"attributes": {
"id": 1701,
"className": "layout-column-1",
"url": "https://d.pr/i/fW6V3V+",
"alt": ""
}
}
]
},
{
"name": "core/column",
"attributes": {},
"innerBlocks": [
{
"name": "core/paragraph",
"attributes": {
"className": "layout-column-2",
"content": "Phosfluorescently morph intuitive relationships rather than customer directed human capital.",
"dropCap": false
}
}
]
}
]
},
{
"name": "core/heading",
"attributes": {
"content": "Columns inside Columns (nested inner blocks)",
"level": 2
}
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"innerBlocks": [
{
"name": "core/column",
"attributes": {},
"innerBlocks": [
{
"name": "core/image",
"attributes": {
"id": 1701,
"className": "layout-column-1",
"url": "https://d.pr/i/fW6V3V+",
"alt": ""
}
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"innerBlocks": [
{
"name": "core/column",
"attributes": {
"width": "33.33%"
},
"innerBlocks": [
{
"name": "core/heading",
"attributes": {
"fontSize": "large",
"content": "Life is so rich",
"level": 2
}
},
{
"name": "core/heading",
"attributes": {
"level": 3,
"content": "Life is so dynamic"
}
}
]
},
{
"name": "core/column",
"attributes": {
"width": "66.66%"
},
"innerBlocks": [
{
"name": "core/paragraph",
"attributes": {
"content": "This rhyming poem is the spark that can reignite the fires within you. It challenges you to go out and live your life in the present moment as a “hero” and leave your mark on this world.",
"dropCap": false
}
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"innerBlocks": [
{
"name": "core/column",
"attributes": {},
"innerBlocks": [
{
"name": "core/image",
"attributes": {
"id": 361,
"sizeSlug": "large",
"linkDestination": "none",
"url": "https://gato-graphql.lndo.site/wp-content/uploads/2022/05/graphql-voyager-public-1024x622.jpg",
"alt": ""
}
}
]
},
{
"name": "core/column",
"attributes": {}
},
{
"name": "core/column",
"attributes": {},
"innerBlocks": [
{
"name": "core/image",
"attributes": {
"id": 362,
"sizeSlug": "large",
"linkDestination": "none",
"url": "https://gato-graphql.lndo.site/wp-content/uploads/2022/05/namespaced-interactive-schema-1024x598.webp",
"alt": ""
}
}
]
}
]
}
]
}
]
}
]
}
]
}
]
}
}
}Filtrare gli elementi di dati dei blocchi
Analogamente a blocks, anche blockDataItems consente di filtrare i blocchi recuperati, tramite l'argomento filterBy.
Questa query:
{
post(by: { id: 1 }) {
id
blockDataItems(
filterBy: {
include: [
"core/heading"
]
}
)
}
}...produrrà:
{
"data": {
"post": {
"blockDataItems": [
{
"name": "core/heading",
"attributes": {
"content": "List Block",
"level": 2
},
"innerBlocks": null
},
{
"name": "core/heading",
"attributes": {
"className": "has-top-margin",
"content": "Columns Block",
"level": 2
},
"innerBlocks": null
},
{
"name": "core/heading",
"attributes": {
"content": "Columns inside Columns (nested inner blocks)",
"level": 2
},
"innerBlocks": null
}
]
}
}
}Si noti che, analogamente a blocks, non tutti i blocchi di tipo core/heading sono stati inclusi: quelli annidati sotto core/column sono stati esclusi, poiché non c'è modo di raggiungerli (dato che i blocchi core/columns e core/column sono essi stessi esclusi).
blockFlattenedDataItems
Entrambi i campi blocks e blockDataItems consentono di filtrare i blocchi recuperati (tramite l'argomento filterBy). In entrambi i casi, se un blocco soddisfa la condizione di inclusione ma è annidato all'interno di un blocco che non la soddisfa, verrà escluso.
Ci sono però situazioni in cui è necessario recuperare tutti i blocchi di un certo tipo dal custom post, indipendentemente da dove si trovino nella gerarchia. Ad esempio, potrebbe essere necessario includere tutti i blocchi di tipo core/image, per recuperare tutte le immagini presenti in un articolo del blog.
Per soddisfare questa esigenza esiste il campo CustomPost.blockFlattenedDataItems. A differenza dei campi blocks e blockDataItems, appiattisce la gerarchia dei blocchi in un unico livello.
Questa query:
{
post(by: { id: 1 }) {
blockFlattenedDataItems
}
}...produrrà:
{
"data": {
"post": {
"blockFlattenedDataItems": [
{
"name": "core/gallery",
"attributes": {
"linkTo": "none",
"className": "alignnone",
"images": [
{
"url": "https://d.pr/i/zd7Ehu+",
"alt": "",
"id": "1706"
},
{
"url": "https://d.pr/i/jXLtzZ+",
"alt": "",
"id": "1705"
}
],
"ids": [],
"shortCodeTransforms": [],
"imageCrop": true,
"fixedHeight": true,
"sizeSlug": "large",
"allowResize": false
},
"innerBlockPositions": null,
"parentBlockPosition": null
},
{
"name": "core/heading",
"attributes": {
"content": "List Block",
"level": 2
},
"innerBlockPositions": null,
"parentBlockPosition": null
},
{
"name": "core/list",
"attributes": {
"ordered": false,
"values": "<li>List item 1</li><li>List item 2</li><li>List item 3</li><li>List item 4</li>"
},
"innerBlockPositions": null,
"parentBlockPosition": null
},
{
"name": "core/heading",
"attributes": {
"className": "has-top-margin",
"content": "Columns Block",
"level": 2
},
"innerBlockPositions": null,
"parentBlockPosition": null
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"innerBlockPositions": [
5,
7
],
"parentBlockPosition": null
},
{
"name": "core/column",
"attributes": {},
"parentBlockPosition": 4,
"innerBlockPositions": [
6
]
},
{
"name": "core/image",
"attributes": {
"id": 1701,
"className": "layout-column-1",
"url": "https://d.pr/i/fW6V3V+",
"alt": ""
},
"parentBlockPosition": 5,
"innerBlockPositions": null
},
{
"name": "core/column",
"attributes": {},
"parentBlockPosition": 4,
"innerBlockPositions": [
8
]
},
{
"name": "core/paragraph",
"attributes": {
"className": "layout-column-2",
"content": "Phosfluorescently morph intuitive relationships rather than customer directed human capital.",
"dropCap": false
},
"parentBlockPosition": 7,
"innerBlockPositions": null
},
{
"name": "core/heading",
"attributes": {
"content": "Columns inside Columns (nested inner blocks)",
"level": 2
},
"innerBlockPositions": null,
"parentBlockPosition": null
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"innerBlockPositions": [
11
],
"parentBlockPosition": null
},
{
"name": "core/column",
"attributes": {},
"parentBlockPosition": 10,
"innerBlockPositions": [
12,
13
]
},
{
"name": "core/image",
"attributes": {
"id": 1701,
"className": "layout-column-1",
"url": "https://d.pr/i/fW6V3V+",
"alt": ""
},
"parentBlockPosition": 11,
"innerBlockPositions": null
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"parentBlockPosition": 11,
"innerBlockPositions": [
14,
17
]
},
{
"name": "core/column",
"attributes": {
"width": "33.33%"
},
"parentBlockPosition": 13,
"innerBlockPositions": [
15,
16
]
},
{
"name": "core/heading",
"attributes": {
"fontSize": "large",
"content": "Life is so rich",
"level": 2
},
"parentBlockPosition": 14,
"innerBlockPositions": null
},
{
"name": "core/heading",
"attributes": {
"level": 3,
"content": "Life is so dynamic"
},
"parentBlockPosition": 14,
"innerBlockPositions": null
},
{
"name": "core/column",
"attributes": {
"width": "66.66%"
},
"parentBlockPosition": 13,
"innerBlockPositions": [
18,
19
]
},
{
"name": "core/paragraph",
"attributes": {
"content": "This rhyming poem is the spark that can reignite the fires within you. It challenges you to go out and live your life in the present moment as a “hero” and leave your mark on this world.",
"dropCap": false
},
"parentBlockPosition": 17,
"innerBlockPositions": null
},
{
"name": "core/columns",
"attributes": {
"isStackedOnMobile": true
},
"parentBlockPosition": 17,
"innerBlockPositions": [
20,
22,
23
]
},
{
"name": "core/column",
"attributes": {},
"parentBlockPosition": 19,
"innerBlockPositions": [
21
]
},
{
"name": "core/image",
"attributes": {
"id": 361,
"sizeSlug": "large",
"linkDestination": "none",
"url": "https://gato-graphql.lndo.site/wp-content/uploads/2022/05/graphql-voyager-public-1024x622.jpg",
"alt": ""
},
"parentBlockPosition": 20,
"innerBlockPositions": null
},
{
"name": "core/column",
"attributes": {},
"parentBlockPosition": 19,
"innerBlockPositions": null
},
{
"name": "core/column",
"attributes": {},
"parentBlockPosition": 19,
"innerBlockPositions": [
24
]
},
{
"name": "core/image",
"attributes": {
"id": 362,
"sizeSlug": "large",
"linkDestination": "none",
"url": "https://gato-graphql.lndo.site/wp-content/uploads/2022/05/namespaced-interactive-schema-1024x598.webp",
"alt": ""
},
"parentBlockPosition": 23,
"innerBlockPositions": null
}
]
}
}
}Si noti come l'attributo innerBlocks sia scomparso, poiché i blocchi non sono più annidati. Al suo posto, la risposta include altri due attributi (che permettono di ricostruire la gerarchia dei blocchi):
parentBlockPosition: la posizione del blocco padre all'interno dell'array restituito, oppurenullse si tratta di un blocco di livello superioreinnerBlockPositions: un array con le posizioni dei blocchi interni del blocco all'interno dell'array restituito
Filtrare gli elementi di dati dei blocchi appiattiti
Ora che la gerarchia dei blocchi è stata appiattita, filtrare per core/heading produrrà tutti questi blocchi (anche se originariamente uno di essi era annidato sotto un blocco che è stato escluso).
Questa query:
{
post(by: { id: 1 }) {
id
blockFlattenedDataItems(
filterBy: {
include: [
"core/heading"
]
}
)
}
}...produrrà:
{
"data": {
"post": {
"blockFlattenedDataItems": [
{
"name": "core/heading",
"attributes": {
"content": "List Block",
"level": 2
}
},
{
"name": "core/heading",
"attributes": {
"className": "has-top-margin",
"content": "Columns Block",
"level": 2
}
},
{
"name": "core/heading",
"attributes": {
"content": "Columns inside Columns (nested inner blocks)",
"level": 2
}
},
{
"name": "core/heading",
"attributes": {
"fontSize": "large",
"content": "Life is so rich",
"level": 2
}
},
{
"name": "core/heading",
"attributes": {
"level": 3,
"content": "Life is so dynamic"
}
}
]
}
}
}Si noti che i due attributi aggiuntivi, parentBlockPosition e innerBlockPositions, vengono rimossi durante il filtraggio, poiché non hanno più senso.