Храните мелкие картинки в CSS

Храните мелкие картинки, которые нельзя засунуть в спрайты, в data:image base64 в CSS — это экономит кучу запросов к вебсерверу.

Кодируем изображение в base64 с помощью онлайн сервисов, вроде сервиса от DailyCoding (очень удобно, ничего лишнего).

Кладем получившуюся строку в CSS файл, заменяя «ТИП» на MIME-тип вашего изображения — jpeg/png/gif или (OMG!) bmp и «КОД» на нужную строку в base64:
.some_background {
background-image: url("data:image/ТИП;base64,КОД");
}

Теперь можно смело подключать нужному элементу стиль some_background и наслаждаться двумя запросами к вебсерверу (html + css), вместо трех (html + css + изображение).

Пример реализации с изображениями:

  • images.html — 361 байт
  • images/images.css — 305 байт
  • images/test1.png — 1 600 байт
  • images/test2.png — 1 143 байт
  • Итого — 3 409 байт


* при клике, картинка увеличится

Пример реализации с base64:

  • base64.html — 361 байт
  • images/base64.css — 3 991 байт
  • Итого — 4 352 байт


* при клике, картинка увеличится

Пример работы готового инструмента Jammit:

  • (Google Chrome, ~60 Мбит/сек):
  • До (102 запроса, 73,23 КБ передано, 3.41 сек):


    * при клике, картинка увеличится

    После (2 запроса, 153,88 КБ передано, 0,94 сек):


    * при клике, картинка увеличится

    Плюсы:

    • уменьшение числа запросов к вебсерверу;
    • меньшее засорение кеша пользователя;
    • иногда уменьшение результативного объема изображения на больших файлах.

    Минусы:

    • сложность обновления изображений;
    • иногда незначительное увеличение результативного объема изображения на мелких файлах.

    Непонятности (непонятно, плюс это или минус):

    • Internet Explorer 5, 6 и 7 не добавляли в друзья base64, но в IE8 работает нормально. Ее можно включить, но не рекомендую это делать, лучше использовать mhtml

    Мне кажется, что увеличение объема CSS файла лучше, чем лишний запрос к вебсерверу, поскольку по-умолчанию браузеры открывают в среднем 8 паралленьных соединений к вебсерверу, а 50—70 изображений это уже очередь, а кто любит очереди? :)

    Использовать этот метод стоит уже после утверждения дизайна и верстки, поскольку обновлять изображения таким методом несколько дольше, чем обычно.

    Для автоматического упаковывания изображений в base64 есть онлайн сервис duris.ru, но можно использовать и PHP скрипт с регуляркой:
    <?php
    echo preg_replace('/images\/[-\w\/\.]*/ie','"data:image/".((substr("\\0",-4)==".png")?"png":"gif").";base64,".base64_encode(file_get_contents("\\0"))',file_get_contents('style.css'));
    ?>

    Он делает из такого CSS файла:

    * {
    	padding:0;
    	margin:0;
    }
    html {
    	display:table;
    	width:100%;
    	height:100%;
    }
    body {
    	margin:auto 0;
    	overflow-y:scroll;
    	background:hsl(0,0%,30%) url(images/background.svg) no-repeat;
    
    }
    .px_sort_0{background:url(images/px/arrow-090-small.png)}
    .px_sort_1{background:url(images/px/arrow-270-small.png)}/* margin:0 5px; */
    .px_status_0{background:url(images/px/minus-circle-frame.png);cursor:pointer}
    .px_status_1{background:url(images/px/plus-circle-frame.png);cursor:pointer}
    .px_delete{background:url(images/px/cross-circle-frame.png);cursor:pointer}
    .px_help{background:url(images/px/question-frame.png);cursor:help}
    .px_info{background:url(images/px/information-frame.png);cursor:help}
    .px_return{background:url(images/px/arrow-skip-180.png);cursor:pointer}
    .px_watch{background:url(images/px/eye.png);cursor:pointer}
    .px_home{background:url(images/px/home.png)}
    [data-beforeAddContent]:before {
    	content:attr(data-beforeAddContent);
    	display:block;
    	color:red;
    	width:100px;
    	height:16px;
    	border:10px solid black;
    }
    

    такой:

    * {
    	padding:0;
    	margin:0;
    }
    html {
    	display:table;
    	width:100%;
    	height:100%;
    }
    body {
    	margin:auto 0;
    	overflow-y:scroll;
    	background:hsl(0,0%,30%) url() no-repeat;
    }
    .px_sort_0{background:url()}
    .px_sort_1{background:url()}/* margin:0 5px; */
    .px_status_0{background:url();cursor:pointer}
    .px_status_1{background:url();cursor:pointer}
    .px_delete{background:url();cursor:pointer}
    .px_help{background:url();cursor:help}
    .px_info{background:url();cursor:help}
    .px_return{background:url();cursor:pointer}
    .px_watch{background:url();cursor:pointer}
    .px_home{background:url()}
    [data-beforeAddContent]:before{
    	content:attr(data-beforeAddContent);
    	display:block;
    	color:red;
    	width:100px;
    	height:16px;
    	border:10px solid black;
    }
    

    Для Ruby on Rails есть примочка Jammit, которая упаковывает изображения в CSS, а так же делает кучу других плюшек.