89 lines
3 KiB
Python
89 lines
3 KiB
Python
|
#!/usr/bin/env python
|
||
|
import sys, re
|
||
|
|
||
|
def cssMinify(css, saveFileName):
|
||
|
# remove comments - this will break a lot of hacks :-P
|
||
|
css = re.sub( r'\s*/\*\s*\*/', "$$HACK1$$", css ) # preserve IE<6 comment hack
|
||
|
css = re.sub( r'/\*[\s\S]*?\*/', "", css )
|
||
|
css = css.replace( "$$HACK1$$", '/**/' ) # preserve IE<6 comment hack
|
||
|
|
||
|
# url() doesn't need quotes
|
||
|
css = re.sub( r'url\((["\'])([^)]*)\1\)', r'url(\2)', css )
|
||
|
|
||
|
# spaces may be safely collapsed as generated content will collapse them anyway
|
||
|
css = re.sub( r'\s+', ' ', css )
|
||
|
|
||
|
# shorten collapsable colors: #aabbcc to #abc
|
||
|
css = re.sub( r'#([0-9a-f])\1([0-9a-f])\2([0-9a-f])\3(\s|;)', r'#\1\2\3\4', css )
|
||
|
|
||
|
# fragment values can loose zeros
|
||
|
css = re.sub( r':\s*0(\.\d+([cm]m|e[mx]|in|p[ctx]))\s*;', r':\1;', css )
|
||
|
|
||
|
for rule in re.findall( r'([^{]+){([^}]*)}', css ):
|
||
|
|
||
|
# we don't need spaces around operators
|
||
|
selectors = [re.sub( r'(?<=[\[\(>+=])\s+|\s+(?=[=~^$*|>+\]\)])', r'', selector.strip() ) for selector in rule[0].split( ',' )]
|
||
|
|
||
|
# order is important, but we still want to discard repetitions
|
||
|
properties = {}
|
||
|
porder = []
|
||
|
for prop in re.findall( '(.*?):(.*?)(;|$)', rule[1] ):
|
||
|
key = prop[0].strip().lower()
|
||
|
if key not in porder: porder.append( key )
|
||
|
properties[ key ] = prop[1].strip()
|
||
|
|
||
|
# output rule if it contains any declarations
|
||
|
if properties:
|
||
|
f = open(saveFileName,'a')
|
||
|
f.write( "%s{%s}" % ( ','.join( selectors ), ''.join(['%s:%s;' % (key, properties[key]) for key in porder])[:-1] ) )
|
||
|
f.close()
|
||
|
|
||
|
def fileMin(ifile):
|
||
|
returnVal = ''
|
||
|
ifSplit = ifile.split('.')
|
||
|
for n, filePart in enumerate(ifSplit):
|
||
|
if n == len(ifSplit)-1:
|
||
|
returnVal = returnVal + "min."
|
||
|
returnVal = returnVal + filePart
|
||
|
if n != len(ifSplit)-1:
|
||
|
returnVal =returnVal + "."
|
||
|
return returnVal
|
||
|
|
||
|
css = ""
|
||
|
files = []
|
||
|
saveFileName = "style.min.css"
|
||
|
flag = False
|
||
|
sepFlag = False
|
||
|
for n, i in enumerate(sys.argv):
|
||
|
if n == 0:
|
||
|
pass
|
||
|
elif i in ("-h", "--help"):
|
||
|
print("Add CSS files space separated to be combined and minified. \n \
|
||
|
\t-h, --help \n \
|
||
|
\t\tDisplay this help message. \n \
|
||
|
\t-o, --output \n \
|
||
|
\t\tDeclare name of output CSS file. \n \
|
||
|
\t-s, --separate \n \
|
||
|
\t\tOutput each individual file as min. i.e.: \n \
|
||
|
\t\tstyle.css will become style.min.css")
|
||
|
break
|
||
|
elif i in ("-o", "--output"):
|
||
|
flag = True
|
||
|
elif i in ("-s", "--separate"):
|
||
|
sepFlag = True
|
||
|
elif flag:
|
||
|
saveFileName = i
|
||
|
flag = False
|
||
|
elif sepFlag:
|
||
|
files.append(i)
|
||
|
else:
|
||
|
css += open( i, 'r' ).read()
|
||
|
|
||
|
if sepFlag:
|
||
|
for cssFile in files:
|
||
|
css = open(cssFile, 'r').read()
|
||
|
cssFile = fileMin(cssFile)
|
||
|
cssMinify(css, cssFile)
|
||
|
else:
|
||
|
cssMinify(css, saveFileName)
|