2010-08-06 01:27:47 +00:00
|
|
|
#include <iostream>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <qpdf/QPDF.hh>
|
2018-06-18 15:06:51 -04:00
|
|
|
#include <qpdf/QPDFPageDocumentHelper.hh>
|
|
|
|
#include <qpdf/QPDFPageObjectHelper.hh>
|
2010-08-06 01:27:47 +00:00
|
|
|
#include <qpdf/QUtil.hh>
|
|
|
|
#include <qpdf/Buffer.hh>
|
|
|
|
#include <qpdf/QPDFWriter.hh>
|
2019-06-20 23:35:23 -04:00
|
|
|
#include <qpdf/QIntC.hh>
|
2010-08-06 01:27:47 +00:00
|
|
|
|
|
|
|
static char const* whoami = 0;
|
|
|
|
|
|
|
|
void usage()
|
|
|
|
{
|
|
|
|
std::cerr << "Usage: " << whoami << " infile.pdf outfile.pdf [in-password]"
|
|
|
|
<< std::endl
|
|
|
|
<< "Invert some images in infile.pdf;"
|
|
|
|
<< " write output to outfile.pdf" << std::endl;
|
|
|
|
exit(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Derive a class from StreamDataProvider to provide updated stream
|
|
|
|
// data. The main purpose of using this object is to avoid having to
|
|
|
|
// allocate memory up front for the objects. A real application might
|
|
|
|
// use temporary files in order to avoid having to allocate all the
|
|
|
|
// memory. Here, we're not going to worry about that since the goal
|
|
|
|
// is really to show how to use this facility rather than to show the
|
|
|
|
// best possible way to write an image inverter. This class still
|
|
|
|
// illustrates dynamic creation of the new stream data.
|
|
|
|
class ImageInverter: public QPDFObjectHandle::StreamDataProvider
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual ~ImageInverter()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
virtual void provideStreamData(int objid, int generation,
|
|
|
|
Pipeline* pipeline);
|
|
|
|
|
2013-06-14 11:58:37 -04:00
|
|
|
// Map [og] = image object
|
|
|
|
std::map<QPDFObjGen, QPDFObjectHandle> image_objects;
|
|
|
|
// Map [og] = image data
|
|
|
|
std::map<QPDFObjGen, PointerHolder<Buffer> > image_data;
|
2010-08-06 01:27:47 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
void
|
|
|
|
ImageInverter::provideStreamData(int objid, int generation,
|
|
|
|
Pipeline* pipeline)
|
|
|
|
{
|
|
|
|
// Use the object and generation number supplied to look up the
|
|
|
|
// image data. Then invert the image data and write the inverted
|
|
|
|
// data to the pipeline.
|
2013-06-14 11:58:37 -04:00
|
|
|
PointerHolder<Buffer> data =
|
|
|
|
this->image_data[QPDFObjGen(objid, generation)];
|
2010-09-24 20:45:18 +00:00
|
|
|
size_t size = data->getSize();
|
|
|
|
unsigned char* buf = data->getBuffer();
|
2010-08-06 01:27:47 +00:00
|
|
|
unsigned char ch;
|
|
|
|
for (size_t i = 0; i < size; ++i)
|
|
|
|
{
|
2019-06-20 23:35:23 -04:00
|
|
|
ch = QIntC::to_uchar(0xff - buf[i]);
|
2010-08-06 01:27:47 +00:00
|
|
|
pipeline->write(&ch, 1);
|
|
|
|
}
|
|
|
|
pipeline->finish();
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char* argv[])
|
|
|
|
{
|
|
|
|
whoami = QUtil::getWhoami(argv[0]);
|
|
|
|
|
|
|
|
// For libtool's sake....
|
|
|
|
if (strncmp(whoami, "lt-", 3) == 0)
|
|
|
|
{
|
|
|
|
whoami += 3;
|
|
|
|
}
|
|
|
|
|
2013-11-29 22:08:25 -05:00
|
|
|
// For test suite
|
|
|
|
bool static_id = false;
|
|
|
|
if ((argc > 1) && (strcmp(argv[1], " --static-id") == 0))
|
|
|
|
{
|
|
|
|
static_id = true;
|
|
|
|
--argc;
|
|
|
|
++argv;
|
|
|
|
}
|
|
|
|
|
2010-08-06 01:27:47 +00:00
|
|
|
if (! ((argc == 3) || (argc == 4)))
|
|
|
|
{
|
|
|
|
usage();
|
|
|
|
}
|
|
|
|
|
|
|
|
char const* infilename = argv[1];
|
|
|
|
char const* outfilename = argv[2];
|
|
|
|
char const* password = (argc == 4) ? argv[3] : "";
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
QPDF qpdf;
|
|
|
|
qpdf.processFile(infilename, password);
|
|
|
|
|
|
|
|
ImageInverter* inv = new ImageInverter;
|
|
|
|
PointerHolder<QPDFObjectHandle::StreamDataProvider> p = inv;
|
|
|
|
|
|
|
|
// For each page...
|
2018-06-18 15:06:51 -04:00
|
|
|
std::vector<QPDFPageObjectHelper> pages =
|
|
|
|
QPDFPageDocumentHelper(qpdf).getAllPages();
|
|
|
|
for (std::vector<QPDFPageObjectHelper>::iterator iter = pages.begin();
|
2010-08-06 01:27:47 +00:00
|
|
|
iter != pages.end(); ++iter)
|
|
|
|
{
|
2018-06-18 15:06:51 -04:00
|
|
|
QPDFPageObjectHelper& page(*iter);
|
2010-08-06 01:27:47 +00:00
|
|
|
// Get all images on the page.
|
|
|
|
std::map<std::string, QPDFObjectHandle> images =
|
|
|
|
page.getPageImages();
|
|
|
|
for (std::map<std::string, QPDFObjectHandle>::iterator iter =
|
|
|
|
images.begin();
|
|
|
|
iter != images.end(); ++iter)
|
|
|
|
{
|
|
|
|
QPDFObjectHandle& image = (*iter).second;
|
|
|
|
QPDFObjectHandle image_dict = image.getDict();
|
|
|
|
QPDFObjectHandle color_space =
|
|
|
|
image_dict.getKey("/ColorSpace");
|
|
|
|
QPDFObjectHandle bits_per_component =
|
|
|
|
image_dict.getKey("/BitsPerComponent");
|
|
|
|
|
|
|
|
// For our example, we can only work with images 8-bit
|
|
|
|
// grayscale images that we can fully decode. Use
|
|
|
|
// pipeStreamData with a null pipeline to determine
|
|
|
|
// whether the image is filterable. Directly inspect
|
|
|
|
// keys to determine the image type.
|
2017-08-19 09:18:14 -04:00
|
|
|
if (image.pipeStreamData(0, qpdf_ef_compress,
|
|
|
|
qpdf_dl_generalized) &&
|
2010-08-06 01:27:47 +00:00
|
|
|
color_space.isName() &&
|
|
|
|
bits_per_component.isInteger() &&
|
|
|
|
(color_space.getName() == "/DeviceGray") &&
|
|
|
|
(bits_per_component.getIntValue() == 8))
|
|
|
|
{
|
|
|
|
// Store information about the images based on the
|
|
|
|
// object and generation number. Recall that a single
|
|
|
|
// image object may be used more than once.
|
2013-06-14 11:58:37 -04:00
|
|
|
QPDFObjGen og = image.getObjGen();
|
|
|
|
if (inv->image_objects.count(og) == 0)
|
2010-08-06 01:27:47 +00:00
|
|
|
{
|
2013-06-14 11:58:37 -04:00
|
|
|
inv->image_objects[og] = image;
|
|
|
|
inv->image_data[og] = image.getStreamData();
|
2010-08-06 01:27:47 +00:00
|
|
|
|
|
|
|
// Register our stream data provider for this
|
|
|
|
// stream. Future calls to getStreamData or
|
|
|
|
// pipeStreamData will use the new
|
|
|
|
// information. Provide null for both filter
|
|
|
|
// and decode parameters. Note that this does
|
|
|
|
// not mean the image data will be
|
|
|
|
// uncompressed when we write the file. By
|
|
|
|
// default, QPDFWriter will use /FlateDecode
|
|
|
|
// for anything that is uncompressed or
|
|
|
|
// filterable in the input QPDF object, so we
|
|
|
|
// don't have to deal with it explicitly here.
|
|
|
|
image.replaceStreamData(
|
|
|
|
p,
|
|
|
|
QPDFObjectHandle::newNull(),
|
2012-07-07 17:33:45 -04:00
|
|
|
QPDFObjectHandle::newNull());
|
2010-08-06 01:27:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write out a new file
|
|
|
|
QPDFWriter w(qpdf, outfilename);
|
2013-11-29 22:08:25 -05:00
|
|
|
if (static_id)
|
2010-08-06 01:27:47 +00:00
|
|
|
{
|
|
|
|
// For the test suite, uncompress streams and use static
|
|
|
|
// IDs.
|
2015-11-01 16:39:15 -05:00
|
|
|
w.setStaticID(true); // for testing only
|
2010-08-06 01:27:47 +00:00
|
|
|
}
|
|
|
|
w.write();
|
|
|
|
std::cout << whoami << ": new file written to " << outfilename
|
|
|
|
<< std::endl;
|
|
|
|
}
|
|
|
|
catch (std::exception &e)
|
|
|
|
{
|
|
|
|
std::cerr << whoami << " processing file " << infilename << ": "
|
|
|
|
<< e.what() << std::endl;
|
|
|
|
exit(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|