Skip to content

Lab MCS2 - Defining Topics

In this lab, you'll learn how to create custom Topics in Microsoft Copilot Studio. Topics are core building blocks of an agent. With topics you can provide a conversational experience to your end users, where the conversation can be single-turn or multi-turn. Topics define how the conversation between the user and the agent can evolve, through discrete interaction paths.

Note

This lab builds on the previous one, Lab MCS1. You should be able to continue working on the same agent, improving its functionalities with new capabilities.

Topics can be created with a graphical designer or describing the intent with natural language. Once you have created a new Topic, you can also edit its definition with a low level code editor, in case you need to apply detailed fine tuning.

There are two different flavors of Topics:

  • System Topics: automatically defined by Microsoft Copilot Studio. Can be disabled, but cannot be deleted.
  • Custom Topics: created by agent authors to provide custom interaction paths.

Additional information about topics

You can find additional information about Topics in agents made with Microsoft Copilot Studio reading the following article. You can also dig into the System Topics reading the article Use system topics.

In this lab you will learn:

  • How to rely on Orchestration based on generative AI
  • How to create a single-turn topic
  • How to create a multi-turn topic
  • How to use adaptive cards to interact with users

Exercise 1 : Orchestration based on Generative AI

In this first exercise you are going to enable the agent that you created in Lab MCS1 to use Orchestration based on Generative AI, which is a new feature under preview at the time of this writing.

Step 1: Enabling Orchestration based on generative AI

One important feature of an agent made with Copilot Studio, is generative orchestration. With generative orchestration, your agent can choose the best knowledge base, topics, and actions to use to interact with the user and answer to the user's queries, or to respond to event triggers.

By default agents use classic orchestration, which means that an agent responds to users by triggering a topic whose trigger phrases match most closely the user's query. With generative orchestration, Copilot Studio understands the intent of the user processing the prompt provided by the user in natural language and determines the best item to trigger.

Disclaimer

Enabling generative orchestration can impact how billing is calculated. Learn more about billing for generative mode. There are key differences between classic and generative orchestration, such as how knowledge is searched, and the supported data sources. Before turning on generative mode for an existing agent, read about the known limitations.

To enable generative orchestration, open a browser and, using the work account of your target Microsoft 365 tenant, go to https://copilotstudio.microsoft.com to start using Microsoft Copilot Studio.

Select to view the list of 1️⃣ Agents and then 2️⃣ edit the agent that you created in the previous Lab MCS1.

The interface of Microsoft Copilot Studio when browsing the agents and selecting one item to edit.

In the Overview tab enable the toggle with name Orchestration like illustrated in the following screenshot.

The interface of Microsoft Copilot Studio with the generative orchestration enabled and highlighted.

It usually takes a while to enable the generative orchestration. Once the setting is applied, publish the agent to confirm the changes.


Exercise 2 : Creating a single-turn topic

In this exercise you are going to create a new topic to collect an input from the user and to provide a feedback based on the collected input. Specifically, the topic collects information about the current user's role and gives back guidance about what to do with the agent based on the provided role.

Step 1: Creating a new single-turn topic

To create a new topic, in the upper side of the screen select the 1️⃣ Topics tab, then select 2️⃣ + Add a topic, and then 3️⃣ From blank to start creating a new custom topic.

Creating topics with Copilot

Notice that you can also create a new topic simply providing a description with natural language, allowing Copilot to draft the topic for you.

The interface of Microsoft Copilot Studio when creating a new topic. There is the **Topics** tab highlighted, with the **+ Add a topic** dropdown menu, and the **From blank** option highlighted.

Copilot Studio will provide you the graphical designer to define the new topic. The very first building block of the topic is the Trigger action, which is used to describe what the topic does. When generative orchestration is enabled, in the action you can find a text area where you can define with natural language what the purpose of the topic is. For the sake of working on this lab, you can provide the following content:

