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.

Anyone out there playing with iText? iText is an open source library that allows for the creation and manipulation of PDFs via Java. I've heard of it before but have never really played with it. I find cfdocument and cfpdf to be good enough for most of my needs, but earlier this week a user wrote in asking about how to use iText to get comments from a PDF. While cfpdf supports DDX, an XML based language that does PDF magic via an embedded Livecycle engine in ColdFusion, it specifically blocks you from running DDX to get comments. I'm not going to complain about this as I'm still surprised as to how much you can do via DDX, but it would be nice to have a work around. iText itself is also bunded within ColdFusion. That means you can use it without installing any additional Jars or using JavaLoader. So an example:

<cfset reader = createObject("java", "com.lowagie.text.pdf.PdfReader").init(expandPath("testa.pdf"))>

I know (or knew, before this week) next to zip about iText but even I can tell what this does. There are full API docs, but a Google search turned up this perfect blog entry: How to extract comments from pdf using Itext. Before going on - read that entry and check out the Java code. Now let's look at a version in ColdFusion:

<cfset reader = CreateObject("java", "com.lowagie.text.pdf.PdfReader").init(expandPath("testa.pdf"))> <cfset pageDict = reader.getPageN(1)>

<cfdump var="#pageDict#" expand="false"> <cfset pdfname = createObject("java","com.lowagie.text.pdf.PdfName")>

<cfset annots = pageDict.getAsArray(pdfname.ANNOTS)> <cfset iterator = annots.listIterator()> <cfloop condition="iterator.hasNext()"> <cfset annot = reader.getPdfObject(iterator.next())> <cfset content = reader.getPdfObject(annot.get(pdfname.CONTENTS))> <cfif not isNull(content)> <cfoutput> content=#content# <p> </cfoutput> <cfelse> it was null <p> </cfif> </cfloop>

<p> done

Again - even if you've never seen iText before, you can take some guesses as to what is going on. I specifically ask for annotations and then iterate over them. It's interesting to note - and I'm again surprised by how complex PDFs are - that comments are merely one type of annotation. The code above works fine (*) for comments, but not for other things like "Pop Up Notes" (which I didn't know existed until I looked at the sample PDF the reader sent me).

Anyway - interesting stuff here. I'm considering perhaps adding to my pdfUtils CFC wrapper. I can imagine a getComments method could be useful?

  • One note on the CFML code. Unlike the Java code demonstrated in the other blog entry, this code only gets one page. That's a trivial enough change to fix though.