Creo que algo como esto podría funcionar:
var hasWebP = false;
(function() {
var img = new Image();
img.onload = function() {
hasWebP = !!(img.height > 0 && img.width > 0);
};
img.onerror = function() {
hasWebP = false;
};
img.src = 'http://www.gstatic.com/webp/gallery/1.webp';
})();
En Firefox e IE, el controlador "onload" simplemente no se llamará en absoluto si la imagen no se puede entender, y en su lugar se llama "onerror".
No mencionaste jQuery, pero como ejemplo de cómo lidiar con la naturaleza asincrónica de esa verificación, podrías devolver un objeto jQuery "Deferred":
function hasWebP() {
var rv = $.Deferred();
var img = new Image();
img.onload = function() { rv.resolve(); };
img.onerror = function() { rv.reject(); };
img.src = 'http://www.gstatic.com/webp/gallery/1.webp';
return rv.promise();
}
Entonces podrías escribir:
hasWebP().then(function() {
}, function() {
});
Aquí hay un ejemplo de jsfiddle.
Un verificador más avanzado: http://jsfiddle.net/JMzj2/29/ . Éste carga imágenes de una URL de datos y comprueba si se carga correctamente. Dado que WebP ahora también admite imágenes sin pérdida, puede verificar si el navegador actual solo admite WebP con pérdida o también WebP sin pérdida. (Nota: esto también comprueba implícitamente el soporte de URL de datos).
var hasWebP = (function() {
var images = {
basic: "",
lossless: ""
};
return function(feature) {
var deferred = $.Deferred();
$("<img>").on("load", function() {
if(this.width === 2 && this.height === 1) {
deferred.resolve();
} else {
deferred.reject();
}
}).on("error", function() {
deferred.reject();
}).attr("src", images[feature || "basic"]);
return deferred.promise();
}
})();
var add = function(msg) {
$("<p>").text(msg).appendTo("#x");
};
hasWebP().then(function() {
add("Basic WebP available");
}, function() {
add("Basic WebP *not* available");
});
hasWebP("lossless").then(function() {
add("Lossless WebP available");
}, function() {
add("Lossless WebP *not* available");
});