PDFをMVCでブラウザーに返す方法は?


120

このiTextSharpのデモコードがあります

    Document document = new Document();
    try
    {
        PdfWriter.GetInstance(document, new FileStream("Chap0101.pdf", FileMode.Create));

        document.Open();

        document.Add(new Paragraph("Hello World"));

    }
    catch (DocumentException de)
    {
        Console.Error.WriteLine(de.Message);
    }
    catch (IOException ioe)
    {
        Console.Error.WriteLine(ioe.Message);
    }

    document.Close();

コントローラーにPDF文書をブラウザーに返すにはどうすればよいですか?

編集:

このコードを実行するとAcrobatが開きますが、「ファイルが破損しており、修復できませんでした」というエラーメッセージが表示されます。

  public FileStreamResult pdf()
    {
        MemoryStream m = new MemoryStream();
        Document document = new Document();
        PdfWriter.GetInstance(document, m);
        document.Open();
        document.Add(new Paragraph("Hello World"));
        document.Add(new Paragraph(DateTime.Now.ToString()));
        m.Position = 0;

        return File(m, "application/pdf");
    }

これが機能しない理由はありますか?



@ mg1075あなたのリンクは死んでいます
thecoolmacdude 2017年

回答:


128

を返しますFileContentResult。コントローラアクションの最後の行は次のようになります。

return File("Chap0101.pdf", "application/pdf");

このPDFを動的に生成する場合はMemoryStream、を使用し、ファイルに保存するのではなく、メモリ内にドキュメントを作成する方がよい場合があります。コードは次のようになります。

Document document = new Document();

MemoryStream stream = new MemoryStream();

try
{
    PdfWriter pdfWriter = PdfWriter.GetInstance(document, stream);
    pdfWriter.CloseStream = false;

    document.Open();
    document.Add(new Paragraph("Hello World"));
}
catch (DocumentException de)
{
    Console.Error.WriteLine(de.Message);
}
catch (IOException ioe)
{
    Console.Error.WriteLine(ioe.Message);
}

document.Close();

stream.Flush(); //Always catches me out
stream.Position = 0; //Not sure if this is required

return File(stream, "application/pdf", "DownloadName.pdf");

@Tony、最初にドキュメントを閉じてストリームをフラッシュする必要があります。
Geoff

2
ジェフ、私はこれを達成しようとしていますが、同様の問題があります。実行時に「閉じたストリームにアクセスできません」というエラーが表示されますが、閉じないと、何も返されません。
littlechris

1
@littlechrisに感謝します。そうです、私はpdfWriter.CloseStream = falseを含むようにコードを編集しました。
Geoff、

1
はい@Geoff stream.Possition = 0; あなたはそれを書いていない場合は、オープンPDFの瞬間に、必要とされるAcrobatは、エラーが「ファイルが破損して」スロー
アルベルト・レオン

3
型 'System.Web.Mvc.FileStreamResult'を暗黙的に 'System.Web.Mvc.FileContentResult'に変換できません
CountMurphy

64

私はそれをこのコードで動作させました。

using iTextSharp.text;
using iTextSharp.text.pdf;

public FileStreamResult pdf()
{
    MemoryStream workStream = new MemoryStream();
    Document document = new Document();
    PdfWriter.GetInstance(document, workStream).CloseStream = false;

    document.Open();
    document.Add(new Paragraph("Hello World"));
    document.Add(new Paragraph(DateTime.Now.ToString()));
    document.Close();

    byte[] byteInfo = workStream.ToArray();
    workStream.Write(byteInfo, 0, byteInfo.Length);
    workStream.Position = 0;

    return new FileStreamResult(workStream, "application/pdf");    
}

Document、PdfWriter、Paragraphが認識されません。どの名前空間を追加しますか?
マイケル

9
usingが見つけることができる例に単一のステートメントがないことを少し心配しています...ここでは必要ありませんか?少なくとも3つの使い捨てオブジェクトがあると思います...
Kobi

はい、ステートメントを使用することは良いことです。これが、たとえば1人以上で使用する
本番環境の

7
FileSteamResultがストリームを閉じます。この回答を参照してください。stackoverflow.com/a/10429907/228770
Ed Spencer

重要なのは、Position = 0に設定することです。ありがとう@TonyBorf
ThanhLD

23

指定する必要があります:

Response.AppendHeader("content-disposition", "inline; filename=file.pdf");
return new FileStreamResult(stream, "application/pdf")

ファイルをダウンロードするのではなく、ブラウザーで直接開く場合


ありがとうございました!私はこれをする方法についてどこでも探していました!!
スコッティ2016

17

FileResultアクションメソッドからを返しFile()、コントローラーで拡張メソッドを使用すると、やりたいことを簡単に実行できます。File()ファイルのバイナリコンテンツ、ファイルへのパス、またはを取得するメソッドにはオーバーライドがありますStream

public FileResult DownloadFile()
{
    return File("path\\to\\pdf.pdf", "application/pdf");
}

11

私は同様の問題に遭遇し、解決策を見つけました。私は2つの投稿を使用しました。1つはダウンロードのために戻るメソッドを示すスタックからの投稿、もう1つはItextSharpおよびMVCの実用的なソリューションを示す投稿です。

