Added a CSS data URI generator

Signed-off-by: Jannis Leidel <jannis@leidel.info>
This commit is contained in:
Atamert Ölçgen
2010-02-17 23:15:59 +02:00
committed by Jannis Leidel
parent 24b67475ac
commit 316033916e
6 changed files with 67 additions and 0 deletions

View File

@@ -18,3 +18,5 @@ if COMPRESS_CSS_FILTERS is None:
if COMPRESS_JS_FILTERS is None:
COMPRESS_JS_FILTERS = []
COMPRESS_DATA_URI_MIN_SIZE = getattr(settings, 'COMPRESS_DATA_URI_MIN_SIZE', 1024)

View File

@@ -0,0 +1,46 @@
import os, re
from base64 import b64encode
from mimetypes import guess_type
from compressor.filters import FilterBase, FilterError
from compressor.conf import settings
class DataUriFilter(FilterBase):
"""Filter for embedding media as data: URIs.
Settings:
COMPRESS_DATA_URI_MIN_SIZE: Only files that are smaller than this
value will be embedded. Unit; bytes.
Don't use this class directly. Use a subclass.
"""
def input(self, filename=None, **kwargs):
if not filename or not filename.startswith(settings.MEDIA_ROOT):
return self.content
output = self.content
for url_pattern in self.url_patterns:
output = url_pattern.sub(self.data_uri_converter, output)
return output
def get_file_path(self, url):
return os.path.join(settings.MEDIA_ROOT, url[len(settings.MEDIA_URL):])
def data_uri_converter(self, matchobj):
url = matchobj.group(1).strip(' \'"')
if not url.startswith('data:'):
path = self.get_file_path(url)
if os.stat(path).st_size <= settings.COMPRESS_DATA_URI_MIN_SIZE:
data = b64encode(open(path, 'rb').read())
return 'url("data:%s;base64,%s")' % (guess_type(path)[0], data)
return 'url("%s")' % url
class CssDataUriFilter(DataUriFilter):
"""Filter for embedding media as data: URIs in CSS files.
See DataUriFilter.
"""
url_patterns = (
re.compile(r'url\(([^\)]+)\)'),
)

View File

@@ -137,6 +137,21 @@ class CssAbsolutizingTestCase(TestCase):
self.assertEqual(out, self.cssNode.hunks)
class CssDataUriTestCase(TestCase):
def setUp(self):
settings.COMPRESS = True
settings.COMPRESS_CSS_FILTERS = ['compressor.filters.datauri.CssDataUriFilter']
settings.MEDIA_URL = '/media/'
self.css = """
<link rel="stylesheet" href="/media/css/datauri.css" type="text/css" charset="utf-8">
"""
self.cssNode = CssCompressor(self.css)
def test_data_uris(self):
out = [u'.add { background-image: url(""); }\n.python { background-image: url("/media/img/python.png"); }\n.datauri { background-image: url(" vr4MkhoXe0rZigAAAABJRU5ErkJggg=="); }\n']
self.assertEqual(out, self.cssNode.hunks)
class CssMediaTestCase(TestCase):
def setUp(self):
self.css = """
@@ -170,6 +185,7 @@ def render(template_string, context_dict=None):
t = Template(template_string)
return t.render(c).strip()
class TemplatetagTestCase(TestCase):
def setUp(self):
settings.COMPRESS = True

View File

@@ -0,0 +1,3 @@
.add { background-image: url("../img/add.png"); }
.python { background-image: url("../img/python.png"); }
.datauri { background-image: url(" vr4MkhoXe0rZigAAAABJRU5ErkJggg=="); }

BIN
tests/media/img/add.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 733 B

BIN
tests/media/img/python.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 KiB