Merged
authorLaurent Bessard
Tue, 14 May 2013 18:43:52 +0200
changeset 1137 c37f8d379ab0
parent 1136 5fac491d3a0e (diff)
parent 1132 28f96aa9c070 (current diff)
child 1138 cf2a6a7c87e8
Merged
--- a/PLCGenerator.py	Tue May 14 18:41:33 2013 +0900
+++ b/PLCGenerator.py	Tue May 14 18:43:52 2013 +0200
@@ -902,7 +902,10 @@
                         block_infos = self.GetBlockType(block_type)
                     if block_infos is None:
                         raise PLCGenException, _("Undefined block type \"%s\" in \"%s\" POU")%(block_type, self.Name)
-                    block_infos["generate"](self, instance, block_infos, body, None)
+                    try:
+                        block_infos["generate"](self, instance, block_infos, body, None)
+                    except ValueError, e:
+                        raise PLCGenException, e.message
                 elif isinstance(instance, plcopen.commonObjects_connector):
                     connector = instance.getname()
                     if self.ComputedConnectors.get(connector, None):
@@ -961,7 +964,10 @@
                     block_infos = self.GetBlockType(block_type)
                 if block_infos is None:
                     raise PLCGenException, _("Undefined block type \"%s\" in \"%s\" POU")%(block_type, self.Name)
-                paths.append(str(block_infos["generate"](self, next, block_infos, body, connection, order, to_inout)))
+                try:
+                    paths.append(str(block_infos["generate"](self, next, block_infos, body, connection, order, to_inout)))
+                except ValueError, e:
+                    raise PLCGenException, e.message
             elif isinstance(next, plcopen.commonObjects_continuation):
                 name = next.getname()
                 computed_value = self.ComputedConnectors.get(name, None)
--- a/editors/CodeFileEditor.py	Tue May 14 18:41:33 2013 +0900
+++ b/editors/CodeFileEditor.py	Tue May 14 18:43:52 2013 +0200
@@ -303,9 +303,11 @@
             # Code completion
             else:
                 self.AutoCompSetIgnoreCase(False)  # so this needs to match
-
-                # Images are specified with a appended "?type"
-                self.AutoCompShow(0, " ".join([word + "?1" for word in self.KEYWORDS]))
+                
+                keywords = self.KEYWORDS + [var["Name"]
+                                            for var in self.Controler.GetVariables()]
+                keywords.sort()
+                self.AutoCompShow(0, " ".join(keywords))
         else:
             event.Skip()
 
--- a/editors/Viewer.py	Tue May 14 18:41:33 2013 +0900
+++ b/editors/Viewer.py	Tue May 14 18:43:52 2013 +0200
@@ -1248,12 +1248,13 @@
                         output_name = output_connector["name"])
                 else:
                     connector = connectors["outputs"][i]
-                if output_connector.get("negated", False):
-                    connector.SetNegated(True)
-                if output_connector.get("edge", "none") != "none":
-                    connector.SetEdge(output_connector["edge"])
-                if connectors["outputs"].index(connector) == i:
-                    connector.SetPosition(wx.Point(*output_connector["position"]))
+                if connector is not None:
+                    if output_connector.get("negated", False):
+                        connector.SetNegated(True)
+                    if output_connector.get("edge", "none") != "none":
+                        connector.SetEdge(output_connector["edge"])
+                    if connectors["outputs"].index(connector) == i:
+                        connector.SetPosition(wx.Point(*output_connector["position"]))
         for i, input_connector in enumerate(instance["inputs"]):
             if i < len(connectors["inputs"]):
                 if isinstance(element, FBD_Block):
@@ -1262,42 +1263,56 @@
                         input_name = input_connector["name"])
                 else:
                     connector = connectors["inputs"][i]
