Adicionar elementos de interface interativos a cards

Esta página explica como adicionar widgets e elementos de interface aos cards para que os usuários possam interagir com seu app do Google Chat, por exemplo, clicando em um botão ou enviando informações.

Os apps de chat podem usar as seguintes interfaces do Chat para criar cards interativos:

  • Mensagens que contêm um ou mais cards.
  • Páginas iniciais, que é um card que aparece na guia Início das mensagens diretas com o app do Chat.
  • Caixas de diálogo, que são cards abertos em uma nova janela de mensagens e páginas iniciais.

Quando os usuários interagem com os cards, os apps do Chat podem usar os dados recebidos para processar e responder de acordo. Para mais detalhes, consulte Coletar e processar informações de usuários do Google Chat.


Use o Card Builder para criar e visualizar mensagens e interfaces de usuário para apps do Chat:

Abra o Card Builder

Pré-requisitos

Um app do Google Chat configurado para receber e responder a eventos de interação. Para criar um app do Chat interativo, conclua um dos seguintes guias de início rápido com base na arquitetura do app que você quer usar:

Adicionar um botão

O widget ButtonList mostra um conjunto de botões. Os botões podem mostrar texto, um ícone ou ambos. Cada Button oferece suporte a uma OnClick ação que ocorre quando os usuários clicam no botão. Exemplo:

  • Abra um hiperlink com OpenLink, para fornecer mais informações aos usuários.
  • Execute um action que executa uma função personalizada, como chamar uma API.

Para acessibilidade, os botões oferecem suporte a texto alternativo.

Adicionar um botão que executa uma função personalizada

A seguir, um card com um widget ButtonList e dois botões. Um botão abre a documentação para desenvolvedores do Google Chat em uma nova guia. O outro botão executa uma função personalizada chamada goToView() e transmite o parâmetro viewType="BIRD EYE VIEW".

Adicionar um botão com o estilo do Material Design

A seguir, mostramos um conjunto de botões em diferentes estilos de botões do Material Design.

Para aplicar o estilo do Material Design, não inclua o atributo "cor".

Adicionar um botão com cor personalizada e um botão desativado

Você pode impedir que os usuários cliquem em um botão definindo "disabled": "true".

A imagem a seguir mostra um card com um widget ButtonList e dois botões. Um botão usa o campo Color para personalizar a cor de segundo plano do botão. O outro botão é desativado com o campo Disabled, que impede que o usuário clique no botão e execute a função.

Adicionar um botão com um ícone

A imagem a seguir mostra um card com um widget ButtonList e dois widgets de ícone Button. Um botão usa o campo knownIcon para mostrar o ícone de e-mail integrado do Google Chat. O outro botão usa o campo iconUrl para mostrar um widget de ícone personalizado.

Adicionar um botão com um ícone e um texto

A seguir, mostramos um card com um widget ButtonList que pede ao usuário para enviar um e-mail. O primeiro botão mostra um ícone de e-mail, e o segundo mostra texto. O usuário pode clicar no ícone ou no botão de texto para executar a função sendEmail.

Personalizar o botão de uma seção recolhível

Personalize o botão de controle que fecha e abre seções em um card. Escolha entre vários ícones ou imagens para representar visualmente o conteúdo da seção, facilitando a compreensão e a interação dos usuários com as informações.

Adicionar um menu flutuante

O Overflow menu pode ser usado em cards do Chat para ter mais opções e ações. Ele permite incluir mais opções sem sobrecarregar a interface do card, garantindo um design limpo e organizado.

Adicionar uma lista de ícones

O widget ChipList é uma maneira versátil e visualmente atraente de mostrar informações. Use listas de ícones para representar tags, categorias ou outros dados relevantes, facilitando a navegação e a interação dos usuários com seu conteúdo.

Coletar informações dos usuários

Esta seção explica como adicionar widgets que coletam informações, como texto ou seleções.

