Create a Table of Contents in PDF Using Power Automate
In this guide, you’ll learn how to add a table of contents to a PDF using Power Automate. For this example, you’ll be adding a table of contents to your PDF in SharePoint, but you can easily customize this example to destinations like Dropbox, Google Drive, OneDrive, or any platform supported by Power Automate.
Prerequisites
Ensure the following prerequisites are in place:
-
An Office 365 subscription with a SharePoint Online license.
-
A full subscription to Muhimbi [Document Converter for Power Automate][]. Note that the free version doesn’t support this functionality.
-
Appropriate privileges to create flows.
-
Working knowledge of SharePoint Online and Microsoft Power Automate (Flow).
How to Create a Table of Contents
The steps to convert and create a table of contents from a document are outlined below. The flow will look as follows:
1. Creating a New Flow
Create a new flow using the When a file is created or modified in a folder SharePoint Online trigger. Fill out the URL for the site collection and select the relevant SharePoint folder.
2. Converting a Document
Insert Muhimbi’s Convert document action and fill it out as shown in the screenshot below. Substitute the Source file name and Source file content fields with suitable values.
-
Source file name — x-ms-file-name-encoded, which is the output of When a file is created or modified in a folder.
-
Source file content — File Content, which is the output of When a file is created or modified in a folder.
-
Output format — PDF
-
Override settings — Refer to the following:
\<?xml version="1.0" encoding="UTF-8"?\> \<Override\> \<ConversionSettings\> \<TOCSettings\> \<MinimumEntries\>0\</MinimumEntries\> \<Bookmark\>Table Of Contents\</Bookmark\> \<Location\>Front\</Location\> \<Properties\> \<NameValuePair\> \<Name\>title\</Name\> \<Value\>Index\</Value\> \</NameValuePair\> \</Properties\> \<Template\>\<\![CDATA\[\<?xml version="1.0" encoding="UTF-8"?\> \<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:msxsl="urn:schemas-microsoft-com:xslt" version="1.0" exclude-result-prefixes="msxsl"\> \<xsl:output method="html" indent="yes" /\> \<xsl:template match="/toc"\> \<html\> \<head\> \<style type="text/css"\>ul.toc { margin: 0; padding: 0; list-style: none; } ol.toc { margin: 0; padding: 0; margin-left: 10px; list-style: none; } ul.toc li, ol.toc li { list-style: none; overflow: hidden; } .bookmark-container { position: relative; width: 100%; } .bookmark-container table { width: 100%; display: inline-table; border-collapse: collapse; } .bookmark-container td { padding:0; } td.page { vertical-align: bottom; text-align: right; } a.toc { text-decoration: none; color: \#000; } span.title { background-color: white; padding-right: 4px; } span.page { text-align: right; padding-left: 4px; background-color: white; } .dots { width: 100%; height: 2px; position: absolute; bottom: 5px; border-bottom: 2px dotted black; z-index: -1; } \</style\> \</head\> \<body\> \<h1\> \<xsl:value-of select="properties/property\[@name='title'\]" /\> \</h1\> \<br /\> \<br /\> \<xsl:apply-templates /\> \</body\> \</html\> \</xsl:template\> \<xsl:template match="topics"\> \<ul class="toc"\> \<xsl:apply-templates /\> \</ul\> \</xsl:template\> \<!-- Empty template so properties are not appearing --\> \<xsl:template match="properties" /\> \<xsl:template match="topic\[@level='0'\]"\> \<li\> \<div class="bookmark-container"\> \<table\> \<tr\> \<td\> \<span class="title" style="font-weight: 900;"\> \<xsl:element name="a"\> \<xsl:attribute name="href"\> \<xsl:value-of select="@target" /\> \</xsl:attribute\> \<xsl:attribute name="class"\>toc\</xsl:attribute\> \<xsl:value-of select="@title" /\> \</xsl:element\> \</span\> \</td\> \<td class="page"\> \<span class="page"\> \<xsl:element name="a"\> \<xsl:attribute name="href"\> \<xsl:value-of select="@target" /\> \</xsl:attribute\> \<xsl:attribute name="class"\>toc\</xsl:attribute\> \<xsl:value-of select="@page" /\> \</xsl:element\> \</span\> \</td\> \</tr\> \</table\> \<div class="dots" style="" /\> \</div\> \</li\> \<ol class="toc"\> \<xsl:apply-templates /\> \</ol\> \</xsl:template\> \<xsl:template match="topic"\> \<li\> \<div class="bookmark-container"\> \<table\> \<tr\> \<td\> \<span class="title"\> \<xsl:element name="a"\> \<xsl:attribute name="href"\> \<xsl:value-of select="@target" /\> \</xsl:attribute\> \<xsl:attribute name="class"\>toc\</xsl:attribute\> \<xsl:value-of select="@title" /\> \</xsl:element\> \</span\> \</td\> \<td class="page"\> \<span class="page"\> \<xsl:element name="a"\> \<xsl:attribute name="href"\> \<xsl:value-of select="@target" /\> \</xsl:attribute\> \<xsl:attribute name="class"\>toc\</xsl:attribute\> \<xsl:value-of select="@page" /\> \</xsl:element\> \</span\> \</td\> \</tr\> \</table\> \<div class="dots" style="" /\> \</div\> \</li\> \<ol class="toc"\> \<xsl:apply-templates /\> \</ol\> \</xsl:template\> \</xsl:stylesheet\>\]\]\>\</Template\> \</TOCSettings\> \</ConversionSettings\> > \</Override\
3. Creating a File
Insert a Create file action to write the converted file with the table of contents back to SharePoint Online.
-
File Name —
Base File name.pdf
, which is the output variable of the Convert document action. -
File Content — Processed file content, which is the output variable of the Convert document action.
4. Publishing the Workflow
Publish the workflow and upload any document. After a few seconds, the flow will trigger and a compressed PDF file will be generated in the target location.
Note: The best way to test this is to convert an MS Word file that contains proper heading definitions (e.g. Heading 1, Heading 2, etc.).
Troubleshooting/Caveats
HTML isn’t the best format for print or PDF conversion purposes. Depending on your exact needs and source material, you may need to experiment with various available settings. If you have any questions or require assistance, please feel free to contact us.
Please consider the following when using Power Automate:
-
Modern views — In SharePoint Online, it’s possible to display lists and libraries in modern or classic mode. Modern view is undergoing constant change (by Microsoft), and some modern view pages will convert fine, while others don’t. As a result, we currently only support the conversion of pages displayed in classic view.
-
Multi-factor authentication (MFA) — An increasing number of SharePoint Online/Office 365 environments are secured using MFA. Although this is a positive development, it gets in the way of automated systems, including our HTML converter. If your environment uses MFA, create an app password for HTML conversions, or disable MFA on your HTML conversion account.
-
HTML conversion option isn’t displayed — The option to convert the current page to PDF is only available in classic SharePoint pages. Modern pages currently don’t provide the ability to extend the personal actions menu. As soon as Microsoft makes this functionality available, we’ll add it.
-
PDF layout doesn’t match — By default, the page being converted is optimized for print/PDF output. If you prefer for a PDF to look exactly like what’s displayed onscreen, including user interface elements, set the Media Type to Screen in the HTML conversion configuration screen.
-
User-modified content isn’t converted — The converter doesn’t have knowledge about the information currently displayed in the user’s browser (e.g. size of the browser window, or custom data entered but not yet submitted to the server). The converter always requests a fresh copy of the URL and converts that information to PDF.
-
User-specific information is missing — HTML conversion is carried out in the context of the configured user account, though the system isn’t aware of the user carrying out the HTML conversion. Don’t specify accounts that display more information than what users need to have access to.