-                if connectors["inputs"].index(connector) == i:
-                    connector.SetPosition(wx.Point(*input_connector["position"]))
-                if input_connector.get("negated", False):
-                    connector.SetNegated(True)
-                if input_connector.get("edge", "none") != "none":
-                    connector.SetEdge(input_connector["edge"])
-                self.CreateWires(connector, instance["id"], input_connector["links"], ids, selection)
+                if connector is not None:
+                    if connectors["inputs"].index(connector) == i:
+                        connector.SetPosition(wx.Point(*input_connector["position"]))
+                    if input_connector.get("negated", False):
+                        connector.SetNegated(True)
+                    if input_connector.get("edge", "none") != "none":
+                        connector.SetEdge(input_connector["edge"])
+                    if not self.CreateWires(connector, instance["id"], input_connector["links"], ids, selection):
+                        element.RefreshModel()
         element.RefreshConnectors()
         if selection is not None and selection[0].get(instance["id"], False):
             self.SelectInGroup(element)
 
     def CreateWires(self, start_connector, id, links, ids, selection=None):
+        links_connected = True
         for link in links:
             refLocalId = link["refLocalId"]
-            if refLocalId is not None:
-                if refLocalId not in ids:
-                    new_instance = self.Controler.GetEditedElementInstanceInfos(self.TagName, refLocalId, debug = self.Debug)
-                    if new_instance is not None:
-                        self.loadInstance(new_instance, ids, selection)
-                connected = self.FindElementById(refLocalId)
-                if connected is not None:
-                    points = link["points"]
-                    end_connector = connected.GetConnector(wx.Point(points[-1][0], points[-1][1]), link["formalParameter"])
-                    if end_connector is not None:
-                        wire = Wire(self)
-                        wire.SetPoints(points)
-                        start_connector.Connect((wire, 0), False)
-                        end_connector.Connect((wire, -1), False)
-                        wire.ConnectStartPoint(None, start_connector)
-                        wire.ConnectEndPoint(None, end_connector)
-                        connected.RefreshConnectors()
-                        self.AddWire(wire)
-                        if selection is not None and (\
-                           selection[1].get((id, refLocalId), False) or \
-                           selection[1].get((refLocalId, id), False)):
-                            self.SelectInGroup(wire)
+            if refLocalId is None:
+                links_connected = False
+                continue
+            
+            if refLocalId not in ids:
+                new_instance = self.Controler.GetEditedElementInstanceInfos(self.TagName, refLocalId, debug = self.Debug)
+                if new_instance is not None:
+                    self.loadInstance(new_instance, ids, selection)
+            
+            connected = self.FindElementById(refLocalId)
+            if connected is None:
+                links_connected = False
+                continue
+            
+            points = link["points"]
+            end_connector = connected.GetConnector(wx.Point(points[-1][0], points[-1][1]), link["formalParameter"])
+            if end_connector is not None:
+                wire = Wire(self)
+                wire.SetPoints(points)
+                start_connector.Connect((wire, 0), False)
+                end_connector.Connect((wire, -1), False)
+                wire.ConnectStartPoint(None, start_connector)
+                wire.ConnectEndPoint(None, end_connector)
+                connected.RefreshConnectors()
+                self.AddWire(wire)
+                if selection is not None and (\
+                   selection[1].get((id, refLocalId), False) or \
+                   selection[1].get((refLocalId, id), False)):
+                    self.SelectInGroup(wire)
+            else:
+                links_connected = False
+        
+        return links_connected
                         
     def IsOfType(self, type, reference):
         return self.Controler.IsOfType(type, reference, self.Debug)
--- a/graphics/FBD_Objects.py	Tue May 14 18:41:33 2013 +0900
+++ b/graphics/FBD_Objects.py	Tue May 14 18:43:52 2013 +0200
@@ -190,7 +190,9 @@
             for output in self.Outputs:
                 if output_name == output.GetName():
                     return output
