2 # -*- coding: utf-8 -*-
4 #This file is part of CanFestival, a library implementing CanOpen Stack.
6 #Copyright (C): Edouard TISSERANT, Francis DUPIN and Laurent BESSARD
8 #See COPYING file for copyrights details.
10 #This library is free software; you can redistribute it and/or
11 #modify it under the terms of the GNU Lesser General Public
12 #License as published by the Free Software Foundation; either
13 #version 2.1 of the License, or (at your option) any later version.
15 #This library is distributed in the hope that it will be useful,
16 #but WITHOUT ANY WARRANTY; without even the implied warranty of
17 #MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 #Lesser General Public License for more details.
20 #You should have received a copy of the GNU Lesser General Public
21 #License along with this library; if not, write to the Free Software
22 #Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
28 import os, re, platform, sys, time, traceback, getopt
30 __version__ = "$Revision: 1.21 $"
32 from nodelist import *
33 from nodemanager import *
34 from subindextable import *
35 from commondialogs import *
36 from doc_index.DS301_index import *
38 ScriptDirectory = os.path.split(os.path.realpath(__file__))[0]
43 EVT_HTML_URL_CLICK = wx.NewId()
45 class HtmlWindowUrlClick(wx.PyEvent):
46 def __init__(self, linkinfo):
47 wx.PyEvent.__init__(self)
48 self.SetEventType(EVT_HTML_URL_CLICK)
49 self.linkinfo = (linkinfo.GetHref(), linkinfo.GetTarget())
51 class UrlClickHtmlWindow(wx.html.HtmlWindow):
52 """ HTML window that generates and OnLinkClicked event.
54 Use this to avoid having to override HTMLWindow
56 def OnLinkClicked(self, linkinfo):
57 wx.PostEvent(self, HtmlWindowUrlClick(linkinfo))
59 def Bind(self, event, handler, source=None, id=wx.ID_ANY, id2=wx.ID_ANY):
60 if event == HtmlWindowUrlClick:
61 self.Connect(-1, -1, EVT_HTML_URL_CLICK, handler)
63 wx.html.HtmlWindow.Bind(event, handler, source=source, id=id, id2=id2)
65 #-------------------------------------------------------------------------------
67 #-------------------------------------------------------------------------------
69 [ID_HTMLFRAME, ID_HTMLFRAMEHTMLCONTENT] = [wx.NewId() for _init_ctrls in range(2)]
71 class HtmlFrame(wx.Frame):
72 def _init_ctrls(self, prnt):
73 # generated method, don't edit
74 wx.Frame.__init__(self, id=ID_HTMLFRAME, name='HtmlFrame',
75 parent=prnt, pos=wx.Point(320, 231), size=wx.Size(853, 616),
76 style=wx.DEFAULT_FRAME_STYLE, title='')
77 self.Bind(wx.EVT_CLOSE, self.OnCloseFrame)
79 self.HtmlContent = UrlClickHtmlWindow(id=ID_HTMLFRAMEHTMLCONTENT,
80 name='HtmlContent', parent=self, pos=wx.Point(0, 0),
81 size=wx.Size(-1, -1), style=wx.html.HW_SCROLLBAR_AUTO|wx.html.HW_NO_SELECTION)
82 self.HtmlContent.Bind(HtmlWindowUrlClick, self.OnLinkClick)
84 def __init__(self, parent, opened):
85 self._init_ctrls(parent)
86 self.HtmlFrameOpened = opened
88 def SetHtmlCode(self, htmlcode):
89 self.HtmlContent.SetPage(htmlcode)
91 def SetHtmlPage(self, htmlpage):
92 self.HtmlContent.LoadPage(htmlpage)
94 def OnCloseFrame(self, event):
95 self.HtmlFrameOpened.remove(self.GetTitle())
98 def OnLinkClick(self, event):
99 url = event.linkinfo[0]
103 wx.MessageBox('Please point your browser at: %s' % url)
112 [ID_NETWORKEDIT, ID_NETWORKEDITNETWORKNODES,
113 ID_NETWORKEDITHELPBAR,
114 ] = [wx.NewId() for _init_ctrls in range(3)]
116 [ID_NETWORKEDITADDMENUITEMS0, ID_NETWORKEDITADDMENUITEMS1,
117 ID_NETWORKEDITADDMENUITEMS2, ID_NETWORKEDITADDMENUITEMS3,
118 ID_NETWORKEDITADDMENUITEMS4, ID_NETWORKEDITADDMENUITEMS5,
119 ] = [wx.NewId() for _init_coll_AddMenu_Items in range(6)]
121 [ID_NETWORKEDITFILEMENUITEMS0, ID_NETWORKEDITFILEMENUITEMS1,
122 ID_NETWORKEDITFILEMENUITEMS2, ID_NETWORKEDITFILEMENUITEMS4,
123 ID_NETWORKEDITFILEMENUITEMS5, ID_NETWORKEDITFILEMENUITEMS6,
124 ] = [wx.NewId() for _init_coll_FileMenu_Items in range(6)]
126 [ID_NETWORKEDITNETWORKMENUITEMS0, ID_NETWORKEDITNETWORKMENUITEMS1,
127 ID_NETWORKEDITNETWORKMENUITEMS3,
128 ] = [wx.NewId() for _init_coll_AddMenu_Items in range(3)]
131 [ID_NETWORKEDITEDITMENUITEMS0, ID_NETWORKEDITEDITMENUITEMS1,
132 ID_NETWORKEDITEDITMENUITEMS2, ID_NETWORKEDITEDITMENUITEMS4,
133 ID_NETWORKEDITEDITMENUITEMS6, ID_NETWORKEDITEDITMENUITEMS7,
134 ID_NETWORKEDITEDITMENUITEMS8,
135 ] = [wx.NewId() for _init_coll_EditMenu_Items in range(7)]
137 [ID_NETWORKEDITHELPMENUITEMS0, ID_NETWORKEDITHELPMENUITEMS1,
138 ID_NETWORKEDITHELPMENUITEMS2,
139 ] = [wx.NewId() for _init_coll_HelpMenu_Items in range(3)]
141 class networkedit(wx.Frame):
142 def _init_coll_menuBar1_Menus(self, parent):
144 parent.Append(menu=self.FileMenu, title='File')
145 parent.Append(menu=self.NetworkMenu, title='Network')
146 parent.Append(menu=self.EditMenu, title='Edit')
147 parent.Append(menu=self.AddMenu, title='Add')
148 parent.Append(menu=self.HelpMenu, title='Help')
150 def _init_coll_EditMenu_Items(self, parent):
151 parent.Append(help='', id=ID_NETWORKEDITEDITMENUITEMS4,
152 kind=wx.ITEM_NORMAL, text='Refresh\tCTRL+R')
153 parent.AppendSeparator()
154 parent.Append(help='', id=ID_NETWORKEDITEDITMENUITEMS1,
155 kind=wx.ITEM_NORMAL, text='Undo\tCTRL+Z')
156 parent.Append(help='', id=ID_NETWORKEDITEDITMENUITEMS0,
157 kind=wx.ITEM_NORMAL, text='Redo\tCTRL+Y')
158 parent.AppendSeparator()
159 parent.Append(help='', id=ID_NETWORKEDITEDITMENUITEMS6,
160 kind=wx.ITEM_NORMAL, text='Node infos')
161 parent.Append(help='', id=ID_NETWORKEDITEDITMENUITEMS2,
162 kind=wx.ITEM_NORMAL, text='DS-301 Profile')
163 parent.Append(help='', id=ID_NETWORKEDITEDITMENUITEMS8,
164 kind=wx.ITEM_NORMAL, text='DS-302 Profile')
165 parent.Append(help='', id=ID_NETWORKEDITEDITMENUITEMS7,
166 kind=wx.ITEM_NORMAL, text='Other Profile')
167 self.Bind(wx.EVT_MENU, self.OnUndoMenu,
168 id=ID_NETWORKEDITEDITMENUITEMS1)
169 self.Bind(wx.EVT_MENU, self.OnRedoMenu,
170 id=ID_NETWORKEDITEDITMENUITEMS0)
171 self.Bind(wx.EVT_MENU, self.OnCommunicationMenu,
172 id=ID_NETWORKEDITEDITMENUITEMS2)
173 self.Bind(wx.EVT_MENU, self.OnRefreshMenu,
174 id=ID_NETWORKEDITEDITMENUITEMS4)
175 self.Bind(wx.EVT_MENU, self.OnNodeInfosMenu,
176 id=ID_NETWORKEDITEDITMENUITEMS6)
177 self.Bind(wx.EVT_MENU, self.OnEditProfileMenu,
178 id=ID_NETWORKEDITEDITMENUITEMS7)
179 self.Bind(wx.EVT_MENU, self.OnOtherCommunicationMenu,
180 id=ID_NETWORKEDITEDITMENUITEMS8)
182 def _init_coll_HelpMenu_Items(self, parent):
183 parent.Append(help='', id=ID_NETWORKEDITHELPMENUITEMS0,
184 kind=wx.ITEM_NORMAL, text='DS-301 Standard\tF1')
185 self.Bind(wx.EVT_MENU, self.OnHelpDS301Menu,
186 id=ID_NETWORKEDITHELPMENUITEMS0)
187 parent.Append(help='', id=ID_NETWORKEDITHELPMENUITEMS1,
188 kind=wx.ITEM_NORMAL, text='CAN Festival Docs\tF2')
189 self.Bind(wx.EVT_MENU, self.OnHelpCANFestivalMenu,
190 id=ID_NETWORKEDITHELPMENUITEMS1)
191 if Html_Window and self.ModeSolo:
192 parent.Append(help='', id=ID_NETWORKEDITHELPMENUITEMS2,
193 kind=wx.ITEM_NORMAL, text='About')
194 self.Bind(wx.EVT_MENU, self.OnAboutMenu,
195 id=ID_NETWORKEDITHELPMENUITEMS2)
197 def _init_coll_FileMenu_Items(self, parent):
198 parent.Append(help='', id=ID_NETWORKEDITFILEMENUITEMS5,
199 kind=wx.ITEM_NORMAL, text='New\tCTRL+N')
200 parent.Append(help='', id=ID_NETWORKEDITFILEMENUITEMS0,
201 kind=wx.ITEM_NORMAL, text='Open\tCTRL+O')
202 parent.Append(help='', id=ID_NETWORKEDITFILEMENUITEMS1,
203 kind=wx.ITEM_NORMAL, text='Save\tCTRL+S')
204 parent.Append(help='', id=ID_NETWORKEDITFILEMENUITEMS2,
205 kind=wx.ITEM_NORMAL, text='Close\tCTRL+W')
206 parent.AppendSeparator()
207 parent.Append(help='', id=ID_NETWORKEDITFILEMENUITEMS4,
208 kind=wx.ITEM_NORMAL, text='Exit')
209 self.Bind(wx.EVT_MENU, self.OnOpenProjectMenu,
210 id=ID_NETWORKEDITFILEMENUITEMS0)
211 self.Bind(wx.EVT_MENU, self.OnSaveProjectMenu,
212 id=ID_NETWORKEDITFILEMENUITEMS1)
213 self.Bind(wx.EVT_MENU, self.OnCloseProjectMenu,
214 id=ID_NETWORKEDITFILEMENUITEMS2)
215 self.Bind(wx.EVT_MENU, self.OnQuitMenu,
216 id=ID_NETWORKEDITFILEMENUITEMS4)
217 self.Bind(wx.EVT_MENU, self.OnNewProjectMenu,
218 id=ID_NETWORKEDITFILEMENUITEMS5)
220 def _init_coll_NetworkMenu_Items(self, parent):
221 parent.Append(help='', id=ID_NETWORKEDITNETWORKMENUITEMS0,
222 kind=wx.ITEM_NORMAL, text='Add Slave Node')
223 parent.Append(help='', id=ID_NETWORKEDITNETWORKMENUITEMS1,
224 kind=wx.ITEM_NORMAL, text='Remove Slave Node')
225 parent.AppendSeparator()
226 parent.Append(help='', id=ID_NETWORKEDITNETWORKMENUITEMS3,
227 kind=wx.ITEM_NORMAL, text='Build Master Dictionary')
228 self.Bind(wx.EVT_MENU, self.OnAddSlaveMenu,
229 id=ID_NETWORKEDITNETWORKMENUITEMS0)
230 self.Bind(wx.EVT_MENU, self.OnRemoveSlaveMenu,
231 id=ID_NETWORKEDITNETWORKMENUITEMS1)
232 ## self.Bind(wx.EVT_MENU, self.OnBuildMasterMenu,
233 ## id=ID_NETWORKEDITNETWORKMENUITEMS3)
235 def _init_coll_AddMenu_Items(self, parent):
236 parent.Append(help='', id=ID_NETWORKEDITADDMENUITEMS0,
237 kind=wx.ITEM_NORMAL, text='SDO Server')
238 parent.Append(help='', id=ID_NETWORKEDITADDMENUITEMS1,
239 kind=wx.ITEM_NORMAL, text='SDO Client')
240 parent.Append(help='', id=ID_NETWORKEDITADDMENUITEMS2,
241 kind=wx.ITEM_NORMAL, text='PDO Transmit')
242 parent.Append(help='', id=ID_NETWORKEDITADDMENUITEMS3,
243 kind=wx.ITEM_NORMAL, text='PDO Receive')
244 parent.Append(help='', id=ID_NETWORKEDITADDMENUITEMS4,
245 kind=wx.ITEM_NORMAL, text='Map Variable')
246 parent.Append(help='', id=ID_NETWORKEDITADDMENUITEMS5,
247 kind=wx.ITEM_NORMAL, text='User Type')
248 self.Bind(wx.EVT_MENU, self.OnAddSDOServerMenu,
249 id=ID_NETWORKEDITADDMENUITEMS0)
250 self.Bind(wx.EVT_MENU, self.OnAddSDOClientMenu,
251 id=ID_NETWORKEDITADDMENUITEMS1)
252 self.Bind(wx.EVT_MENU, self.OnAddPDOTransmitMenu,
253 id=ID_NETWORKEDITADDMENUITEMS2)
254 self.Bind(wx.EVT_MENU, self.OnAddPDOReceiveMenu,
255 id=ID_NETWORKEDITADDMENUITEMS3)
256 self.Bind(wx.EVT_MENU, self.OnAddMapVariableMenu,
257 id=ID_NETWORKEDITADDMENUITEMS4)
258 self.Bind(wx.EVT_MENU, self.OnAddUserTypeMenu,
259 id=ID_NETWORKEDITADDMENUITEMS5)
261 def _init_coll_HelpBar_Fields(self, parent):
262 parent.SetFieldsCount(3)
264 parent.SetStatusText(number=0, text='')
265 parent.SetStatusText(number=1, text='')
266 parent.SetStatusText(number=2, text='')
268 parent.SetStatusWidths([100, 110, -1])
270 def _init_utils(self):
271 self.menuBar1 = wx.MenuBar()
272 self.menuBar1.SetEvtHandlerEnabled(True)
275 self.FileMenu = wx.Menu(title='')
276 self.NetworkMenu = wx.Menu(title='')
277 self.EditMenu = wx.Menu(title='')
278 self.AddMenu = wx.Menu(title='')
279 self.HelpMenu = wx.Menu(title='')
281 self._init_coll_menuBar1_Menus(self.menuBar1)
283 self._init_coll_FileMenu_Items(self.FileMenu)
284 self._init_coll_NetworkMenu_Items(self.NetworkMenu)
285 self._init_coll_EditMenu_Items(self.EditMenu)
286 self._init_coll_AddMenu_Items(self.AddMenu)
287 self._init_coll_HelpMenu_Items(self.HelpMenu)
289 def _init_ctrls(self, prnt):
290 wx.Frame.__init__(self, id=ID_NETWORKEDIT, name='networkedit',
291 parent=prnt, pos=wx.Point(149, 178), size=wx.Size(1000, 700),
292 style=wx.DEFAULT_FRAME_STYLE, title='Networkedit')
294 self.SetClientSize(wx.Size(1000, 700))
295 self.SetMenuBar(self.menuBar1)
296 self.Bind(wx.EVT_CLOSE, self.OnCloseFrame)
297 if not self.ModeSolo:
298 self.Bind(wx.EVT_MENU, self.OnSaveProjectMenu, id=ID_NETWORKEDITFILEMENUITEMS1)
299 accel = wx.AcceleratorTable([wx.AcceleratorEntry(wx.ACCEL_CTRL, 83, ID_NETWORKEDITFILEMENUITEMS1)])
300 self.SetAcceleratorTable(accel)
302 self.NetworkNodes = wx.Notebook(id=ID_NETWORKEDITNETWORKNODES,
303 name='NetworkNodes', parent=self, pos=wx.Point(0, 0),
304 size=wx.Size(0, 0), style=wx.NB_LEFT)
305 self.NetworkNodes.Bind(wx.EVT_NOTEBOOK_PAGE_CHANGED,
306 self.OnNodeSelectedChanged, id=ID_NETWORKEDITNETWORKNODES)
308 self.HelpBar = wx.StatusBar(id=ID_NETWORKEDITHELPBAR, name='HelpBar',
309 parent=self, style=wx.ST_SIZEGRIP)
310 self._init_coll_HelpBar_Fields(self.HelpBar)
311 self.SetStatusBar(self.HelpBar)
313 def __init__(self, parent, nodelist = None, projectOpen = None):
314 self.ModeSolo = nodelist == None
315 self._init_ctrls(parent)
316 self.HtmlFrameOpened = []
319 icon = wx.Icon(os.path.join(ScriptDirectory,"networkedit.ico"),wx.BITMAP_TYPE_ICO)
323 self.Manager = NodeManager()
325 self.NodeList = NodeList(self.Manager)
326 result = self.NodeList.LoadProject(projectOpen)
328 self.RefreshNetworkNodes()
332 self.NodeList = nodelist
333 self.Manager = self.NodeList.GetManager()
334 self.NodeList.SetCurrentSelected(0)
335 self.RefreshNetworkNodes()
336 self.RefreshProfileMenu()
337 self.NetworkNodes.SetFocus()
339 self.RefreshBufferState()
341 self.RefreshMainMenu()
343 def SetBusId(self, bus_id):
349 def GetCurrentNodeId(self):
350 selected = self.NetworkNodes.GetSelection()
351 # At init selected = -1
353 window = self.NetworkNodes.GetPage(selected)
354 return window.GetIndex()
358 def OnCloseFrame(self, event):
359 if not self.ModeSolo and getattr(self, "_onclose", None) != None:
363 def OnChar(self, event):
364 if event.ControlDown() and event.GetKeyCode() == 83 and getattr(self, "_onsave", None) != None:
368 def OnQuitMenu(self, event):
372 def OnAddSDOServerMenu(self, event):
373 self.Manager.AddSDOServerToCurrent()
374 self.RefreshBufferState()
375 self.RefreshCurrentIndexList()
378 def OnAddSDOClientMenu(self, event):
379 self.Manager.AddSDOClientToCurrent()
380 self.RefreshBufferState()
381 self.RefreshCurrentIndexList()
384 def OnAddPDOTransmitMenu(self, event):
385 self.Manager.AddPDOTransmitToCurrent()
386 self.RefreshBufferState()
387 self.RefreshCurrentIndexList()
390 def OnAddPDOReceiveMenu(self, event):
391 self.Manager.AddPDOReceiveToCurrent()
392 self.RefreshBufferState()
393 self.RefreshCurrentIndexList()
396 def OnAddMapVariableMenu(self, event):
397 self.AddMapVariable()
400 def OnAddUserTypeMenu(self, event):
404 def OnNodeSelectedChanged(self, event):
405 selected = event.GetSelection()
406 # At init selected = -1
408 window = self.NetworkNodes.GetPage(selected)
409 self.NodeList.SetCurrentSelected(window.GetIndex())
410 wx.CallAfter(self.RefreshMainMenu)
411 wx.CallAfter(self.RefreshStatusBar)
414 #-------------------------------------------------------------------------------
415 # Load and Save Funtions
416 #-------------------------------------------------------------------------------
418 def OnNewProjectMenu(self, event):
420 defaultpath = os.path.dirname(self.NodeList.GetRoot())
422 defaultpath = os.getcwd()
423 dialog = wx.DirDialog(self , "Choose a project", defaultpath, wx.DD_NEW_DIR_BUTTON)
424 if dialog.ShowModal() == wx.ID_OK:
425 projectpath = dialog.GetPath()
426 if os.path.isdir(projectpath) and len(os.listdir(projectpath)) == 0:
427 manager = NodeManager()
428 nodelist = NodeList(manager)
429 result = nodelist.LoadProject(projectpath)
431 self.Manager = manager
432 self.NodeList = nodelist
433 self.NodeList.SetCurrentSelected(0)
435 self.RefreshNetworkNodes()
436 self.RefreshBufferState()
438 self.RefreshProfileMenu()
439 self.RefreshMainMenu()
441 message = wx.MessageDialog(self, result, "ERROR", wx.OK|wx.ICON_ERROR)
446 def OnOpenProjectMenu(self, event):
448 defaultpath = os.path.dirname(self.NodeList.GetRoot())
450 defaultpath = os.getcwd()
451 dialog = wx.DirDialog(self , "Choose a project", defaultpath, 0)
452 if dialog.ShowModal() == wx.ID_OK:
453 projectpath = dialog.GetPath()
454 if os.path.isdir(projectpath):
455 manager = NodeManager()
456 nodelist = NodeList(manager)
457 result = nodelist.LoadProject(projectpath)
459 self.Manager = manager
460 self.NodeList = nodelist
461 self.NodeList.SetCurrentSelected(0)
463 self.RefreshNetworkNodes()
464 self.RefreshBufferState()
466 self.RefreshProfileMenu()
467 self.RefreshMainMenu()
469 message = wx.MessageDialog(self, result, "Error", wx.OK|wx.ICON_ERROR)
475 def OnSaveProjectMenu(self, event):
476 if not self.ModeSolo and getattr(self, "_onsave", None) != None:
479 result = self.NodeList.SaveProject()
481 message = wx.MessageDialog(self, result, "Error", wx.OK|wx.ICON_ERROR)
486 def OnCloseProjectMenu(self, event):
488 if self.NodeList.HasChanged():
489 dialog = wx.MessageDialog(self, "There are changes, do you want to save?", "Close Project", wx.YES_NO|wx.CANCEL|wx.ICON_QUESTION)
490 answer = dialog.ShowModal()
492 if answer == wx.ID_YES:
493 result = self.NodeList.SaveProject()
495 message = wx.MessageDialog(self, result, "Error", wx.OK|wx.ICON_ERROR)
498 elif answer == wx.ID_NO:
499 self.NodeList.ForceChanged(False)
500 if not self.NodeList.HasChanged():
503 self.RefreshNetworkNodes()
505 self.RefreshMainMenu()
508 #-------------------------------------------------------------------------------
509 # Slave Nodes Management
510 #-------------------------------------------------------------------------------
512 def OnAddSlaveMenu(self, event):
513 dialog = AddSlaveDialog(self)
514 dialog.SetNodeList(self.NodeList)
515 if dialog.ShowModal() == wx.ID_OK:
516 values = dialog.GetValues()
517 result = self.NodeList.AddSlaveNode(values["slaveName"], values["slaveNodeID"], values["edsFile"])
519 new_editingpanel = EditingPanel(self.NetworkNodes, self, self.NodeList, False)
520 new_editingpanel.SetIndex(values["slaveNodeID"])
521 idx = self.NodeList.GetOrderNumber(values["slaveNodeID"])
522 self.NetworkNodes.InsertPage(idx, new_editingpanel, "")
523 self.NodeList.SetCurrentSelected(idx)
524 self.NetworkNodes.SetSelection(idx)
525 self.RefreshBufferState()
527 message = wx.MessageDialog(self, result, "Error", wx.OK|wx.ICON_ERROR)
533 def OnRemoveSlaveMenu(self, event):
534 slavenames = self.NodeList.GetSlaveNames()
535 slaveids = self.NodeList.GetSlaveIDs()
536 dialog = wx.SingleChoiceDialog(self, "Choose a slave to remove", "Remove slave", slavenames)
537 if dialog.ShowModal() == wx.ID_OK:
538 choice = dialog.GetSelection()
539 result = self.NodeList.RemoveSlaveNode(slaveids[choice])
542 current = self.NetworkNodes.GetSelection()
543 self.NetworkNodes.DeletePage(choice + 1)
544 if self.NetworkNodes.GetPageCount() > 0:
545 new_selection = min(current, self.NetworkNodes.GetPageCount() - 1)
546 self.NetworkNodes.SetSelection(new_selection)
547 if new_selection > 0:
548 self.NodeList.SetCurrentSelected(slaveids[new_selection - 1])
549 self.RefreshBufferState()
551 message = wx.MessageDialog(self, result, "Error", wx.OK|wx.ICON_ERROR)
556 #-------------------------------------------------------------------------------
558 #-------------------------------------------------------------------------------
560 def RefreshTitle(self):
561 if self.NodeList != None:
562 self.SetTitle("Networkedit - %s"%self.NodeList.GetNetworkName())
564 self.SetTitle("Networkedit")
566 def OnRefreshMenu(self, event):
567 self.RefreshCurrentIndexList()
570 def RefreshCurrentIndexList(self):
571 selected = self.NetworkNodes.GetSelection()
573 window = self.NetworkNodes.GetPage(selected)
574 window.RefreshIndexList()
578 def RefreshNetworkNodes(self):
579 if self.NetworkNodes.GetPageCount() > 0:
580 self.NetworkNodes.DeleteAllPages()
582 new_editingpanel = EditingPanel(self.NetworkNodes, self, self.Manager)
583 new_editingpanel.SetIndex(0)
584 self.NetworkNodes.AddPage(new_editingpanel, "")
585 for idx in self.NodeList.GetSlaveIDs():
586 new_editingpanel = EditingPanel(self.NetworkNodes, self, self.NodeList, False)
587 new_editingpanel.SetIndex(idx)
588 self.NetworkNodes.AddPage(new_editingpanel, "")
590 def RefreshStatusBar(self):
592 selected = self.NetworkNodes.GetSelection()
593 if self.HelpBar and selected >= 0:
594 window = self.NetworkNodes.GetPage(selected)
595 selection = window.GetSelection()
597 index, subIndex = selection
598 if self.NodeList.IsCurrentEntry(index):
599 self.HelpBar.SetStatusText("Index: 0x%04X"%index, 0)
600 self.HelpBar.SetStatusText("Subindex: 0x%02X"%subIndex, 1)
601 entryinfos = self.NodeList.GetEntryInfos(index)
602 name = entryinfos["name"]
603 category = "Optional"
604 if entryinfos["need"]:
605 category = "Mandatory"
608 if entryinfos["struct"] & OD_IdenticalIndexes:
609 number = " possibly defined %d times"%entryinfos["nbmax"]
610 if entryinfos["struct"] & OD_IdenticalSubindexes:
612 elif entryinfos["struct"] & OD_MultipleSubindexes:
614 text = "%s: %s entry of struct %s%s."%(name,category,struct,number)
615 self.HelpBar.SetStatusText(text, 2)
618 self.HelpBar.SetStatusText("", i)
621 self.HelpBar.SetStatusText("", i)
623 def RefreshMainMenu(self):
625 self.NetworkMenu.Enable(ID_NETWORKEDITNETWORKMENUITEMS3, False)
626 if self.NodeList == None:
628 self.menuBar1.EnableTop(1, False)
629 self.menuBar1.EnableTop(2, False)
630 self.menuBar1.EnableTop(3, False)
632 self.FileMenu.Enable(ID_NETWORKEDITFILEMENUITEMS1, False)
633 self.FileMenu.Enable(ID_NETWORKEDITFILEMENUITEMS2, False)
635 self.menuBar1.EnableTop(0, False)
636 self.menuBar1.EnableTop(1, False)
637 self.menuBar1.EnableTop(2, False)
640 self.menuBar1.EnableTop(1, True)
642 self.FileMenu.Enable(ID_NETWORKEDITFILEMENUITEMS1, True)
643 self.FileMenu.Enable(ID_NETWORKEDITFILEMENUITEMS2, True)
644 if self.NetworkNodes.GetSelection() == 0:
645 self.menuBar1.EnableTop(2, True)
646 self.menuBar1.EnableTop(3, True)
648 self.menuBar1.EnableTop(2, False)
649 self.menuBar1.EnableTop(3, False)
651 self.menuBar1.EnableTop(0, True)
652 if self.NetworkNodes.GetSelection() == 0:
653 self.menuBar1.EnableTop(1, True)
654 self.menuBar1.EnableTop(2, True)
656 self.menuBar1.EnableTop(1, False)
657 self.menuBar1.EnableTop(2, False)
659 def RefreshProfileMenu(self):
661 profile = self.Manager.GetCurrentProfileName()
662 edititem = self.EditMenu.FindItemById(ID_NETWORKEDITEDITMENUITEMS7)
664 length = self.AddMenu.GetMenuItemCount()
665 for i in xrange(length-6):
666 additem = self.AddMenu.FindItemByPosition(6)
667 self.AddMenu.Delete(additem.GetId())
668 if profile not in ("None", "DS-301"):
669 edititem.SetText("%s Profile"%profile)
670 edititem.Enable(True)
671 self.AddMenu.AppendSeparator()
672 for text, indexes in self.Manager.GetCurrentSpecificMenu():
674 self.AddMenu.Append(help='', id=new_id, kind=wx.ITEM_NORMAL, text=text)
675 self.Bind(wx.EVT_MENU, self.GetProfileCallBack(text), id=new_id)
677 edititem.SetText("Other Profile")
678 edititem.Enable(False)
680 def GetProfileCallBack(self, text):
681 def ProfileCallBack(event):
682 self.Manager.AddSpecificEntryToCurrent(text)
683 self.RefreshBufferState()
684 self.RefreshCurrentIndexList()
686 return ProfileCallBack
688 #-------------------------------------------------------------------------------
690 #-------------------------------------------------------------------------------
692 def RefreshBufferState(self):
694 nodeID = self.Manager.GetCurrentNodeID()
696 nodename = "0x%2.2X %s"%(nodeID, self.Manager.GetCurrentNodeName())
698 nodename = self.Manager.GetCurrentNodeName()
699 self.NetworkNodes.SetPageText(0, nodename)
700 for idx, name in enumerate(self.NodeList.GetSlaveNames()):
701 self.NetworkNodes.SetPageText(idx + 1, name)
704 def OnUndoMenu(self, event):
705 self.Manager.LoadCurrentPrevious()
706 self.RefreshCurrentIndexList()
707 self.RefreshBufferState()
710 def OnRedoMenu(self, event):
711 self.Manager.LoadCurrentNext()
712 self.RefreshCurrentIndexList()
713 self.RefreshBufferState()
716 #-------------------------------------------------------------------------------
718 #-------------------------------------------------------------------------------
720 def OnHelpDS301Menu(self, event):
722 selected = self.NetworkNodes.GetSelection()
724 window = self.NetworkNodes.GetPage(selected)
725 result = window.GetSelection()
728 index, subIndex = result
729 result = OpenPDFDocIndex(index, ScriptDirectory)
730 if type(result) == StringType:
731 message = wx.MessageDialog(self, result, "ERROR", wx.OK|wx.ICON_ERROR)
735 result = OpenPDFDocIndex(None, ScriptDirectory)
736 if type(result) == StringType:
737 message = wx.MessageDialog(self, result, "ERROR", wx.OK|wx.ICON_ERROR)
742 def OnHelpCANFestivalMenu(self, event):
743 #self.OpenHtmlFrame("CAN Festival Reference", os.path.join(ScriptDirectory, "doc/canfestival.html"), wx.Size(1000, 600))
744 if wx.Platform == '__WXMSW__':
745 readerpath = get_acroversion()
746 readerexepath = os.path.join(readerpath,"AcroRd32.exe")
747 if(os.path.isfile(readerexepath)):
748 os.spawnl(os.P_DETACH, readerexepath, "AcroRd32.exe", '"%s"'%os.path.join(ScriptDirectory, "doc","manual_en.pdf"))
750 os.system("xpdf -remote CANFESTIVAL %s %d &"%(os.path.join(ScriptDirectory, "doc/manual_en.pdf"),16))
753 def OnAboutMenu(self, event):
754 self.OpenHtmlFrame("About CAN Festival", os.path.join(ScriptDirectory, "doc/about.html"), wx.Size(500, 450))
757 def OpenHtmlFrame(self, title, file, size):
758 if title not in self.HtmlFrameOpened:
759 self.HtmlFrameOpened.append(title)
760 window = HtmlFrame(self, self.HtmlFrameOpened)
761 window.SetTitle(title)
762 window.SetHtmlPage(file)
763 window.SetClientSize(size)
766 #-------------------------------------------------------------------------------
767 # Editing Profiles functions
768 #-------------------------------------------------------------------------------
770 def OnCommunicationMenu(self, event):
771 dictionary,current = self.Manager.GetCurrentCommunicationLists()
772 self.EditProfile("Edit DS-301 Profile", dictionary, current)
775 def OnOtherCommunicationMenu(self, event):
776 dictionary,current = self.Manager.GetCurrentDS302Lists()
777 self.EditProfile("Edit DS-301 Profile", dictionary, current)
780 def OnEditProfileMenu(self, event):
781 title = "Edit %s Profile"%self.Manager.GetCurrentProfileName()
782 dictionary,current = self.Manager.GetCurrentProfileLists()
783 self.EditProfile(title, dictionary, current)
786 def EditProfile(self, title, dictionary, current):
787 dialog = CommunicationDialog(self)
788 dialog.SetTitle(title)
789 dialog.SetIndexDictionary(dictionary)
790 dialog.SetCurrentList(current)
791 dialog.RefreshLists()
792 if dialog.ShowModal() == wx.ID_OK:
793 new_profile = dialog.GetCurrentList()
796 for index in new_profile:
797 if index not in current:
798 addinglist.append(index)
799 for index in current:
800 if index not in new_profile:
801 removinglist.append(index)
802 self.Manager.ManageEntriesOfCurrent(addinglist, removinglist)
803 self.Manager.BufferCurrentNode()
804 self.RefreshBufferState()
805 self.RefreshCurrentIndexList()
808 #-------------------------------------------------------------------------------
809 # Edit Node informations function
810 #-------------------------------------------------------------------------------
812 def OnNodeInfosMenu(self, event):
813 dialog = NodeInfosDialog(self)
814 name, id, type, description = self.Manager.GetCurrentNodeInfos()
815 dialog.SetValues(name, id, type, description)
816 if dialog.ShowModal() == wx.ID_OK:
817 name, id, type, description = dialog.GetValues()
818 self.Manager.SetCurrentNodeInfos(name, id, type, description)
819 self.RefreshBufferState()
820 self.RefreshProfileMenu()
821 selected = self.NetworkNodes.GetSelection()
823 window = self.NetworkNodes.GetPage(selected)
824 window.RefreshTable()
828 #-------------------------------------------------------------------------------
829 # Add User Types and Variables
830 #-------------------------------------------------------------------------------
832 def AddMapVariable(self):
833 index = self.Manager.GetCurrentNextMapIndex()
835 dialog = MapVariableDialog(self)
836 dialog.SetIndex(index)
837 if dialog.ShowModal() == wx.ID_OK:
838 index, name, struct, number = dialog.GetValues()
839 result = self.Manager.AddMapVariableToCurrent(index, name, struct, number)
840 if type(result) != StringType:
841 self.RefreshBufferState()
842 self.RefreshCurrentIndexList()
844 message = wx.MessageDialog(self, result, "Error", wx.OK|wx.ICON_ERROR)
849 message = wx.MessageDialog(self, result, "No map variable index left!", wx.OK|wx.ICON_ERROR)
853 def AddUserType(self):
854 dialog = UserTypeDialog(self)
855 dialog.SetTypeList(self.Manager.GetCustomisableTypes())
856 if dialog.ShowModal() == wx.ID_OK:
857 type, min, max, length = dialog.GetValues()
858 result = self.Manager.AddUserTypeToCurrent(type, min, max, length)
859 if not IsOfType(result, StringType):
860 self.RefreshBufferState()
861 self.RefreshCurrentIndexList()
863 message = wx.MessageDialog(self, result, "Error", wx.OK|wx.ICON_ERROR)
868 def OpenMasterDCFDialog(self, node_id):
869 self.NetworkNodes.SetSelection(0)
870 self.NetworkNodes.GetPage(0).OpenDCFDialog(node_id)
872 #-------------------------------------------------------------------------------
874 #-------------------------------------------------------------------------------
876 Max_Traceback_List_Size = 20
878 def Display_Exception_Dialog(e_type,e_value,e_tb):
880 for i,line in enumerate(traceback.extract_tb(e_tb)):
881 trcbck = " " + str(i+1) + ". "
882 if line[0].find(os.getcwd()) == -1:
883 trcbck += "file : " + str(line[0]) + ", "
885 trcbck += "file : " + str(line[0][len(os.getcwd()):]) + ", "
886 trcbck += "line : " + str(line[1]) + ", " + "function : " + str(line[2])
887 trcbck_lst.append(trcbck)
890 cap = wx.Window_GetCapture()
894 dlg = wx.SingleChoiceDialog(None,
898 Click on OK for saving an error report.
900 Please contact LOLITech at:
902 bugs_networkedit@lolitech.fr
907 str(e_type) + " : " + str(e_value),
911 res = (dlg.ShowModal() == wx.ID_OK)
917 def Display_Error_Dialog(e_value):
918 message = wx.MessageDialog(None, str(e_value), "Error", wx.OK|wx.ICON_ERROR)
922 def get_last_traceback(tb):
928 def format_namespace(d, indent=' '):
929 return '\n'.join(['%s%s: %s' % (indent, k, repr(v)[:10000]) for k, v in d.iteritems()])
932 ignored_exceptions = [] # a problem with a line in a module is only reported once per session
934 def AddExceptHook(path, app_version='[No version]'):#, ignored_exceptions=[]):
936 def handle_exception(e_type, e_value, e_traceback):
937 traceback.print_exception(e_type, e_value, e_traceback) # this is very helpful when there's an exception in the rest of this func
938 last_tb = get_last_traceback(e_traceback)
939 ex = (last_tb.tb_frame.f_code.co_filename, last_tb.tb_frame.f_lineno)
940 if str(e_value).startswith("!!!"):
941 Display_Error_Dialog(e_value)
942 elif ex not in ignored_exceptions:
943 ignored_exceptions.append(ex)
944 result = Display_Exception_Dialog(e_type,e_value,e_traceback)
947 'app-title' : wx.GetApp().GetAppName(), # app_title
948 'app-version' : app_version,
949 'wx-version' : wx.VERSION_STRING,
950 'wx-platform' : wx.Platform,
951 'python-version' : platform.python_version(), #sys.version.split()[0],
952 'platform' : platform.platform(),
955 'date' : time.ctime(),
959 info['traceback'] = ''.join(traceback.format_tb(e_traceback)) + '%s: %s' % (e_type, e_value)
960 last_tb = get_last_traceback(e_traceback)
961 exception_locals = last_tb.tb_frame.f_locals # the locals at the level of the stack trace where the exception actually occurred
962 info['locals'] = format_namespace(exception_locals)
963 if 'self' in exception_locals:
964 info['self'] = format_namespace(exception_locals['self'].__dict__)
966 output = open(path+os.sep+"bug_report_"+info['date'].replace(':','-').replace(' ','_')+".txt",'w')
970 output.write(a+":\n"+str(info[a])+"\n\n")
972 #sys.excepthook = lambda *args: wx.CallAfter(handle_exception, *args)
973 sys.excepthook = handle_exception
975 if __name__ == '__main__':
977 print "\nUsage of networkedit.py :"
978 print "\n %s [Projectpath]\n"%sys.argv[0]
981 opts, args = getopt.getopt(sys.argv[1:], "h", ["help"])
982 except getopt.GetoptError:
983 # print help information and exit:
988 if o in ("-h", "--help"):
995 projectOpen = args[0]
1000 app = wx.PySimpleApp()
1001 wx.InitAllImageHandlers()
1003 # Install a exception handle for bug reports
1004 AddExceptHook(os.getcwd(),__version__)
1006 frame = networkedit(None, projectOpen=projectOpen)