02021-03-06 stream: convert XForms select1 to a FreeCad PropertyEnumeration
This commit is contained in:
parent
864ea087c8
commit
78146228fc
|
@ -4,8 +4,12 @@ import FreeCADGui
|
||||||
from lxml import etree
|
from lxml import etree
|
||||||
import ExternalAppsList
|
import ExternalAppsList
|
||||||
from ToolXML import *
|
from ToolXML import *
|
||||||
|
from collections import namedtuple
|
||||||
import re
|
import re
|
||||||
|
|
||||||
|
XFormsInput = namedtuple('XFormsInput', ['input', 'modelElement', 'type', 'maybeEnum'])
|
||||||
|
XFormsEnum = namedtuple('XFormsEnum', ['labels', 'values'])
|
||||||
|
|
||||||
def CreateCommand(appName, toolName):
|
def CreateCommand(appName, toolName):
|
||||||
App.ActiveDocument.openTransaction('Create parametric %s from %s'%(toolName, appName))
|
App.ActiveDocument.openTransaction('Create parametric %s from %s'%(toolName, appName))
|
||||||
FreeCADGui.addModule("XternalAppsParametricTool")
|
FreeCADGui.addModule("XternalAppsParametricTool")
|
||||||
|
@ -25,10 +29,12 @@ typeToFreeCADTypeDict = {
|
||||||
'xsd:string': 'App::PropertyString',
|
'xsd:string': 'App::PropertyString',
|
||||||
}
|
}
|
||||||
|
|
||||||
def typeToFreeCADType(type):
|
def typeToFreeCADType(type, maybeEnum):
|
||||||
if type.startswith('mime:'):
|
if maybeEnum is not None:
|
||||||
|
return 'App::PropertyEnumeration'
|
||||||
|
elif type.startswith('mime:'):
|
||||||
return MIMETypeToFreeCADType(MIMEType[5:])
|
return MIMETypeToFreeCADType(MIMEType[5:])
|
||||||
if type in typeToFreeCADTypeDict:
|
elif type in typeToFreeCADTypeDict:
|
||||||
return typeToFreeCADTypeDict[type]
|
return typeToFreeCADTypeDict[type]
|
||||||
else:
|
else:
|
||||||
raise ArgumentException('Unsupported XForms type')
|
raise ArgumentException('Unsupported XForms type')
|
||||||
|
@ -47,10 +53,24 @@ class XternalAppsParametricTool():
|
||||||
obj.Proxy = self
|
obj.Proxy = self
|
||||||
self.createPropertiesFromXML(obj)
|
self.createPropertiesFromXML(obj)
|
||||||
|
|
||||||
|
def interpretFormElement(self, xmlXFormsElement, xml, instanceDocument, types):
|
||||||
|
# TODO: is it safe to pass input unprotected here?
|
||||||
|
modelElement = instanceDocument.find(xmlXFormsElement.attrib['ref'],
|
||||||
|
namespaces=xmlXFormsElement.nsmap)
|
||||||
|
if modelElement is None:
|
||||||
|
raise Exception('Could not find ' + xmlXFormsElement.attrib['ref'] \
|
||||||
|
+ ' in instance document with namespaces=' + repr(xmlXFormsElement.nsmap))
|
||||||
|
type = types.get(instanceDocument.getpath(modelElement))
|
||||||
|
if type is None:
|
||||||
|
raise Exception('Could not find type for ' + instanceDocument.getpath(modelElement))
|
||||||
|
path = xml.getpath(xmlXFormsElement)
|
||||||
|
return (path, xmlXFormsElement, modelElement, type)
|
||||||
|
|
||||||
def interpretXML(self):
|
def interpretXML(self):
|
||||||
"""Parse the self.Tool.XForms document, and return the parsed xml,
|
"""Parse the self.Tool.XForms document, and return
|
||||||
a dictionary types[path] = "type", and a dictionary
|
* the parsed xml,
|
||||||
inputs[path] = (xml_input_element, xml_model_element, type)."""
|
* a dictionary types[path] = "type"
|
||||||
|
* a dictionary inputs[path] = (xml_input_element, xml_model_element, type)."""
|
||||||
types = {}
|
types = {}
|
||||||
modelInstance = {}
|
modelInstance = {}
|
||||||
inputs = {}
|
inputs = {}
|
||||||
|
@ -77,24 +97,33 @@ class XternalAppsParametricTool():
|
||||||
# register all inputs to inputs[pathToElement]
|
# register all inputs to inputs[pathToElement]
|
||||||
for group in xml.findall('./xforms:group', ns):
|
for group in xml.findall('./xforms:group', ns):
|
||||||
for input in group.findall('./xforms:input', ns):
|
for input in group.findall('./xforms:input', ns):
|
||||||
# TODO: is it safe to pass input unprotected here?
|
path, xmlXFormsElement, modelElement, type = self.interpretFormElement(input, xml, instanceDocument, types)
|
||||||
modelElement = instanceDocument.find(input.attrib['ref'], namespaces=input.nsmap)
|
inputs[path] = XFormsInput(input=xmlXFormsElement, modelElement=modelElement, type=type, maybeEnum=None)
|
||||||
if modelElement is None:
|
for select1 in group.findall('./xforms:select1', ns):
|
||||||
raise Exception('Could not find ' + input.attrib['ref'] \
|
path, xmlXFormsElement, modelElement, type = self.interpretFormElement(select1, xml, instanceDocument, types)
|
||||||
+ ' in instance document with namespaces=' + repr(input.nsmap))
|
# Gather the allowed elements for the enum
|
||||||
type = types[instanceDocument.getpath(modelElement)]
|
enum = {}
|
||||||
inputs[xml.getpath(input)] = (input, modelElement, type)
|
for item in select1.findall('./xforms:item', ns):
|
||||||
|
enum[item.attrib['label']] = item.attrib['value']
|
||||||
|
inputs[path] = XFormsInput(input=xmlXFormsElement, modelElement=modelElement, type=type, maybeEnum=enum)
|
||||||
return (xml, types, modelInstance, inputs)
|
return (xml, types, modelInstance, inputs)
|
||||||
|
|
||||||
|
def toSimpleName(self, name):
|
||||||
|
return re.sub(r'( |[^-a-zA-Z0-9])+', ' ', name).title().replace(' ', '')
|
||||||
|
|
||||||
def createPropertiesFromXML(self, obj):
|
def createPropertiesFromXML(self, obj):
|
||||||
xml, types, modelInstance, inputs = self.interpretXML()
|
xml, types, modelInstance, inputs = self.interpretXML()
|
||||||
for (input, modelElement, type) in inputs.values():
|
for (input, modelElement, type, maybeEnum) in inputs.values():
|
||||||
simpleName = re.sub(r'( |[^-a-zA-Z0-9])+', ' ', input.attrib['label']).title().replace(' ', '')
|
simpleName = self.toSimpleName(input.attrib['label'])
|
||||||
group = "/".join(input.xpath('ancestor-or-self::xforms:group/xforms:label/text()', namespaces=ns)) or None
|
group = "/".join(input.xpath('ancestor-or-self::xforms:group/xforms:label/text()', namespaces=ns)) or None
|
||||||
obj.addProperty(typeToFreeCADType(type),
|
print((simpleName, typeToFreeCADType(type, maybeEnum), maybeEnum))
|
||||||
|
obj.addProperty(typeToFreeCADType(type, maybeEnum),
|
||||||
simpleName,
|
simpleName,
|
||||||
group,
|
group,
|
||||||
input.attrib['label'] + '\nA value of type ' + type)
|
input.attrib['label'] + '\nA value of type ' + type)
|
||||||
|
if maybeEnum is not None:
|
||||||
|
setattr(obj, simpleName, [self.toSimpleName(k) for k in maybeEnum.keys()])
|
||||||
|
# TODO: have a converter from the labels to the values
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def Tool(self):
|
def Tool(self):
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
<xforms:bind ref="my:svgfile" type="mime:image/svg+xml" required="true()"/>
|
<xforms:bind ref="my:svgfile" type="mime:image/svg+xml" required="true()"/>
|
||||||
<xforms:bind ref="my:option1" type="xsd:decimal" required="true()"/>
|
<xforms:bind ref="my:option1" type="xsd:decimal" required="true()"/>
|
||||||
<xforms:bind ref="my:option2" type="xsd:string" required="true()"/>
|
<xforms:bind ref="my:option2" type="xsd:string" required="true()"/>
|
||||||
|
<xforms:bind ref="my:option3" type="xsd:string" required="true()"/>
|
||||||
<!--<xforms:submission action="myTool.py" method="exec-double-dash" />-->
|
<!--<xforms:submission action="myTool.py" method="exec-double-dash" />-->
|
||||||
<XternalApps:command medhod="exec" style="double-dash">
|
<XternalApps:command medhod="exec" style="double-dash">
|
||||||
<XternalApps:exception ref="my:svgfile" style="pipe" />
|
<XternalApps:exception ref="my:svgfile" style="pipe" />
|
||||||
|
|
Loading…
Reference in New Issue
Block a user