using System.Collections.Generic;
using System.Diagnostics;
internal static class ImageHelper
const string errorMessage = "Could not recognise image format.";
private static Dictionary<byte[], Func<BinaryReader, Size>> imageFormatDecoders = new Dictionary<byte[], Func<BinaryReader, Size>>()
{ new byte[] { 0x42, 0x4D }, DecodeBitmap },
{ new byte[] { 0x47, 0x49, 0x46, 0x38, 0x37, 0x61 }, DecodeGif },
{ new byte[] { 0x47, 0x49, 0x46, 0x38, 0x39, 0x61 }, DecodeGif },
{ new byte[] { 0x89, 0x50, 0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A }, DecodePng },
{ new byte[] { 0xff, 0xd8 }, DecodeJfif },
{ new byte[] { 0x52, 0x49, 0x46, 0x46 }, DecodeWebP },
public static Size GetDimensions(BinaryReader binaryReader)
int maxMagicBytesLength = imageFormatDecoders.Keys.OrderByDescending(x => x.Length).First().Length;
byte[] magicBytes = new byte[maxMagicBytesLength];
for (int i = 0; i < maxMagicBytesLength; i += 1)
magicBytes[i] = binaryReader.ReadByte();
foreach (var kvPair in imageFormatDecoders)
if (StartsWith(magicBytes, kvPair.Key))
Console.WriteLine(kvPair.Value.Method);
return kvPair.Value(binaryReader);
throw new ArgumentException(errorMessage, "binaryReader");
public static Size GetDimensions(string path)
using (BinaryReader binaryReader = new BinaryReader(File.OpenRead(path)))
return GetDimensions(binaryReader);
catch (ArgumentException e)
if (e.Message.StartsWith(errorMessage))
throw new ArgumentException(errorMessage, "path", e);
public static Size GetDimensions(MemoryStream ms)
using (BinaryReader binaryReader = new BinaryReader(ms))
return GetDimensions(binaryReader);
catch (ArgumentException e)
if (e.Message.StartsWith(errorMessage))
throw new ArgumentException(errorMessage, "path", e);
private static bool StartsWith(byte[] thisBytes, byte[] thatBytes)
for (int i = 0; i < thatBytes.Length; i += 1)
if (thisBytes[i] != thatBytes[i])
private static short ReadLittleEndianInt16(BinaryReader binaryReader)
byte[] bytes = new byte[sizeof(short)];
for (int i = 0; i < sizeof(short); i += 1)
bytes[sizeof(short) - 1 - i] = binaryReader.ReadByte();
return BitConverter.ToInt16(bytes, 0);
private static int ReadLittleEndianInt32(BinaryReader binaryReader)
byte[] bytes = new byte[sizeof(int)];
for (int i = 0; i < sizeof(int); i += 1)
bytes[sizeof(int) - 1 - i] = binaryReader.ReadByte();
return BitConverter.ToInt32(bytes, 0);
private static Size DecodeBitmap(BinaryReader binaryReader)
binaryReader.ReadBytes(16);
int width = binaryReader.ReadInt32();
int height = binaryReader.ReadInt32();
return new Size(width, height);
private static Size DecodeGif(BinaryReader binaryReader)
int width = binaryReader.ReadInt16();
int height = binaryReader.ReadInt16();
return new Size(width, height);
private static Size DecodePng(BinaryReader binaryReader)
binaryReader.ReadBytes(8);
int width = ReadLittleEndianInt32(binaryReader);
int height = ReadLittleEndianInt32(binaryReader);
return new Size(width, height);
private static Size DecodeJfif(BinaryReader binaryReader)
while (binaryReader.ReadByte() == 0xff)
byte marker = binaryReader.ReadByte();
short chunkLength = ReadLittleEndianInt16(binaryReader);
if (marker == 0xc0 || marker == 0xc2)
int height = ReadLittleEndianInt16(binaryReader);
int width = ReadLittleEndianInt16(binaryReader);
return new Size(width, height);
ushort uchunkLength = (ushort)chunkLength;
binaryReader.ReadBytes(uchunkLength - 2);
binaryReader.ReadBytes(chunkLength - 2);
throw new ArgumentException(errorMessage);
private static Size DecodeWebP(BinaryReader binaryReader)
var size = binaryReader.ReadUInt32();
var webp = binaryReader.ReadBytes(4);
var type = binaryReader.ReadBytes(4);
string VP8Type = System.Text.Encoding.UTF8.GetString(type);
Console.WriteLine("VP8Type=\""+VP8Type+"\"");
binaryReader.ReadBytes(4);
byte Flags = binaryReader.ReadByte();
bool bitR3C = (((Flags>> bitPos) & 1) != 0);
var bitAniC = (((Flags>> bitPos) & 1) != 0);
Console.WriteLine("has Animantion ? {0}", bitAniC);
var bitXMPC = (((Flags>> bitPos) & 1) != 0);
Console.WriteLine("has XMP ? {0}", bitXMPC);
bool bitExifC = (((Flags>> bitPos) & 1) != 0);
Console.WriteLine("has EXIF ? {0}", bitExifC);
var bitAlphaC = (((Flags>> bitPos) & 1) != 0);
Console.WriteLine("has Alpha ? {0}", bitAlphaC);
var bitICCC = (((Flags>> bitPos) & 1) != 0);
Console.WriteLine("has ICC ? {0}", bitAniC);
var bitR2C = (((Flags>> bitPos) & 1) != 0);
var bitR1C = (((Flags>> bitPos) & 1) != 0);
binaryReader.ReadBytes(3);
byte[] w = binaryReader.ReadBytes(3);
x = 1 + (w[2] << 16 | w[1] << 8 | w[0]);
byte[] h = binaryReader.ReadBytes(3);
y = 1 + (h[2] << 16 | h[1] << 8 | h[0]);
else if (VP8Type == "VP8L")
binaryReader.ReadBytes(4);
byte[] sig = binaryReader.ReadBytes(1);
if (sig[0] != 47) new Size(0, 0);
byte[] wh = binaryReader.ReadBytes(4);
x = 1 + (((wh[1] & 0x3F) << 8) | wh[0]);
y = 1 + (((wh[3] & 0xF) << 10) | (wh[2] << 2) | ((wh[1] & 0xC0) >> 6));
else if (VP8Type == "VP8 ")
binaryReader.ReadBytes(7);
byte[] frameTag = binaryReader.ReadBytes(3);
if (frameTag[0] != 157 && frameTag[0] != 1 && frameTag[0] != 42) return new Size(0, 0);
x = binaryReader.ReadUInt16() & 0x3fff;
y = binaryReader.ReadUInt16() & 0x3fff;
public static string GetFileNameFromURL(string hrefLink)
string[] parts = hrefLink.Split('/');
string fileName = string.Empty;
fileName = parts[parts.Length - 1];
public static void Main()
Stopwatch sw = new Stopwatch();
string webpURL = "https://www.gstatic.com/webp/gallery3/1_webp_a.webp";
string webpfile = GetFileNameFromURL(webpURL);
Size webpSize = new Size();
WebClient wc = new WebClient();
using (MemoryStream stream = new MemoryStream(wc.DownloadData(webpURL)))
webpSize = ImageHelper.GetDimensions(stream);
Console.WriteLine("File \"{0}\" has dimensions [{1}w X {2}h] in {3} ms.", webpfile, webpSize.Width, webpSize.Height, sw.ElapsedMilliseconds);