Buenas! Una pregunta que últimamente parece que se pregunta varias veces en los foros de ASP.NET MVC es como previsualizar una imagen que se quiere subir al servidor.
Antes que nada aclarar que, técnicamente, la pregunta está mal hecha: no es posible previsualizar la imagen antes de que sea subida. Antiguamente en algunos navegadores, y con un poco de javascript, eso era posible, pero ahora por suerte eso ya no funciona 🙂
Básicamente previsualizar una imagen consiste en:
- Recibir los datos de la imagen
- Guardarla en algún sitio “temporal”
- Mandarla de vuelta al navegador
- Borrar la imagen del sitio “temporal” si el usuario no la acepta
En este primer post vamos a ver como hacerlo sin usar Ajax. Luego habrá un segundo post y veremos como hacerlo usando Ajax (no es tan sencillo porque XMLHttpRequest no soporta el formato de datos multipart/form-data que es el que se usa cuando se mandan ficheros).
Bien, vamos a verlo rápidamente, ya veréis cuan sencillo es 🙂
Lo primero es tener la vista que tenga el formulario para enviar los datos. La siguiente servirá (si queréis más detalles de como hacer upload de ficheros en MVC mirad mi post al respecto):
<h2>Index</h2>
<form enctype="multipart/form-data" method="post" action="@Url.Action("SendImage")">
<label for="img">Seleccionar imagen:</label>
<input type="file" name="img" /> <br />
<input type="submit" />
</form>
Trivial: un form con multipart/form-data que enviará sus datos a la acción SendImage. La acción podría ser algo como:
[HttpPost]
public ActionResult SendImage(HttpPostedFileBase img)
{
var data = new byte[img.ContentLength];
img.InputStream.Read(data, 0, img.ContentLength);
var path = ControllerContext.HttpContext.Server.MapPath("/");
var filename = Path.Combine(path, Path.GetFileName(img.FileName));
System.IO.File.WriteAllBytes(Path.Combine(path, filename), data);
ViewBag.ImageUploaded = filename;
return View("Index");
}
Vale, fijaos que recibimos el HttpPostedFileBase (llamado img como el atributo name del input type=”file”). Lo que hacemos en la acción es muy simple:
- Guardamos la imagen en disco (en este caso por pura pereza uso el directorio raíz de la aplicación web. Por supuesto lo suyo es usar un directorio configurado en web.config y con los permisos NTFS correspondientes).
- Coloco en el ViewBag el nombre de la imagen que se ha guardado (luego vemos porque).
- Devuelvo la vista “Index” (la misma de la cual venimos)
Ok, tal y como lo tenemos ahora, si lo probáis veréis que podéis hacer un upload de la imagen, y la imagen se graba en el disco (en el directorio raíz de la aplicación web), pero no se previsualiza nada. Vamos a modificar la vista Index para que haya esa previsualización.
Para previsualizar la imagen basta con añadir el siguiente código, despues del </form> en la vista:
@if (!string.IsNullOrEmpty(ViewBag.ImageUploaded))
{
<img src="@Url.Action("Preview", new {file=ViewBag.ImageUploaded})" />
}
Simple, no? Si en el ViewBag existe la entrada ImageUploaded generamos un tag <img> cuya dirección es la acción “Preview” y le pasamos el parámetro file con el nombre de la imagen. Y como es la acción Preview? Pues super sencilla:
public ActionResult Preview(string file)
{
var path = ControllerContext.HttpContext.Server.MapPath("/");
if (System.IO.File.Exists(Path.Combine(path, file)))
{
return File(Path.Combine(path, file), "image/jpeg");
}
return new HttpNotFoundResult();
}
Simplemente leemos la imagen guardada y la devolvemos usando un FilePathResult. Simple, eh? 😉
Con eso, si subís una imagen, cuando le deis a submit, aparecerá de nuevo la vista, pero ahora previsualizando la imagen.
Ahora sólo nos queda el punto final: Que el usuario pueda aceptar esa imagen como correcta. Para ello lo más simple es hacer lo siguiente:
- Si el usuario está previsualizando una imagen y NO ha seleccionado otra, cuando hace el submit se entiende que acepta dicha imagen.
- Si el usuario NO está previsualizando una imagen, o bien está previsualizando una pero selecciona otra, al hacer el submit se entiende que descarta la imagen anterior y quiere previsualizar la nueva.
Para ello, tenemos que hacer que la vista le indique al controlador si se está previsualizando una imagen, y cual és. Por suerte eso es muy sencillo. Basta con modificar el <form> de la vista para que su atributo action quede como:
<form enctype="multipart/form-data" method="post"
action="@Url.Action("SendImage", new {previewed=ViewBag.ImageUploaded})">
Fijaos que añadimos un parámetro previewed que la vista mandará a la acción y que será el nombre de la imagen que se está previsualizando.
Vamos a modificar la acción para que reciba ese parámetro y actúe en consecuencia:
[HttpPost]
public ActionResult SendImage(HttpPostedFileBase img, string previewed)
{
if (!string.IsNullOrEmpty(previewed) && img == null)
{
ViewBag.ImageName = previewed;
return View("Step2");
}
// Código tal cual estaba
}
Añadimos ese if al inicio: Si el usuario está previsualizando una imagen y no ha seleccionado otra, entendemos que acepta esta imagen. Entonces guardamos el nombre en el ViewBag y lo mandamos a la siguiente vista (Step2).
La vista es muy sencilla:
<h2>Step2</h2>
Siguiente paso. El usuario ha aceptado la imagen: <br />
<img src="@Url.Action("Preview", new {file=ViewBag.ImageName})" />
Simplemente mostramos la imagen. Fijaos que de nuevo usamos la acción Preview, puesto que la imagen ya la tenemos guardada en el servidor.
Notas finales
Faltarían, al menos, dos cosas para que dicho proyecto funcionase “de forma aceptable”:
- Borrar las imagenes descartadas por el usuario (en la acción SendImage si se recibe una imagen nueva y se estaba previsualizando una, borrar esta ya que el usuario la ha descartado).
- La acción Preview siempre devuelve content-type a image/jpeg, eso debería hacerse según la extensión de la imagen, o guardarlo en algún sitio.
Ambas cosas son triviales y no las añado porque lo único que consiguiría es liar el código un poco más 😉
En el próximo post… lo mismo pero usando Ajax.
Saludos!
Buenas! Donde dije digo, digo Diego… Sí, ya sé que dije que el segundo post sería como hacerlo con Ajax
muy buen post, gracias por compartirlo, resolviste lo del VS2010 SP1? Yo lo resolví con el SP1 Oficial y el SDK 1.4 de Azure.
Hola! Antes que nada, gracias por el ejemplo, me resulto muy útil. Me gustaría hacerte una consulta, sería posible usar un esquema similar con vistas parciales? Por ejemplo para un escenario donde subir la imagen es sólo una parte de la función del formulario?
@Laura
Si que es posible, pero usando Ajax para ello. Estoy preparando el post donde explica como implementar el upload via Ajax 😉
@phito
Buenoooo… al final tuve que desinstalar cosas obsoletas (visual studio 2008) para tener espacio suficiente para instalar el sp1 de vs 2010, que todavía no se porque, pide 5GB LIBRES antes de empezar a instalarse! :S
En fin…
Gracias por vuestros comentarios! 😀