Para saber como processar o que os usuários inserem, consulte Coletar e processar informações de usuários do Google Chat.

Coletar texto

O widget TextInput fornece um campo em que os usuários podem inserir texto. O widget oferece suporte a sugestões, que ajudam os usuários a inserir dados uniformes, e ações de mudança, que são Actions executadas quando ocorre uma mudança no campo de entrada de texto, como um usuário adicionando ou excluindo texto.

Quando você precisar coletar dados abstratos ou desconhecidos dos usuários, use este widget TextInput. Para coletar dados definidos dos usuários, use o widget SelectionInput.

Confira a seguir um card com um widget TextInput:

Coletar datas ou horários

O widget DateTimePicker permite que os usuários insiram uma data, um horário ou ambos. Ou os usuários podem usar o seletor para escolher datas e horários. Se os usuários inserirem uma data ou hora inválida, o seletor vai mostrar um erro que pede para eles inserirem as informações corretamente.

A seguir, mostramos um card com três tipos diferentes de widgets DateTimePicker:

Permitir que os usuários selecionem itens

O widget SelectionInput fornece um conjunto de itens selecionáveis, como caixas de seleção, botões de opção, chaves ou um menu suspenso. Você pode usar esse widget para coletar dados definidos e padronizados dos usuários. Para coletar dados indefinidos dos usuários, use o widget TextInput.

O widget SelectionInput oferece suporte a sugestões, que ajudam os usuários a inserir dados uniformes, e ações de mudança, que são Actions executadas quando ocorre uma mudança em um campo de entrada de seleção, como um usuário selecionando ou desmarcando um item.

Os apps de chat podem receber e processar o valor dos itens selecionados. Para detalhes sobre como trabalhar com entradas de formulário, consulte Processar informações inseridas pelos usuários.

Esta seção apresenta exemplos de cards que usam o widget SelectionInput. Os exemplos usam diferentes tipos de entradas de seção:

Adicionar uma caixa de seleção

A seguir, uma ficha de informações pede que o usuário especifique se um contato é profissional, pessoal ou ambos, com um widget SelectionInput que usa caixas de seleção:

Adicionar um botão de opção

A seguir, mostramos um card que pede ao usuário para especificar se um contato é profissional ou pessoal com um widget SelectionInput que usa botões de opção:

Adicionar uma chave

O exemplo a seguir mostra um card que pede ao usuário para especificar se um contato é profissional, pessoal ou ambos com um widget SelectionInput que usa chaves:

O exemplo a seguir mostra um card que pede ao usuário para especificar se um contato é profissional ou pessoal com um widget SelectionInput que usa um menu suspenso:

Adicionar um menu de seleção múltipla

A seguir, um card pede que o usuário selecione contatos em um menu de seleção múltipla:

É possível preencher itens para um menu de seleção múltipla com as seguintes fontes de dados no Google Workspace:

  • Usuários do Google Workspace: só é possível preencher usuários na mesma organização do Google Workspace.
  • Espaços do Chat: o usuário que insere itens no menu de seleção múltipla só pode ver e selecionar espaços a que pertence na organização do Google Workspace.

Para usar fontes de dados do Google Workspace, especifique o campo platformDataSource. Ao contrário de outros tipos de entrada de seleção, você omite objetos SelectionItem porque esses itens de seleção são originados dinamicamente do Google Workspace.

O código a seguir mostra um menu de seleção múltipla de usuários do Google Workspace. Para preencher os usuários, a entrada de seleção define commonDataSource como USER:

JSON

{
  "selectionInput": {
    "name": "contacts",
    "type": "MULTI_SELECT",
    "label": "Selected contacts",
    "multiSelectMaxSelectedItems": 5,
    "multiSelectMinQueryLength": 1,
    "platformDataSource": {
      "commonDataSource": "USER"
    }
  }
}