-        return self.FindNearestConnector(position, self.Inputs + self.Outputs)
+        if input_name is None and output_name is None:
+            return self.FindNearestConnector(position, self.Inputs + self.Outputs)
+        return None
         
     def GetInputTypes(self):
         return tuple([input.GetType(True) for input in self.Inputs if input.GetName() != "EN"])
--- a/plcopen/structures.py	Tue May 14 18:41:33 2013 +0900
+++ b/plcopen/structures.py	Tue May 14 18:43:52 2013 +0200
@@ -46,38 +46,59 @@
     name = block.getinstanceName()
     type = block.gettypeName()
     executionOrderId = block.getexecutionOrderId()
+    input_variables = block.inputVariables.getvariable()
+    output_variables = block.outputVariables.getvariable()
     inout_variables = {}
-    for input_variable in block.inputVariables.getvariable():
-        for output_variable in block.outputVariables.getvariable():
+    for input_variable in input_variables:
+        for output_variable in output_variables:
             if input_variable.getformalParameter() == output_variable.getformalParameter():
                 inout_variables[input_variable.getformalParameter()] = ""
+    input_names = [input[0] for input in block_infos["inputs"]]
+    output_names = [output[0] for output in block_infos["outputs"]]
     if block_infos["type"] == "function":
-        output_variables = block.outputVariables.getvariable()
         if not generator.ComputedBlocks.get(block, False) and not order:
             generator.ComputedBlocks[block] = True
-            vars = []
+            connected_vars = []
+            input_connected = dict([("EN", None)] + 
+                                   [(input_name, None) for input_name in input_names])
+            for variable in input_variables:
+                parameter = variable.getformalParameter()
+                if input_connected.has_key(parameter):
+                    input_connected[parameter] = variable
+            if input_connected["EN"] is None:
+                input_connected.pop("EN")
+                input_parameters = input_names
+            else:
+                input_parameters = ["EN"] + input_names
             one_input_connected = False
-            for i, variable in enumerate(block.inputVariables.getvariable()):
-                input_info = (generator.TagName, "block", block.getlocalId(), "input", i)
-                connections = variable.connectionPointIn.getconnections()
-                if connections is not None:
-                    parameter = variable.getformalParameter()
-                    if parameter != "EN":
-                        one_input_connected = True
-                    if inout_variables.has_key(parameter):
-                        value = generator.ComputeExpression(body, connections, executionOrderId > 0, True)
-                        inout_variables[parameter] = value
+            all_input_connected = True
+            for i, parameter in enumerate(input_parameters):
+                variable = input_connected.get(parameter)
+                if variable is not None:
+                    input_info = (generator.TagName, "block", block.getlocalId(), "input", i)
+                    connections = variable.connectionPointIn.getconnections()
+                    if connections is not None:
+                        if parameter != "EN":
+                            one_input_connected = True
+                        if inout_variables.has_key(parameter):
+                            value = generator.ComputeExpression(body, connections, executionOrderId > 0, True)
+                            inout_variables[parameter] = value
+                        else:
+                            value = generator.ComputeExpression(body, connections, executionOrderId > 0)
+                        connected_vars.append(([(parameter, input_info), (" := ", ())],
+                                               generator.ExtractModifier(variable, value, input_info)))
                     else:
-                        value = generator.ComputeExpression(body, connections, executionOrderId > 0)
-                    if len(output_variables) > 1:
-                        vars.append([(parameter, input_info),
-                                     (" := ", ())] + generator.ExtractModifier(variable, value, input_info))
-                    else:
-                        vars.append(generator.ExtractModifier(variable, value, input_info))
+                        all_input_connected = False
+                else:
+                    all_input_connected = False
+            if len(output_variables) > 1 or not all_input_connected:
+                vars = [name + value for name, value in connected_vars]
+            else:
+                vars = [value for name, value in connected_vars]
             if one_input_connected:
                 for i, variable in enumerate(output_variables):
                     parameter = variable.getformalParameter()
