simpleResponse, basicCard, carouselBrowse, mediaResponse, tableCard
| Rich Response Items |
|---|
Rich visual elements that deliver resources & visually-interesting content to user
| Visual selection responses |
|---|
AKA Carousel or List, don't use both in a single response
https://developers.google.com/assistant/conversational/selection-responses
| Samples |
|---|
| "Kitchen sink" intent handler |
| Kitchen Sink JSON Rich Response |
| Skeleton JSON Rich Response |
- https://github.com/dialogflow/fulfillment-webhook-json/tree/master/responses/v2/ActionsOnGoogle/RichResponses
- https://www.npmjs.com/package/actions-on-google
- https://github.com/googleapis/nodejs-dialogflow
- https://github.com/actions-on-google/actions-on-google-nodejs/blob/master/src/service/dialogflow/dialogflow.ts
- https://developers.google.com/assistant/actions/reference/conversation-api-playground
- https://github.com/valgaze/dialogflow-speedrun/blob/master/docs/resources.md
- https://cloud.google.com/dialogflow/docs/reference/rest/v2/projects.agent.sessions/detectIntent
- https://cloud.google.com/dialogflow/docs/reference/rest/v2/DetectIntentResponse
- https://developers.google.com/assistant/actions/build/json/dialogflow-webhook-json
- https://actions-on-google.github.io/actions-on-google-nodejs/interfaces/dialogflow_api_v2.googleclouddialogflowv2queryresult.html#webhookpayload
- https://github.com/valgaze/dialogflow-speedrun/blob/master/docs/resources.md
{
"webhookPayload": {
"google": {
"systemIntent": {
"intent": "actions.intent.OPTION",
"data": {
"@type": "type.googleapis.com/google.actions.v2.OptionValueSpec",
"listSelect": {
"title": "List Title",
"items": []
},
"carouselSelect": {
"items": []
}
},
"richResponse": {
"items": [],
"suggestions": [],
"linkOutSuggestion": {}
}
}
}
}
}https://developers.google.com/assistant/conversational/simple-responses#SimpleResponseSamples
Note: This must be included as a fallback if any rich responses are used as a fallback
Note: You can optionally pass "speech" for voice to text (in that case, text is available on textToSpeech rather than displayText)
Usage
conv.ask(new SimpleResponse({
speech: 'this is speech too! [textToSpeech]'
}))
conv.add('this is speech too! [textToSpeech]')
conv.ask('this is speech too! [textToSpeech]')Response (webhookPayload.google.richResponse.items)
All 3 above will return this output
{
"simpleResponse": {
"textToSpeech": "this is speech too! [textToSpeech]"
}
}Usage
conv.ask(
new SimpleResponse({
text: `Abcdefg Hijklmop`
})
);Response (webhookPayload.google.richResponse.items)
{
"simpleResponse": {
"displayText": "Abcdefg Hijklmop"
}
}Usage
conv.ask(
new SimpleResponse({
speech: 'This is speech [textToSpeech]',
text: 'This is display text [displayText]'
})
)
**Response (webhookPayload.google.richResponse.items)**
```json
{
"simpleResponse": {
"textToSpeech": "This is speech [textToSpeech]",
"displayText": "This is display text [displayText]"
}
}https://developers.google.com/assistant/conversational/rich-responses#BasicCardSamples
Usage
app.intent('my intent name', (conv, parameters) => {
const flavor = 'mint'
const size = 'large'
const imageURL = `https://i.imgur.com/W9Eeuu1.jpg`
conv.ask(
new BasicCard({
text: `Here is your 🍦!!! This type of ${flavor} ice cream is great \nEverybody after this line break loves it.`,
subtitle: "Here's your subtitle",
title: `Here's your ${size} ${flavor}`,
buttons: new Button({
title: `Learn more about ${flavor}`,
url: `https://duckduckgo.com/?q=${flavor}+ice+cream`
}),
image: new Image({
url: imageURL,
alt: `${flavor} mmm...`
})
})
);
})Response (webhookPayload.google.richResponse.items)
{
"basicCard": {
"title": "Here's your large mint",
"subtitle": "Here's your subitle",
"formattedText": "Here is your 🍦!!! This type of mint ice cream is great",
"image": {
"url": "https://i.imgur.com/W9Eeuu1.jpg",
"accessibilityText": "mint mmm..."
},
"buttons": [{
"title": "Learn more about mint",
"openUrlAction": {
"url": "https://duckduckgo.com/?q=mint+ice+cream"
}
}]
}
}Note: Images on their own get wrapped in a card
Note: Image URLs must be publicly accessible (or you can base64, ex )
Usage
app.intent('my intent name', (conv, parameters) => {
conv.ask(new Image({
url:
"http://storage.googleapis.com/automotive-media/album_art.jpg",
alt: "Image becomes a card"
}))
})Response (webhookPayload.google.richResponse.items)
{
"basicCard": {
"image": {
"url": "http://storage.googleapis.com/automotive-media/album_art.jpg",
"accessibilityText": "Image becomes a card"
}
}
}https://developers.google.com/assistant/conversational/rich-responses#suggestion_chips
Note: Images on their own get wrapped in a card
Usage
app.intent('my intent name', (conv, parameters) => {
// Suggestions
conv.ask(new Suggestions(['Suggestion 1', 'Suggestion 2']));
})Response (webhookPayload.google.richResponse.suggestions)
{
"suggestions": [{
"title": "Suggestion 1"
}, {
"title": "Suggestion 2"
}]
}Note: only one linkout suggestion per response. URL must be valid
Usage
app.intent('my intent name', (conv, parameters) => {
// LinkOut Suggestion
conv.ask(new LinkOutSuggestion({
name: 'Suggestion bongo',
url: 'https://assistant.google.com/',
}));
})Response (webhookPayload.google.richResponse.suggestions)
{
"suggestions": [],
"linkOutSuggestion": {
"destinationName": "Jams",
"url": "https://www.youtube.com/watch?v=UzYibo3igGU"
}
}https://developers.google.com/assistant/conversational/rich-responses#TableCardSamples
Usage
app.intent('my intent name', (conv, parameters) => {
// Table
conv.ask(new Table({
dividers: true,
columns: ['header 1', 'header 2', 'header 3'],
rows: [
['row 1 item 1', 'row 1 item 2', 'row 1 item 3'],
['row 2 item 1', 'row 2 item 2', 'row 2 item 3'],
],
}));
})Response (webhookPayload.google.richResponse.suggestions)
{
"tableCard": {
"rows": [{
"cells": [{
"text": "row 1 item 1"
}, {
"text": "row 1 item 2"
}, {
"text": "row 1 item 3"
}],
"dividerAfter": true
}, {
"cells": [{
"text": "row 2 item 1"
}, {
"text": "row 2 item 2"
}, {
"text": "row 2 item 3"
}],
"dividerAfter": true
}],
"columnProperties": [{
"header": "header 1"
}, {
"header": "header 2"
}, {
"header": "header 3"
}]
}
}https://developers.google.com/assistant/conversational/rich-responses#MediaResponseSamples
Note: Resource URL must be publicly accessible
Usage
app.intent('my intent name', (conv, parameters) => {
// Media Object
conv.ask(
new MediaObject({
name: "Jazz in Paris",
url:
"http://storage.googleapis.com/automotive-media/Jazz_In_Paris.mp3",
description: "A funky Jazz tune",
icon: new Image({
url:
"http://storage.googleapis.com/automotive-media/album_art.jpg",
alt: "Media icon"
})
})
)
})Response (webhookPayload.google.richResponse.items)
{
"mediaResponse": {
"mediaType": "AUDIO",
"mediaObjects": [{
"contentUrl": "http://storage.googleapis.com/automotive-media/Jazz_In_Paris.mp3",
"description": "A funky Jazz tune",
"icon": {
"url": "http://storage.googleapis.com/automotive-media/album_art.jpg",
"accessibilityText": "Media icon"
},
"name": "Jazz in Paris"
}]
}
}https://developers.google.com/assistant/conversational/rich-responses#BrowsingCarouselSamples
Note: Resource URL must be publicly accessible
Usage
app.intent('my intent name', (conv, parameters) => {
// Browse Carousel
conv.ask(new BrowseCarousel({
items: [
new BrowseCarouselItem({
title: 'Title of item 1',
url: 'https://example.com',
description: 'Description of item 1',
image: new Image({
url: 'https://storage.googleapis.com/actionsresources/logo_assistant_2x_64dp.png',
alt: 'Image alternate text',
}),
footer: 'Item 1 footer',
}),
new BrowseCarouselItem({
title: 'Title of item 2',
url: 'https://example.com',
description: 'Description of item 2',
image: new Image({
url: 'https://storage.googleapis.com/actionsresources/logo_assistant_2x_64dp.png',
alt: 'Image alternate text',
}),
footer: 'Item 2 footer',
}),
],
}));
})Response (webhookPayload.google.richResponse.items)
{
"carouselBrowse": {
"items": [
{
"title": "Title of item 1",
"openUrlAction": {
"url": "https://example.com"
},
"description": "Description of item 1",
"footer": "Item 1 footer",
"image": {
"url": "https://storage.googleapis.com/actionsresources/logo_assistant_2x_64dp.png",
"accessibilityText": "Image alternate text"
}
},
{
"title": "Title of item 2",
"openUrlAction": {
"url": "https://example.com"
},
"description": "Description of item 2",
"footer": "Item 2 footer",
"image": {
"url": "https://storage.googleapis.com/actionsresources/logo_assistant_2x_64dp.png",
"accessibilityText": "Image alternate text"
}
}
]
}
}Note: Taps should emit event actions_intent_OPTION (hopefully an intent "listening" for that to handle)
-
http://kristianbrimble.com/dynamic-carousels-from-your-dialogflow-webhook/
-
https://developers.google.com/assistant/conversational/selection-responses
Usage
app.intent('my intent name', (conv, parameters) => {
// Constants for list and carousel selection
const SELECTION_KEY_GOOGLE_ASSISTANT = "googleAssistant";
const SELECTION_KEY_GOOGLE_PAY = "googlePay";
const SELECTION_KEY_GOOGLE_PIXEL = "googlePixel";
const SELECTION_KEY_GOOGLE_HOME = "googleHome";
// Constant for image URLs
const IMG_URL_AOG =
"https://storage.googleapis.com/actionsresources/logo_assistant_2x_64dp.png";
const IMG_URL_GOOGLE_PAY =
"https://storage.googleapis.com/actionsresources/logo_pay_64dp.png";
const IMG_URL_GOOGLE_PIXEL =
"https://storage.googleapis.com/madebygoog/v1/Pixel/Pixel_ColorPicker/Pixel_Device_Angled_Black-720w.png";
const IMG_URL_GOOGLE_HOME =
"https://lh3.googleusercontent.com/Nu3a6F80WfixUqf_ec_vgXy_c0-0r4VLJRXjVFF_X_CIilEu8B9fT35qyTEj_PEsKw";
// Carousel select
conv.ask(
new Carousel({
items: {
// Add the first item to the carousel
[SELECTION_KEY_GOOGLE_ASSISTANT]: {
synonyms: ["Assistant", "Google Assistant"],
title: "Item #1",
description: "Description of Item #1",
image: new Image({
url: IMG_URL_AOG,
alt: "Google Assistant logo"
})
},
// Add the second item to the carousel
[SELECTION_KEY_GOOGLE_PAY]: {
synonyms: ["Transactions", "Google Payments"],
title: "Item #2",
description: "Description of Item #2",
image: new Image({
url: IMG_URL_GOOGLE_PAY,
alt: "Google Pay logo"
})
},
// Add third item to the carousel
[SELECTION_KEY_GOOGLE_PIXEL]: {
synonyms: ["Pixel", "Google Pixel phone"],
title: "Item #3",
description: "Description of Item #3",
image: new Image({
url: IMG_URL_GOOGLE_PIXEL,
alt: "Google Pixel phone"
})
},
// Add last item of the carousel
[SELECTION_KEY_GOOGLE_HOME]: {
title: "Item #4",
synonyms: ["Google Home"],
description: "Description of Item #4",
image: new Image({
url: IMG_URL_GOOGLE_HOME,
alt: "Google Home"
})
}
}
})
);
})Response (webhookPayload.google.systemIntent.data.carouselSelect)
{
"webhookPayload": {
"google": {
"richResponse": {
"items": [],
"suggestions": []
},
"systemIntent": {
"intent": "actions.intent.OPTION",
"data": {
"@type": "type.googleapis.com/google.actions.v2.OptionValueSpec",
"carouselSelect": {
"items": [{
"optionInfo": {
"key": "googleAssistant",
"synonyms": [
"Assistant",
"Google Assistant"
]
},
"description": "Description of Item #1",
"image": {
"url": "https://storage.googleapis.com/actionsresources/logo_assistant_2x_64dp.png",
"accessibilityText": "Google Assistant logo"
},
"title": "Item #1"
},
{
"optionInfo": {
"key": "googlePay",
"synonyms": [
"Transactions",
"Google Payments"
]
},
"description": "Description of Item #2",
"image": {
"url": "https://storage.googleapis.com/actionsresources/logo_pay_64dp.png",
"accessibilityText": "Google Pay logo"
},
"title": "Item #2"
},
{
"optionInfo": {
"key": "googlePixel",
"synonyms": [
"Pixel",
"Google Pixel phone"
]
},
"description": "Description of Item #3",
"image": {
"url": "https://storage.googleapis.com/madebygoog/v1/Pixel/Pixel_ColorPicker/Pixel_Device_Angled_Black-720w.png",
"accessibilityText": "Google Pixel phone"
},
"title": "Item #3"
},
{
"optionInfo": {
"key": "googleHome",
"synonyms": [
"Google Home"
]
},
"description": "Description of Item #4",
"image": {
"url": "https://lh3.googleusercontent.com/Nu3a6F80WfixUqf_ec_vgXy_c0-0r4VLJRXjVFF_X_CIilEu8B9fT35qyTEj_PEsKw",
"accessibilityText": "Google Home"
},
"title": "Item #4"
}
]
}
}
}
}
}
}Note: Taps should emit event actions_intent_OPTION (hopefully an intent "listening" for that to handle)
Note: title & option keys are required fields
-
http://kristianbrimble.com/dynamic-carousels-from-your-dialogflow-webhook/
-
https://developers.google.com/assistant/conversational/selection-responses
Usage
app.intent('my intent name', (conv, parameters) => {
// Constants for list and carousel selection
const SELECTION_KEY_GOOGLE_ASSISTANT = "googleAssistant";
const SELECTION_KEY_GOOGLE_PAY = "googlePay";
const SELECTION_KEY_GOOGLE_PIXEL = "googlePixel";
const SELECTION_KEY_GOOGLE_HOME = "googleHome";
// Constant for image URLs
const IMG_URL_AOG =
"https://storage.googleapis.com/actionsresources/logo_assistant_2x_64dp.png";
const IMG_URL_GOOGLE_PAY =
"https://storage.googleapis.com/actionsresources/logo_pay_64dp.png";
const IMG_URL_GOOGLE_PIXEL =
"https://storage.googleapis.com/madebygoog/v1/Pixel/Pixel_ColorPicker/Pixel_Device_Angled_Black-720w.png";
const IMG_URL_GOOGLE_HOME =
"https://lh3.googleusercontent.com/Nu3a6F80WfixUqf_ec_vgXy_c0-0r4VLJRXjVFF_X_CIilEu8B9fT35qyTEj_PEsKw";
conv.ask(new List({
title: 'List Title',
items: {
// Add the first item to the list
[SELECTION_KEY_GOOGLE_ASSISTANT]: {
synonyms: [
'Assistant',
'Google Assistant',
],
title: 'Item #1',
description: 'Description of Item #1',
image: new Image({
url: 'https://www.gstatic.com/images/branding/product/2x/assistant_48dp.png',
alt: 'Google Assistant logo',
}),
},
// Add the second item to the list
[SELECTION_KEY_GOOGLE_PAY]: {
synonyms: [
'Transactions',
'Google Payments',
'Google Pay',
],
title: 'Item #2',
description: 'Description of Item #2',
image: new Image({
url: 'https://www.gstatic.com/images/branding/product/2x/pay_48dp.png',
alt: 'Google Pay logo',
}),
},
// Add the third item to the list
[SELECTION_KEY_GOOGLE_PIXEL]: {
synonyms: [
'Pixel',
'Google Pixel',
'Pixel phone'
],
title: 'Item #3',
description: 'Description of Item #3',
image: new Image({
url: 'https://storage.googleapis.com/madebygoog/v1/Pixel/Pixel_ColorPicker/Pixel_Device_Angled_Black-720w.png',
alt: 'Google Pixel phone',
}),
},
// Add the last item to the list
[SELECTION_KEY_GOOGLE_HOME]: {
title: 'Item #4',
synonyms: [
'Home',
'Google Home',
],
description: 'Description of Item #4',
image: new Image({
url: 'https://lh3.googleusercontent.com/Nu3a6F80WfixUqf_ec_vgXy_c0-0r4VLJRXjVFF_X_CIilEu8B9fT35qyTEj_PEsKw',
alt: 'Google Home',
}),
},
},
}));
})Response (webhookPayload.google.systemIntent.data.listSelect)
{
"webhookPayload": {
"google": {
"richResponse": {
"items": [],
"suggestions": []
},
"systemIntent": {
"intent": "actions.intent.OPTION",
"data": {
"@type": "type.googleapis.com/google.actions.v2.OptionValueSpec",
"listSelect": {
"title": "List Title",
"items": [{
"optionInfo": {
"key": "googleAssistant",
"synonyms": [
"Assistant",
"Google Assistant"
]
},
"description": "Description of Item #1",
"image": {
"url": "https://www.gstatic.com/images/branding/product/2x/assistant_48dp.png",
"accessibilityText": "Google Assistant logo"
},
"title": "Item #1"
},
{
"optionInfo": {
"key": "googlePay",
"synonyms": [
"Transactions",
"Google Payments",
"Google Pay"
]
},
"description": "Description of Item #2",
"image": {
"url": "https://www.gstatic.com/images/branding/product/2x/pay_48dp.png",
"accessibilityText": "Google Pay logo"
},
"title": "Item #2"
},
{
"optionInfo": {
"key": "googlePixel",
"synonyms": [
"Pixel",
"Google Pixel",
"Pixel phone"
]
},
"description": "Description of Item #3",
"image": {
"url": "https://storage.googleapis.com/madebygoog/v1/Pixel/Pixel_ColorPicker/Pixel_Device_Angled_Black-720w.png",
"accessibilityText": "Google Pixel phone"
},
"title": "Item #3"
},
{
"optionInfo": {
"key": "googleHome",
"synonyms": [
"Home",
"Google Home"
]
},
"description": "Description of Item #4",
"image": {
"url": "https://lh3.googleusercontent.com/Nu3a6F80WfixUqf_ec_vgXy_c0-0r4VLJRXjVFF_X_CIilEu8B9fT35qyTEj_PEsKw",
"accessibilityText": "Google Home"
},
"title": "Item #4"
}
]
}
}
}
}
}
}
https://github.com/DefinitelyTyped/DefinitelyTyped/blob/master/types/dialogflow-fulfillment/index.d.ts
https://gist.github.com/iamarcel/f39a08820f6f9181b92eee042c53c908
{ "fulfillmentText": "replicated message", "fulfillmentMessages": [ { "platform": "FACEBOOK", "text": { "text": [ "replicated message" ] }, "message": "text" }, { "platform": "ACTIONS_ON_GOOGLE", "simpleResponses": { "simpleResponses": [ { "textToSpeech": "replicated message", "ssml": "", "displayText": "" } ] }, "message": "simpleResponses" }, { "platform": "FACEBOOK", "payload": { "fields": { "flag": { "stringValue": "followup", "kind": "stringValue" } } }, "message": "payload" }, { "platform": "ACTIONS_ON_GOOGLE", "payload": { "fields": { "flag": { "stringValue": "followup", "kind": "stringValue" } } }, "message": "payload" }, { "platform": "FACEBOOK", "text": { "text": [ "This is a response for facebook messenger" ] }, "message": "text" }, { "platform": "ACTIONS_ON_GOOGLE", "simpleResponses": { "simpleResponses": [ { "textToSpeech": ">>SIMPLE RESPONSE CONFIGURED FROM DIALOGFLOW<<", "ssml": "", "displayText": ">>SIMPLE RESPONSE CONFIGURED FROM DIALOGFLOW<<" } ] }, "message": "simpleResponses" }, { "platform": "ACTIONS_ON_GOOGLE", "listSelect": { "items": [ { "info": { "synonyms": [], "key": "chocolateoptionkey" }, "title": "choclate title", "description": "chocolate desc", "image": { "imageUri": "https://i.imgur.com/uMtF8ah.jpg", "accessibilityText": "chocolate" } }, { "info": { "synonyms": [], "key": "mintoptionkey" }, "title": "mint title", "description": "mint desc", "image": { "imageUri": "https://i.imgur.com/W9Eeuu1.jpg", "accessibilityText": "mint" } }, { "info": { "synonyms": [], "key": "strawberryoptionkey" }, "title": "strawberry title", "description": "strawberry desc", "image": { "imageUri": "https://i.imgur.com/GSZEI39.jpg", "accessibilityText": "mint accessibility" } }, { "info": { "synonyms": [], "key": "vanillaoptionkey" }, "title": "vanilla title", "description": "vanilla desc", "image": { "imageUri": "https://i.imgur.com/zqKeYU5.jpg", "accessibilityText": "vanilla accessibility" } } ], "title": "List 123", "subtitle": "" }, "message": "listSelect" }, { "platform": "ACTIONS_ON_GOOGLE", "suggestions": { "suggestions": [ { "title": "aaa" }, { "title": "bbbb" }, { "title": "ccc" } ] }, "message": "suggestions" }, { "platform": "PLATFORM_UNSPECIFIED", "text": { "text": [ "replicated message" ] }, "message": "text" }, { "platform": "PLATFORM_UNSPECIFIED", "payload": { "fields": { "flag": { "stringValue": "followup", "kind": "stringValue" } } }, "message": "payload" } ] }