Base64 encoding and embedding images in CSS

By on

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.