Archivo categoría ASP.NET 2.0 / 3.5

BC30002: Type ‘System.Data.Linq.Mapping.DatabaseAttribute’ is not defined.

Si estás utilizando LINQ en tu aplicación de asp.net y te sale este error:

BC30002: Type 'System.Data.Linq.Mapping.DatabaseAttribute' is not defined.

Es probable que no hayas hecho referencia del assembly en el archivo web.config.
Entre el tag assemblies y /assemblies de tu archivo web.config coloca lo siguiente:

XML:
  1. <add assembly="System.Data.Linq, Version=3.5.0.0, Culture=neutral, PublicKeyToken=B77A5C561934E089"/>

¡Suerte!

No hay Comentarios

Cómo obtener el UserId del usuario logueado

Para obtener el UserID del usuario actualmente logueado, utiliza el siguiente código:

VB.NET:
  1. Dim UsuarioLogueado As MembershipUser = Membership.GetUser()
  2. Dim ElGUID As Guid = UsuarioLogueado.ProviderUserKey
  3. 'Si lo necesitas como string utiliza ElGUID.ToString

En el ejemplo el UserId queda almacenado en la variable ElGUID que es de tipo Guid (Globally unique identifier).

No hay Comentarios

Utilizando múltiples controles UpdateProgress en una sola página

Si tienes dos o más controles UpdatePanel que quieres que se actualicen por separado, quizás también quieras tener diferentes indicadores “un momento por favor…” para cada uno de ellos, con la finalidad de que se muestren en diferentes posiciones.

Antes de darte la solución considera este código:

ASP:
  1. <asp:ScriptManager ID="ScriptManager1" runat="server" />
  2.     <asp:UpdatePanel ID="UpdatePanel1" runat="server">
  3.         <Triggers>
  4.             <asp:AsyncPostBackTrigger ControlID="Boton1" EventName="Click" />
  5.         </Triggers>
  6.         <ContentTemplate>
  7.             <asp:UpdateProgress ID="UpdateProgress1" runat="server" DisplayAfter="0">
  8.                 <ProgressTemplate>
  9.                     Un momento por favor...
  10.                 </ProgressTemplate>
  11.             </asp:UpdateProgress>
  12.             <asp:Button ID="Boton1" runat="server" Text="Botón 1" />
  13.             <asp:Label ID="Hora1" runat="server" Text="<br />"></asp:Label>
  14.         </ContentTemplate>
  15.     </asp:UpdatePanel>
  16.     <br />
  17.     <br />
  18.     <asp:UpdatePanel ID="UpdatePanel2" runat="server">
  19.         <Triggers>
  20.             <asp:AsyncPostBackTrigger ControlID="Boton2" EventName="Click" />
  21.         </Triggers>
  22.         <ContentTemplate>
  23.             <asp:UpdateProgress ID="UpdateProgress2" runat="server" DisplayAfter="0">
  24.                 <ProgressTemplate>
  25.                     Un momento por favor...
  26.                 </ProgressTemplate>
  27.             </asp:UpdateProgress>
  28.             <asp:Button ID="Boton2" runat="server" Text="Botón 2" />
  29.             <asp:Label ID="Hora2" runat="server" Text=""></asp:Label>
  30.         </ContentTemplate>
  31.     </asp:UpdatePanel>

VB.NET:
  1. Protected Sub Boton1_Click(ByVal sender As Object, ByVal e As System.EventArgs) Handles Boton1.Click
  2.         Threading.Thread.Sleep(1000)
  3.         Hora1.Text = "La hora es " & Now.ToString
  4.     End Sub
  5.  
  6.     Protected Sub Boton2_Click(ByVal sender As Object, ByVal e As System.EventArgs) Handles Boton2.Click
  7.         Threading.Thread.Sleep(1000)
  8.         Hora2.Text = "La hora es " & Now.ToString
  9.     End Sub

El cual genera lo siguiente:

image

Y al hacer clic en el botón 1 (o en el 2) ocurre lo siguiente:

image

Los dos controles UpdateProgress aparecieron porque es su comportamiento natural cuando se hace un postBack. Obviamente queremos que cada uno aparezca únicamente cuando se haga clic en su botón correspondiente.

Para hacerlo, utiliza la propiedad AssociatedUpdatePanelID del control UpdateProgress, de tal forma que tu código quede así:

