Есть ли прямой способ добавить один документ PDF в другой с помощью iTextSharp? - PullRequest
7 голосов
/ 19 февраля 2009

Я искал в интернете примеры того, как это сделать. Я нашел несколько, которые кажутся немного более вовлеченными, чем они должны быть. Итак, мой вопрос, используя iTextSharp, есть ли достаточно лаконичный способ добавить один документ PDF в другой?

Оптимально, это НЕ будет включать в себя третий файл. Просто откройте первый PDF-документ, добавьте второй PDF-документ к первому, а затем закройте их оба.

Ответы [ 4 ]

13 голосов
/ 03 марта 2010

Я действительно могу что-то упустить, но я сделал что-то намного проще. Я допускаю, что это решение, вероятно, не будет обновлять закладки (как в лучшем ответе здесь пока), но оно работает для меня безупречно. Поскольку я объединял документы с заполняемыми формами, я использовал PdfCopyFields вместо PdfCopy.

Вот код (я удалил всю обработку ошибок, чтобы сделать реальный код более заметным, добавьте попытку .. окончательно закрыть открытые ресурсы, если вы планируете использовать код):

    void MergePdfStreams(List<Stream> Source, Stream Dest)
    {
        PdfCopyFields copy = new PdfCopyFields(Dest);

        foreach (Stream source in Source)
        {
            PdfReader reader = new PdfReader(source);
            copy.AddDocument(reader);
        }

        copy.Close();
    }

Вы можете передавать любой поток, будь то FileStream, MemoryStream (полезно при чтении PDF из баз данных, не требуются временные файлы и т. Д.)

Пример использования:

    void TestMergePdfStreams()
    {
        List<Stream> sources = new List<Stream>()
        {
            new FileStream("template1.pdf", FileMode.Open),
            new FileStream("template2.pdf", FileMode.Open),
            new MemoryStream((byte[])someDataRow["PDF_COLUMN_NAME"])
        };

        MergePdfStreams(sources, new FileStream("MergedOutput.pdf", FileMode.Create));
    }
5 голосов
/ 20 февраля 2009

Хорошо, это не так просто, но работает и удивительно быстро. (И он использует 3-й файл, такого понятия, как открытие и добавление нет.) Я «обнаружил» это в документах / примерах. Вот код:

private void CombineMultiplePDFs( string[] fileNames, string outFile ) {
    int pageOffset = 0;
    ArrayList master = new ArrayList();
    int f = 0;

    Document document = null;
    PdfCopy writer = null;
    while ( f < fileNames.Length ) {
        // we create a reader for a certain document
        PdfReader reader = new PdfReader( fileNames[ f ] );
        reader.ConsolidateNamedDestinations();
        // we retrieve the total number of pages
        int n = reader.NumberOfPages;
        ArrayList bookmarks = SimpleBookmark.GetBookmark( reader );
        if ( bookmarks != null ) {
            if ( pageOffset != 0 ) {
                SimpleBookmark.ShiftPageNumbers( bookmarks, pageOffset, null );
            }
            master.AddRange( bookmarks );
        }
        pageOffset += n;

        if ( f == 0 ) {
            // step 1: creation of a document-object
            document = new Document( reader.GetPageSizeWithRotation( 1 ) );
            // step 2: we create a writer that listens to the document
            writer = new PdfCopy( document, new FileStream( outFile, FileMode.Create ) );
            // step 3: we open the document
            document.Open();
        }
        // step 4: we add content
        for ( int i = 0; i < n; ) {
            ++i;
            if ( writer != null ) {
                PdfImportedPage page = writer.GetImportedPage( reader, i );
                writer.AddPage( page );
            }
        }
        PRAcroForm form = reader.AcroForm;
        if ( form != null && writer != null ) {
            writer.CopyAcroForm( reader );
        }
        f++;
    }
    if ( master.Count > 0 && writer != null ) {
        writer.Outlines = master;
    }
    // step 5: we close the document
    if ( document != null ) {
        document.Close();
    }
}
1 голос
/ 20 февраля 2009

Да. Я видел класс под названием PdfManipulation, опубликованный на форуме iText. Однако использование этого класса потребует третьего файла.

Класс изначально находится в VB.Net. Я скачал его из поста на vbforums.com . Очевидно, что у него нет функции слияния файлов, поэтому я написал ее на основе кода в этом классе.

Это было написано на машине без iTextSharp. Это может иметь ошибки. Я даже не уверен, являются ли номера страниц 0 или 1. Но попробуйте.

public static void MergePdfFiles(IEnumerable<string> files, string output) {
    iTextSharp.text.Document doc;
    iTextSharp.text.pdf.PdfCopy pdfCpy;

    doc = new iTextSharp.text.Document();
    pdfCpy = new iTextSharp.text.pdf.PdfCopy(doc, new System.IO.FileStream(output, System.IO.FileMode.Create));
    doc.Open();

    foreach (string file in files) {
        // initialize a reader
        iTextSharp.text.pdf.PdfReader reader = new iTextSharp.text.pdf.PdfReader(file);
        int pageCount = reader.NumberOfPages;

        // set page size for the documents
        doc.SetPageSize(reader.GetPageSizeWithRotation(1));

        for (int pageNum = 1; pageNum <= pageCount; pageNum++) {
            iTextSharp.text.pdf.PdfImportedPage page = pdfCpy.GetImportedPage(reader, pageNum);
            pdfCpy.AddPage(page);
        }

        reader.Close();
    }

    doc.Close();
}
0 голосов
/ 20 февраля 2009

Я не знаю, как это сделать для файлов PDF, но для postscript вы просто объединяете файлы. Если у вас установлены pdf2ps и ps2pdf, вам подойдет следующее:

pdf2ps file1.pdf file1.ps
pdf2ps file2.pdf file2.ps
cat file1.ps file2.ps > combined.ps
ps2pdf combined.ps combined.pdf

Я не эксперт по pdf2ps или ps2pdf. Я только когда-либо использовал ps2pdf, и когда я это делаю, он оставляет текст в виде текста (я все еще могу выбрать и скопировать текст из полученного PDF). Когда я делаю вышеупомянутые шаги (pdf-> ps, объединить, ps-> pdf), я получаю в результате pdf, который похож на изображение. Понятия не имею, почему.

...