csscom.py 4.1 KB

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