O código a seguir mostra um menu de seleção múltipla de espaços do Chat. Para preencher espaços, a entrada de seleção especifica o campo hostAppDataSource. O menu de seleção múltipla também define defaultToCurrentSpace como true, o que faz com que o espaço atual seja a seleção padrão no menu:

JSON

{
  "selectionInput": {
    "name": "spaces",
    "type": "MULTI_SELECT",
    "label": "Selected contacts",
    "multiSelectMaxSelectedItems": 3,
    "multiSelectMinQueryLength": 1,
    "platformDataSource": {
      "hostAppDataSource": {
        "chatDataSource": {
          "spaceDataSource": {
            "defaultToCurrentSpace": true
          }
        }
      }
    }
  }
}

Os menus de seleção múltipla também podem preencher itens de uma fonte de dados externa ou de terceiros. Por exemplo, você pode usar menus de seleção múltipla para ajudar um usuário a escolher em uma lista de leads de vendas de um sistema de gestão de relacionamento com o cliente (CRM).

Para usar uma fonte de dados externa, use o campo externalDataSource para especificar uma função que retorne itens da fonte de dados.

Para reduzir as solicitações a uma fonte de dados externa, inclua itens sugeridos que aparecem no menu de seleção múltipla antes que os usuários digitem no menu. Por exemplo, você pode preencher os contatos pesquisados recentemente para o usuário. Para preencher itens sugeridos de uma fonte de dados externa, especifique objetos SelectionItem.

O código a seguir mostra um menu de seleção múltipla de itens de um conjunto externo de contatos para o usuário. O menu mostra um contato por padrão e executa a função getContacts para recuperar e preencher itens da fonte de dados externa:

Node.js

node/selection-input/index.js
selectionInput: {
  name: "contacts",
  type: "MULTI_SELECT",
  label: "Selected contacts",
  multiSelectMaxSelectedItems: 3,
  multiSelectMinQueryLength: 1,
  externalDataSource: { function: "getContacts" },
  // Suggested items loaded by default.
  // The list is static here but it could be dynamic.
  items: [getContact("3")]
}

Python

python/selection-input/main.py
'selectionInput': {
  'name': "contacts",
  'type': "MULTI_SELECT",
  'label': "Selected contacts",
  'multiSelectMaxSelectedItems': 3,
  'multiSelectMinQueryLength': 1,
  'externalDataSource': { 'function': "getContacts" },
  # Suggested items loaded by default.
  # The list is static here but it could be dynamic.
  'items': [get_contact("3")]
}

Java

java/selection-input/src/main/java/com/google/chat/selectionInput/App.java
.setSelectionInput(new GoogleAppsCardV1SelectionInput()
  .setName("contacts")
  .setType("MULTI_SELECT")
  .setLabel("Selected contacts")
  .setMultiSelectMaxSelectedItems(3)
  .setMultiSelectMinQueryLength(1)
  .setExternalDataSource(new GoogleAppsCardV1Action().setFunction("getContacts"))
  .setItems(List.of(getContact("3")))))))))));

Apps Script

apps-script/selection-input/selection-input.gs
selectionInput: {
  name: "contacts",
  type: "MULTI_SELECT",
  label: "Selected contacts",
  multiSelectMaxSelectedItems: 3,
  multiSelectMinQueryLength: 1,
  externalDataSource: { function: "getContacts" },
  // Suggested items loaded by default.
  // The list is static here but it could be dynamic.
  items: [getContact("3")]
}

Para fontes de dados externas, também é possível preencher automaticamente os itens que os usuários começam a digitar no menu de multisseleção. Por exemplo, se um usuário começar a digitar Atl para um menu que preenche cidades nos Estados Unidos, seu app do Chat poderá sugerir automaticamente Atlanta antes que o usuário termine de digitar. É possível usar o preenchimento automático em até 100 itens.

