add the ability to read from XML into RNA for rna_xml module

This commit is contained in:
2012-01-01 08:09:30 +00:00
parent 6a1643ec12
commit e32e6004e4

View File

@@ -20,27 +20,16 @@
# <pep8 compliant>
import bpy
# easier to read
PRETTY_INTEND = True
invalid_classes = (
bpy.types.Operator,
bpy.types.Panel,
bpy.types.KeyingSet,
bpy.types.Header,
)
def build_property_typemap():
def build_property_typemap(skip_classes):
property_typemap = {}
for attr in dir(bpy.types):
cls = getattr(bpy.types, attr)
if issubclass(cls, invalid_classes):
if issubclass(cls, skip_classes):
continue
properties = cls.bl_rna.properties.keys()
@@ -54,9 +43,31 @@ def print_ln(data):
print(data, end="")
def rna2xml(fw=print_ln, ident_val=" "):
def rna2xml(fw=print_ln,
root_node="",
root_rna=None, # must be set
root_rna_skip=set(),
ident_val=" ",
skip_classes=(bpy.types.Operator,
bpy.types.Panel,
bpy.types.KeyingSet,
bpy.types.Header,
),
pretty_format=True,
method='DATA'):
from xml.sax.saxutils import quoteattr
property_typemap = build_property_typemap()
property_typemap = build_property_typemap(skip_classes)
def number_to_str(val, val_type):
if val_type == int:
return "%d" % val
elif val_type == float:
return "%.6g" % val
elif val_type == bool:
return "TRUE" if val else "FALSE"
else:
raise NotImplemented("this type is not a number %s" % val_type)
def rna2xml_node(ident, value, parent):
ident_next = ident + ident_val
@@ -68,7 +79,7 @@ def rna2xml(fw=print_ln, ident_val=" "):
value_type = type(value)
if issubclass(value_type, invalid_classes):
if issubclass(value_type, skip_classes):
return
# XXX, fixme, pointcache has eternal nested pointer to its self.
@@ -81,12 +92,8 @@ def rna2xml(fw=print_ln, ident_val=" "):
subvalue = getattr(value, prop)
subvalue_type = type(subvalue)
if subvalue_type == int:
node_attrs.append("%s=\"%d\"" % (prop, subvalue))
elif subvalue_type == float:
node_attrs.append("%s=\"%.4g\"" % (prop, subvalue))
elif subvalue_type == bool:
node_attrs.append("%s=\"%s\"" % (prop, "TRUE" if subvalue else "FALSE"))
if subvalue_type in (int, bool, float):
node_attrs.append("%s=\"%s\"" % (prop, number_to_str(subvalue, subvalue_type)))
elif subvalue_type is str:
node_attrs.append("%s=%s" % (prop, quoteattr(subvalue)))
elif subvalue_type == set:
@@ -95,7 +102,7 @@ def rna2xml(fw=print_ln, ident_val=" "):
node_attrs.append("%s=\"NONE\"" % prop)
elif issubclass(subvalue_type, bpy.types.ID):
# special case, ID's are always referenced.
node_attrs.append("%s=%s" % (prop, quoteattr(subvalue_type.__name__ + ":" + subvalue.name)))
node_attrs.append("%s=%s" % (prop, quoteattr(subvalue_type.__name__ + "::" + subvalue.name)))
else:
try:
subvalue_ls = list(subvalue)
@@ -110,8 +117,9 @@ def rna2xml(fw=print_ln, ident_val=" "):
if type(subvalue_rna).__name__ == "bpy_prop_array":
# TODO, multi-dim!
def str_recursive(s):
if type(s) in (int, float, bool):
return str(s)
subsubvalue_type = type(s)
if subsubvalue_type in (int, float, bool):
return number_to_str(s, subsubvalue_type)
else:
return " ".join([str_recursive(si) for si in s])
@@ -120,21 +128,22 @@ def rna2xml(fw=print_ln, ident_val=" "):
nodes_lists.append((prop, subvalue_ls, subvalue_type))
# declare + attributes
if PRETTY_INTEND:
if pretty_format:
tmp_str = "<%s " % value_type_name
tmp_ident = "\n" + ident + (" " * len(tmp_str))
fw("%s%s%s>\n" % (ident, tmp_str, tmp_ident.join(node_attrs)))
del tmp_str
del tmp_ident
else:
fw("%s<%s %s>\n" % (ident, value_type_name, " ".join(node_attrs)))
# unique members
for prop, subvalue, subvalue_type in nodes_items:
rna2xml_node(ident_next, subvalue, value)
fw("%s<%s>\n" % (ident_next, prop)) # XXX, this is awkward, how best to solve?
rna2xml_node(ident_next + ident_val, subvalue, value)
fw("%s</%s>\n" % (ident_next, prop)) # XXX, need to check on this.
# list members
for prop, subvalue, subvalue_type in nodes_lists:
@@ -146,40 +155,125 @@ def rna2xml(fw=print_ln, ident_val=" "):
fw("%s</%s>\n" % (ident, value_type_name))
fw("<root>\n")
for attr in dir(bpy.data):
# -------------------------------------------------------------------------
# needs re-workign to be generic
# exceptions
if attr.startswith("_"):
continue
elif attr == "window_managers":
continue
if root_node:
fw("<%s>\n" % root_node)
value = getattr(bpy.data, attr)
try:
ls = value[:]
except:
ls = None
# bpy.data
if method == 'DATA':
for attr in dir(root_rna):
if type(ls) == list:
fw("%s<%s>\n" % (ident_val, attr))
for blend_id in ls:
rna2xml_node(ident_val + ident_val, blend_id, None)
fw("%s</%s>\n" % (ident_val, attr))
# exceptions
if attr.startswith("_"):
continue
elif attr in root_rna_skip:
continue
fw("</root>\n")
value = getattr(root_rna, attr)
try:
ls = value[:]
except:
ls = None
if type(ls) == list:
fw("%s<%s>\n" % (ident_val, attr))
for blend_id in ls:
rna2xml_node(ident_val + ident_val, blend_id, None)
fw("%s</%s>\n" % (ident_val, attr))
# any attribute
elif method == 'ATTR':
rna2xml_node("", root_rna, None)
if root_node:
fw("</%s>\n" % root_node)
def main():
filename = bpy.data.filepath.rstrip(".blend") + ".xml"
file = open(filename, 'w')
rna2xml(file.write)
file.close()
def xml2rna(root_xml,
root_rna=None, # must be set
):
# read back.
from xml.dom.minidom import parse
xml_nodes = parse(filename)
print("Written:", filename)
def rna2xml_node(xml_node, value):
# print("evaluating:", xml_node.nodeName)
if __name__ == "__main__":
main()
# ---------------------------------------------------------------------
# Simple attributes
for attr in xml_node.attributes.keys():
# print(" ", attr)
subvalue = getattr(value, attr, Ellipsis)
if subvalue is Ellipsis:
print("%s.%s not found" % (type(value).__name__, attr))
else:
value_xml = xml_node.attributes[attr].value
subvalue_type = type(subvalue)
tp_name = 'UNKNOWN'
if subvalue_type == float:
value_xml_coerce = float(value_xml)
tp_name = 'FLOAT'
elif subvalue_type == int:
value_xml_coerce = int(value_xml)
tp_name = 'INT'
elif subvalue_type == bool:
value_xml_coerce = {'TRUE': True, 'FALSE': False}[value_xml]
tp_name = 'BOOL'
elif subvalue_type == str:
value_xml_coerce = value_xml
tp_name = 'STR'
elif hasattr(subvalue, "__len__"):
value_xml_split = value_xml.split()
try:
value_xml_coerce = [int(v) for v in value_xml_split]
except ValueError:
value_xml_coerce = [float(v) for v in value_xml_split]
tp_name = 'ARRAY'
# print(" %s.%s (%s) --- %s" % (type(value).__name__, attr, tp_name, subvalue_type))
setattr(value, attr, value_xml_coerce)
# ---------------------------------------------------------------------
# Complex attributes
for child_xml in xml_node.childNodes:
if child_xml.nodeType == child_xml.ELEMENT_NODE:
# print()
# print(child_xml.nodeName)
subvalue = getattr(value, child_xml.nodeName, None)
if subvalue is not None:
elems = []
for child_xml_real in child_xml.childNodes:
if child_xml_real.nodeType == child_xml_real.ELEMENT_NODE:
elems.append(child_xml_real)
del child_xml_real
if hasattr(subvalue, "__len__"):
# Collection
if len(elems) != len(subvalue):
print("Size Mismatch! collection:", child_xml.nodeName)
else:
for i in range(len(elems)):
child_xml_real = elems[i]
subsubvalue = subvalue[i]
if child_xml_real is None or subsubvalue is None:
print("None found %s - %d collection:", (child_xml.nodeName, i))
else:
rna2xml_node(child_xml_real, subsubvalue)
else:
# print(elems)
if len(elems) == 1:
# sub node named by its type
child_xml_real, = elems
# print(child_xml_real, subvalue)
rna2xml_node(child_xml_real, subvalue)
else:
# empty is valid too
pass
rna2xml_node(root_xml, root_rna)