csscom.py 4.9 KB

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