--- a/ProjectController.py Thu Nov 08 11:20:35 2018 +0100
+++ b/ProjectController.py Thu Nov 08 14:33:35 2018 +0100
@@ -55,7 +55,7 @@
from editors.ProjectNodeEditor import ProjectNodeEditor
from editors.IECCodeViewer import IECCodeViewer
from editors.DebugViewer import DebugViewer, REFRESH_PERIOD
-from dialogs import DiscoveryDialog
+from dialogs import UriEditor from PLCControler import PLCControler
from plcopen.structures import IEC_KEYWORDS
from plcopen.types_enums import ComputeConfigurationResourceName, ITEM_CONFNODE
@@ -1768,7 +1768,7 @@
# Launch Service Discovery dialog
- dialog = DiscoveryDialog(self.AppFrame)
+ dialog = UriEditor(self.AppFrame) answer = dialog.ShowModal()
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/controls/DiscoveryPanel.py Thu Nov 08 14:33:35 2018 +0100
@@ -0,0 +1,233 @@
+# This file is part of Beremiz, a Integrated Development Environment for +# programming IEC 61131-3 automates supporting plcopen standard and CanFestival. +# Copyright (C) 2007: Edouard TISSERANT and Laurent BESSARD +# Copyright (C) 2017: Andrey Skvortsov <andrej.skvortzov@gmail.com> +# See COPYING file for copyrights details. +# This program is free software; you can redistribute it and/or +# modify it under the terms of the GNU General Public License +# as published by the Free Software Foundation; either version 2 +# of the License, or (at your option) any later version. +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. +from __future__ import absolute_import +import wx.lib.mixins.listctrl as listmix +from zeroconf import ServiceBrowser, Zeroconf +service_type = '_PYRO._tcp.local.' +class AutoWidthListCtrl(wx.ListCtrl, listmix.ListCtrlAutoWidthMixin): + def __init__(self, parent, id, name, pos=wx.DefaultPosition, + size=wx.DefaultSize, style=0): + wx.ListCtrl.__init__(self, parent, id, pos, size, style, name=name) + listmix.ListCtrlAutoWidthMixin.__init__(self) +class DiscoveryPanel(wx.Panel, listmix.ColumnSorterMixin): + def _init_coll_MainSizer_Items(self, parent): + parent.AddWindow(self.staticText1, 0, border=20, flag=wx.TOP | wx.LEFT | wx.RIGHT | wx.GROW) + parent.AddWindow(self.ServicesList, 0, border=20, flag=wx.LEFT | wx.RIGHT | wx.GROW) + parent.AddSizer(self.ButtonGridSizer, 0, border=20, flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.GROW) + def _init_coll_MainSizer_Growables(self, parent): + parent.AddGrowableCol(0) + parent.AddGrowableRow(1) + def _init_coll_ButtonGridSizer_Items(self, parent): + parent.AddWindow(self.RefreshButton, 0, border=0, flag=0) + parent.AddWindow(self.ByIPCheck, 0, border=0, flag=0) + def _init_coll_ButtonGridSizer_Growables(self, parent): + parent.AddGrowableCol(0) + parent.AddGrowableRow(0) + def _init_sizers(self): + self.MainSizer = wx.FlexGridSizer(cols=1, hgap=0, rows=3, vgap=10) + self.ButtonGridSizer = wx.FlexGridSizer(cols=2, hgap=5, rows=1, vgap=0) + self._init_coll_MainSizer_Items(self.MainSizer) + self._init_coll_MainSizer_Growables(self.MainSizer) + self._init_coll_ButtonGridSizer_Items(self.ButtonGridSizer) + self._init_coll_ButtonGridSizer_Growables(self.ButtonGridSizer) + self.SetSizer(self.MainSizer) + def _init_list_ctrl(self): + self.ServicesList = AutoWidthListCtrl( + name='ServicesList', parent=self, pos=wx.Point(0, 0), size=wx.Size(0, 0), + style=wx.LC_REPORT | wx.LC_EDIT_LABELS | wx.LC_SORT_ASCENDING | wx.LC_SINGLE_SEL) + self.ServicesList.InsertColumn(0, _('NAME')) + self.ServicesList.InsertColumn(1, _('TYPE')) + self.ServicesList.InsertColumn(2, _('IP')) + self.ServicesList.InsertColumn(3, _('PORT')) + self.ServicesList.SetColumnWidth(0, 150) + self.ServicesList.SetColumnWidth(1, 150) + self.ServicesList.SetColumnWidth(2, 150) + self.ServicesList.SetColumnWidth(3, 150) + self.ServicesList.SetInitialSize(wx.Size(-1, 300)) + self.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnItemSelected, id=listID) + self.Bind(wx.EVT_LIST_ITEM_ACTIVATED, self.OnItemActivated, id=listID) + def _init_ctrls(self, prnt): + self.staticText1 = wx.StaticText( + label=_('Services available:'), name='staticText1', parent=self, + pos=wx.Point(0, 0), size=wx.DefaultSize, style=0) + self.RefreshButton = wx.Button( + label=_('Refresh'), name='RefreshButton', parent=self, + pos=wx.Point(0, 0), size=wx.DefaultSize, style=0) + self.Bind(wx.EVT_BUTTON, self.OnRefreshButton, id=refreshID) + self.ByIPCheck = wx.CheckBox(self, label=_("Use IP instead of Service Name")) + def __init__(self, parent): + wx.Panel.__init__(self, parent) + listmix.ColumnSorterMixin.__init__(self, 4) + self._init_ctrls(parent) + self.ZeroConfInstance = Zeroconf() + self.LatestSelection = None + if self.Browser is not None: + self.ZeroConfInstance.close() + if self.Browser is not None: + self.Browser = ServiceBrowser(self.ZeroConfInstance, service_type, self) + def OnRefreshButton(self, event): + self.ServicesList.DeleteAllItems() + # Used by the ColumnSorterMixin, see wx/lib/mixins/listctrl.py + return self.ServicesList + def getColumnText(self, index, col): + item = self.ServicesList.GetItem(index, col) + def OnItemSelected(self, event): + self.SetURI(event.m_itemIndex) + def OnItemActivated(self, event): + self.SetURI(event.m_itemIndex) + self.EndModal(wx.ID_OK) +# def SetURI(self, idx): +# connect_type = self.getColumnText(idx, 1) +# connect_address = self.getColumnText(idx, 2) +# connect_port = self.getColumnText(idx, 3) +# self.URI = "%s://%s:%s"%(connect_type, connect_address, connect_port) + self.LatestSelection = idx + svcname = self.getColumnText(idx, 0) + connect_type = self.getColumnText(idx, 1) + self.URI = "%s://%s" % (connect_type, svcname + '.' + service_type) + if self.LatestSelection is not None: + if self.ByIPCheck.IsChecked(): + self.URI = "%s://%s:%s" % tuple( + map(lambda col:self.getColumnText(self.LatestSelection, col), + def remove_service(self, zeroconf, _type, name): + wx.CallAfter(self._removeService, name) + def _removeService(self, name): + called when a service with the desired type goes offline. + # loop through the list items looking for the service that went offline + for idx in xrange(self.ServicesList.GetItemCount()): + # this is the unique identifier assigned to the item + item_id = self.ServicesList.GetItemData(idx) + # this is the full typename that was received by addService + item_name = self.itemDataMap[item_id][4] + self.ServicesList.DeleteItem(idx) + def add_service(self, zeroconf, _type, name): + wx.CallAfter(self._addService, _type, name) + def _addService(self, _type, name): + called when a service with the desired type is discovered. + info = self.ZeroConfInstance.get_service_info(_type, name) + svcname = name.split(".")[0] + typename = _type.split(".")[0][1:] + ip = str(socket.inet_ntoa(info.address)) + num_items = self.ServicesList.GetItemCount() + # display the new data in the list + new_item = self.ServicesList.InsertStringItem(num_items, svcname) + self.ServicesList.SetStringItem(new_item, 1, "%s" % typename) + self.ServicesList.SetStringItem(new_item, 2, "%s" % ip) + self.ServicesList.SetStringItem(new_item, 3, "%s" % port) + # record the new data for the ColumnSorterMixin + # we assign every list item a unique id (that won't change when items + # are added or removed) + self.ServicesList.SetItemData(new_item, self.nextItemId) + # the value of each column has to be stored in the itemDataMap + # so that ColumnSorterMixin knows how to sort the column. + # "name" is included at the end so that self.removeService + self.itemDataMap[self.nextItemId] = [svcname, typename, ip, port, name] --- a/controls/__init__.py Thu Nov 08 11:20:35 2018 +0100
+++ b/controls/__init__.py Thu Nov 08 14:33:35 2018 +0100
@@ -44,3 +44,4 @@
from controls.LogViewer import LogViewer
from controls.CustomStyledTextCtrl import CustomStyledTextCtrl
from controls.CustomToolTip import CustomToolTip
--- a/dialogs/DiscoveryDialog.py Thu Nov 08 11:20:35 2018 +0100
+++ /dev/null Thu Jan 01 00:00:00 1970 +0000
@@ -1,266 +0,0 @@
-# This file is part of Beremiz, a Integrated Development Environment for
-# programming IEC 61131-3 automates supporting plcopen standard and CanFestival.
-# Copyright (C) 2007: Edouard TISSERANT and Laurent BESSARD
-# Copyright (C) 2017: Andrey Skvortsov <andrej.skvortzov@gmail.com>
-# See COPYING file for copyrights details.
-# This program is free software; you can redistribute it and/or
-# modify it under the terms of the GNU General Public License
-# as published by the Free Software Foundation; either version 2
-# of the License, or (at your option) any later version.
-# This program is distributed in the hope that it will be useful,
-# but WITHOUT ANY WARRANTY; without even the implied warranty of
-# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-# GNU General Public License for more details.
-# You should have received a copy of the GNU General Public License
-# along with this program; if not, write to the Free Software
-# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
-from __future__ import absolute_import
-import wx.lib.mixins.listctrl as listmix
-from zeroconf import ServiceBrowser, Zeroconf
-service_type = '_PYRO._tcp.local.'
-class AutoWidthListCtrl(wx.ListCtrl, listmix.ListCtrlAutoWidthMixin):
- def __init__(self, parent, id, name, pos=wx.DefaultPosition,
- size=wx.DefaultSize, style=0):
- wx.ListCtrl.__init__(self, parent, id, pos, size, style, name=name)
- listmix.ListCtrlAutoWidthMixin.__init__(self)
- ID_DISCOVERYDIALOG, ID_DISCOVERYDIALOGSTATICTEXT1,
- ID_DISCOVERYDIALOGSERVICESLIST, ID_DISCOVERYDIALOGREFRESHBUTTON,
- ID_DISCOVERYDIALOGLOCALBUTTON, ID_DISCOVERYDIALOGIPBUTTON,
-] = [wx.NewId() for _init_ctrls in range(6)]
-class DiscoveryDialog(wx.Dialog, listmix.ColumnSorterMixin):
- def _init_coll_MainSizer_Items(self, parent):
- parent.AddWindow(self.staticText1, 0, border=20, flag=wx.TOP | wx.LEFT | wx.RIGHT | wx.GROW)
- parent.AddWindow(self.ServicesList, 0, border=20, flag=wx.LEFT | wx.RIGHT | wx.GROW)
- parent.AddSizer(self.ButtonGridSizer, 0, border=20, flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.GROW)
- def _init_coll_MainSizer_Growables(self, parent):
- parent.AddGrowableCol(0)
- parent.AddGrowableRow(1)
- def _init_coll_ButtonGridSizer_Items(self, parent):
- parent.AddWindow(self.RefreshButton, 0, border=0, flag=0)
- parent.AddWindow(self.LocalButton, 0, border=0, flag=0)
- parent.AddWindow(self.IpButton, 0, border=0, flag=0)
- parent.AddSizer(self.ButtonSizer, 0, border=0, flag=0)
- def _init_coll_ButtonGridSizer_Growables(self, parent):
- parent.AddGrowableCol(0)
- parent.AddGrowableCol(1)
- parent.AddGrowableRow(0)
- def _init_sizers(self):
- self.MainSizer = wx.FlexGridSizer(cols=1, hgap=0, rows=3, vgap=10)
- self.ButtonGridSizer = wx.FlexGridSizer(cols=4, hgap=5, rows=1, vgap=0)
- self._init_coll_MainSizer_Items(self.MainSizer)
- self._init_coll_MainSizer_Growables(self.MainSizer)
- self._init_coll_ButtonGridSizer_Items(self.ButtonGridSizer)
- self._init_coll_ButtonGridSizer_Growables(self.ButtonGridSizer)
- self.SetSizer(self.MainSizer)
- def _init_list_ctrl(self):
- self.ServicesList = AutoWidthListCtrl(
- id=ID_DISCOVERYDIALOGSERVICESLIST,
- name='ServicesList', parent=self, pos=wx.Point(0, 0), size=wx.Size(0, 0),
- style=wx.LC_REPORT | wx.LC_EDIT_LABELS | wx.LC_SORT_ASCENDING | wx.LC_SINGLE_SEL)
- self.ServicesList.InsertColumn(0, _('NAME'))
- self.ServicesList.InsertColumn(1, _('TYPE'))
- self.ServicesList.InsertColumn(2, _('IP'))
- self.ServicesList.InsertColumn(3, _('PORT'))
- self.ServicesList.SetColumnWidth(0, 150)
- self.ServicesList.SetColumnWidth(1, 150)
- self.ServicesList.SetColumnWidth(2, 150)
- self.ServicesList.SetColumnWidth(3, 150)
- self.ServicesList.SetInitialSize(wx.Size(-1, 300))
- self.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnItemSelected, id=ID_DISCOVERYDIALOGSERVICESLIST)
- self.Bind(wx.EVT_LIST_ITEM_ACTIVATED, self.OnItemActivated, id=ID_DISCOVERYDIALOGSERVICESLIST)
- def _init_ctrls(self, prnt):
- self.staticText1 = wx.StaticText(
- id=ID_DISCOVERYDIALOGSTATICTEXT1,
- label=_('Services available:'), name='staticText1', parent=self,
- pos=wx.Point(0, 0), size=wx.DefaultSize, style=0)
- self.RefreshButton = wx.Button(
- id=ID_DISCOVERYDIALOGREFRESHBUTTON,
- label=_('Refresh'), name='RefreshButton', parent=self,
- pos=wx.Point(0, 0), size=wx.DefaultSize, style=0)
- self.Bind(wx.EVT_BUTTON, self.OnRefreshButton, id=ID_DISCOVERYDIALOGREFRESHBUTTON)
- self.LocalButton = wx.Button(
- id=ID_DISCOVERYDIALOGLOCALBUTTON,
- label=_('Local'), name='LocalButton', parent=self,
- pos=wx.Point(0, 0), size=wx.DefaultSize, style=0)
- self.Bind(wx.EVT_BUTTON, self.OnLocalButton, id=ID_DISCOVERYDIALOGLOCALBUTTON)
- self.IpButton = wx.Button(
- id=ID_DISCOVERYDIALOGIPBUTTON,
- label=_('Add IP'), name='IpButton', parent=self,
- pos=wx.Point(0, 0), size=wx.DefaultSize, style=0)
- self.Bind(wx.EVT_BUTTON, self.OnIpButton, id=ID_DISCOVERYDIALOGIPBUTTON)
- self.ButtonSizer = self.CreateButtonSizer(wx.OK | wx.CANCEL | wx.CENTER)
- def __init__(self, parent):
- self, id=ID_DISCOVERYDIALOG,
- name='DiscoveryDialog', parent=parent,
- style=wx.DEFAULT_DIALOG_STYLE,
- title=_('Service Discovery'))
- listmix.ColumnSorterMixin.__init__(self, 4)
- self._init_ctrls(parent)
- self.ZeroConfInstance = Zeroconf()
- self.LatestSelection = None
- if self.Browser is not None:
- self.ZeroConfInstance.close()
- if self.Browser is not None:
- self.Browser = ServiceBrowser(self.ZeroConfInstance, service_type, self)
- def OnRefreshButton(self, event):
- self.ServicesList.DeleteAllItems()
- def OnLocalButton(self, event):
- self.EndModal(wx.ID_OK)
- def OnIpButton(self, event):
- return self.getColumnText(self.LatestSelection, col)
- if self.LatestSelection is not None:
- self.URI = "%s://%s:%s" % tuple(map(GetColText, (1, 2, 3)))
- self.EndModal(wx.ID_OK)
- # Used by the ColumnSorterMixin, see wx/lib/mixins/listctrl.py
- return self.ServicesList
- def getColumnText(self, index, col):
- item = self.ServicesList.GetItem(index, col)
- def OnItemSelected(self, event):
- self.SetURI(event.m_itemIndex)
- def OnItemActivated(self, event):
- self.SetURI(event.m_itemIndex)
- self.EndModal(wx.ID_OK)
-# def SetURI(self, idx):
-# connect_type = self.getColumnText(idx, 1)
-# connect_address = self.getColumnText(idx, 2)
-# connect_port = self.getColumnText(idx, 3)
-# self.URI = "%s://%s:%s"%(connect_type, connect_address, connect_port)
- self.LatestSelection = idx
- svcname = self.getColumnText(idx, 0)
- connect_type = self.getColumnText(idx, 1)
- self.URI = "%s://%s" % (connect_type, svcname + '.' + service_type)
- def remove_service(self, zeroconf, _type, name):
- wx.CallAfter(self._removeService, name)
- def _removeService(self, name):
- called when a service with the desired type goes offline.
- # loop through the list items looking for the service that went offline
- for idx in xrange(self.ServicesList.GetItemCount()):
- # this is the unique identifier assigned to the item
- item_id = self.ServicesList.GetItemData(idx)
- # this is the full typename that was received by addService
- item_name = self.itemDataMap[item_id][4]
- self.ServicesList.DeleteItem(idx)
- def add_service(self, zeroconf, _type, name):
- wx.CallAfter(self._addService, _type, name)
- def _addService(self, _type, name):
- called when a service with the desired type is discovered.
- info = self.ZeroConfInstance.get_service_info(_type, name)
- svcname = name.split(".")[0]
- typename = _type.split(".")[0][1:]
- ip = str(socket.inet_ntoa(info.address))
- num_items = self.ServicesList.GetItemCount()
- # display the new data in the list
- new_item = self.ServicesList.InsertStringItem(num_items, svcname)
- self.ServicesList.SetStringItem(new_item, 1, "%s" % typename)
- self.ServicesList.SetStringItem(new_item, 2, "%s" % ip)
- self.ServicesList.SetStringItem(new_item, 3, "%s" % port)
- # record the new data for the ColumnSorterMixin
- # we assign every list item a unique id (that won't change when items
- # are added or removed)
- self.ServicesList.SetItemData(new_item, self.nextItemId)
- # the value of each column has to be stored in the itemDataMap
- # so that ColumnSorterMixin knows how to sort the column.
- # "name" is included at the end so that self.removeService
- self.itemDataMap[self.nextItemId] = [svcname, typename, ip, port, name]
--- a/dialogs/UriEditor.py Thu Nov 08 11:20:35 2018 +0100
+++ b/dialogs/UriEditor.py Thu Nov 08 14:33:35 2018 +0100
@@ -2,10 +2,11 @@
from connectors import ConnectorSchemes, EditorClassFromScheme
+from controls.DiscoveryPanel import DiscoveryPanel class UriEditor(wx.Dialog):
def _init_ctrls(self, parent):
- self.UriTypeChoice = wx.Choice(parent=self, choices=self.URITYPES)
+ self.UriTypeChoice = wx.Choice(parent=self, choices=self.choices) self.UriTypeChoice.SetSelection(0)
self.Bind(wx.EVT_CHOICE, self.OnTypeChoice, self.UriTypeChoice)
self.editor_sizer = wx.BoxSizer(wx.HORIZONTAL)
@@ -26,11 +27,11 @@
- def __init__(self, parent, uri):
+ def __init__(self, parent, uri=''): name='UriEditor', parent=parent,
- self.URITYPES = [_("- Select URI Scheme -")] + ConnectorSchemes()
+ self.choices = [_("- Search local network -")] + ConnectorSchemes() @@ -45,17 +46,24 @@
scheme, loc = uri.strip().split("://",1)
+ scheme = scheme.upper()
- scheme = scheme.upper()
if scheme in ConnectorSchemes():
self.UriTypeChoice.SetStringSelection(scheme)
- self._replaceSchemeEditor(scheme)
+ self.UriTypeChoice.SetSelection(0) + self._replaceSchemeEditor(scheme) self.scheme_editor.SetLoc(loc)
- if self.scheme_editor is None:
+ if self.scheme is None: + return self.scheme_editor.GetURI() return self.scheme+"://"+self.scheme_editor.GetLoc()
@@ -67,11 +75,16 @@
self.scheme_editor.Destroy()
self.scheme_editor = None
- EditorClass = EditorClassFromScheme(scheme)
- if EditorClass is not None:
+ if scheme is not None : + EditorClass = EditorClassFromScheme(scheme) self.scheme_editor = EditorClass(scheme,self)
- self.editor_sizer.Add(self.scheme_editor)
- self.scheme_editor.Refresh()
+ # None is for searching local network + self.scheme_editor = DiscoveryPanel(self) + self.editor_sizer.Add(self.scheme_editor) + self.scheme_editor.Refresh() self.editor_sizer.Layout()
--- a/dialogs/__init__.py Thu Nov 08 11:20:35 2018 +0100
+++ b/dialogs/__init__.py Thu Nov 08 14:33:35 2018 +0100
@@ -48,5 +48,4 @@
from dialogs.PouActionDialog import PouActionDialog
from dialogs.FindInPouDialog import FindInPouDialog
from dialogs.BrowseValuesLibraryDialog import BrowseValuesLibraryDialog
-from dialogs.DiscoveryDialog import DiscoveryDialog
from dialogs.UriEditor import UriEditor