# HG changeset patch # User Edouard Tisserant # Date 1724111468 -7200 # Node ID 22ba103801ee8c04988c8a72da05bd30b1afbb5e # Parent 86e39be014d8174730617fe94cee5e15d350a6bd# Parent 2b0f8c4c8d02013078c7207a1d548875057aa0d6 Merge remote-tracking branch 'origin/python3' into python3 diff -r 2b0f8c4c8d02 -r 22ba103801ee CodeFileTreeNode.py --- a/CodeFileTreeNode.py Fri Aug 16 16:30:50 2024 +0200 +++ b/CodeFileTreeNode.py Tue Aug 20 01:51:08 2024 +0200 @@ -107,7 +107,7 @@ filepath = self.CodeFileName() if os.path.isfile(filepath): - xmlfile = open(filepath, 'r') + xmlfile = open(filepath, 'r', encoding='utf-8', errors='backslashreplace') codefile_xml = xmlfile.read() xmlfile.close() diff -r 2b0f8c4c8d02 -r 22ba103801ee IDEFrame.py --- a/IDEFrame.py Fri Aug 16 16:30:50 2024 +0200 +++ b/IDEFrame.py Tue Aug 20 01:51:08 2024 +0200 @@ -1116,7 +1116,7 @@ printout2 = GraphicPrintout(window, page_size, margins, True) preview = wx.PrintPreview(printout, printout2, data) - if preview.Ok(): + if preview.IsOk(): preview_frame = wx.PreviewFrame(preview, self, _("Print preview"), style=wx.DEFAULT_FRAME_STYLE | wx.FRAME_FLOAT_ON_PARENT) preview_frame.Initialize() @@ -2599,7 +2599,7 @@ self.PageSize = page_size if self.PageSize[0] == 0 or self.PageSize[1] == 0: self.PageSize = (1050, 1485) - self.Preview = preview + self.IsPreview = lambda *_x : preview self.Margins = margins self.FontSize = 5 self.TextMargin = 3 @@ -2620,9 +2620,9 @@ def OnBeginDocument(self, startPage, endPage): dc = self.GetDC() - if not self.Preview and isinstance(dc, wx.PostScriptDC): + if not self.IsPreview() and isinstance(dc, wx.PostScriptDC): dc.SetResolution(720) - super(GraphicPrintout, self).OnBeginDocument(startPage, endPage) + return super(GraphicPrintout, self).OnBeginDocument(startPage, endPage) def OnPrintPage(self, page): dc = self.GetDC() @@ -2630,21 +2630,21 @@ dc.Clear() dc.SetUserScale(1.0, 1.0) dc.SetDeviceOrigin(0, 0) - dc.printing = not self.Preview + dc.printing = not self.IsPreview() # Get the size of the DC in pixels ppiPrinterX, ppiPrinterY = self.GetPPIPrinter() pw, ph = self.GetPageSizePixels() - dw, dh = dc.GetSizeTuple() + dw, dh = dc.GetSize().Get() Xscale = (dw * ppiPrinterX) / (pw * 25.4) Yscale = (dh * ppiPrinterY) / (ph * 25.4) - fontsize = self.FontSize * Yscale - - margin_left = self.Margins[0].x * Xscale - margin_top = self.Margins[0].y * Yscale - area_width = dw - self.Margins[1].x * Xscale - margin_left - area_height = dh - self.Margins[1].y * Yscale - margin_top + fontsize = round(self.FontSize * Yscale) + + margin_left = round(self.Margins[0].x * Xscale) + margin_top = round(self.Margins[0].y * Yscale) + area_width = dw - round(self.Margins[1].x * Xscale) - margin_left + area_height = dh - round(self.Margins[1].y * Yscale) - margin_top dc.SetPen(MiterPen(wx.BLACK)) dc.SetBrush(wx.TRANSPARENT_BRUSH) @@ -2667,7 +2667,7 @@ # Set the scale and origin dc.SetDeviceOrigin(-posX + margin_left, -posY + margin_top) - dc.SetClippingRegion(posX, posY, self.PageSize[0] * scale, self.PageSize[1] * scale) + dc.SetClippingRegion(posX, posY, round(self.PageSize[0] * scale), round(self.PageSize[1] * scale)) dc.SetUserScale(scale, scale) self.Viewer.DoDrawing(dc, True) diff -r 2b0f8c4c8d02 -r 22ba103801ee LocalRuntimeMixin.py --- a/LocalRuntimeMixin.py Fri Aug 16 16:30:50 2024 +0200 +++ b/LocalRuntimeMixin.py Tue Aug 20 01:51:08 2024 +0200 @@ -51,7 +51,10 @@ # shutdown local runtime self.local_runtime.kill(gently=False) # clear temp dir - shutil.rmtree(self.local_runtime_tmpdir) + try: + shutil.rmtree(self.local_runtime_tmpdir) + except: + pass self.local_runtime = None diff -r 2b0f8c4c8d02 -r 22ba103801ee PLCControler.py --- a/PLCControler.py Fri Aug 16 16:30:50 2024 +0200 +++ b/PLCControler.py Tue Aug 20 01:51:08 2024 +0200 @@ -1373,7 +1373,8 @@ return False def IsEndType(self, typename): - if typename is not None: + # Check if the type is a base type + if type(typename) == str: return not typename.startswith("ANY") return True diff -r 2b0f8c4c8d02 -r 22ba103801ee ProjectController.py --- a/ProjectController.py Fri Aug 16 16:30:50 2024 +0200 +++ b/ProjectController.py Tue Aug 20 01:51:08 2024 +0200 @@ -508,6 +508,9 @@ self._setBuildPath(BuildPath) # get confnodes bloclist (is that usefull at project creation?) self.RefreshConfNodesBlockLists() + # default scaling + for iec_lang in ["FBD", "LD", "SFC"]: + PLCControler.SetProjectProperties(self, properties={"scaling": {iec_lang: (8, 8)}}) # this will create files base XML files self.SaveProject() return None diff -r 2b0f8c4c8d02 -r 22ba103801ee bacnet/BacnetSlaveEditor.py --- a/bacnet/BacnetSlaveEditor.py Fri Aug 16 16:30:50 2024 +0200 +++ b/bacnet/BacnetSlaveEditor.py Tue Aug 20 01:51:08 2024 +0200 @@ -828,7 +828,7 @@ # use only to enable drag'n'drop # self.VariablesGrid.SetDropTarget(VariableDropTarget(self)) self.VariablesGrid.Bind( - wx.grid.EVT_GRID_CELL_CHANGING, self.OnVariablesGridCellChange) + wx.grid.EVT_GRID_CELL_CHANGED, self.OnVariablesGridCellChange) # self.VariablesGrid.Bind(wx.grid.EVT_GRID_CELL_LEFT_CLICK, self.OnVariablesGridCellLeftClick) # self.VariablesGrid.Bind(wx.grid.EVT_GRID_EDITOR_SHOWN, self.OnVariablesGridEditorShown) self.MainSizer.Add(self.VariablesGrid, flag=wx.GROW) diff -r 2b0f8c4c8d02 -r 22ba103801ee controls/LocationCellEditor.py --- a/controls/LocationCellEditor.py Fri Aug 16 16:30:50 2024 +0200 +++ b/controls/LocationCellEditor.py Tue Aug 20 01:51:08 2024 +0200 @@ -188,9 +188,7 @@ var_type = self.CellControl.GetVarType() if var_type is not None: self.Table.SetValueByName(row, 'Type', var_type) - else: - wx.CallAfter(self.Table.Parent.ShowErrorMessage, - _("Selected location is identical to previous one")) + self.CellControl.Disable() return changed diff -r 2b0f8c4c8d02 -r 22ba103801ee controls/SearchResultPanel.py --- a/controls/SearchResultPanel.py Fri Aug 16 16:30:50 2024 +0200 +++ b/controls/SearchResultPanel.py Tue Aug 20 01:51:08 2024 +0200 @@ -172,7 +172,7 @@ "type": ITEM_PROJECT, "data": None, "text": None, - "matches": None, + "matches": 0, } search_results_tree_children = search_results_tree_infos.setdefault("children", []) for tagname in self.ElementsOrder: @@ -291,7 +291,7 @@ start_idx = start[1] end_idx = reduce(lambda x, y: x + y, [len(x) + 1 for x in text_lines[:end[0] - start[0]]], end[1] + 1) style = wx.TextAttr(wx.BLACK, wx.Colour(206, 204, 247)) - elif infos["type"] is not None and infos["matches"] > 1: + elif infos["type"] is not None and infos["matches"] > 0: text = _("(%d matches)") % infos["matches"] start_idx, end_idx = 0, len(text) style = wx.TextAttr(wx.Colour(0, 127, 174)) diff -r 2b0f8c4c8d02 -r 22ba103801ee dialogs/ActionBlockDialog.py --- a/dialogs/ActionBlockDialog.py Fri Aug 16 16:30:50 2024 +0200 +++ b/dialogs/ActionBlockDialog.py Tue Aug 20 01:51:08 2024 +0200 @@ -148,7 +148,7 @@ self.ActionsGrid = CustomGrid(self, size=wx.Size(-1, 250), style=wx.VSCROLL) self.ActionsGrid.DisableDragGridSize() self.ActionsGrid.EnableScrolling(False, True) - self.ActionsGrid.Bind(wx.grid.EVT_GRID_CELL_CHANGING, + self.ActionsGrid.Bind(wx.grid.EVT_GRID_CELL_CHANGED, self.OnActionsGridCellChange) main_sizer.Add(self.ActionsGrid, border=20, flag=wx.GROW | wx.LEFT | wx.RIGHT) diff -r 2b0f8c4c8d02 -r 22ba103801ee dialogs/FBDVariableDialog.py --- a/dialogs/FBDVariableDialog.py Fri Aug 16 16:30:50 2024 +0200 +++ b/dialogs/FBDVariableDialog.py Tue Aug 20 01:51:08 2024 +0200 @@ -155,10 +155,8 @@ # Get variable expression and select corresponding value in name list # box if it exists selected = self.Expression.GetValue() - if selected != "" and self.VariableName.FindString(selected) != wx.NOT_FOUND: - self.VariableName.SetStringSelection(selected) - else: - self.VariableName.SetSelection(wx.NOT_FOUND) + self.VariableName.SetSelection( + wx.NOT_FOUND if selected == "" else self.VariableName.FindString(selected, True)) # Disable name list box if no name present inside self.VariableName.Enable(self.VariableName.GetCount() > 0) @@ -185,10 +183,7 @@ # Set expression text control value self.Expression.ChangeValue(value) # Select corresponding text in name list box if it exists - if self.VariableName.FindString(value) != wx.NOT_FOUND: - self.VariableName.SetStringSelection(value) - else: - self.VariableName.SetSelection(wx.NOT_FOUND) + self.VariableName.SetSelection(self.VariableName.FindString(value, True)) # Parameter is variable execution order elif name == "executionOrder": @@ -265,7 +260,7 @@ """ # Select the corresponding value in name list box if it exists self.VariableName.SetSelection( - self.VariableName.FindString(self.Expression.GetValue())) + self.VariableName.FindString(self.Expression.GetValue(), True)) self.Refresh() event.Skip() diff -r 2b0f8c4c8d02 -r 22ba103801ee editors/TextViewer.py --- a/editors/TextViewer.py Fri Aug 16 16:30:50 2024 +0200 +++ b/editors/TextViewer.py Tue Aug 20 01:51:08 2024 +0200 @@ -877,13 +877,15 @@ lineText = self.Editor.GetTextRange(start_pos, end_pos).replace("\t", " ") # Code completion - if key == wx.WXK_SPACE and event.ControlDown(): + if key == wx.WXK_SPACE and event.RawControlDown(): words = lineText.split(" ") words = [word for i, word in enumerate(words) if word != '' or i == len(words) - 1] kw = [] + self.RefreshVariableTree() + if self.TextSyntax == "IL": if len(words) == 1: kw = self.Keywords @@ -898,13 +900,17 @@ kw = self.Keywords + list(self.Variables.keys()) + list(self.Functions.keys()) if len(kw) > 0: if len(words[-1]) > 0: - kw = [keyword for keyword in kw if keyword.startswith(words[-1])] + kw = [keyword for keyword in kw if keyword.startswith(words[-1].upper())] + if len(kw) > 0: kw.sort() self.Editor.AutoCompSetIgnoreCase(True) self.Editor.AutoCompShow(len(words[-1]), " ".join(kw)) key_handled = True elif key == wx.WXK_RETURN or key == wx.WXK_NUMPAD_ENTER: - if self.TextSyntax in ["ST", "ALL"]: + if self.Editor.AutoCompActive(): + self.Editor.AutoCompComplete() + key_handled = True + elif self.TextSyntax in ["ST", "ALL"]: indent = self.Editor.GetLineIndentation(line) if LineStartswith(lineText.strip(), self.BlockStartKeywords): indent = (indent // 2 + 1) * 2 diff -r 2b0f8c4c8d02 -r 22ba103801ee editors/Viewer.py --- a/editors/Viewer.py Fri Aug 16 16:30:50 2024 +0200 +++ b/editors/Viewer.py Tue Aug 20 01:51:08 2024 +0200 @@ -1542,15 +1542,18 @@ # Popup menu functions # ------------------------------------------------------------------------------- - def GetForceVariableMenuFunction(self, iec_path, element): - iec_type = self.GetDataType(iec_path) - - def ForceVariableFunction(event): - if iec_type is not None: - dialog = ForceVariableDialog(self.ParentWindow, iec_type, str(element.GetValue())) - if dialog.ShowModal() == wx.ID_OK: - self.ParentWindow.AddDebugVariable(iec_path) - self.ForceDataValue(iec_path, dialog.GetValue()) + def GetForceVariableMenuFunction(self, iec_path, iec_type, value, immediate = False): + + def ForceVariableFunction(event, value=value): + if not immediate: + # use value as default value in dialog + dialog = ForceVariableDialog(self.ParentWindow, iec_type, str(value)) + if dialog.ShowModal() != wx.ID_OK: + return + value = dialog.GetValue() + self.ParentWindow.AddDebugVariable(iec_path) + self.ForceDataValue(iec_path, value) + return ForceVariableFunction def GetReleaseVariableMenuFunction(self, iec_path): @@ -1570,13 +1573,39 @@ def PopupForceMenu(self): iec_path = self.GetElementIECPath(self.SelectedElement) + + if iec_path is None: + # GetElementIECPath() does not work for variables and coils + # In such case get the IEC path using the instance path + for ElementType in [FBD_Variable, LD_Coil]: + if isinstance(self.SelectedElement, ElementType): + instance_path = self.GetInstancePath(True) + iec_path = "%s.%s" % (instance_path, self.SelectedElement.GetName()) + menu = wx.Menu(title='') + break + if iec_path is not None: menu = wx.Menu(title='') - item = self.AppendItem(menu, - _("Force value"), - self.GetForceVariableMenuFunction( - iec_path.upper(), - self.SelectedElement)) + iec_type = self.GetDataType(iec_path) + if iec_type == "BOOL": + self.AppendItem(menu, + _("Force Toggle"), + self.GetForceVariableMenuFunction( + iec_path.upper(), iec_type, not(self.SelectedElement.GetValue()), True)) + self.AppendItem(menu, + _("Force True"), + self.GetForceVariableMenuFunction( + iec_path.upper(), iec_type, True, True)) + self.AppendItem(menu, + _("Force False"), + self.GetForceVariableMenuFunction( + iec_path.upper(), iec_type, False, True)) + else: + self.AppendItem(menu, + _("Force value"), + self.GetForceVariableMenuFunction( + iec_path.upper(), iec_type, + self.SelectedElement.GetValue())) ritem = self.AppendItem(menu, _("Release value"), @@ -1585,6 +1614,7 @@ ritem.Enable(True) else: ritem.Enable(False) + if self.Editor.HasCapture(): self.Editor.ReleaseMouse() self.Editor.PopupMenu(menu) diff -r 2b0f8c4c8d02 -r 22ba103801ee etherlab/ConfigEditor.py --- a/etherlab/ConfigEditor.py Fri Aug 16 16:30:50 2024 +0200 +++ b/etherlab/ConfigEditor.py Tue Aug 20 01:51:08 2024 +0200 @@ -672,7 +672,7 @@ self.ProcessVariablesGrid = CustomGrid(self.EthercatMasterEditor, style=wx.VSCROLL) self.ProcessVariablesGrid.SetMinSize(wx.Size(0, 150)) self.ProcessVariablesGrid.SetDropTarget(ProcessVariableDropTarget(self)) - self.ProcessVariablesGrid.Bind(wx.grid.EVT_GRID_CELL_CHANGING, + self.ProcessVariablesGrid.Bind(wx.grid.EVT_GRID_CELL_CHANGED, self.OnProcessVariablesGridCellChange) self.ProcessVariablesGrid.Bind(wx.grid.EVT_GRID_CELL_LEFT_CLICK, self.OnProcessVariablesGridCellLeftClick) @@ -697,7 +697,7 @@ self.StartupCommandsGrid = CustomGrid(self.EthercatMasterEditor, style=wx.VSCROLL) self.StartupCommandsGrid.SetDropTarget(StartupCommandDropTarget(self)) self.StartupCommandsGrid.SetMinSize(wx.Size(0, 150)) - self.StartupCommandsGrid.Bind(wx.grid.EVT_GRID_CELL_CHANGING, + self.StartupCommandsGrid.Bind(wx.grid.EVT_GRID_CELL_CHANGED, self.OnStartupCommandsGridCellChange) self.StartupCommandsGrid.Bind(wx.grid.EVT_GRID_EDITOR_SHOWN, self.OnStartupCommandsGridEditorShow) diff -r 2b0f8c4c8d02 -r 22ba103801ee graphics/GraphicCommons.py --- a/graphics/GraphicCommons.py Fri Aug 16 16:30:50 2024 +0200 +++ b/graphics/GraphicCommons.py Tue Aug 20 01:51:08 2024 +0200 @@ -349,10 +349,10 @@ posx, posy = self.GetPosition() min_width, min_height = self.GetMinSize() if width < min_width: - self.Pos.x = max(0, self.Pos.x - (width - min_width) * x_factor) + self.Pos.x = max(0, round(self.Pos.x - (width - min_width) * x_factor)) width = min_width if height < min_height: - self.Pos.y = max(0, self.Pos.y - (height - min_height) * y_factor) + self.Pos.y = max(0, round(self.Pos.y - (height - min_height) * y_factor)) height = min_height if scaling is not None: self.Pos.x = round_scaling(self.Pos.x, scaling[0]) @@ -1026,16 +1026,16 @@ """ # Create a new connector - def __init__(self, parent, name, type, position, direction, negated=False, edge="none", onlyone=False): + def __init__(self, parent, name, Type, position, direction, negated=False, edge="none", onlyone=False): DebugDataConsumer.__init__(self) ToolTipProducer.__init__(self, parent.Parent) self.ParentBlock = parent self.Name = name - self.Type = type + self.Type = Type self.Pos = position self.Direction = direction self.Wires = [] - if self.ParentBlock.IsOfType("BOOL", type): + if self.ParentBlock.IsOfType("BOOL", Type): self.Negated = negated self.Edge = edge else: @@ -1141,8 +1141,8 @@ return self.ParentBlock.IsOfType(type, reference) or self.ParentBlock.IsOfType(reference, type) # Changes the connector name - def SetType(self, type): - self.Type = type + def SetType(self, Type): + self.Type = Type for wire, _handle in self.Wires: wire.SetValid(wire.IsConnectedCompatible()) diff -r 2b0f8c4c8d02 -r 22ba103801ee graphics/LD_Objects.py --- a/graphics/LD_Objects.py Fri Aug 16 16:30:50 2024 +0200 +++ b/graphics/LD_Objects.py Tue Aug 20 01:51:08 2024 +0200 @@ -593,11 +593,11 @@ dc.SetBrush(wx.Brush(HIGHLIGHTCOLOR)) dc.SetLogicalFunction(wx.AND) # Draw two rectangles for representing the contact - left_left = (self.Pos.x - 1) * scalex - 2 - right_left = (self.Pos.x + self.Size[0] - 2) * scalex - 2 - top = (self.Pos.y - 1) * scaley - 2 - width = 4 * scalex + 5 - height = (self.Size[1] + 3) * scaley + 5 + left_left = round((self.Pos.x - 1) * scalex) - 2 + right_left = round((self.Pos.x + self.Size[0] - 2) * scalex) - 2 + top = round((self.Pos.y - 1) * scaley) - 2 + width = round(4 * scalex + 5) + height = round((self.Size[1] + 3) * scaley) + 5 dc.DrawRectangle(left_left, top, width, height) dc.DrawRectangle(right_left, top, width, height) @@ -974,31 +974,24 @@ elif self.Type == COIL_FALLING: typetext = "N" - if getattr(dc, "printing", False) and not isinstance(dc, wx.PostScriptDC): - # Draw an clipped ellipse for representing the coil - clipping_box = dc.GetClippingBox() - dc.SetClippingRegion(self.Pos.x - 1, self.Pos.y, self.Size[0] + 2, self.Size[1] + 1) - dc.DrawEllipse(self.Pos.x, self.Pos.y - int(self.Size[1] * (sqrt(2) - 1.) / 2.) + 1, self.Size[0], int(self.Size[1] * sqrt(2)) - 1) - dc.DestroyClippingRegion() - if clipping_box != (0, 0, 0, 0): - dc.SetClippingRegion(*clipping_box) - name_size = dc.GetTextExtent(self.Name) - if typetext != "": - type_size = dc.GetTextExtent(typetext) - else: - # Draw a two ellipse arcs for representing the coil - dc.DrawEllipticArc(self.Pos.x, self.Pos.y - int(self.Size[1] * (sqrt(2) - 1.) / 2.) + 1, self.Size[0], int(self.Size[1] * sqrt(2)) - 1, 135, 225) - dc.DrawEllipticArc(self.Pos.x, self.Pos.y - int(self.Size[1] * (sqrt(2) - 1.) / 2.) + 1, self.Size[0], int(self.Size[1] * sqrt(2)) - 1, -45, 45) - # Draw a point to avoid hole in left arc - if not getattr(dc, "printing", False): - if self.Value is not None and self.Value: - dc.SetPen(MiterPen(wx.GREEN)) - else: - dc.SetPen(MiterPen(wx.BLACK)) - dc.DrawPoint(self.Pos.x + 1, self.Pos.y + self.Size[1] // 2 + 1) - name_size = self.NameSize - if typetext != "": - type_size = self.TypeSize + printing = getattr(dc, "printing", False) + # Draw a two ellipse arcs for representing the coil + pos = (self.Pos.x, + self.Pos.y - round(self.Size[1] * (sqrt(2) - 1.) / 2.) + 1, + self.Size[0], round(self.Size[1] * sqrt(2)) - 1) + + if printing: + # workaround for printing bug with DrawEllipticArc + # add an offset to the y position proportional to the height of the ellipse + # sqrt(2) ratio obtained heuristically + pos = (pos[0], pos[1] + round(sqrt(2)*pos[3]), pos[2], pos[3]) + + dc.DrawEllipticArc(*pos, 135, 225) + dc.DrawEllipticArc(*pos, -45, 45) + + name_size = self.NameSize + if typetext != "": + type_size = self.TypeSize # Draw coil name name_pos = (self.Pos.x + (self.Size[0] - name_size[0]) // 2, diff -r 2b0f8c4c8d02 -r 22ba103801ee requirements.txt --- a/requirements.txt Fri Aug 16 16:30:50 2024 +0200 +++ b/requirements.txt Tue Aug 20 01:51:08 2024 +0200 @@ -24,7 +24,7 @@ lxml==4.9.2 matplotlib==3.7.1 msgpack==1.0.5 -Nevow @ git+https://git@github.com/beremiz/nevow-py3.git@6deba7284e159af46a412699f046e060d7026cf9 +Nevow @ git+https://git@github.com/beremiz/nevow-py3.git@81c2eaeaaa20022540a98a3106f72e0199fbcc1b numpy==1.24.3 packaging==23.1 Pillow==9.5.0 diff -r 2b0f8c4c8d02 -r 22ba103801ee util/ProcessLogger.py --- a/util/ProcessLogger.py Fri Aug 16 16:30:50 2024 +0200 +++ b/util/ProcessLogger.py Tue Aug 20 01:51:08 2024 +0200 @@ -136,7 +136,7 @@ if _debug and self.logger: self.logger.write("(DEBUG) launching:\n" + self.Command_str + "\n") - self.Proc = subprocess.Popen(self.Command, encoding="utf-8", **popenargs) + self.Proc = subprocess.Popen(self.Command, encoding="utf-8", errors="backslashreplace", **popenargs) self.outt = outputThread( self.Proc,