csscom.py 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. #!/usr/bin/env python
  2. from argparse import ArgumentParser
  3. import logging
  4. from parse import parse_groups
  5. from generate import generate_group
  6. def compress_css(css, combine_blocks=True, compress_whitespace=True,
  7. compress_color=True, compress_font=True,
  8. compress_dimension=True, sort_properties=True, tab='\t'):
  9. groups = parse_groups(css)
  10. options = dict(combine_blocks=combine_blocks,
  11. compress_whitespace=compress_whitespace,
  12. compress_color=compress_color,
  13. compress_font=compress_font,
  14. compress_dimension=compress_dimension,
  15. sort_properties=sort_properties,
  16. tab=tab)
  17. compressed_groups = [generate_group(selectors, blocks, **options)
  18. for selectors, blocks in groups]
  19. newlines = '' if compress_whitespace else '\n\n'
  20. return newlines.join(compressed_groups)
  21. def parse_options():
  22. parser = ArgumentParser(description='Just another CSS compressor. '
  23. 'If none of the compression options below (those starting with '
  24. '"-c") are specified, all are enabled by default. If any are '
  25. 'specified, the others are not enabled.')
  26. parser.add_argument('files', metavar='FILE', nargs='+',
  27. help='CSS files to compress')
  28. parser.add_argument('-cw', '--compress-whitespace', action='store_true',
  29. help='omit unnecessary whitespaces and semicolons')
  30. parser.add_argument('-cc', '--compress-color', action='store_true',
  31. help='replace color codes/names with shorter synonyms')
  32. parser.add_argument('-cf', '--compress-font', action='store_true',
  33. help='replace separate font statements with shortcut '
  34. 'font statement where possible')
  35. parser.add_argument('-cd', '--compress-dimension', action='store_true',
  36. help='replace separate margin/padding statements with '
  37. 'shortcut statements where possible')
  38. parser.add_argument('-cb', '--combine-blocks', action='store_true',
  39. help='combine or split blocks into blocks with '
  40. 'comma-separated selectors if it results in less '
  41. 'css code')
  42. parser.add_argument('-nc', '--no-compression', action='store_true',
  43. help='don\'t apply any compression, just generate CSS')
  44. parser.add_argument('-ns', '--no-sort', action='store_false',
  45. dest='sort_properties', help='sort property names')
  46. parser.add_argument('-s', '--spaces', type=int, metavar='NUMBER=4',
  47. nargs='?', const=4,
  48. help='number of spaces to use for indenting (indent '
  49. 'defaults to a single tab [\\t])')
  50. parser.add_argument('-o', '--output', metavar='FILE',
  51. help='filename for compressed output (default is '
  52. 'stdout)')
  53. parser.add_argument('-ow', '--overwrite', action='store_true',
  54. help='use the first CSS file as output file')
  55. parser.add_argument('-v', '--verbose', action='store_true',
  56. help='show which compressions are performed')
  57. parser.add_argument('-d', '--debug', action='store_true',
  58. help='show debug statements')
  59. parser.add_argument('--logfile', metavar='FILE',
  60. help='file to write verbose/debug statements to '
  61. '(defaults to stderr)')
  62. args = parser.parse_args()
  63. # Enable all compression options if none are explicitely enabled
  64. if not any([args.compress_whitespace, args.compress_color,
  65. args.compress_font, args.compress_dimension,
  66. args.combine_blocks]) and not args.no_compression:
  67. args.compress_whitespace = args.compress_color = args.compress_font \
  68. = args.compress_dimension = args.combine_blocks = True
  69. return args
  70. def _content(filename):
  71. handle = open(filename, 'r')
  72. content = handle.read()
  73. handle.close()
  74. return content
  75. if __name__ == '__main__':
  76. args = parse_options()
  77. options = dict(args._get_kwargs())
  78. files = options.pop('files')
  79. del options['no_compression']
  80. spaces = options.pop('spaces')
  81. options['tab'] = '\t' if spaces is None else spaces * ' '
  82. outfile = options.pop('output')
  83. if options.pop('overwrite'):
  84. outfile = files[0]
  85. log_level = logging.WARNING
  86. if options.pop('verbose'):
  87. log_level = logging.INFO
  88. if options.pop('debug'):
  89. log_level = logging.DEBUG
  90. logging.basicConfig(level=log_level, filename=options.pop('logfile'),
  91. #format='%(levelname)s: %(message)s')
  92. format='%(filename)s, line %(lineno)s: %(levelname)s: %(message)s')
  93. try:
  94. css = '\n'.join(_content(filename) for filename in files)
  95. compressed = compress_css(css, **options)
  96. if outfile:
  97. open(outfile, 'w').write(compressed)
  98. else:
  99. print compressed,
  100. except IOError as e:
  101. print e