forked from mypaint/libmypaint
-
Notifications
You must be signed in to change notification settings - Fork 1
/
generate.py
235 lines (191 loc) · 6.97 KB
/
generate.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
#!/usr/bin/env python2
# libmypaint - The MyPaint Brush Library
# Copyright (C) 2007-2012 Martin Renold <[email protected]>
# Copyright (C) 2012-2016 by the MyPaint Development Team.
#
# Permission to use, copy, modify, and/or distribute this software for any
# purpose with or without fee is hereby granted, provided that the above
# copyright notice and this permission notice appear in all copies.
#
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
"Code generator, part of the build process."
from __future__ import absolute_import, division, print_function
import os
import sys
from os.path import basename
import json
from collections import namedtuple
_SETTINGS = [] # brushsettings.settings
_SETTING_ORDER = [
"internal_name", # cname
"displayed_name", # name
"constant",
"minimum", # min
"default",
"maximum", # max
"tooltip",
]
_INPUTS = [] # brushsettings.inputs
_INPUT_ORDER = [
"id", # name
"hard_minimum", # hard_min
"soft_minimum", # soft_min
"normal",
"soft_maximum", # soft_max
"hard_maximum", # hard_max
"displayed_name", # dname
"tooltip",
]
_STATES = [] # brushsettings.states
class _BrushSetting (namedtuple("_BrushSetting", _SETTING_ORDER)):
def validate(self):
msg = "Failed to validate %s: %r" % (self.internal_name, self)
if self.minimum and self.maximum:
assert (self.minimum <= self.default), msg
assert (self.maximum >= self.default), msg
assert (self.minimum < self.maximum), msg
assert self.default is not None
class _BrushInput (namedtuple("_BrushInput", _INPUT_ORDER)):
def validate(self):
msg = "Failed to validate %s: %r" % (self.id, self)
if self.hard_maximum is not None:
assert (self.hard_maximum >= self.soft_maximum), msg
assert (self.hard_maximum >= self.normal), msg
if self.hard_minimum is not None:
assert (self.hard_minimum <= self.soft_minimum), msg
assert (self.hard_minimum <= self.normal), msg
if None not in (self.hard_maximum, self.hard_minimum):
assert (self.hard_minimum < self.hard_maximum), msg
assert self.normal is not None
assert (self.soft_minimum < self.soft_maximum), msg
assert (self.soft_minimum <= self.normal), msg
assert (self.soft_maximum >= self.normal), msg
def _init_globals_from_json(filename):
"""Populate global variables above from the canonical JSON definition."""
with open(filename, "rb") as fp:
defs = json.load(fp)
for input_def in defs["inputs"]:
input = _BrushInput(**input_def)
input.validate()
_INPUTS.append(input)
for setting_def in defs["settings"]:
setting = _BrushSetting(**setting_def)
setting.validate()
_SETTINGS.append(setting)
for state_name in defs["states"]:
_STATES.append(state_name)
def writefile(filename, s):
"""Write generated code if changed."""
bn = basename(sys.argv[0])
s = '// DO NOT EDIT - autogenerated by ' + bn + '\n\n' + s
if os.path.exists(filename) and open(filename).read() == s:
print('Checked {}: up to date, not rewritten'.format(filename))
else:
print('Writing {}'.format(filename))
open(filename, 'w').write(s)
def generate_enum(enum_name, enum_prefix, count_name, items):
indent = " " * 4
begin = "typedef enum {\n"
end = "} %s;\n" % enum_name
entries = []
for idx, name in items:
entries.append(indent + enum_prefix + name)
entries.append(indent + count_name)
return begin + ",\n".join(entries) + "\n" + end
def generate_static_struct_array(struct_type, instance_name, entries_list):
indent = " " * 4
begin = "static %s %s[] = {\n" % (struct_type, instance_name)
end = "};\n"
entries = []
for entry in entries_list:
entries.append(indent + "{%s}" % (", ".join(entry)))
entries.append("\n")
return begin + ", \n".join(entries) + end
def stringify(value):
value = value.replace("\n", "\\n")
value = value.replace('"', '\\"')
return "\"%s\"" % value
def floatify(value, positive_inf=True):
if value is None:
return "FLT_MAX" if positive_inf else "-FLT_MAX"
return str(value)
def gettextify(value):
return "N_(%s)" % stringify(value)
def boolify(value):
return str("TRUE") if value else str("FALSE")
def input_info_struct(i):
return (
stringify(i.id),
floatify(i.hard_minimum, positive_inf=False),
floatify(i.soft_minimum, positive_inf=False),
floatify(i.normal),
floatify(i.soft_maximum),
floatify(i.hard_maximum),
gettextify(i.displayed_name),
gettextify(i.tooltip),
)
def settings_info_struct(s):
return (
stringify(s.internal_name),
gettextify(s.displayed_name),
boolify(s.constant),
floatify(s.minimum, positive_inf=False),
floatify(s.default),
floatify(s.maximum),
gettextify(s.tooltip),
)
def generate_internal_settings_code():
content = ''
content += generate_static_struct_array(
"MyPaintBrushSettingInfo",
"settings_info_array",
[settings_info_struct(i) for i in _SETTINGS],
)
content += "\n"
content += generate_static_struct_array(
"MyPaintBrushInputInfo",
"inputs_info_array",
[input_info_struct(i) for i in _INPUTS],
)
return content
def generate_public_settings_code():
content = ''
content += generate_enum(
"MyPaintBrushInput",
"MYPAINT_BRUSH_INPUT_",
"MYPAINT_BRUSH_INPUTS_COUNT",
enumerate([i.id.upper() for i in _INPUTS]),
)
content += '\n'
content += generate_enum(
"MyPaintBrushSetting",
"MYPAINT_BRUSH_SETTING_",
"MYPAINT_BRUSH_SETTINGS_COUNT",
enumerate([i.internal_name.upper() for i in _SETTINGS]),
)
content += '\n'
content += generate_enum(
"MyPaintBrushState",
"MYPAINT_BRUSH_STATE_",
"MYPAINT_BRUSH_STATES_COUNT",
enumerate([i.upper() for i in _STATES]),
)
content += '\n'
return content
if __name__ == '__main__':
_init_globals_from_json("brushsettings.json")
script = sys.argv[0]
try:
public_header_file, internal_header_file = sys.argv[1:]
except:
msg = "usage: {} PUBLICdotH INTERNALdotH".format(script)
print(msg, file=sys.stderr)
sys.exit(2)
writefile(public_header_file, generate_public_settings_code())
writefile(internal_header_file, generate_internal_settings_code())