9 changed files with 18 additions and 568 deletions
@ -1,202 +0,0 @@ |
|||
#!/usr/bin/env python |
|||
# -*- coding: utf-8 -*- |
|||
|
|||
# `cssmin.py` - A Python port of the YUI CSS compressor. |
|||
|
|||
|
|||
from StringIO import StringIO # The pure-Python StringIO supports unicode. |
|||
import re |
|||
|
|||
|
|||
__version__ = '0.1.1' |
|||
|
|||
|
|||
def remove_comments(css): |
|||
"""Remove all CSS comment blocks.""" |
|||
|
|||
iemac = False |
|||
preserve = False |
|||
comment_start = css.find("/*") |
|||
while comment_start >= 0: |
|||
# Preserve comments that look like `/*!...*/`. |
|||
# Slicing is used to make sure we don"t get an IndexError. |
|||
preserve = css[comment_start + 2:comment_start + 3] == "!" |
|||
|
|||
comment_end = css.find("*/", comment_start + 2) |
|||
if comment_end < 0: |
|||
if not preserve: |
|||
css = css[:comment_start] |
|||
break |
|||
elif comment_end >= (comment_start + 2): |
|||
if css[comment_end - 1] == "\\": |
|||
# This is an IE Mac-specific comment; leave this one and the |
|||
# following one alone. |
|||
comment_start = comment_end + 2 |
|||
iemac = True |
|||
elif iemac: |
|||
comment_start = comment_end + 2 |
|||
iemac = False |
|||
elif not preserve: |
|||
css = css[:comment_start] + css[comment_end + 2:] |
|||
else: |
|||
comment_start = comment_end + 2 |
|||
comment_start = css.find("/*", comment_start) |
|||
|
|||
return css |
|||
|
|||
|
|||
def remove_unnecessary_whitespace(css): |
|||
"""Remove unnecessary whitespace characters.""" |
|||
|
|||
def pseudoclasscolon(css): |
|||
|
|||
""" |
|||
Prevents 'p :link' from becoming 'p:link'. |
|||
|
|||
Translates 'p :link' into 'p ___PSEUDOCLASSCOLON___link'; this is |
|||
translated back again later. |
|||
""" |
|||
|
|||
regex = re.compile(r"(^|\})(([^\{\:])+\:)+([^\{]*\{)") |
|||
match = regex.search(css) |
|||
while match: |
|||
css = ''.join([ |
|||
css[:match.start()], |
|||
match.group().replace(":", "___PSEUDOCLASSCOLON___"), |
|||
css[match.end():]]) |
|||
match = regex.search(css) |
|||
return css |
|||
|
|||
css = pseudoclasscolon(css) |
|||
# Remove spaces from before things. |
|||
css = re.sub(r"\s+([!{};:>+\(\)\],])", r"\1", css) |
|||
|
|||
# If there is a `@charset`, then only allow one, and move to the beginning. |
|||
css = re.sub(r"^(.*)(@charset \"[^\"]*\";)", r"\2\1", css) |
|||
css = re.sub(r"^(\s*@charset [^;]+;\s*)+", r"\1", css) |
|||
|
|||
# Put the space back in for a few cases, such as `@media screen` and |
|||
# `(-webkit-min-device-pixel-ratio:0)`. |
|||
css = re.sub(r"\band\(", "and (", css) |
|||
|
|||
# Put the colons back. |
|||
css = css.replace('___PSEUDOCLASSCOLON___', ':') |
|||
|
|||
# Remove spaces from after things. |
|||
css = re.sub(r"([!{}:;>+\(\[,])\s+", r"\1", css) |
|||
|
|||
return css |
|||
|
|||
|
|||
def remove_unnecessary_semicolons(css): |
|||
"""Remove unnecessary semicolons.""" |
|||
|
|||
return re.sub(r";+\}", "}", css) |
|||
|
|||
|
|||
def remove_empty_rules(css): |
|||
"""Remove empty rules.""" |
|||
|
|||
return re.sub(r"[^\}\{]+\{\}", "", css) |
|||
|
|||
|
|||
def normalize_rgb_colors_to_hex(css): |
|||
"""Convert `rgb(51,102,153)` to `#336699`.""" |
|||
|
|||
regex = re.compile(r"rgb\s*\(\s*([0-9,\s]+)\s*\)") |
|||
match = regex.search(css) |
|||
while match: |
|||
colors = match.group(1).split(",") |
|||
hexcolor = '#%.2x%.2x%.2x' % tuple(map(int, colors)) |
|||
css = css.replace(match.group(), hexcolor) |
|||
match = regex.search(css) |
|||
return css |
|||
|
|||
|
|||
def condense_zero_units(css): |
|||
"""Replace `0(px, em, %, etc)` with `0`.""" |
|||
|
|||
return re.sub(r"([\s:])(0)(px|em|%|in|cm|mm|pc|pt|ex)", r"\1\2", css) |
|||
|
|||
|
|||
def condense_multidimensional_zeros(css): |
|||
"""Replace `:0 0 0 0;`, `:0 0 0;` etc. with `:0;`.""" |
|||
|
|||
css = css.replace(":0 0 0 0;", ":0;") |
|||
css = css.replace(":0 0 0;", ":0;") |
|||
css = css.replace(":0 0;", ":0;") |
|||
|
|||
# Revert `background-position:0;` to the valid `background-position:0 0;`. |
|||
css = css.replace("background-position:0;", "background-position:0 0;") |
|||
|
|||
return css |
|||
|
|||
|
|||
def condense_floating_points(css): |
|||
"""Replace `0.6` with `.6` where possible.""" |
|||
|
|||
return re.sub(r"(:|\s)0+\.(\d+)", r"\1.\2", css) |
|||
|
|||
|
|||
def condense_hex_colors(css): |
|||
"""Shorten colors from #AABBCC to #ABC where possible.""" |
|||
|
|||
regex = re.compile(r"([^\"'=\s])(\s*)#([0-9a-fA-F])([0-9a-fA-F])([0-9a-fA-F])([0-9a-fA-F])([0-9a-fA-F])([0-9a-fA-F])") |
|||
match = regex.search(css) |
|||
while match: |
|||
first = match.group(3) + match.group(5) + match.group(7) |
|||
second = match.group(4) + match.group(6) + match.group(8) |
|||
if first.lower() == second.lower(): |
|||
css = css.replace(match.group(), match.group(1) + match.group(2) + '#' + first) |
|||
match = regex.search(css, match.end() - 3) |
|||
else: |
|||
match = regex.search(css, match.end()) |
|||
return css |
|||
|
|||
|
|||
def condense_whitespace(css): |
|||
"""Condense multiple adjacent whitespace characters into one.""" |
|||
|
|||
return re.sub(r"\s+", " ", css) |
|||
|
|||
|
|||
def condense_semicolons(css): |
|||
"""Condense multiple adjacent semicolon characters into one.""" |
|||
|
|||
return re.sub(r";;+", ";", css) |
|||
|
|||
|
|||
def wrap_css_lines(css, line_length): |
|||
"""Wrap the lines of the given CSS to an approximate length.""" |
|||
|
|||
lines = [] |
|||
line_start = 0 |
|||
for i, char in enumerate(css): |
|||
# It's safe to break after `}` characters. |
|||
if char == '}' and (i - line_start >= line_length): |
|||
lines.append(css[line_start:i + 1]) |
|||
line_start = i + 1 |
|||
|
|||
if line_start < len(css): |
|||
lines.append(css[line_start:]) |
|||
return '\n'.join(lines) |
|||
|
|||
|
|||
def cssmin(css, wrap = None): |
|||
css = remove_comments(css) |
|||
css = condense_whitespace(css) |
|||
# A pseudo class for the Box Model Hack |
|||
# (see http://tantek.com/CSS/Examples/boxmodelhack.html) |
|||
css = css.replace('"\\"}\\""', "___PSEUDOCLASSBMH___") |
|||
#css = remove_unnecessary_whitespace(css) |
|||
css = remove_unnecessary_semicolons(css) |
|||
css = condense_zero_units(css) |
|||
css = condense_multidimensional_zeros(css) |
|||
css = condense_floating_points(css) |
|||
css = normalize_rgb_colors_to_hex(css) |
|||
css = condense_hex_colors(css) |
|||
if wrap is not None: |
|||
css = wrap_css_lines(css, wrap) |
|||
css = css.replace("___PSEUDOCLASSBMH___", '"\\"}\\""') |
|||
css = condense_semicolons(css) |
|||
return css.strip() |
@ -1,218 +0,0 @@ |
|||
#!/usr/bin/python |
|||
|
|||
# This code is original from jsmin by Douglas Crockford, it was translated to |
|||
# Python by Baruch Even. The original code had the following copyright and |
|||
# license. |
|||
# |
|||
# /* jsmin.c |
|||
# 2007-05-22 |
|||
# |
|||
# Copyright (c) 2002 Douglas Crockford (www.crockford.com) |
|||
# |
|||
# Permission is hereby granted, free of charge, to any person obtaining a copy of |
|||
# this software and associated documentation files (the "Software"), to deal in |
|||
# the Software without restriction, including without limitation the rights to |
|||
# use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies |
|||
# of the Software, and to permit persons to whom the Software is furnished to do |
|||
# so, subject to the following conditions: |
|||
# |
|||
# The above copyright notice and this permission notice shall be included in all |
|||
# copies or substantial portions of the Software. |
|||
# |
|||
# The Software shall be used for Good, not Evil. |
|||
# |
|||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
|||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
|||
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
|||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
|||
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
|||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
|||
# SOFTWARE. |
|||
# */ |
|||
|
|||
from StringIO import StringIO |
|||
|
|||
def jsmin(js): |
|||
ins = StringIO(js) |
|||
outs = StringIO() |
|||
JavascriptMinify().minify(ins, outs) |
|||
str = outs.getvalue() |
|||
if len(str) > 0 and str[0] == '\n': |
|||
str = str[1:] |
|||
return str |
|||
|
|||
def isAlphanum(c): |
|||
"""return true if the character is a letter, digit, underscore, |
|||
dollar sign, or non-ASCII character. |
|||
""" |
|||
return ((c >= 'a' and c <= 'z') or (c >= '0' and c <= '9') or |
|||
(c >= 'A' and c <= 'Z') or c == '_' or c == '$' or c == '\\' or (c is not None and ord(c) > 126)); |
|||
|
|||
class UnterminatedComment(Exception): |
|||
pass |
|||
|
|||
class UnterminatedStringLiteral(Exception): |
|||
pass |
|||
|
|||
class UnterminatedRegularExpression(Exception): |
|||
pass |
|||
|
|||
class JavascriptMinify(object): |
|||
|
|||
def _outA(self): |
|||
self.outstream.write(self.theA) |
|||
def _outB(self): |
|||
self.outstream.write(self.theB) |
|||
|
|||
def _get(self): |
|||
"""return the next character from stdin. Watch out for lookahead. If |
|||
the character is a control character, translate it to a space or |
|||
linefeed. |
|||
""" |
|||
c = self.theLookahead |
|||
self.theLookahead = None |
|||
if c == None: |
|||
c = self.instream.read(1) |
|||
if c >= ' ' or c == '\n': |
|||
return c |
|||
if c == '': # EOF |
|||
return '\000' |
|||
if c == '\r': |
|||
return '\n' |
|||
return ' ' |
|||
|
|||
def _peek(self): |
|||
self.theLookahead = self._get() |
|||
return self.theLookahead |
|||
|
|||
def _next(self): |
|||
"""get the next character, excluding comments. peek() is used to see |
|||
if a '/' is followed by a '/' or '*'. |
|||
""" |
|||
c = self._get() |
|||
if c == '/': |
|||
p = self._peek() |
|||
if p == '/': |
|||
c = self._get() |
|||
while c > '\n': |
|||
c = self._get() |
|||
return c |
|||
if p == '*': |
|||
c = self._get() |
|||
while 1: |
|||
c = self._get() |
|||
if c == '*': |
|||
if self._peek() == '/': |
|||
self._get() |
|||
return ' ' |
|||
if c == '\000': |
|||
raise UnterminatedComment() |
|||
|
|||
return c |
|||
|
|||
def _action(self, action): |
|||
"""do something! What you do is determined by the argument: |
|||
1 Output A. Copy B to A. Get the next B. |
|||
2 Copy B to A. Get the next B. (Delete A). |
|||
3 Get the next B. (Delete B). |
|||
action treats a string as a single character. Wow! |
|||
action recognizes a regular expression if it is preceded by ( or , or =. |
|||
""" |
|||
if action <= 1: |
|||
self._outA() |
|||
|
|||
if action <= 2: |
|||
self.theA = self.theB |
|||
if self.theA == "'" or self.theA == '"': |
|||
while 1: |
|||
self._outA() |
|||
self.theA = self._get() |
|||
if self.theA == self.theB: |
|||
break |
|||
if self.theA <= '\n': |
|||
raise UnterminatedStringLiteral() |
|||
if self.theA == '\\': |
|||
self._outA() |
|||
self.theA = self._get() |
|||
|
|||
|
|||
if action <= 3: |
|||
self.theB = self._next() |
|||
if self.theB == '/' and (self.theA == '(' or self.theA == ',' or |
|||
self.theA == '=' or self.theA == ':' or |
|||
self.theA == '[' or self.theA == '?' or |
|||
self.theA == '!' or self.theA == '&' or |
|||
self.theA == '|' or self.theA == ';' or |
|||
self.theA == '{' or self.theA == '}' or |
|||
self.theA == '\n'): |
|||
self._outA() |
|||
self._outB() |
|||
while 1: |
|||
self.theA = self._get() |
|||
if self.theA == '/': |
|||
break |
|||
elif self.theA == '\\': |
|||
self._outA() |
|||
self.theA = self._get() |
|||
elif self.theA <= '\n': |
|||
raise UnterminatedRegularExpression() |
|||
self._outA() |
|||
self.theB = self._next() |
|||
|
|||
|
|||
def _jsmin(self): |
|||
"""Copy the input to the output, deleting the characters which are |
|||
insignificant to JavaScript. Comments will be removed. Tabs will be |
|||
replaced with spaces. Carriage returns will be replaced with linefeeds. |
|||
Most spaces and linefeeds will be removed. |
|||
""" |
|||
self.theA = '\n' |
|||
self._action(3) |
|||
|
|||
while self.theA != '\000': |
|||
if self.theA == ' ': |
|||
if isAlphanum(self.theB): |
|||
self._action(1) |
|||
else: |
|||
self._action(2) |
|||
elif self.theA == '\n': |
|||
if self.theB in ['{', '[', '(', '+', '-']: |
|||
self._action(1) |
|||
elif self.theB == ' ': |
|||
self._action(3) |
|||
else: |
|||
if isAlphanum(self.theB): |
|||
self._action(1) |
|||
else: |
|||
self._action(2) |
|||
else: |
|||
if self.theB == ' ': |
|||
if isAlphanum(self.theA): |
|||
self._action(1) |
|||
else: |
|||
self._action(3) |
|||
elif self.theB == '\n': |
|||
if self.theA in ['}', ']', ')', '+', '-', '"', '\'']: |
|||
self._action(1) |
|||
else: |
|||
if isAlphanum(self.theA): |
|||
self._action(1) |
|||
else: |
|||
self._action(3) |
|||
else: |
|||
self._action(1) |
|||
|
|||
def minify(self, instream, outstream): |
|||
self.instream = instream |
|||
self.outstream = outstream |
|||
self.theA = '\n' |
|||
self.theB = None |
|||
self.theLookahead = None |
|||
|
|||
self._jsmin() |
|||
self.instream.close() |
|||
|
|||
if __name__ == '__main__': |
|||
import sys |
|||
jsm = JavascriptMinify() |
|||
jsm.minify(sys.stdin, sys.stdout) |
Loading…
Reference in new issue