Imagen rota

Imágenes que no dicen onLoad

JavaScript y la etiqueta <img> siempre han sido buenos amigos, desde aquellos días que  gracias a un pequeño script se precargaban todas las imágenes de una página, como por ejemplo los rollovers de ciertos botones.

var preload_imgs = ["image1.jpg", "image2.gif", "image3.jpg"];
for(var i=0; i<preload_imgs.length; i++)
{
	var img = new Image();
	img.src = preload_imgs[i];
}

¿Pero que ocurre cuando se crean o se tratan imágenes dinámicamente, y se quiere controlar el preciso momento en que están son (des)cargadas?

La respuesta parece sencilla: registrar el evento load de la imagen según el navegador del usuario (addEventListener para clientes modernos, attachEvent para IE arcanos y onload para el resto) y esperamos a que se dispare. El contrapunto es que esto último no siempre ocurre, y las causas pueden ser muy variadas.

Tomemos el siguiente código de ejemplo:

var loadHandler = function()
{
	document.getElementById('container').appendChild( this );
}

var img = new Image();
img.src = "carcharoth.jpg";
// Se evita deliberadamente la lógica de descubrir el registrador
// correcto por navegador para ahorrarnos código
img.onload = loadHandler;

La primera sensación es que debería funcionar: se añade al documento la imagen una vez cargada; y en efecto lo hace en la mayoría de navegadores de escritorio. Menos en Internet Explorer, versiones 8 e inferiores. El porqué se debe a que se registra el evento onload después de asignar un nuevo origen a la imagen (img.src = "carcharoth.jpg"), y en el caso que la imagen esté en la caché del navegador (ya se ha descargado previamente), IE ejecutará el evento load inmediatamente después de la asignación sin dar tiempo a que se registre.

Para corregirlo, simplemente se ha de colocar el registrador de eventos antes de cambiar la URL de la imagen:

var img = new Image();
img.onload = loadHandler;
img.src = "carcharoth.jpg";

O comprobar si la imagen se ha «completado» (img.complete) y en consecuencia invocar a pelo el método registrado:

var img = new Image();
img.onload = loadHandler;
if( img.complete )
	img.onload();

Otra situación: tenemos una imagen en el documento, y cambiamos su origen URL a ese mismo origen vía JavaScript. ¿Debería lanzar un evento load? Así es y así se hace en todos los navegadores a excepción de Chrome; el navegador de Google tiene un pequeño bug que impide lanzar el evento cuando se carga el mismo origen para una imagen.

Para solucionarlo, como explican en éste hilo, se ha de “vaciar” el origen URL de la imagen antes de asignarle el mismo.

// La URL de la imagen es "shenglong.jpg"
var img = document.getElementById('myimg');
img.src = "";
img.addEventListener("load", loadHandler, false);
img.src = "shenglong.jpg";

 

Una manera efectiva de evitar estos comportamientos es utilizar algún tipo de framework que vigile por la buena salud de nuestras imágenes. jQuery, como no, hace un buen trabajo gracias al método .load(), aunque en la página de la referencia avisan que tampoco es perfecto.

Por suerte, un buen samaritano en los comentarios de esa página ofrece un truquillo para que funcione en la mayoría de escenarios:

// Crea una imagen vacía
var $img = $('<img>');
// Registra el evento load
$img.load( loadHandler );
// Asigna la URL de la imagen para que el evento load
// sea correctamente capturado
$img.attr( 'src', 'mventris.jpg' );

Y si todo falla, y no hay manera que esas jodidas imágenes de la galería alerten cuando se han cargado completamente, explorando en GitHub encontramos el plugin para jQuery imagesLoaded, creado por Paul Irish (el tío ese que parece que está desarrollando HTML5 el solito). Partiendo de un contenedor, el plugin es capaz de detectar cuándo las imágenes que contiene se han cargado y cuántos errores han ocurrido. Y por las pruebas parece que es muy eficaz.

var $container = $('#container');
var preload_imgs = ["image1.jpg", "image2.gif", "image3.jpg"];

for(var i=0; i<preload_imgs.length; i++)
{
	// Crea las imágenes y las añade al contenedor
	var $img = $('<img>');
	$img.attr( 'src', preload_imgs[i] );
	$container.append( $img );
}

// Se invocará una vez todas las imágenes del contenedor
// se hayan cargado con o sin error
$container.imagesLoaded( function( $images, $proper, $broken )
{
	$container.append(
	'<p>' + $proper.length + ' images loaded of ' + $images.length
	+ '<br>Errors: ' + $broken.length + '</p>' );
});

 

¿Conoces más formas de detectar vía JavaScript cuando una imagen se ha cargado?

Publicado por

Raúl Parralejo

Raúl Parralejo

Frontend developer, especializado en el desarrollo de aplicaciones web y juegos para móvil y escritorio.

Deja una respuesta

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