Added a CSS data URI generator
Signed-off-by: Jannis Leidel <jannis@leidel.info>
This commit is contained in:
committed by
Jannis Leidel
parent
24b67475ac
commit
316033916e
@@ -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)
|
||||
|
||||
46
compressor/filters/datauri.py
Normal file
46
compressor/filters/datauri.py
Normal 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\(([^\)]+)\)'),
|
||||
)
|
||||
@@ -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
|
||||
|
||||
3
tests/media/css/datauri.css
Normal file
3
tests/media/css/datauri.css
Normal 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
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
BIN
tests/media/img/python.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 11 KiB |
Reference in New Issue
Block a user