Acceder a PageMethods con jQuery

[ING] Ir al sitio web de jQueryMuy interesante el artículo Using jQuery to directly call ASP.NET AJAX page methods, en el que se demuestra que es posible, y además realmente sencillo, invocar métodos estáticos de página (PageMethods) utilizando esta magnífica librería javascript.

Como ya sabemos los PageMethods son métodos estáticos definidos dentro de la clase de una página, es decir, en su codebehind, y que son accesibles desde cliente utilizando Ajax (en minúsculas!).

Hace más de un año ya estuve comentando cómo hacerlo utilizando la propia infraestructura de ASP.NET Ajax y sus librerías de scripting para hacerlo muy fácilmente. De forma muy breve, todo consistía en crear el método estático al que se deseaba acceder, decorarlo con el atributo WebMethod, e incluir en la página un ScriptManager con estableciéndole la propiedad EnablePageMethods=true; a partir de ese momento, podíamos invocarlo desde cliente usando javascript de forma muy directa.

Sin embargo, el uso de estándares como JSON hace posible la invocación de WebMethods desde scripting sin necesidad de recurrir a la magia del ScriptManager. De hecho, y como vamos a ver más adelante, vamos a realizar la llamada al método utilizando Ajax desde una página HTML pura, sin controles de servidor ni otros edulcorantes proporcionados por ASP.NET.

Desarrollaremos un ejemplo completo (esta vez en VB.NET, por cambiar un poco), que demuestre cómo podemos crear un PageMethod en una página (default.aspx), e invocarlo desde otra (pagina.htm) usando la librería jQuery para comunicar ambos componentes.

1. Definición del PageMethod


Como ya comenté en el post al que hacía referencia antes, un PageMethod se define en la clase asociada a una página .aspx (su code-behind) y es un método normal y corriente, aunque obligatoriamente será estático y público. Además, debe presentar el atributo WebMethod(), que lo identifica como accesible a través llamadas HTTP.

El siguiente código muestra el PageMethod que vamos a utilizar. Se supone que partimos de una página Default.aspx totalmente vacía (de hecho, no vamos a crear nada más dentro), y que el siguiente código corresponde al archivo Default.aspx.vb:

Imports System.Web.Services

Partial Public Class _Default
Inherits System.Web.UI.Page

<WebMethod()> _
Public Shared Function Saludame(ByVal Nombre As String) As String
Return «Hola a » + Nombre _
+ » desde el servidor, son las » _
+ DateTime.Now.ToString(«hh:mm:ss»)

End Function
End Class


Se puede observar que el método recibirá un parámetro, el nombre del destinatario del mensaje de saludo. Una vez que lo tenemos definido de esta forma, siempre que el proyecto esté correctamente configurado, el método es accesible en la dirección «Default.aspx/Saludame» para todo aquél que quiera utilizarlo, siempre que siga ciertas reglas.


2. Creamos la página «cliente»


Interfaz del ejemploPara probar la invocación al método definido anteriormente montaremos un interfaz muy simple, un cuadro de edición en el que el usuario introducirá un nombre y un botón que solicitará al servidor el saludo. Las sucesivas respuestas se irán añadiendo al final de la página, tras la etiqueta «Mensajes obtenidos».

En lugar de mostrar el código fuente de la página HTML (pagina.html) completo, voy a hacerlo por partes, y saltándome algunas porciones que no tienen demasiado interés. En primer lugar va la correspondiente al interfaz, que es bastante simple:

<body>
<form>
<input type=»text» id=»nombre» />
<input type=»button» value=»¡Pulsa!»
onclick=»llamar();» />
<br />
<strong>Mensajes obtenidos:</strong><br />
<label id=»lblMensajes» />
</form>
</body>


Como se puede intuir, la función llamar(), invocada al hacer click sobre el botón, será la que realice la conexión con el servidor, le envíe el nombre contenido en el cuadro de edición, e introduzca la respuesta en la etiqueta lblMensajes, que actuará como contenedor.

Vamos ahora con la porción de página donde se desarrolla la acción. Primero se incluye jQuery en la página, y acto seguido creamos la función llamar() que hemos comentado anteriormente:

<script type=»text/javascript» 
src=»scripts/jquery-1.2.6.min.js» ></script>
<script type=»text/javascript»>
function llamar()
{
$.ajax({
type: «POST»,
data: «{ Nombre: ‘» + $(«#nombre»).val() + «‘}» ,
url: «Default.aspx/Saludame»,
contentType: «application/json; charset=utf-8»,
dataType: «json»,
success: function(msg) {
$(«#lblMensajes»).append(msg);
$(«#lblMensajes»).append(«<br />»);
},
error: function(msg) { alert(«Algún problema debe haber…»); }
});
}
</script>


Los parámetros de llamada a la función ajax() de jQuery son los siguientes:



  • type, la petición HTTP es de tipo POST. Es obligatorio para acceder a los PageMethods por motivos de seguridad.
  • data contiene una representación JSON de los parámetros enviados al método. En este caso, vemos que se compone de forma dinámica inyectándole el valor del cuadro de edición (obtenido a su vez utilizando un selector jQuery y el método val()). Fijaos que esta representación define un objeto anónimo cuyas propiedades deberán coincidir con los parámetros del método; para este ejemplo, un valor correcto de data sería «{ Nombre: ‘Juan’ }».
  • url contiene la dirección del método. Observad que es el nombre de la página seguido del nombre del método.
  • El content-type indica el tipo de contenido que enviamos al servidor; en este caso, dado que le vamos a enviar JSON codificado en utf-8 debe ser «application/json; charset=utf-8».
  • Con dataType indicamos al servidor qué tipo de respuesta estamos esperando. Dado que la trataremos también desde script, será más sencillo si es JSON.
  • success contiene una función anónima que recibe como parámetro la respuesta del servidor. En este caso, hemos implementado su inclusión en la etiqueta que habíamos preparado para mostrar los mensajes del servidor.
  • de la misma forma, en error implementamos lo que queremos que haga el sistema cuando se produzca un problema. En este caso sólo avisamos.


Y… ¡listo para funcionar!

Fijaos en un detalle importante: nada impide la invocación de PageMethods desde páginas distintas a donde se han definido, cosa que con el comportamiento estándar del ScriptManager y su EnablePageMethods=true sería imposible.

Finalmente, recordar que para que todo esto funcione el proyecto web debe estar correctamente configurado para trabajar con las extensiones Ajax. Las plantillas de proyectos de Visual Studio 2008 ya vienen preparadas de forma predeterminada, pero si usas VS2005 deberás crear el proyecto usando las plantillas para aplicaciones Web con Ajax, o retocar a mano el web.config.

Puedes descargar el proyecto de prueba para VS2005.

Publicado en: www.variablenotfound.com.

4 comentarios sobre “Acceder a PageMethods con jQuery”

  1. Hola, JM, estoy intentando implementar esto en un antiguo proyecto web de FMWK 1.1 y me salta el error parsererror, el mismo codigo en otro de FMWK 3.5 me funciona perfectamente. Estoy buscando algo por internet pero de momento no hay manera. Sabes si se puede hacer y si hay que añadir algo más?
    Muchas Gracias.

  2. Hola, Carlos!

    Me temo que pocas pistas voy a poder darte; no recuerdo si esa característica estaba implementada con la versión 1.1 del framework.

    Lo que supongo que sí podrías hacer es lo mismo pero sin usar pagemethods, es decir, servicios web (.asmx), o, llevándolo al extremo, crear un aspx que retorne al cliente renderizando de forma manual (como en MVC) la información que necesite.

    Saludos!

  3. Si, bueno de momento me he montado la respuesta Json de forma un tanto «artesanal» en el Load de la página y haciendo un response con el contentType adecuado si parece que lo coge bien, ya había probado con .asmx, pero el problema es que parece que no llega a entrar por el WebMethod, ya que no devuelve el resultado de forma correcta, seguiré investigando, muchas gracias por responder tan rápido.

  4. Hola, estoy tratando de implementar tu ejemplo, pero tengo algunos inconvenietes (Mensaje de error).
    este es codigo:
    function llamar()
    {
    $.ajax({
    type: «POST»,
    data: «»,
    url: «frmCtaCteUtiliza.aspx/Grabar»,
    contentType: «application/json; charset=utf-8»,
    dataType: «json»,
    success: function() {
    },
    error: function() { alert(«Algún problema debe haber…»); }
    });

Deja un comentario

Tu dirección de correo electrónico no será publicada. Los campos obligatorios están marcados con *