This topic can handle queries like these: collect user's role and provide feedback, 
give me a feedback based on my role, what's your feedback for my role?

If you use classic orchestration, you can specify between 5 and 10 triggering phrases or sentences, instead of a single descriptive text.

The interface of Microsoft Copilot Studio when designing a new topic. There is a **Trigger** action with the value suggested in this exercise step as the trigger condition. There is also the button to add new actions highlighted.


Step 2: Collecting user's input

Select the + button in the middle of the screen to add new actions or steps to the current topic. By selecting the + button, a list of all the available actions appears. The main options are:

  • Send a message: to send a message to the user. The message can be text, an image, a video, an adaptive card, etc.
  • Ask a question: to ask an input to the user. The input can be text, an image, a video, an attachment, an adaptive card, etc.
  • Ask with adaptive card: to collect content from the user utilizing an Adaptive Card.
  • Add a condition: to add branches to the topic based on comparison between variables and/or constant values.
  • Variable management: to manage variables, which can be scoped at topic level, global, system, or environment.
  • Topic management: to manage the lifecycle of the current topic.
  • Add an action: to consume external actions like Power Automate flows, custom connectors, or other agents in a multi-agent scenario.
  • Advanced: to provide advanced capabilities like consuming an external HTTP REST API, using generative answers, sending events or activities, etc.

The menu to select actions to add to the current topic. There available options are: send a message, ask a question, ask with adaptive card, add a condition, variable management, topic management, add an action, advanced.

In order to collect user's input, add an Ask a question action. Since we want to ask the user what her/his role is, let's provide the following value for the question text.

What is your role?

Notice that by default Copilot Studio assigns a Multiple choice options data type to the collected input, as highlighted in the Identify configuration field. Select the + New option command just below the Identify configuration field and add one by one the following three values:

  • Candidate
  • Employee
  • HR staff member

The action automatically stores the value selected by the user into a variable stored at topic-level. You can select the three dots in the upper right corner of the action and select the Properties command to customize the action, or you can select the variable in the lower part of the action to update its name and other settings.

The context menu of the action with commands to: see properties, rename the action, delete the action, add a comment to the action.

For example, you can rename the variable to UserRole. You can see how the action looks like once it is fully configured.

The action fully configured with all the settings and commands highlighted. There is the question text, the data type for the result, the options, the variable to store the selected option, and the scope of the variable.


Step 3: Providing feedback to the user

You will notice that, while adding new options to the Ask a question action, Copilot Studio automatically creates a set of conditions for all the branches matching each of the provided options, plus one last condition for All other conditions. Inside each branch, you can specify some custom logic to provide a specialized feedback to the user. To do so, select the + command below each of the Condition branches and add one action of type Send a message. You can eventually add more than one action to each branch.

For the three branches, you could for example provide the following feedback messages:

  • Candidate
You are a new candidate, as such you can submit your resume for evaluation.
  • Employee
As an employee, you can ask me about how to improve your career or about 
how to learn more about your growth mindset.
  • HR staff member
As an HR staff member, you can manage the list of candidates and all the hiring procedures

For the All other conditions branch, you can configure a Redirect action, available under the Topic management group, to fallback to the System Topic with name Fallback.

The condition branches with messages sent to the user for each option and a redirection to the **Fallback** topic in case the user did not provide any of the supported options.

The very simple logic of the topic is now ready.


Step 4: Ending the current topic

To properly complete the flow of the custom topic, add a new action of type End current topic, which is available under the Topic management group. This last action will instruct Copilot Studio about the fact that your topic conversation is completed.

The **End current topic** action inserted in the current topic flow.


Step 5: Testing the current topic

You are now ready to save and test the topic. Select the Save button in the upper right corner of the designer, provide a name for the topic through the dialog window that will appear, and select the Save button again.

The dialog window to assign a name to the topic and to save it.

For example, name the topic as Collect user's role. Now open the test panel by selecting the Test command in the upper right corner of the designer. Type the following prompt:

What's your feedback for my role?

The agent will prompt you to choose a role and, based on your selection, you will get a specific feedback like illustrated in the following screenshot.

The agent in action in the test panel inside Microsoft Copilot Studio.

Exercise 2 is now completed, well done! You can now move to Exercise 3 to create a multi-turn topic.


Exercise 3 : Creating a multi-turn topic

For simple interactions, you create single-turn conversations with just one question and one answer. But more substantive topics require a multi-turn conversation with multiple back-and-forth interactions between the user and the agent. In this exercise, you are going to create a multi-turn topic to collect data about a potential candidate for a new role.

Step 1: Creating a new multi-turn topic

Imagine that you want to create a topic to collect the following information about a new candidate:

  • First name
  • Last name
  • E-mail
  • Current role

To collect the above information, create a new topic following the instructions provided in Exercise 2 Step 1. The Trigger description for the topic could be:

This topic helps to collect information about a new candidate to process. Trigger sentences can be: 
register a new candidate, create a new candidate, add a new candidate.

Then, for each of the above information fields, you need to prompt the user with an action of type Ask a question. However, in this exercise the value of the answer will vary depending on the requested field. For example: first name, last name, and current role will be simple text fields., while the e-mail field will need to be a valid e-mail.

So for the first name, last name, and current role fields you can select the Identify property of the Ask a question action and you can select User's entire response as the actual entity type. This will get the text value provided by the user as the actual collected value. The variable type will be automatically updated to be a string by Copilot Studio. Give a meaningful name to each of the variables holding the values provided by the user. In the following screenshot you can see how the first name input action can be defined. You can do the same for last name and current job role.

The **Ask a question** action configured to collect the candidate's first name and store it into a variable of type string, accepting any value provided by the user.

For the e-mail field, choose Email as the entity for the Identify property, so that Copilot Studio will automatically validate the user's input against the e-mail validation rules. The underlying variable will still be a string.

The **Ask a question** action configured to collect the candidate's e-mail accepting only values of type email provided by the user.

Now you are ready to collect all of the information about the candidate and you can give a feedback to the user.


Step 2: Giving a feedback to the user

Based on the collected input, you can now send a message to the user confirming the collected data. Add a new action of type Send a message and build the content of the message using the variables where the collected inputs were stored. To add variables to the message, simply select the {x} command in the toolbar of the Send a message action and select the variable that you are looking for.

The action **Send a message** with the insert variable command highlighted and the list of variables available in the current topic.

You can insert variables defined in the current topic, system variables, or environment variables. Once you have configured the recap message with all the variables, it should look like the following screenshot.

The action **Send a message** with all the referenced variables.

Just for the sake of having a final confirmation from the user, insert an action of type Ask a question, provide the following message:

Is it ok for you to insert this new candidate?

And configure the action to support the answers Yes and No. As like as it happened in Exercise 2 Step 3, you can now configure the various branches for each of the outcomes. For the sake of simplicity, you can simply rely on a couple of Send a message actions, one for each branch, and put a thumb up or a thumb down emoji as the actual content on the message, depending on the user's feedback. Lastly, add an action of type End current topic to complete the topic.

The final part of the topic with the last **Ask a question** action, three branches to manage the user's input and one final action to **End current topic**.

You can now save the topic, give it a name like for example Register a new candidate and test it in the integrated testing interface. Here follows a screenshot of the interaction with the multi-turn topic. You can also notice that when the value provided for the e-mail field is not correct, Copilot Studio automatically re-prompts the user to provide a correct value.

The interaction with the multi-turn topic, where there are a set of questions and answers to collect all the candidate data.


Exercise 4 : Using Adaptive Cards

Collecting inputs using multiple instances of Ask a question actions is definitely one option that you have. However, when you need to collect a lot of data or when you want to have a nice looking interaction with the user you can consider using an Adaptive Card.