-                    if not inout_variables.has_key(parameter):
+                    if not inout_variables.has_key(parameter) and parameter in output_names + ["ENO"]:
                         if variable.getformalParameter() == "":
                             variable_name = "%s%d"%(type, block.getlocalId())
                         else:
@@ -103,67 +124,98 @@
                 generator.Program += [(");\n", ())]
             else:
                 generator.Warnings.append(_("\"%s\" function cancelled in \"%s\" POU: No input connected")%(type, generator.TagName.split("::")[-1]))
-        if link:
-            connectionPoint = link.getposition()[-1]
-        else:
-            connectionPoint = None
-        for i, variable in enumerate(output_variables):
-            blockPointx, blockPointy = variable.connectionPointOut.getrelPositionXY()
-            if not connectionPoint or block.getx() + blockPointx == connectionPoint.getx() and block.gety() + blockPointy == connectionPoint.gety():
-                output_info = (generator.TagName, "block", block.getlocalId(), "output", i)
-                parameter = variable.getformalParameter()
-                if inout_variables.has_key(parameter):
-                    output_value = inout_variables[parameter]
-                else:
-                    if parameter == "":
-                        output_name = "%s%d"%(type, block.getlocalId())
-                    else:
-                        output_name = "%s%d_%s"%(type, block.getlocalId(), parameter)
-                    output_value = [(output_name, output_info)]
-                return generator.ExtractModifier(variable, output_value, output_info)
     elif block_infos["type"] == "functionBlock":
         if not generator.ComputedBlocks.get(block, False) and not order:
             generator.ComputedBlocks[block] = True
             vars = []
-            for i, variable in enumerate(block.inputVariables.getvariable()):
-                input_info = (generator.TagName, "block", block.getlocalId(), "input", i)
-                connections = variable.connectionPointIn.getconnections()
-                if connections is not None:
-                    parameter = variable.getformalParameter()
-                    value = generator.ComputeExpression(body, connections, executionOrderId > 0, inout_variables.has_key(parameter))
-                    vars.append([(parameter, input_info),
-                                 (" := ", ())] + generator.ExtractModifier(variable, value, input_info))
+            offset_idx = 0
+            for variable in input_variables:
+                parameter = variable.getformalParameter()
+                if parameter in input_names or parameter == "EN":
+                    if parameter == "EN":
+                        input_idx = 0
+                        offset_idx = 1
+                    else:
+                        input_idx = offset_idx + input_names.index(parameter)
+                    input_info = (generator.TagName, "block", block.getlocalId(), "input", input_idx)
+                    connections = variable.connectionPointIn.getconnections()
+                    if connections is not None:
+                        value = generator.ComputeExpression(body, connections, executionOrderId > 0, inout_variables.has_key(parameter))
+                        vars.append([(parameter, input_info),
+                                     (" := ", ())] + generator.ExtractModifier(variable, value, input_info))
             generator.Program += [(generator.CurrentIndent, ()), 
                                   (name, (generator.TagName, "block", block.getlocalId(), "name")),
                                   ("(", ())]
             generator.Program += JoinList([(", ", ())], vars)
             generator.Program += [(");\n", ())]
