Sample code for Microsoft Teams application with SSO, Bots, Messaging extensions and other capabilities.
This lab is part of extending app capabilities for your teams app which begins with a Northwind Orders core application using the AAD
path. The core app is the boilerplate application with which you will do this lab.
Complete labs A01-A03 for deeper understanding of how the core application works, to set up AAD application registration etc. to update the
.env
file as per the.env_sample
. This configuration is required for the success of the lab.
The completed lab is here
Deep links help the user to directly navigate to the content. In this lab we will create deep link to entities in Teams so the user can navigate to contents within the app’s tab.
In this exercise you will learn new concepts as below:
Use below syntax, to create the deep link for this lab.
https://teams.microsoft.com/l/entity/<app-id>/<entitiyId>?context={"subEntityId": "<subEntityId>"}
staticTabs
object for the particular entity (tab). In our case this is the entity id Orders
of My Orders
tab.1. client\page\orderDetail.html
Let’s add the copy to clipboard button and a div to display a message to show if the copy was successful.
Add below block of code and paste it above orderDetails
div element.
<div id="copySection" style="display: none;">
<div> <button id="btnCopyOrderUrl">Copy order url</button></div>
<div style="flex-grow: 1;padding:5px;" id="copyMessage">Copy to clipboard</div>
</div>
2. client\page\orderDetail.js
Import the teams SDK module as well as the environment variable we exported to use.
Paste below code above the displayUI() function definition.
import { env } from '/modules/env.js';
Replace the displayUI() function with below definition:
async function displayUI() {
const displayElement = document.getElementById('content');
const detailsElement = document.getElementById('orderDetails');
const copyUrlElement = document.getElementById('btnCopyOrderUrl');
const copyMsgElement = document.getElementById('copyMessage');
const copySectionElement = document.getElementById('copySection');
const errorMsgElement = document.getElementById('message');
try {
const searchParams = new URLSearchParams(window.location.search);
if (searchParams.has('orderId')) {
const orderId = searchParams.get('orderId');
const order = await getOrder(orderId);
displayElement.innerHTML = `
<h1>Order ${order.orderId}</h1>
<p>Customer: ${order.customerName}<br />
Contact: ${order.contactName}, ${order.contactTitle}<br />
Date: ${new Date(order.orderDate).toDateString()}<br />
${order.employeeTitle}: ${order.employeeName} (${order.employeeId})
</p>
`;
order.details.forEach(item => {
const orderRow = document.createElement('tr');
orderRow.innerHTML = `<tr>
<td>${item.quantity}</td>
<td><a href="/pages/productDetail.html?productId=${item.productId}">${item.productName}</a></td>
<td>${item.unitPrice}</td>
<td>${item.discount}</td>
</tr>`;
detailsElement.append(orderRow);
});
copySectionElement.style.display = "flex";
copyUrlElement.addEventListener('click', async ev => {
try {
//temp textarea for copy to clipboard functionality
var textarea = document.createElement("textarea");
const encodedContext = encodeURI(`{"subEntityId": "${order.orderId}"}`);
//form the deeplink
const deeplink = `https://teams.microsoft.com/l/entity/${env.TEAMS_APP_ID}/Orders?&context=${encodedContext}`;
textarea.value = deeplink;
document.body.appendChild(textarea);
textarea.select();
document.execCommand("copy"); //deprecated but there is an issue with navigator.clipboard api
document.body.removeChild(textarea);
copyMsgElement.innerHTML = "Link copied!"
} catch (err) {
console.error('Failed to copy: ', err);
}
});
} else {
errorMsgElement.innerText = `No order to show`;
displayElement.style.display = "none";
orderDetails.style.display = "none";
}
}
catch (error) { // If here, we had some other error
errorMsgElement.innerText = `Error: ${JSON.stringify(error)}`;
}
}
The deep link is created in the copyUrlElement.addEventListener()
function.
The deep link will navigate:
The App ID is included in the deep link to help Teams display the correct application. The teams app id is taken from .env
file, which is the id in the manifest file.
encodedContext
is a JSON constant that defines the parameter(subEntityId) to be passed to the tab; in this case it’s the order ID.
The entityId identifies the tab; in this case it is Orders
, which is the id for My Orders
tab. This is set in the manifest.json file; the relevant section is here for your reference:
....
"staticTabs": [
{
"entityId": "Orders",
"name": "My Orders",
"contentUrl": "https://<HOSTNAME>/pages/myOrders.html",
"websiteUrl": "https://<HOSTNAME>/pages/myOrders.html",
"scopes": [
"personal"
]
},
{
"entityId": "Products",
"name": "Products",
"contentUrl": "https://<HOSTNAME>/pages/categories.html",
"websiteUrl": "https://<HOSTNAME>/pages/categories.html",
"scopes": [
"personal"
]
}
],
....
4. client\myOrders.js
Add import statements for the Microsoft Teams SDK and the inTeams helper function.
import 'https://statics.teams.cdn.office.net/sdk/v1.11.0/js/MicrosoftTeams.min.js';
import { inTeams } from '../modules/teamsHelpers.js';
Using the My Orders
tab as the base, we will redirect the deeplink to Order details
page to show the order only if the subEntitiyId is present in the teams context. In the displayUI()
function, at the top of the try
block, add code to check for a subEntityId and do the redirect if it’s found.
// Handle incoming deep links by redirecting to the selected order
if (await inTeams()) {
microsoftTeams.initialize(async () => {
microsoftTeams.getContext(async (context) => {
if (context.subEntityId) {
window.location.href = `/pages/orderDetail.html?orderId=${context.subEntityId}`;
}
});
});
}
The updated function definition looks like below:
async function displayUI() {
const displayElement = document.getElementById('content');
const ordersElement = document.getElementById('orders');
const messageDiv = document.getElementById('message');
try {
// Handle incoming deep links by redirecting to the selected order
if (await inTeams()) {
microsoftTeams.initialize(async () => {
microsoftTeams.getContext(async (context) => {
if (context.subEntityId) {
window.location.href = `/pages/orderDetail.html?orderId=${context.subEntityId}`;
}
});
});
}
// Display order data
const employee = await getLoggedInEmployee();
if (employee) {
displayElement.innerHTML = `<h3>Orders for ${employee.displayName}<h3>`;
employee.orders.forEach(order => {
const orderRow = document.createElement('tr');
orderRow.innerHTML = `<tr>
<td><a href="/pages/orderDetail.html?orderId=${order.orderId}">${order.orderId}</a></td>
<td>${(new Date(order.orderDate)).toDateString()}</td>
<td>${order.shipName}</td>
<td>${order.shipAddress}, ${order.shipCity} ${order.shipRegion || ''} ${order.shipPostalCode || ''} ${order.shipCountry}</td>
</tr>`;
ordersElement.append(orderRow);
});
}
}
catch (error) { // If here, we had some other error
messageDiv.innerText = `Error: ${JSON.stringify(error)}`;
}
}
5. server\server.js
We need the teams app id from the manifest (which is added based on the value in the .env file). We’ll need to expose this id in the client side code.
Update the request app.get('/modules/env.js')
and add TEAMS_APP_ID as below:
app.get('/modules/env.js', (req, res) => { res.contentType("application/javascript"); res.send(` export const env = { HOSTNAME: "${process.env.HOSTNAME}", TENANT_ID: "${process.env.TENANT_ID}", CLIENT_ID: "${process.env.CLIENT_ID}", TEAMS_APP_ID: "${process.env.TEAMS_APP_ID}" }; `); });
6. manifest\manifest.template.json
Update the version number so it’s greater than it was; for example if your manifest was version 1.4, make it 1.4.1 or 1.5.0. This is required in order for you to update the app in Teams.
"version": "1.5.0"
Now that you have applied all code changes, let’s test the features. This lab assumes you already completed the Core lab and registered an Azure AD application, built and uploaded your Teams package, etc. So at this point, all you should need to do is run
npm start
Once you are in the application, go to My orders
page and select any order.
Select Copy order url.
On selection, the message next to button changes from Copy to clipboard to Link copied!
Login as another user who has Northwind Order app installed in their teams. Open the link in the browser. It should open in the personal tab with the order information displayed. This animated picture shows the solution in action.
After completing this lab, you may continue with any of the following labs.