What are Adaptive Cards? Adaptive Cards are platform-independent UI snippets authored in JSON that can be exchanged between apps and services. Once delivered to an app, the JSON transforms into native UI that automatically adapts to its environment. This enables the design and integration of lightweight UI across major platforms and frameworks.
Adaptive cards are everywhere

Step 1: Collecting input with Adaptive Cards

Imagine that you want to create yet another topic to collect the following information about a new candidate:

  • First name
  • Last name
  • E-mail
  • Current role
  • Spoken languages
  • Skills

Specifically, spoken languages and skills are multi-select lists of values.

So open the Topics tab, and disable the topic that you created in Exercise 3 to avoid any overlapping of trigger conditions. Then, create a new topic following the instructions provided in Exercise 2 Step 1. The Trigger description for the topic could be:

This topic helps to collect information about a new candidate to process. Trigger sentences can be: 
register a new candidate, create a new candidate, add a new candidate.

Now add a new action of type Ask with adaptive card, 1️⃣ select the body of the new action and 2️⃣ type the following JSON into the text area available in the property pane of the action.

The interface of the topic when adding an **Ask with adaptive card** action, with the side panel open to define the JSON of the adaptive card.

{
    "type": "AdaptiveCard",
    "body": [
        {
            "type": "TextBlock",
            "size": "Medium",
            "weight": "Bolder",
            "text": "New HR Candidate Information"
        },
        {
            "type": "Input.Text",
            "id": "firstname",
            "placeholder": "First Name"
        },
        {
            "type": "Input.Text",
            "id": "lastname",
            "placeholder": "Last Name"
        },
        {
            "type": "Input.Text",
            "id": "email",
            "placeholder": "Email"
        },
        {
            "type": "Input.Text",
            "id": "current_role",
            "placeholder": "Current Role"
        },
        {
            "type": "Input.ChoiceSet",
            "id": "spoken_languages",
            "placeholder": "Spoken Languages",
            "isMultiSelect": true,
            "choices": [
                {
                    "title": "English",
                    "value": "English"
                },
                {
                    "title": "French",
                    "value": "French"
                },
                {
                    "title": "German",
                    "value": "German"
                },
                {
                    "title": "Italian",
                    "value": "Italian"
                },
                {
                    "title": "Portuguese",
                    "value": "Portuguese"
                },
                {
                    "title": "Spanish",
                    "value": "Spanish"
                }
            ]
        },
        {
            "type": "Input.ChoiceSet",
            "id": "skills",
            "placeholder": "Skills",
            "isMultiSelect": true,
            "choices": [
                {
                    "title": "Computer Programming",
                    "value": "Computer Programming"
                },
                {
                    "title": "Project Management",
                    "value": "Project Management"
                },
                {
                    "title": "Team Lead",
                    "value": "Team Lead"
                }
            ]
        }
    ],
    "actions": [
        {
            "type": "Action.Submit",
            "title": "Submit"
        }
    ],
    "$schema": "http://adaptivecards.io/schemas/adaptive-card.json",
    "version": "1.5"
}

Once you updated the JSON of the adaptive card, Copilot Studio renders the preview of the card in the UI of the topic designer, as you can see in the following screenshot. Copilot Studio also automatically defines a set of topic-level variables to collect the values provided by the user.

The interface of the topic when adding an **Ask with adaptive card** action, with the side panel open and a proper JSON of the adaptive card defined. On the designer there is a preview of the actual adaptive card and a list of output arguments to collect the values provided by the user.


Step 2: Providing feedback with Adaptive Cards

You can now use an adaptive card also to provide feedback to the user, showing a recap of the collected data. Select the + command to add a new action of type Send a message. Then select + Add in the upper left corner of the new action and select Adaptive card to switch to a message of type adaptive card.

The command box to add an adaptive card as the message to send with the **Send a message** action.

A side panel shows up allowing you to define the content of the adaptive card. Copy and paste the following JSON into the text area.