Para preencher automaticamente os itens, crie uma função que consulte a fonte de dados externa e retorne itens sempre que um usuário digitar no menu de seleção múltipla. A função precisa fazer o seguinte:

  • Transmita um objeto de evento que represente a interação do usuário com o menu.
  • Identifique se o valor invokedFunction do evento de interação corresponde à função do campo externalDataSource.
  • Quando as funções corresponderem, retorne os itens sugeridos da fonte de dados externa. Para sugerir itens com base no que o usuário digita, receba o valor da chave autocomplete_widget_query. Esse valor representa o que o usuário digita no menu.

O código a seguir preenche automaticamente itens de um recurso de dados externo. Usando o exemplo anterior, o app de chat sugere itens com base em quando a função getContacts é acionada:

Node.js

node/selection-input/index.js
/**
 * Responds to a WIDGET_UPDATE event in Google Chat.
 *
 * @param {Object} event The event object from Chat API.
 * @return {Object} Response from the Chat app.
 */
function onWidgetUpdate(event) {
  if (event.common["invokedFunction"] === "getContacts") {
    const query = event.common.parameters["autocomplete_widget_query"];
    return { actionResponse: {
      type: "UPDATE_WIDGET",
      updatedWidget: { suggestions: { items: [
        // The list is static here but it could be dynamic.
        getContact("1"), getContact("2"), getContact("3"), getContact("4"), getContact("5")
      // Only return items based on the query from the user
      ].filter(e => !query || e.text.includes(query))}}
    }};
  }
}

/**
 * Generate a suggested contact given an ID.
 *
 * @param {String} id The ID of the contact to return.
 * @return {Object} The contact formatted as a suggested item for selectors.
 */
function getContact(id) {
  return {
    value: id,
    startIconUri: "https://www.gstatic.com/images/branding/product/2x/contacts_48dp.png",
    text: "Contact " + id
  };
}

Python

python/selection-input/main.py
def on_widget_update(event: dict) -> dict:
  """Responds to a WIDGET_UPDATE event in Google Chat."""
  if "getContacts" == event.get("common").get("invokedFunction"):
    query = event.get("common").get("parameters").get("autocomplete_widget_query")
    return { 'actionResponse': {
      'type': "UPDATE_WIDGET",
      'updatedWidget': { 'suggestions': { 'items': list(filter(lambda e: query is None or query in e["text"], [
        # The list is static here but it could be dynamic.
        get_contact("1"), get_contact("2"), get_contact("3"), get_contact("4"), get_contact("5")
      # Only return items based on the query from the user
      ]))}}
    }}


def get_contact(id: str) -> dict:
  """Generate a suggested contact given an ID."""
  return {
    'value': id,
    'startIconUri': "https://www.gstatic.com/images/branding/product/2x/contacts_48dp.png",
    'text': "Contact " + id
  }

Java

java/selection-input/src/main/java/com/google/chat/selectionInput/App.java
// Responds to a WIDGET_UPDATE event in Google Chat.
Message onWidgetUpdate(JsonNode event) {
  if ("getContacts".equals(event.at("/invokedFunction").asText())) {
    String query = event.at("/common/parameters/autocomplete_widget_query").asText();
    return new Message().setActionResponse(new ActionResponse()
      .setType("UPDATE_WIDGET")
      .setUpdatedWidget(new UpdatedWidget()
        .setSuggestions(new SelectionItems().setItems(List.of(
          // The list is static here but it could be dynamic.
          getContact("1"), getContact("2"), getContact("3"), getContact("4"), getContact("5")
        // Only return items based on the query from the user
        ).stream().filter(e -> query == null || e.getText().indexOf(query) > -1).toList()))));
  }
  return null;
}

// Generate a suggested contact given an ID.
GoogleAppsCardV1SelectionItem getContact(String id) {
  return new GoogleAppsCardV1SelectionItem()
    .setValue(id)
    .setStartIconUri("https://www.gstatic.com/images/branding/product/2x/contacts_48dp.png")
    .setText("Contact " + id);
}

Apps Script

