开发者

Determine which UpdatePanel causes the partial (asynchronous) PostBack?

In a page contains two UpdatePanels, How can I know which UpdatePanel causes the partial PostBack ?

I mean in the Page_Load event handler.

This is my code:

 <asp:ScriptManager ID="ScriptManager1" runat="server">
 </asp:ScriptManager>
 <asp:UpdatePanel ID="UpdatePanel1" runat="server" UpdateMode="Conditional" 
     onprerender="UpdatePanel1_PreRender">
     <ContentTemplate>
         <A:u1 ID="u1" runat="server" />
     </ContentTemplate>
 </asp:UpdatePanel>
 <asp:UpdatePanel ID="UpdatePanel2" runat="server" UpdateMode="Conditional" 
     onprerender="UpdatePanel2_PreRender">
     <ContentTemplate&开发者_如何学Cgt;
         <A:u2 ID="u2" runat="server" />
     </ContentTemplate>
 </asp:UpdatePanel>

I tried this code, but it isn't worked alost!

protected void Page_Load(object sender, EventArgs e)
{
    if (ScriptManager.GetCurrent(Page).IsInAsyncPostBack)
    {
        if (UpdatePanel1.IsInPartialRendering)
        {
            // never enter to here
        }
        if (UpdatePanel2.IsInPartialRendering)
        {
            // neither here
        }
    }
}

Any help!


You can use the IsInPartialRendering property of the UpdatePanel class to determine if a specific panel caused the partial postback:

protected void Page_Render(object sender, EventArgs e)
{
    if (ScriptManager.GetCurrent(Page).IsInAsyncPostBack) {
        if (yourFirstUpdatePanel.IsInPartialRendering) {
            // The first UpdatePanel caused the partial postback.
        } else if (yourSecondUpdatePanel.IsInPartialRendering) {
            // The second UpdatePanel caused the partial postback.
        }
    }
}

EDIT: It appears that IsInPartialRendering is always false before the Render phase. Since you want that information during the Load phase, it won't work as expected. See this bug.

There's a workaround documented here that consists in deriving your own class from UpdatePanel to access its protected RequiresUpdate property:

public class ExtendedUpdatePanel : UpdatePanel
{
    public bool IsUpdating
    {
        get {
            return RequiresUpdate;
        }
    }
}

After replacing asp:UpdatePanel with ExtendedUpdatePanel in your page markup, the code above becomes:

protected void Page_Load(object sender, EventArgs e)
{
    if (ScriptManager.GetCurrent(Page).IsInAsyncPostBack) {
        if (yourFirstUpdatePanel.IsUpdating) {
            // The first UpdatePanel caused the partial postback.
        } else if (yourSecondUpdatePanel.IsUpdating) {
            // The second UpdatePanel caused the partial postback.
        }
    }
}


Try this:

ScriptManager.GetCurrent().AsyncPostBackSourceElementID


If you don't want to extend the original UpdatePanel class, you can also use this workaround:

string id = ScriptManager.GetCurrent(Page).AsyncPostBackSourceElementID;

Thus, you can examine the resulting id and add some conditional clauses to determine which code should execute. This id should be equal to the first parameter that was passed through the javascript function __doPostBack('someid', '').

For example, I have a user control in my Update Panel: this control contains a bunch of link buttons that trigger the UpdatePanel.) I also can manually update this panel from some external links (using something like __doPostBack('myUpdatePanelClientId', '');

That is, in my case, I see three different ways for loading my UpdatePanel:

  • First page load;
  • Link button (or any other type of button) clicked inside my UpdatePanel;
  • PostBack triggered outside the UpdatePanel.

Every scenario gives me a different id. The first one gives me an empty string (since this is the first page load, there hasn't been yet any postback triggered with the __doPostBack function.)

The second one gives me the UniqueId of the button that was pushed inside the user control (this is the original and expected ASP.NET behavior.)

The third one gives me exactly what I passed as the first argument when I coded the method (ie: the ClientId of the UpdatePanel.)

Here's how I succeeded to implement my UpdatePanel use case (assuming I'm using the partial rendering mode.) This is not perfect but it works as intended.

protected void myUpdatePanel_Load(object sender, EventArgs e)
{
    string id = ScriptManager.GetCurrent(Page).AsyncPostBackSourceElementID;

    bool firstLoad = (String.IsNullOrEmpty(id));
    bool triggerFromUpdatePanel = !firstLoad && (id.Contains(userControlInsideMyUpdatePanel.UniqueID));
    bool triggerFromExternalControl = !firstLoad && (id == myUpdatePanel.ClientID);

    // case 1, 2, 3.
    if ((firstLoad)  || (triggerFromUpdatePanel)  || (triggerFromExternalControl ))
    {
        // do something
    }
    else 
    {
        // do nothing!
    }


}


If the asyncpostbackelementid is set then you could check that updatepanel's uniqueid starts with that id, meaning that is inside it since updatepanels are inaming containers.


In client side use:

function EndRequestHandler(sender, args) { if (Sys.WebForms.PageRequestManager.getInstance()._postBackSettings.asyncTarget == 'Id of element do postback') { // do something . . . } } Sys.WebForms.PageRequestManager.getInstance().add_endRequest(EndRequestHandler);


is possible to determine the object inside an update panel an execute the code needed

If (ScriptManager.GetCurrent(Page).IsInAsyncPostBack) Then
                Dim id As String = ScriptManager.GetCurrent(Page).AsyncPostBackSourceElementID
                Dim Obj = UpdatePanel1.FindControlRecursive(id)
                If Not IsNothing(Obj) Then
                    a = 1
                End If
End If

below the function used to find de object inside the update panel. Is an extension of System.Web.UI.Control

a=1 Is the desired code.

 Public Module thisExtensions
        <System.Runtime.CompilerServices.Extension> _
        Public Function FindControlRecursive(control As System.Web.UI.Control, id As String) As System.Web.UI.Control
            If control Is Nothing Then
                Return Nothing
            End If
            'try to find the control at the current level
            Dim ctrl As Control = control.FindControl(id)

            If ctrl Is Nothing Then
                'search the children
                For Each child As Control In control.Controls
                    ctrl = FindControlRecursive(child, id)

                    If ctrl IsNot Nothing Then
                        Exit For
                    End If
                Next
            End If
            Return ctrl
        End Function
    End Module
0

上一篇:

下一篇:

精彩评论

暂无评论...
验证码 换一张
取 消

最新问答

问答排行榜