{
  "type": "AdaptiveCard",
  "body":
    [
      {
        "type": "TextBlock",
        "size": "Medium",
        "weight": "Bolder",
        "text": "Candidate Summary"
      },
      {
        "type": "FactSet",
        "facts":
          [
            { "title": "First Name:", "value": "firstname" },
            { "title": "Last Name:", "value": "lastname" },
            { "title": "Email:", "value": "email" },
            { "title": "Current Role:", "value": "current_role" },
            { "title": "Spoken Languages:", "value": "spoken_languages" },
            { "title": "Skills:", "value": "skills" }
          ]
      }
    ],
  "$schema": "http://adaptivecards.io/schemas/adaptive-card.json",
  "version": "1.3"
}

It is very important that, when pasting the JSON of the adaptive card, you have the Edit JSON option selected (which is by the way the default one) at the top of the text area. That means that you are editing the actual JSON of the adaptive card. As soon as you will move the focus out of the text area where you pasted the JSON, the Send a message action will start previewing the adaptive card. As you can see the adaptive card is rendering static values for all the variables (firstname, lastname, etc.).

The **Send a message** action configured to render an adaptive card. In the side panel on the right side there is the JSON of the adaptive card. In the body of the action there is a preview of the adaptive card.

Now, select the drop down with Edit JSON and switch to Edit Formula in order to replace the static values with the actual variables defined in the current Topic.

The **Send a message** action configured to render an adaptive card. In the side panel on the right side there is the JSON of the adaptive card. In the body of the action there is a preview of the adaptive card.

Select the button to expand the editor and you can start replacing the static values with formulas that will refer to the actual value of the topic-level variables.

The side panel of the **Send a message** action while editing the adaptive card JSON in **Edit formula** mode. There is a button to expand the editor.

You will see a dialog popping up and providing an advanced editor with intellisense helping you to reference variables and PowerFx functions, like illustrated in the following screenshot.

The adaptive card editor with intellisense to reference variables and PowerFx functions.

One by one, you can replace all the static values with the actual variables. In particular, the spoken languages and the skills are list of values (variables of type Table) so, in order to render their value, you need to rely on the PowerFx Concat function and then on the PowerFx Text function to convert the result into an actual string. Here you can find the content of the adaptive card JSON when all the formulas are in place.

{
  type: "AdaptiveCard",
  body: [
    {
      type: "TextBlock",
      size: "Medium",
      weight: "Bolder",
      text: "Candidate Summary"
    },
    {
      type: "FactSet",
      facts: [
        {
          title: "First Name:",
          value: Topic.firstname
        },
        {
          title: "Last Name:",
          value: Topic.lastname
        },
        {
          title: "E-mail:",
          value: Topic.email
        },
        {
          title: "Current Role:",
          value: Topic.current_role
        },
        {
          title: "Spoken Languages:",
          value: Text(Concat(Topic.spoken_languages, Value, ", "))
        },
        {
          title: "Skills:",
          value: Text(Concat(Topic.skills, Value, ", "))
        }
      ]
    }
  ],
  '$schema': "http://adaptivecards.io/schemas/adaptive-card.json",
  version: "1.3"
}

Additional information about PowerFx functions

You can find additional information about the PowerFx functions reading the article Create expressions using Power Fx


Step 3: Testing the current topic

Now add an End current topic action at the end of the topic, save it, give it a name like for example Register new candidate with adaptive cards, and test it in the test panel on the right side of the agent designer. In the following screenshot you can see how the topic interacts with a user.

The topic collecting information about a new candidate using an adaptive card and providing feedback through another adaptive card.

Now your topic is ready and fully functional. In the upcoming labs you will learn how to create a new candidate record for real, storing data in an external HR service.


CONGRATULATIONS!

Now your agent supports multiple conversation paths through different topics. In the next labs you will learn how to work with custom Actions.

Start here with Lab MCS3, to define actions in your agent with Copilot Studio.