apps-script/selection-input/selection-input.gs
/**
 * Responds to a WIDGET_UPDATE event in Google Chat.
 *
 * @param {Object} event The event object from Chat API.
 * @return {Object} Response from the Chat app.
 */
function onWidgetUpdate(event) {
  if (event.common["invokedFunction"] === "getContacts") {
    const query = event.common.parameters["autocomplete_widget_query"];
    return { actionResponse: {
      type: "UPDATE_WIDGET",
      updatedWidget: { suggestions: { items: [
        // The list is static here but it could be dynamic.
        getContact("1"), getContact("2"), getContact("3"), getContact("4"), getContact("5")
      // Only return items based on the query from the user
      ].filter(e => !query || e.text.includes(query))}}
    }};
  }
}

/**
 * Generate a suggested contact given an ID.
 *
 * @param {String} id The ID of the contact to return.
 * @return {Object} The contact formatted as a suggested item for selectors.
 */
function getContact(id) {
  return {
    value: id,
    startIconUri: "https://www.gstatic.com/images/branding/product/2x/contacts_48dp.png",
    text: "Contact " + id
  };
}

Validar os dados inseridos nos cards

Nesta página, explicamos como validar os dados inseridos em um action e widgets de card. Por exemplo, é possível validar se um campo de entrada de texto tem texto inserido pelo usuário ou se ele contém um determinado número de caracteres.

Definir widgets obrigatórios para ações

Como parte do action do card, adicione nomes de widgets que uma ação precisa à lista requiredWidgets.

Se algum widget listado aqui não tiver um valor quando essa ação for invocada, o envio da ação do formulário será cancelado.

Quando "all_widgets_are_required": "true" é definido para uma ação, todos os widgets no card são obrigatórios para essa ação.

Definir uma ação all_widgets_are_required na multisseleção

JSON

{
  "sections": [
    {
      "header": "Select contacts",
      "widgets": [
        {
          "selectionInput": {
            "type": "MULTI_SELECT",
            "label": "Selected contacts",
            "name": "contacts",
            "multiSelectMaxSelectedItems": 3,
            "multiSelectMinQueryLength": 1,
            "onChangeAction": {
              "all_widgets_are_required": true
            },
            "items": [
              {
                "value": "contact-1",
                "startIconUri": "https://www.gstatic.com/images/branding/product/2x/contacts_48dp.png",
                "text": "Contact 1",
                "bottomText": "Contact one description",
                "selected": false
              },
              {
                "value": "contact-2",
                "startIconUri": "https://www.gstatic.com/images/branding/product/2x/contacts_48dp.png",
                "text": "Contact 2",
                "bottomText": "Contact two description",
                "selected": false
              },
              {
                "value": "contact-3",
                "startIconUri": "https://www.gstatic.com/images/branding/product/2x/contacts_48dp.png",
                "text": "Contact 3",
                "bottomText": "Contact three description",
                "selected": false
              },
              {
                "value": "contact-4",
                "startIconUri": "https://www.gstatic.com/images/branding/product/2x/contacts_48dp.png",
                "text": "Contact 4",
                "bottomText": "Contact four description",
                "selected": false
              },
              {
                "value": "contact-5",
                "startIconUri": "https://www.gstatic.com/images/branding/product/2x/contacts_48dp.png",
                "text": "Contact 5",
                "bottomText": "Contact five description",
                "selected": false
              }
            ]
          }
        }
      ]
    }
  ]
}
Definir uma ação all_widgets_are_required em dateTimePicker

JSON