ASP:
  1. <asp:ScriptManager ID="ScriptManager1" runat="server" />
  2.     <asp:UpdatePanel ID="UpdatePanel1" runat="server">
  3.         <Triggers>
  4.             <asp:AsyncPostBackTrigger ControlID="Boton1" EventName="Click" />
  5.         </Triggers>
  6.         <ContentTemplate>
  7.             <asp:UpdateProgress ID="UpdateProgress1" runat="server" DisplayAfter="0" AssociatedUpdatePanelID="UpdatePanel1">
  8.                 <ProgressTemplate>
  9.                     Un momento por favor...
  10.                 </ProgressTemplate>
  11.             </asp:UpdateProgress>
  12.             <asp:Button ID="Boton1" runat="server" Text="Botón 1" />
  13.             <asp:Label ID="Hora1" runat="server" Text="<br />"></asp:Label>
  14.         </ContentTemplate>
  15.     </asp:UpdatePanel>
  16.     <br />
  17.     <br />
  18.     <asp:UpdatePanel ID="UpdatePanel2" runat="server">
  19.         <Triggers>
  20.             <asp:AsyncPostBackTrigger ControlID="Boton2" EventName="Click" />
  21.         </Triggers>
  22.         <ContentTemplate>
  23.             <asp:UpdateProgress ID="UpdateProgress2" runat="server" DisplayAfter="0" AssociatedUpdatePanelID="UpdatePanel2">
  24.                 <ProgressTemplate>
  25.                     Un momento por favor...
  26.                 </ProgressTemplate>
  27.             </asp:UpdateProgress>
  28.             <asp:Button ID="Boton2" runat="server" Text="Botón 2" />
  29.             <asp:Label ID="Hora2" runat="server" Text=""></asp:Label>
  30.         </ContentTemplate>
  31.     </asp:UpdatePanel>

El código VB quedaría igual

De esta forma al oprimir el primer botón ocurrirá esto:

image

Y al oprimir el segundo botón verás esto:

image

De esta forma puedes colocar varios indicadores para cada una de las partes en las que dividas visualmente tu página.

No hay Comentarios

Cómo asignar un mismo procedimiento a múltiples botones

Supongamos que por alguna razón quieres colocar dos botones en una página ASP.NET que ejecuten el mismo procedimiento. Por ejemplo, un largo formulario con dos botones para guardar los cambios.

Solamente tienes que escribir el código para uno de los botones, como normalmente lo harías.

Y para el segundo, establece el evento "OnClick" al procedimiento del primer botón.

Aquí está un ejemplo con los dos botones:

ASP:
  1. <asp:Button ID="Guardar" runat="server" Text="Guardar cambios" />
  2. <asp:Button ID="Guardar2" runat="server" Text="Guardar cambios" OnClick="Guardar_Click" />

¡Suerte!

No hay Comentarios

Internet Explorer deja un espacio entre imágenes

A veces ASP.NET deja pequeños espacios de uno o dos pixeles entre una imagen y otra, como si se tratara de un “padding-bottom”.  He encontrado que estos espacios en ocasiones se deben a saltos de línea (aunque suene ridículo) que se dejan en el código fuente.

Ejemplo:

ASP:
  1. <tr>
  2.    <td>
  3.       <asp:Image Id="Imagen1" runat="server">
  4.    </td>
  5. </tr>
  6. <tr>
  7.    <td>
  8.       <asp:Image Id="Imagen2" runat="server">
  9.    </td>
  10. </tr>

Si lo anterior deja un espacio entre la imagen 1 y la imagen 2, intenta esto:

ASP:
  1. <tr><td><asp:Image Id="Imagen1" runat="server"></td></tr>
  2. <tr><td><asp:Image Id="Imagen2" runat="server"></td></tr>

Claro que no debería ser, pero al parecer es algún bug de Internet Explorer.

Ahora bien, si lo que quieres es una solución más confiable, utiliza el siguiente estilo en tu hoja de estilos:

CSS:
  1. img { display: block; }

Como por arte de magia, el problema desaparece sin tener que estar eliminando saltos de línea en nuestro código.

Tip obtenido de mattPealing.

2 Comentarios

Exclusive access could not be obtained because the database is in use

Intentas restaurar una base de datos desde un backup utilizando Microsoft SQL Server Management Studio y te aparece el siguiente error:

Restore failed for Server 'TuServidor'.  (Microsoft.SqlServer.SmoExtended)

System.Data.SqlClient.SqlError: Exclusive access could not be obtained because the database is in use. (Microsoft.SqlServer.Smo)

Lo que ocurre es que la base de datos está abierta por otro proceso y debes cerrar todas las conexiones. Para hacerlo, abre una ventana de Query y coloca lo siguiente:

SQL:
  1. USE master
  2. go
  3. ALTER DATABASE nombre_de_tu_base_de_datos SET single_user WITH rollback immediate
  4. ALTER DATABASE nombre_de_tu_base_de_datos SET multi_user

Este script fue tomado de un excelente post escrito por Sergio Tarrillo, en el cual puedes encontrar soluciones adicionales para este problema.

8 Comentarios

Cómo hacer una página que cierre la sesión

Para cerrar sesión en una aplicación de ASP.NET puedes utilizar el control LoginStatus, sin embargo en ocasiones querrás que esto ocurra al acceder una página, por ejemplo CerrarSesion.aspx.  Para hacerlo coloca lo siguiente en el code-behind de dicha página:

VB.NET:
  1. Protected Sub Page_Load(ByVal sender As Object, ByVal e As System.EventArgs) Handles Me.Load
  2.          FormsAuthentication.SignOut()
  3.          Session.Clear()
  4.          Session.Abandon()
  5.          Response.Redirect(FormsAuthentication.LoginUrl)
  6.      End Sub

