I have found existing questions about the Table of Contents macro, but none quite match this question.
Background to this question: what I'm doing
I have recently returned to using Confluence frequently after a long period of infrequent use. I'm relearning what Confluence does, and doesn't, do out of the box.
I'm using Confluence to write policies for my employer. We have decided to use an ISO standard as a template for the policy heading structure. The heading structure contains three levels. I have "chunked" the structure into Confluence pages at the highest heading level, giving me 14 pages. Those pages will grow as I add content to the skeleton heading structure.
On returning to Confluence, after spending time in other documentation systems, I had two nasty surprises. I'd be very happy to be wrong about both of them:
I want a single table of contents that includes all of the policy headings. For now, as a workaround, I've created a "single page" version of the policy that consists of a "Table of Contents" macro followed by an "Include Page" macro for each of the 14 "chunks". Each "Include Page" macro is preceded by a Heading 1 paragraph that duplicates the "chunk" page title. A useful side-effect is that I can use this page to search the entire policy using my browser's built-in "Find" function.
On a few occasions, I've needed to perform search'n'replace across the 14 pages of the policy. For now, without a better solution—and I'm not interested in petitioning my employer to buy another for-money Confluence plugin—I've done that by manually editing each of the 14 pages. That's not been fun, but so far I've been too busy writing and editing content to want to look into writing my own search'n'replace code. It's occurred to me to try WebDAV with an external editor, but after making some enquiries to my local Confluence administrator, I dropped that idea (I can explain, if asked).
I'd like to split those 14 pages into smaller chunks, perhaps even right down to the lowest (third) heading level, but I baulk at the thought of:
One feature of the "Table of Contents" macro that I really like is the ability to include or exclude headings based on regular expressions. I'm using that feature to create different versions of the table of contents, based on the presence or absence of strings that I append to the headings, such as "(TODO)" or "(N/A)".
What I want
I want a version of the Table of Contents macro that offers two "checkbox" settings:
Does anyone have this, or something similar, as a user macro that they could share with me?
Failing that: (where, in the installed Confluence product code) can I get the source code of the supplied Table of Contents macro, so that I can use it as the basis for my own version with these additional settings? I have some experience writing Confluence user macros, but that was several years ago.
I'd also like a search-and-replace macro that recurses into descendant pages, but I should ask a separate question about that (I mention it here because it's so closely related to my use case for this TOC macro).
Hi,
I want to know how to include only the headings starting with number
I think you can get what you are looking for using some other macros in conjunction with the TOC.
On your parent page set up a children display macro like this.
On your child pages setup your page like this.
This is what the final result looks like on the parent page.
If you are on the server, as opposed to cloud, version of Confluence you could then add some CSS styling to make the outline look the way you wont on the parent page. Is this what you are looking for.
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
Thank you very much for your answer.
I have tested it with the following page tree hierarchy (note the various levels of "chunking"):
Result:
Perfect!
Notes:
Is it what I'm looking for? I'm in a quandary:
So, I'm not sure whether to accept your answer or not. I sincerely appreciate your answer, but it's not quite what I'm looking for.
I'd still like a single macro, or: can you tell me whether and where, in the installed Confluence product code, I can get the source code of the supplied Table of Contents macro, so that I can use it as the basis for my own version with these additional settings?
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
Re:
I think you can get what you are looking for using some other macros in conjunction with the TOC.
On that note: it occurs to me that I don't necessarily need an enhanced version of the Table of Contents macro to get what I want. As long as Confluence—possibly more accurately, my browser—can cope with the potentially large resultant page for viewing, it occurs to me that a macro that does something similar to the "Include Page" macro, but for all descendants of the current page, and that includes descendant page titles as headings, would do. ("Include Descendant Pages"? Or, in Confluence-speak, "Include Child Pages", with an "Include Descendants" checkbox?) Still, because of that "potentially large" issue, a TOC macro would be preferable.
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
With regards to your concern about many people working on the documentation you could create a page template that already has the TOC and excerpt included in the template so your users won't have to remember to put it in.
As for a macro that will include children I actually wrote a user macro a few years ago that does that very thing. Here is the code. Another option is the Include Content add-on in the marketplace. It has an include children macro as well. My user macro has a few more features
Macro Name:
include_children
Macro Title:
Include Children
Macro Body Processing:
No macro body
Template:
## Macro Name: include_children ## Visibility: ## Macro Title: Include Children ## Description: This will include the content from from the child pages onto the parent page that this macro is placed on. ## Categories: Confluence content ## Icon URL: ## Documentation URL: https://bitbucket.org/fredclown/confluence-user-macros ## Macro Body Processing: No macro body ############################## ## Template: ## ############################## ## Developed By: Davin Studer ## Date Created: 12/4/2014 ## Updated by: Milo Grika ## Added: show/hide title, link/unlink title. ## Updated by: Gregor Mirai ## Date updated: 26/1/2016 ## Added: page separator, PDF split page marker options. ## Updated by: Davin Studer ## Date updated: 4/13/2018 ## Added: Include Descendants checkbox ## Updated by: Davin Studer ## Date updated: 12/3/2018 ## Added: Added an option to specify the parent page. ## @param ShowTitle:title=Show Title|type=boolean|desc=Deselect to remove the title of the child page.|default=true ## @param LinkTitle:title=Link Title|type=boolean|desc=Select to turn the titles of the child pages into links to those pages (Show Title must be selected).|default=false ## @param IncludeDescendants:title=Include Descendants|type=boolean|desc=Select to include descendants.|default=false ## @param PageSeparator:title=Page Separator|type=boolean|desc=Separate pages with horizontal ruler.|default=false ## @param SplitPages:title=Split Pages|type=boolean|desc=Split pages marker for PDF export will be inserted.|default=false ## @param Collapsible:title=Collapsible|type=boolean|desc=Should the included pages be collapsible?|default=false ## @param FilterLabel:title=Filter on Page Label|type=string|desc=Include only subpages with the specified label. ## @param Page:title=Page title|type=confluence-content|required=false|desc=If not specified, the current page is used. ## @param Order:title=Order|type=enum|enumValues=Nav Order,Reverse Nav Order,Alphabetical,Reverse Alphabetical,Create Date,Reverse Create Date|default=nav order|desc=In what order should the child pages be displayed? Nav Order refers to the order of the child pages in the page tree. Reverse Nav Order simply reverses that. #set( $containerManagerClass=$content.class.forName('com.atlassian.spring.container.ContainerManager') ) #set( $getInstanceMethod=$containerManagerClass.getDeclaredMethod('getInstance',null) ) #set( $containerManager=$getInstanceMethod.invoke(null,null) ) #set( $containerContext=$containerManager.containerContext ) #set( $pageManager=$containerContext.getComponent('pageManager') ) #if( $paramPage && $paramPage != "" ) ########################## ## Find the page ## ########################## #set( $parts = $paramPage.split(":") ) #set( $i = 0 ) #set ( $len = 0 ) #set( $key = "" ) #set( $name = "" ) ##Having trouble finding out the length/size of $parts ... brute force it #foreach ( $part in $parts ) #set ( $len = $len + 1 ) #end #if ( $len == 1 ) #set ( $key = $content.spaceKey ) #set ( $name = $paramPage ) #else #foreach ( $part in $parts ) #if ( $i == 0 ) #set ( $key = $part ) #set ( $i = $i + 1 ) #else #set ( $name = $part ) #end #end #end #set ( $tempcontent = "" ) #if ( $pageManager ) #set ( $tempcontent = $pageManager.getPage($key, $name) ) #end #if ( $tempcontent && $tempcontent != "" ) #set ( $content = $tempcontent ) #else The page "$paramPage" was not found in this space. #set ( $content = "" ) #end #end #set( $data = "" ) #childPages ( $content.id ) #macro ( childPages $pageid ) ## Find the array of (sorted) child pages. #if( $paramOrder == "Nav Order" || $paramOrder == "Reverse Nav Order" ) #set( $children = $pageManager.getPage($pageid).getSortedChildren() ) #elseif( $paramOrder == "Alphabetical" || $paramOrder == "Reverse Alphabetical" ) #set( $array = $pageManager.getPage($pageid).getChildren() ) ############################################################### ## Could not find a method to get them in Alphabetical order ## ## Must sort them myself ... Bubble sort ## ############################################################### #set( $size = $array.size() ) #foreach( $junk in $array ) #set( $count = -1 ) #foreach( $line in $array ) #set( $count = $count + 1 ) #if( $velocityCount < $size ) #if( $line.getTitle().compareToIgnoreCase($array.get($velocityCount).getTitle()) > 0 ) #set( $tmp = $array.get($velocityCount) ) #set( $junk = $array.set($velocityCount, $line) ) #set( $junk = $array.set($count, $tmp) ) #end #end #end #end #set( $children = $array ) #elseif( $paramOrder == "Create Date" || $paramOrder == "Reverse Create Date" ) #set( $children = $pageManager.getPage($pageid).getChildren() ) #end ## Prepare children data. #foreach( $child in $children ) ## Create the map of labels for the page. #set( $labelsMap = {} ) #foreach( $label in $child.getLabels() ) #set( $dummy = $labelsMap.put($label.toString(), 1) ) #end ## Include all pages (when no filter label is specified) or pages with a specific label. #if( !$paramFilterLabel || $paramFilterLabel && $labelsMap.containsKey($paramFilterLabel) ) #set( $include = "" ) #if ( $paramCollapsible == true ) #set( $include = $include + '<ac:structured-macro ac:name="expand">' ) #set( $include = $include + '<ac:parameter ac:name="title">' + $child.getTitle() + '</ac:parameter>' ) #set( $include = $include + '<ac:rich-text-body>' ) #set( $include = $include + '<p>' ) #end #set( $include = $include + '<div class="included-child-page">' ) ## Show title and links. #if( $paramShowTitle == true ) #if( $paramLinkTitle == true ) #set( $include = $include + '<h1 class="included-child-page-title"><a href="' + $child.getUrlPath() + '">' + $child.getTitle() + '</a></h1>' ) #else #set( $include = $include + '<h1 class="included-child-page-title">' + $child.getTitle() + '</h1>' ) #end #end ## Include sub page. #set( $include = $include + '<div class="included-child-page-body">' ) #set( $include = $include + '<ac:structured-macro ac:name="include">' ) #set( $include = $include + '<ac:parameter ac:name="">' ) #set( $include = $include + '<ac:link>' ) #set( $include = $include + '<ri:page ri:content-title="' + $child.getTitle() + '" ri:space-key="' + $content.spaceKey + '"/>' ) #set( $include = $include + '</ac:link>' ) #set( $include = $include + '</ac:parameter>' ) #set( $include = $include + '</ac:structured-macro>' ) #set( $include = $include + '</div>' ) #set( $include = $include + '</div>' ) #if ( $paramCollapsible == true ) #set( $include = $include + '</p>' ) #set( $include = $include + '</ac:rich-text-body>' ) #set( $include = $include + '</ac:structured-macro>' ) #end ## Define sorting order. #set( $inOrder = false ) #if( $paramOrder == "Nav Order" || $paramOrder == "Alphabetical" || $paramOrder == "Create Date" ) #set( $inOrder = true ) #end ## Include page separator or include page split marker used for PDF exports. #if( $paramPageSeparator == true ) #set( $include = $include + '<hr/><br/>' ) #end #if( $paramSplitPages == true ) #set( $include = $include + '<div style="page-break-before:always;"></div>' ) #end ## Add page content in order or in reverse order. #if( $inOrder == true ) #set( $data = $data + $include ) #else #set( $data = $include + $data) #end #end #if($paramIncludeDescendants == true) #childPages( $child.id ) #end #end #end $data <style type="text/css"> .included-child-page { margin-bottom: 30px; overflow: hidden; } </style>
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
Hi Davin
That's a great macro. I'm challanged with a requirement similar to this.
Instead of showing the whole page, my users would like to see only the headings of the child-pages, where the links should be usable. Is this possible? Do you have a hint on which element i have to change or to enhance?
Thanks for your help.
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
This seems to have stopped working for us after upgrading to Confluence 7.8. Do you have an update by any chance?
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.
You must be a registered user to add a comment. If you've already registered, sign in. Otherwise, register and sign in.