In November 2020, I was happy to release my side project "Snooze Bot" as a
free app on the Microsoft Teams store:
https://appsource.microsoft.com/en-us/product/office/WA200002297
I had been working on it for a few weeks. The fact that all of us were under
lockdown gave me some extra time in the evenings and weekends to focus on
learning the Microsoft Teams platform and create an app on it which addressed
a gap which I noticed in my day to day use.
We all get a lot of Teams messages daily and need a way to manage them or come
back to them at a later time. Snooze bot helps us do exactly that. It lets us
Snooze message which we want to deal with later. When a message is snoozed, we
get an option to select the duration after which Snooze Bot should remind us
about the message. When the time arrives, the bot will send you personal
message in teams reminding about the snoozed message.
If you haven't checked out Snooze Bot yet, feel free to install it and give it
a try. I am happy to hear any feedback and potential improvements.
One of my goals when creating the app was to learn about the Microsoft Teams
developer platform and also blog about the interesting things I came across.
So in this series of posts, let's outline some Microsoft Teams development
concepts which I found really useful. The first one being posting an Adaptive
Card carousel as a welcome message when the bot is added by the user.
It's always recommended as a good practice to send a welcome message when the
user adds the bot. According to the Microsoft docs:
In personal contexts, welcome messages set your bot's tone. The message
includes a greeting, what the bot can do, and some suggestions for how to
interact (for example, “Try asking me about …”). If possible, these
suggestions should return stored responses without having to sign in.
Also, sending the welcome message one of the requirements before the app is accepted in AppSource by the validation team.
So we can send a simple chat message from the bot to the user as a welcome
message. So why go for an Adaptive Card carousel? This is because adding too
much information in a single message can get overwhelming for the user and
they might be tempted to just skip it. Also if your bot has a
lot of functionality you need a way to efficiently present that information to
the user. This is where carousels created by Adaptive Cards some into
play:
The Adaptive Card json:
First, we need to define the Adaptive cards which will show up in the welcome message. I am storing mine as json files in my solution. The cards contain helpful text and also links to images which show the functionality of Snooze Bot
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
{ | |
"type": "AdaptiveCard", | |
"version": "1.2", | |
"body": [ | |
{ | |
"type": "TextBlock", | |
"size": "extraLarge", | |
"text": "Snooze Bot" | |
}, | |
{ | |
"type": "TextBlock", | |
"text": "You can use me to snooze teams messages in personal chats, groups chats and channels.", | |
"wrap": true | |
}, | |
{ | |
"type": "Image", | |
"url": "https://welcomemessagestrg.blob.core.windows.net/images/welcomeuser.png" | |
}, | |
{ | |
"type": "TextBlock", | |
"text": "Click on the **three dots** next to any Microsoft Teams chat message, then **More actions** and select **Snooze message**", | |
"wrap": true | |
} | |
] | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
{ | |
"type": "AdaptiveCard", | |
"version": "1.2", | |
"body": [ | |
{ | |
"type": "TextBlock", | |
"text": "When the selected time arrives, I will send you a personal card reminding you about the snoozed message. From the personal card, either you can go directly to the Teams message or you can snooze the message again", | |
"wrap": true | |
}, | |
{ | |
"type": "Image", | |
"url": "https://welcomemessagestrg.blob.core.windows.net/images/welcomeuser2.PNG" | |
} | |
] | |
} |
The Bot:
2) Get the Adaptive cards from the json files
3) Insert the adaptive cards into a Bot Framework carousel and send it to the user
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using AdaptiveCards; | |
using Microsoft.Bot.Builder; | |
using Microsoft.Bot.Builder.Teams; | |
using Microsoft.Bot.Schema; | |
using System; | |
using System.Collections.Generic; | |
using System.IO; | |
using System.Threading; | |
using System.Threading.Tasks; | |
namespace TeamsBlogBot | |
{ | |
public class SnoozeBot : TeamsActivityHandler | |
{ | |
//Fired when the Teams app is added by the user | |
protected override async Task OnMembersAddedAsync(IList<ChannelAccount> membersAdded, ITurnContext<IConversationUpdateActivity> turnContext, CancellationToken cancellationToken) | |
{ | |
foreach (var member in membersAdded) | |
{ | |
//membersAdded also contains the id of the bot. So confirm if the user being added is not the bot itself. | |
if (member.Id != turnContext.Activity.Recipient.Id) | |
{ | |
//Create a list out of the Adaptive Cards | |
var welcomeAttachments = new List<Attachment>(); | |
var adaptiveCardAttachment = GetCardAttachment("WelcomeCard1.json"); | |
welcomeAttachments.Add(adaptiveCardAttachment); | |
var adaptiveCardAttachment2 = GetCardAttachment("WelcomeCard2.json"); | |
welcomeAttachments.Add(adaptiveCardAttachment2); | |
//Send the carousel back to Teams | |
await turnContext.SendActivityAsync(MessageFactory.Carousel(welcomeAttachments)); | |
} | |
} | |
} | |
internal Attachment GetCardAttachment(string cardFileName) | |
{ | |
string path = Path.Combine(Environment.CurrentDirectory, @"AdaptiveCardJson\", cardFileName); | |
string json = GetCardJSON(path); | |
var adaptiveCardResult = AdaptiveCard.FromJson(json); | |
return new Attachment | |
{ | |
Content = adaptiveCardResult.Card, | |
ContentType = AdaptiveCard.ContentType, | |
}; | |
} | |
public static string GetCardJSON(string path) | |
{ | |
if (!File.Exists(path)) | |
return string.Empty; | |
using (var f = File.OpenText(path)) | |
{ | |
return f.ReadToEnd(); | |
} | |
} | |
} | |
} |
No comments:
Post a Comment