28
28
/* It is autogenerated by spriteutils. */
31
def __init__(self, margin=150):
31
def __init__(self, css_template_file, group_name,
32
url_prefix_substitutions=None, margin=150):
33
"""Initialize with the specified css template file.
35
:param css_template_file: (str) Name of the file containing
37
css rules with a background-image style that needs to be
38
combined into the sprite file, a comment allowing sprites to
39
be grouped into different image files, and a
40
background-repeat style if necessary. Currently, "repeat-y"
41
is not supported since the file is combined vertically, so
42
repeat-y would show the entire combined image file.
46
background-image: url(../edit.png)
47
/* sprite-ref: group1 */
50
background-image: url(../blue-bar.png)
51
/* sprite-ref: group1 */
52
background-repeat: repeat-x
55
:param group_name: (str) Only add sprites to the
56
combined image file whose sprite-ref comment in the
57
css template match this group-name.
59
:param url_prefix_substitutions: (dict) The css template
60
will contain references to image files by their url
61
path, but the filesystem path relative to the css
64
:param margin: (int) The number of pixels between each sprite.
65
Be aware that Firefox will ignore extremely large images,
66
for example 64x34000 pixels.
68
If the css_template_file has been modified, a new
69
css file using an existing combined image and positioning
70
file can be generated using:
71
sprite_util = SpriteUtil(...)
72
sprite_util.loadPositioning(...)
73
sprite_util.saveConvertedCSS(...)
75
If a new image file needs to be added to the combined image
76
and the positioning file, they can be regenerated with:
77
sprite_util = SpriteUtil(...)
78
sprite_util.combineImages(...)
79
sprite_util.savePNG(...)
80
sprite_util.savePositioning(...)
82
If the image file is regenerated any time the css file is
83
regenerated, then the step for saving and loading the positioning
84
information could be removed. For example:
85
sprite_util = SpriteUtil(...)
86
sprite_util.combineImages(...)
87
sprite_util.savePNG(...)
88
sprite_util.saveConvertedCSS(...)
33
90
self.combined_image = None
34
91
self.positions = None
35
self.css_object = None
92
self.group_name = group_name
94
self._loadCSSTemplate(
95
css_template_file, group_name, url_prefix_substitutions)
38
def loadCSSTemplate(self, css_file, group_name,
97
def _loadCSSTemplate(self, css_template_file, group_name,
39
98
url_prefix_substitutions=None):
40
100
smartsprites_exp = re.compile(
41
101
r'/\*+([^*]*sprite-ref: [^*]*)\*/')
42
self.css_object = cssutils.parseFile(css_file)
102
self.css_object = cssutils.parseFile(css_template_file)
44
104
for rule in self.css_object:
45
105
if rule.cssText is None:
51
111
if parameters['sprite-ref'] == group_name:
52
112
filename = self._getSpriteImagePath(
53
113
rule, url_prefix_substitutions)
54
sprites.append(dict(filename=filename, rule=rule))
55
self.sprites = sprites
56
self.group_name = group_name
114
self.sprites.append(dict(filename=filename, rule=rule))
58
116
def _getSpriteImagePath(self, rule, url_prefix_substitutions=None):
117
"""Convert the url path to a filesystem path."""
59
118
# Remove url() from string.
60
119
filename = rule.style.backgroundImage[4:-1]
61
120
# Convert urls to paths relative to the css
128
193
self.combined_image = combined_image
130
195
def savePNG(self, filename):
196
"""Save the PIL image object to disk."""
131
197
self.combined_image.save(filename, format='png', optimize=True)
133
199
def savePositioning(self, filename):
200
"""Save the positions of sprites in the combined image.
202
This allows the final css to be generated after making
203
changes to the css template without recreating the combined
134
206
fp = open(filename, 'w')
135
207
fp.write(self.EDIT_WARNING)
136
208
simplejson.dump(self.positions, fp=fp, indent=4)
138
210
def loadPositioning(self, filename):
211
"""Load file with the positions of sprites in the combined image."""
139
212
json = open(filename).read()
140
213
# Remove comments from the beginning of the file.
141
214
start = json.index('{')
142
215
json = json[start:]
143
216
self.positions = simplejson.loads(json)
145
def saveConvertedCSS(self, css_file, combined_image_file):
218
def saveConvertedCSS(self, css_file, combined_image_url_path):
219
"""Generate new css from the template and the positioning info.
221
Example css template:
222
background-image: url(../edit.png); /* sprite-ref: group1 */
224
background-image: url(combined_image_url_path)
225
background-position: 0px 2344px
146
227
for sprite in self.sprites:
147
228
rule = sprite['rule']
148
rule.style.backgroundImage = 'url(%s)' % combined_image_file
229
rule.style.backgroundImage = 'url(%s)' % combined_image_url_path
149
230
position = self.positions[sprite['filename']]
150
231
rule.style.backgroundPosition = '%dpx %dpx' % tuple(position)