-        if link:
-            connectionPoint = link.getposition()[-1]
+    
+    if link:
+        connectionPoint = link.getposition()[-1]
+        output_parameter = link.getformalParameter()
+    else:
+        connectionPoint = None
+        output_parameter = None
+    
+    output_variable = None
+    output_idx = 0
+    if output_parameter is not None:
+        if output_parameter in output_names or output_parameter == "ENO":
+            for variable in output_variables:
+                if variable.getformalParameter() == output_parameter:
+                    output_variable = variable
+                    if output_parameter != "ENO":
+                        output_idx = output_names.index(output_parameter)
+    else:
+        for i, variable in enumerate(output_variables):
+            blockPointx, blockPointy = variable.connectionPointOut.getrelPositionXY()
+            if (not connectionPoint or 
+                block.getx() + blockPointx == connectionPoint.getx() and 
+                block.gety() + blockPointy == connectionPoint.gety()):
+                output_variable = variable
+                output_parameter = variable.getformalParameter()
+                output_idx = i
+    
+    if output_variable is not None:
+        if block_infos["type"] == "function":
+            output_info = (generator.TagName, "block", block.getlocalId(), "output", output_idx)
+            if inout_variables.has_key(output_parameter):
+                output_value = inout_variables[output_parameter]
+            else:
+                if output_parameter == "":
+                    output_name = "%s%d"%(type, block.getlocalId())
+                else:
+                    output_name = "%s%d_%s"%(type, block.getlocalId(), output_parameter)
+                output_value = [(output_name, output_info)]
+            return generator.ExtractModifier(output_variable, output_value, output_info)
+        
+        if block_infos["type"] == "functionBlock":
+            output_info = (generator.TagName, "block", block.getlocalId(), "output", output_idx)
+            output_name = generator.ExtractModifier(output_variable, [("%s.%s"%(name, output_parameter), output_info)], output_info)
+            if to_inout:
+                variable_name = "%s_%s"%(name, output_parameter)
+                if not generator.IsAlreadyDefined(variable_name):
+                    if generator.Interface[-1][0] != "VAR" or generator.Interface[-1][1] is not None or generator.Interface[-1][2]:
+                        generator.Interface.append(("VAR", None, False, []))
+                    if variable.connectionPointOut in generator.ConnectionTypes:
+                        generator.Interface[-1][3].append(
+                            (generator.ConnectionTypes[output_variable.connectionPointOut], variable_name, None, None))
+                    else:
+                        generator.Interface[-1][3].append(("ANY", variable_name, None, None))
+                    generator.Program += [(generator.CurrentIndent, ()),
+                                          ("%s := "%variable_name, ())]
+                    generator.Program += output_name
+                    generator.Program += [(";\n", ())]
+                return [(variable_name, ())]
+            return output_name 
+    if link is not None:
+        if output_parameter is None:
+            output_parameter = ""
+        if name:
+            blockname = "%s(%s)" % (name, type)
         else:
-            connectionPoint = None
-        for i, variable in enumerate(block.outputVariables.getvariable()):
-            blockPointx, blockPointy = variable.connectionPointOut.getrelPositionXY()
-            if not connectionPoint or block.getx() + blockPointx == connectionPoint.getx() and block.gety() + blockPointy == connectionPoint.gety():
-                output_info = (generator.TagName, "block", block.getlocalId(), "output", i)
-                output_name = generator.ExtractModifier(variable, [("%s.%s"%(name, variable.getformalParameter()), output_info)], output_info)
-                if to_inout:
-                    variable_name = "%s_%s"%(name, variable.getformalParameter())
-                    if not generator.IsAlreadyDefined(variable_name):
-                        if generator.Interface[-1][0] != "VAR" or generator.Interface[-1][1] is not None or generator.Interface[-1][2]:
-                            generator.Interface.append(("VAR", None, False, []))
-                        if variable.connectionPointOut in generator.ConnectionTypes:
-                            generator.Interface[-1][3].append((generator.ConnectionTypes[variable.connectionPointOut], variable_name, None, None))
-                        else:
-                            generator.Interface[-1][3].append(("ANY", variable_name, None, None))
-                        generator.Program += [(generator.CurrentIndent, ()),
-                                              ("%s := "%variable_name, ())]
-                        generator.Program += output_name
-                        generator.Program += [(";\n", ())]
-                    return [(variable_name, ())]
-                return output_name 
-    if link is not None:
-        raise ValueError, _("No output variable found")
+            blockname = type
+        raise ValueError, _("No output %s variable found in block %s in POU %s. Connection must be broken")  % \
+                          (output_parameter, blockname, generator.Name)
 
 def initialise_block(type, name, block = None):
     return [(type, name, None, None)]