Base64 encoding and embedding images in CSS

One problem I always have with mobile development is making images load faster. Having lots of images can make the page extremely slow, and with the iphone 4 using double rez images, its even slower.

Theres a couple steps to take to optimize the images.

First I recommend using no image tags, and only using the url() property in CSS to display your images.

.img-social-facebook { background: url(/assets/images/gurgo/social-icon-facebook.png) no-repeat; background-size: 100%; width: 32px; height: 32px; }

Second, encode all your url entries using base64 encode. This allows each image to be inside your main css.

.img-social-facebook { background: url() no-repeat; background-size: 100%; width: 32px; height: 32px; }

The problem with this is encoding all the images into a css file can be hella time consuming. The solution: write a script to do it for you!

./cssbuild.php gurgo.src.css gurgo.css

Thats all you need to convert a css any time you modify it. This script will read though the css for whatever regex you want, encode the images, and output it to the second file. Source:

#!/usr/bin/php <?php

$file = dirname(FILE).'/../public_html/assets/css/'.$argv[1];

$lines = file($file); foreach ($lines as $key => $line) {

if (preg_match('/url\(\/assets\/images.*\)/i',$line)) {
  $css = preg_replace('/^.*url\(\/assets\/images\/(.*)\).*$/','\\1',$line);
  $file = new Caffeine_View_Helper_ImageBase64(dirname(__FILE__).'/../public_html/assets/images/'.trim($css));
  $lines[$key] = preg_replace('/url\((.*)\)/','url('.$file.')',$lines[$key]);
}

}

$writefile = dirname(FILE).'/../public_html/assets/css/'.$argv[2]; file_put_contents($writefile, $lines); echo "done\n";

class Caffeine_View_Helper_ImageBase64 { public function __construct($image) { $this->_file = $image; $fp = fopen($this->_file,'rb', 0); if (!$fp) throw new Exception('Could not open '.$this->_file.' for reading'); $picture = fread($fp,filesize($this->_file)); fclose($fp); $this->_encoded = base64_encode($picture); }

public function output() {
  return 'data:'.mime_content_type($this->_file).';base64,'.$this->_encoded;
}

public function __toString() {
  return $this->output();
}

}

Reasons NOT to use this method: If you have a page that has images only used on a single page, I instead recommend using a base64 encoded img tag on that page. If that image is going to be reused, using a normal image is most likely your best bet.