Hire Me! I'm currently looking for my next role in developer relations and advocacy. If you've got an open role and think I'd be a fit, please reach out. You can also find me on LinkedIn.

Today's PDF entry is all about merging. ColdFusion 8 allows us to merge any number of PDFs, whether from files or directly in memory. What are some usage examples? Your site could have a standard disclaimer that you want added to the front of each PDF you create. You may have a standard credits page you want to add to the end. Whatever the need - ColdFusion makes it pretty simple, so let's take a look.

As I mentioned above, you can work with PDFs on the file system or with PDFs in memory. Let's first take a look at PDFs on the file system. The CFPDF tag takes a directory attribute. This directory consists of the PDF files you want to merge. By default ColdFusion will merge all files in the folder. There are three things to consider when working with a folder:

  1. CFPDF will sort your PDFs by timestamp first. You can supply the order attribute to change this to name.
  2. CFPDF will sort your PDFs in reverse order. So if you use name, PDFs will be sorted Z to A. You can change this by using the ascending attribute. The default value is no.
  3. CFPDF will try to merge every file in a folder. If a folder contains non-PDF files, ColdFusion will ignore it. If you do not want ColdFusion to ignore non-PDF files, use stopOnError=true.

So let's look at a simple example:

<cfdocument name="pdf1" format="pdf"> <cfoutput> This is PDF 1 at #timeFormat(now())# </cfoutput> </cfdocument>

<cfdocument name="pdf2" format="pdf"> <cfoutput> This is PDF 2 at #timeFormat(now())# </cfoutput> </cfdocument>

<cfset savedFolder = expandPath("./pdfs")>

<cffile action="write" file="#savedFolder#/pdf1.pdf" output="#pdf1#"> <cffile action="write" file="#savedFolder#/pdf2.pdf" output="#pdf2#">

<cfpdf action="merge" directory="#savedFolder#" name="mergedpdf">

<cfcontent type="application/pdf" reset="true" variable="#toBinary(mergedpdf)#">

The code begins by simply creating two PDFs. These PDFs are stored to the file system in a subfolder named pdfs. The important line is here:

<cfpdf action="merge" directory="#savedFolder#" name="mergedpdf">

I simply specify a directory and in my case, a name variable to store the result in memory. Lastly I serve up the PDF with the cfcontent tag. If you run this you will notice that the PDF seems backwards. PDF2 is on page 1, and PDF1 is on page 2. This makes sense if you remember the above notes. The default order is by time, descending, and PDF2 was written out first.

Now let's take it up a notch and introduce a new tag, cfpdfparam. The cfpdfparam tag is only used with merging PDFs. It lets you do all kinds of fun things. It gives you the power to provide more control over the order. It lets you specify a page range for each PDF. (So for example, merge pages 1-10 in pdf 1, pages 13-19 in pdf 2, and pages 90-100 in pdf 3.) You can also supply passwords for individual PDFs that need them. Pretty cool, eh? Here is a simple example:

<cfdocument name="pdf1" format="pdf"> <cfoutput> This is PDF 1 at #timeFormat(now())# </cfoutput> </cfdocument>

<cfdocument name="pdf2" format="pdf"> <cfoutput> This is PDF 2 at #timeFormat(now())# </cfoutput> </cfdocument>

<cfpdf action="merge" name="mergedpdf"> <cfpdfparam source="pdf1"> <cfpdfparam source="pdf2"> </cfpdf>

<cfcontent type="application/pdf" reset="true" variable="#toBinary(mergedpdf)#">

This example is much like the first one. I create two PDFs with cfdocument. This time though I don't bother saving them to the file system. I then do the merge operation, but note the use of cfpdfparam. Now my order will work correctly because I explicitly specified the proper order. I could have used filenames as well. (And let me thank Adobe again for supporting relative paths!)

One final note - another option for merging PDFs is "keepBookmark". This tells CFPDF to keep the bookmarks in the source PDF files. I'll be talking about bookmarks more in the next entry.

Please let me know if you are enjoying this series. The last entry didn't get any comments so I want to make sure folks are still getting it. :)