{
  "sections": [
    {
      "widgets": [
        {
          "textParagraph": {
            "text": "A datetime picker widget with both date and time:"
          }
        },
        {
          "divider": {}
        },
        {
          "dateTimePicker": {
            "name": "date_time_picker_date_and_time",
            "label": "meeting",
            "type": "DATE_AND_TIME"
          }
        },
        {
          "textParagraph": {
            "text": "A datetime picker widget with just date:"
          }
        },
        {
          "divider": {}
        },
        {
          "dateTimePicker": {
            "name": "date_time_picker_date_only",
            "label": "Choose a date",
            "type": "DATE_ONLY",
            "onChangeAction":{
              "all_widgets_are_required": true
            }
          }
        },
        {
          "textParagraph": {
            "text": "A datetime picker widget with just time:"
          }
        },
        {
          "divider": {}
        },
        {
          "dateTimePicker": {
            "name": "date_time_picker_time_only",
            "label": "Select a time",
            "type": "TIME_ONLY"
          }
        }
      ]
    }
  ]
}
Definir uma ação all_widgets_are_required no menu suspenso

JSON

{
  "sections": [
    {
      "header": "Section Header",
      "collapsible": true,
      "uncollapsibleWidgetsCount": 1,
      "widgets": [
        {
          "selectionInput": {
            "name": "location",
            "label": "Select Color",
            "type": "DROPDOWN",
            "onChangeAction": {
              "all_widgets_are_required": true
            },
            "items": [
              {
                "text": "Red",
                "value": "red",
                "selected": false
              },
              {
                "text": "Green",
                "value": "green",
                "selected": false
              },
              {
                "text": "White",
                "value": "white",
                "selected": false
              },
              {
                "text": "Blue",
                "value": "blue",
                "selected": false
              },
              {
                "text": "Black",
                "value": "black",
                "selected": false
              }
            ]
          }
        }
      ]
    }
  ]
}

Definir a validação de um widget de entrada de texto

No campo de validação do widget textInput, é possível especificar o limite de caracteres e o tipo de entrada para esse widget de entrada de texto.

Definir um limite de caracteres para um widget de entrada de texto

JSON

{
  "sections": [
    {
      "header": "Tell us about yourself",
      "collapsible": true,
      "uncollapsibleWidgetsCount": 2,
      "widgets": [
        {
          "textInput": {
            "name": "favoriteColor",
            "label": "Favorite color",
            "type": "SINGLE_LINE",
            "validation": {"character_limit":15},
            "onChangeAction":{
              "all_widgets_are_required": true
            }
          }
        }
      ]
    }
  ]
}
Definir o tipo de entrada para um widget de entrada de texto

JSON

{
  "sections": [
    {
      "header": "Validate text inputs by input types",
      "collapsible": true,
      "uncollapsibleWidgetsCount": 2,
      "widgets": [
        {
          "textInput": {
            "name": "mailing_address",
            "label": "Please enter a valid email address",
            "type": "SINGLE_LINE",
            "validation": {
              "input_type": "EMAIL"
            },
            "onChangeAction": {
              "all_widgets_are_required": true
            }
          }
        },
        {
          "textInput": {
            "name": "validate_integer",
            "label": "Please enter a number",
              "type": "SINGLE_LINE",
            "validation": {
              "input_type": "INTEGER"
            }
          }
        },
        {
          "textInput": {
            "name": "validate_float",
            "label": "Please enter a number with a decimal",
            "type": "SINGLE_LINE",
            "validation": {
              "input_type": "FLOAT"
            }
          }
        }
      ]
    }
  ]
}

Resolver problemas

Quando um app ou card do Google Chat retorna um erro, a interface do Chat mostra a mensagem "Ocorreu um erro". ou "Não foi possível processar sua solicitação". Às vezes, a interface do Chat não mostra nenhuma mensagem de erro, mas o app ou card do Chat produz um resultado inesperado. Por exemplo, uma mensagem do card pode não aparecer.

Embora uma mensagem de erro não apareça na interface do Chat, mensagens de erro descritivas e dados de registro estão disponíveis para ajudar você a corrigir erros quando o registro de erros para apps do Chat está ativado. Para ajuda com a visualização, depuração e correção de erros, consulte Resolver e corrigir erros do Google Chat.