public FileStreamResult About()
{
    // Set up the document and the MS to write it to and create the PDF writer instance
    MemoryStream ms = new MemoryStream();
    Document document = new Document(PageSize.A4.Rotate());
    PdfWriter writer = PdfWriter.GetInstance(document, ms);

    // Open the PDF document
    document.Open();

    // Set up fonts used in the document
    Font font_heading_1 = FontFactory.GetFont(FontFactory.TIMES_ROMAN, 19, Font.BOLD);
    Font font_body = FontFactory.GetFont(FontFactory.TIMES_ROMAN, 9);

    // Create the heading paragraph with the headig font
    Paragraph paragraph;
    paragraph = new Paragraph("Hello world!", font_heading_1);

    // Add a horizontal line below the headig text and add it to the paragraph
    iTextSharp.text.pdf.draw.VerticalPositionMark seperator = new iTextSharp.text.pdf.draw.LineSeparator();
    seperator.Offset = -6f;
    paragraph.Add(seperator);

    // Add paragraph to document
    document.Add(paragraph);

    // Close the PDF document
    document.Close();

    // Hat tip to David for his code on stackoverflow for this bit
    // /programming/779430/asp-net-mvc-how-to-get-view-to-generate-pdf
    byte[] file = ms.ToArray();
    MemoryStream output = new MemoryStream();
    output.Write(file, 0, file.Length);
    output.Position = 0;

    HttpContext.Response.AddHeader("content-disposition","attachment; filename=form.pdf");


    // Return the output stream
    return File(output, "application/pdf"); //new FileStreamResult(output, "application/pdf");
}

素晴らしい例!これはまさに私が探していたものでした!-ピート-
DigiOzマルチメディア

2
使用?閉じる?廃棄?流す?誰がメモリリークを気にしますか?
vbullinger 2012


3

私はこの質問が古いことを知っていますが、同様のものを見つけることができなかったので、私はこれを共有すると思いました。

Razor使用して通常どおりビュー/モデルを作成し、それらをPdfsとしてレンダリングしたかったのです。

このようにして、iTextSharpを使用してドキュメントをレイアウトする方法を理解するのではなく、標準のHTML出力を使用してPDFプレゼンテーションを制御しました。

プロジェクトとソースコードは、nugetのインストール手順とともにここにあります。

https://github.com/andyhutch77/MvcRazorToPdf

Install-Package MvcRazorToPdf

3

FileStreamResult確かに動作します。ただし、Microsoft Docsを見ると、ActionResult -> FileResult別の派生クラスを持つfromを継承していFileContentResultます。「バイナリファイルの内容を応答に送信します」。そのためbyte[]、すでにがある場合は、FileContentResult代わりに使用する必要があります。

public ActionResult DisplayPDF()
{
    byte[] byteArray = GetPdfFromWhatever();

    return new FileContentResult(byteArray, "application/pdf");
}

2

通常は、Response.Flushの後にResponse.Closeを実行しますが、何らかの理由でiTextSharpライブラリはこれを好まないようです。データは通過しません。AdobeはPDFが破損していると考えています。Response.Close関数を省略して、結果が優れているかどうかを確認します。

Response.Clear();
Response.ContentType = "application/pdf";
Response.AppendHeader("Content-disposition", "attachment; filename=file.pdf"); // open in a new window
Response.OutputStream.Write(outStream.GetBuffer(), 0, outStream.GetBuffer().Length);
Response.Flush();

// For some reason, if we close the Response stream, the PDF doesn't make it through
//Response.Close();

2
HttpContext.Response.AddHeader("content-disposition","attachment; filename=form.pdf");

ファイル名が動的に生成されている場合、ここでファイル名を定義する方法は、ここでGUIDを通じて生成されます。


1

DBからvar-binaryデータを返し、ポップアップまたはブラウザにPDFを表示する場合は、次のコードに従ってください。

ページを表示:

@using (Html.BeginForm("DisplayPDF", "Scan", FormMethod.Post))
    {
        <a href="javascript:;" onclick="document.forms[0].submit();">View PDF</a>
    }

スキャンコントローラ:

public ActionResult DisplayPDF()
        {
            byte[] byteArray = GetPdfFromDB(4);
            MemoryStream pdfStream = new MemoryStream();
            pdfStream.Write(byteArray, 0, byteArray.Length);
            pdfStream.Position = 0;
            return new FileStreamResult(pdfStream, "application/pdf");
        }

        private byte[] GetPdfFromDB(int id)
        {
            #region
            byte[] bytes = { };
            string constr = System.Configuration.ConfigurationManager.ConnectionStrings["Connection"].ConnectionString;
            using (SqlConnection con = new SqlConnection(constr))
            {
                using (SqlCommand cmd = new SqlCommand())
                {
                    cmd.CommandText = "SELECT Scan_Pdf_File FROM PWF_InvoiceMain WHERE InvoiceID=@Id and Enabled = 1";
                    cmd.Parameters.AddWithValue("@Id", id);
                    cmd.Connection = con;
                    con.Open();
                    using (SqlDataReader sdr = cmd.ExecuteReader())
                    {
                        if (sdr.HasRows == true)
                        {
                            sdr.Read();
                            bytes = (byte[])sdr["Scan_Pdf_File"];
                        }
                    }
                    con.Close();
                }
            }

            return bytes;
            #endregion
        }
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.