Y listo, con eso cierras la autenticación, limpias y abandonas la sesión, y redireccionas a la página de login.

1 Comentario

Saving changes is not permitted en SQL Server

Trabajando en una instalación limpia de SQL Server 2008, si intentas modificar la estructura de una tabla, puedes encontrarte este error al intentarla guardar:

Saving changes is not permitted. The changes you have made require the following tables to be dropped and re-created. You have either made changes to a table that can't be re-created or enabled the option Prevent saving changes that require the table to be re-created.

saving-not-permitted

Si oprimes Cancel, no se guarda ningún cambio.

La solución es muy sencilla, abre Microsoft SQL Server Management Studio y haz clic en el menú Tools, y luego en Options.

Después en la columna izquierda selecciona Designers y desmarca la casilla Prevent saving changes that require table re-creation.

prevent

Y listo, ahora podrás modificar tu tabla y guardar los cambios.

2 Comentarios

Cómo posicionar verticalmente una página después de un PostBack

En ocasiones tenemos un control de ASP.NET (por ejemplo un DataGrid o un DataPager) que está posicionado verticalmente en la mitad de una página, lo que significa que el visitante tuvo que haber desplazado la barra de desplazamiento (scrollbar) hacia abajo para llegar a ese punto. Si este control realiza un PostBack, cuando la página regresa al navegador, se muestra la parte superior de la página, sacando al visitante de contexto ya que tendría que volver a utilizar la barra de desplazamiento para visualizar el control. Esto no es nada amigable.

Para realizar el salto al punto vertical que decidas, coloca en ese lugar un ancla:

ASP:
  1. <a name="proyectos" />

Después agrega en el code-behind de tu página este procedimiento:

VB.NET:
  1. Private Sub RegistraScriptParaAncla(ByVal strAnchorID As String)
  2.         Dim Script As String = ""
  3.         Script &= "<script language=JavaScript id='BookMarkScript'>" & vbCrLf
  4.         Script &= "var hashValue='#" & strAnchorID & "';" & vbCrLf
  5.         Script &= "if(location.hash!=hashValue)" & vbCrLf
  6.         Script &= "location.hash=hashValue;" & vbCrLf
  7.         Script &= "</script>" & vbCrLf
  8.         If (Not Page.ClientScript.IsStartupScriptRegistered("BookMarkScript")) Then
  9.             Page.ClientScript.RegisterStartupScript(Me.GetType(), "BookMarkScript", Script)
  10.         End If
  11. End Sub

Ahora, en alguna parte de tu code-behind que se ejecute en el PostBack, por ejemplo, al presionar un botón o la liga de un paginador, manda a llamar la función, de esta manera:

VB.NET:
  1. RegistraScriptParaAncla("proyectos")

Puedes tener varias anclas y hacer que la página se desplace a diferentes puntos basado en diferentes condiciones.

Este truco funciona perfectamente y el crédito es para el autor (y los que comentaron) de este post.

No hay Comentarios

User, group, or role already exists in the current database.

Seguramente te ha pasado que al migrar una base de datos de SQL Server de un servidor a otro, haces tu backup usando Microsoft SQL Server Management Studio, transfieres el archivo, y haces el restore en el nuevo servidor. Todo parece perfecto, pero cuando vas a crear tu usuario obtienes el siguiente error:

error1

User, group, or role '' already exists in the current database. (Microsoft SQL Server, Error: 15023)

Entonces expandes tu base de datos, vas al nodo Security -> Users ->, buscas el usuario, y lo intentas borrar, pero entonces aparece otro error:

error2

Drop failed for User ''. (Microsoft.SqlServer.Smo). The database principal owns a database role and cannot be dropped. (Microsoft SQL Server, Error: 15421)

En este punto te das cuenta de que el usuario está "huérfano" (orphaned), y una solución para darle la vuelta al problema es crear un nuevo usuario con otro nombre, asignarlo a la base de datos, y modificar el nombre del usuario en el Connection String de tu aplicación.

Pero existe una manera de solucionar las cosas correcta y fácilmente.

Abre una ventana de query en tu base de datos y teclea lo siguiente para ver los usuarios huérfanos:

SQL:
  1. EXEC sp_change_users_login 'Report'

Si ves a tu usuario ahí significa que efectivamente está huérfano y para deshuerfanizarlo ;) teclea lo siguiente:

SQL:
  1. EXEC sp_change_users_login 'Auto_Fix', 'user'

Modificando user por el nombre de tu usuario.

Y listo, asunto totalmente solucionado, no tienes que crear nuevos usuarios ni modificar las Connection Strings de tus aplicaciones.

Por último, si quieres crear un nuevo usuario para esta base de datos, usa:

SQL:
  1. EXEC sp_change_users_login 'Auto_Fix', 'user', 'login', 'password'

Referencia: Stupid Programmer.

No hay Comentarios