web-dev-qa-db-ja.com

asp.netの更新パネル内でダウンロード機能が機能しない

FormViewを含むWebユーザーコントロールがあります。 formviewは求職者の詳細を表示します。 「履歴書をダウンロード」リンクのボタンを用意しました。これにより、管理者/ HRは履歴書をダウンロードできます。このコントロールを、UpdatePanelを含むaspxページに配置しました。ダウンロードリンク以外はすべて正常に動作します。

ダウンロードリンクボタンにコマンドを指定しましたが、ダウンロードを開始するコマンドに関数が関連付けられています。

以下は私が実装したコードです-

//Command on 'Download' link button within FormView
protected void lnkDownload_Command(object sender, CommandEventArgs e)
{
    if (e.CommandName.Equals("Download"))
    {
        StartDownload(e.CommandArgument.ToString());
    }
}

//My routine to download document
//sFileInfo contains filepath$==$mimetype
protected void StartDownload(string sFileInfo)
{
    string[] d = sFileInfo.ToString().Split((new string[] { "$==$" }), StringSplitOptions.None);
    string filename = d[0];
    string docType = d[1];

    System.IO.FileInfo file = new System.IO.FileInfo(d[0]);

    if (file.Exists)
    {
        Response.Clear();
        Response.AddHeader("Content-Disposition", "attachment; filename=" + d[0]);
        Response.AddHeader("Content-Length", file.Length.ToString());
        Response.ContentType = d[1];
        Response.WriteFile(file.FullName);
        Response.End();
    }
    else
    {
        Server.Transfer("~/Mesgbox.aspx?cat=2");
    }
}

更新パネルを削除するとコードは完全に機能しますが、更新パネルを使用するとスクリプトエラーが発生します。

助言がありますか....?

あなたの時間を共有してくれてありがとう。

27
IrfanRaza

全ページポストバックを開始するには、更新パネルにポストバックトリガーを追加します。

<asp:UpdatePanel runat="server">
    <Triggers>
        <asp:PostBackTrigger ControlID="YourControlID" />
    </Triggers>
    <ContentTemplate>
        .....
60
Mitul

ScriptManagerは結果を使用して(応答全体ではなく)DIVを更新するため、UpdatePanel部分ポストバックで添付ファイルを返すことはできません。あなたがしようとしていることの最も簡単な修正は、ダウンロードボタンをポストバックコントロールにすることです。これにより、そのボタンで完全なポストバックが開始されます。以下は、Page_Loadに含めるコードです。

ScriptManager.GetCurrent(this.Page).RegisterPostBackControl(this.lnkDownload);
23
Ken Melton

更新パネル内からもダウンロードドキュメントをトリガーできます。

更新パネルがあり、中には3つのネストされたリピーターがあります。最も内側のリピーターでは、LinkBut​​tonsを使用して一連のダウンロードリンクを作成します。各リンクには、webserviceを介してドキュメントをフェッチしてディスパッチするコマンドが含まれています。

各リピーターにはOnItemDataBoundメソッドがあります。最後のリピーターには次のものがあります

        protected void LinkDocRepeaterOnItemDataBound(object sender, RepeaterItemEventArgs e) {
            if(!(e.Item.ItemType == ListItemType.Item || e.Item.ItemType == ListItemType.AlternatingItem)) {
            return;
                }
            LinkButton linkButton = (LinkButton)e.Item.FindControlRecursive("LinkId");
            var scriptManager = ScriptManager.GetCurrent(this.Page);
            if (scriptManager != null) {
               scriptManager.RegisterPostBackControl(linkButton);
            }
        }

各リンクボタンがドキュメントをダウンロードするようになりました。

5
carlosJ

私の状況:

SQLデータから長時間実行されているExcelファイルを読み込んでいたため、ファイルの作成中にスピナーgifを表示するように更新進行状況パネルに表示し、更新パネル内からファイルをダウンロードしました。思ったよりトリッキーでした。

このリンクは検索で高くなり、それを回避しようとした後、iframeが私にとって有用であることがわかりました。

iframe非同期ダウンロード

これが最終的に機能したものです。(この正確なコードはテストされていません)

MyPage.aspx ...(更新パネル内、トリガーなし)

<asp:Button runat="server" ID="btnExcelExport" Text="Export to Excel" OnClick="btnExcelExport_Click" />

<iframe runat="server" id="ifmExcel" width="0" height="0" marginheight="0" marginwidth="0"
                    frameborder="0" />

MyPage.aspx.cs

        protected void btnExcelExport_Click(object sender, EventArgs e)
        {
            //long running process here, taking advantage of the update progress panel
            var bytes = GetExcelFile();
            //generate a key to pass to the download page to access the file bytes
            var cacheKey = Guid.NewGuid().ToString("N");//N means no hyphens
            //placing the result in cache for a few seconds so the download page can grab it             
            Context.Cache.Insert(key: cacheKey, value: bytes, dependencies: null, absoluteExpiration: DateTime.Now.AddSeconds(30), slidingExpiration: System.Web.Caching.Cache.NoSlidingExpiration);
            ifmExcel.Attributes.Add("src", String.Format("MyDownloadPage.aspx?cacheKey={0}", cacheKey));
        }

MyDownloadPage.aspx.cs ...

        protected void Page_Load(object sender, EventArgs e)
    {
        if (!IsPostBack)
        {
            var bytes = Context.Cache.Get(Request.QueryString.Get("cacheKey")) as byte[];
            Response.Clear();
            Response.AddHeader(
                "content-disposition", string.Format("attachment; filename={0}.xlsx", "Invoice"));
            Response.ContentType = "application/xlsx";
            Response.BinaryWrite(bytes);
            Response.End();
        }
    }

他の非同期ポストバックと同じように期待どおりに動作するようです。

3
TheDev6

'runat = "server" OnClick = "DownloadFile">' runat = "server" OnClick = "DeleteFile" /> <%----%>

protected void UploadFile(object sender、EventArgs e){if(FileUpload1.HasFile){string FileName = Path.GetFileName(FileUpload1.PostedFile.FileName);文字列拡張子= "。" + FileName.Split( '。')[1] .ToString(); string FileName_Guid = Convert.ToString(Guid.NewGuid())+拡張; FileUpload1.PostedFile.SaveAs(@ "C:\ Uploads \" + FileName_Guid); string Platform_Config_ID = PlatformConfigID.Value; DataTable dt = new DataTable(); dt = DAL.Upload_File(FileName_Guid、FileName、Platform_Config_ID);

            gv_Files.DataSource = dt;
            gv_Files.DataBind();
            ScriptManager scriptManager = ScriptManager.GetCurrent(this.Page);
            scriptManager.RegisterPostBackControl(gv_Files);
        }
    }

    protected void DownloadFile(object sender, EventArgs e)
    {
        try
        {                
            LinkButton lnkDownload = (LinkButton)sender;
            GridViewRow row = (GridViewRow)lnkDownload.NamingContainer;
            LinkButton download = row.FindControl("lnkDownload") as LinkButton;
            ScriptManager.GetCurrent(this).RegisterPostBackControl(download);
            string FileName = (sender as LinkButton).CommandArgument.Split(';')[0].ToString();
            string OriginalFileName = (sender as LinkButton).CommandArgument.Split(';')[1].ToString();
            string FilePath = @"C:\Uploads\" + FileName.ToString();
            FileInfo file = new FileInfo(FilePath);
            if (file.Exists)
            {                    
                Response.ContentType = ContentType;
                Response.AppendHeader("Content-Disposition", "attachment; filename=" + OriginalFileName);
                Response.Headers.Set("Cache-Control", "private, max-age=0");
                Response.WriteFile(FilePath);                    
                Response.End();                                     
            }                
        }
        catch (Exception ex)
        {
            // do nothing
        }
    }


    protected void DeleteFile(object sender, EventArgs e)
    {
        string FileName_Guid = (sender as LinkButton).CommandArgument.Split(';')[0].ToString();
        string File_ID = (sender as LinkButton).CommandArgument.Split(';')[1].ToString();
        string Filename = (sender as LinkButton).CommandArgument.Split(';')[2].ToString();
        string Platform_Config_ID = (sender as LinkButton).CommandArgument.Split(';')[3].ToString();
        string FilePath = @"C:\Uploads\" + FileName_Guid;
        File.Delete(FilePath);

        DataTable dt = new DataTable();
        dt = DAL.Delete_File(File_ID, Filename, Platform_Config_ID);

        gv_Files.DataSource = dt;
        gv_Files.DataBind();
    }

次の手順を試してください。

  1. _<asp:UpdatePanel ID="UpdatePanel5" runat="server" l**UpdateMode="Conditional"**>
    <asp:GridView ID="gv_Files" runat="server" AllowPaging="True" PageSize="7" OnPageIndexChanging="gv_Files_PageIndexChanging" AutoGenerateColumns="false" EmptyDataText="No files uploaded" GridLines="None">
    _
  2. (コードビハインド)UploadFile()

    _ScriptManager scriptManager = ScriptManager.GetCurrent(this.Page);
    scriptManager.RegisterPostBackControl(gv_Files);
    _
  3. DownloadFile()

    _DownloadFile() --> LinkButton download = row.FindControl("lnkDownload") as LinkButton;
    ScriptManager.GetCurrent(this).RegisterPostBackControl(download);
    _

更新パネルではResponseオブジェクトを使用できません。

0
Anuraj