author | laurent |
Fri, 09 Dec 2011 11:26:10 +0100 | |
changeset 600 | 7db729686416 |
parent 586 | 9aa96a36cf33 |
child 632 | 3ea55a5db68e |
permissions | -rw-r--r-- |
301 | 1 |
#!/usr/bin/env python |
2 |
# -*- coding: utf-8 -*- |
|
3 |
||
4 |
#This file is part of PLCOpenEditor, a library implementing an IEC 61131-3 editor |
|
5 |
#based on the plcopen standard. |
|
6 |
# |
|
7 |
#Copyright (C) 2007: Edouard TISSERANT and Laurent BESSARD |
|
8 |
# |
|
9 |
#See COPYING file for copyrights details. |
|
10 |
# |
|
11 |
#This library is free software; you can redistribute it and/or |
|
12 |
#modify it under the terms of the GNU General Public |
|
13 |
#License as published by the Free Software Foundation; either |
|
14 |
#version 2.1 of the License, or (at your option) any later version. |
|
15 |
# |
|
16 |
#This library is distributed in the hope that it will be useful, |
|
17 |
#but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
18 |
#MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
19 |
#General Public License for more details. |
|
20 |
# |
|
21 |
#You should have received a copy of the GNU General Public |
|
22 |
#License along with this library; if not, write to the Free Software |
|
23 |
#Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
|
24 |
||
25 |
import wx |
|
26 |
import wx.lib.plot as plot |
|
361
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
27 |
from graphics.GraphicCommons import DebugViewer |
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
28 |
from controls import EditorPanel |
301 | 29 |
|
30 |
colours = ['blue', 'red', 'green', 'yellow', 'orange', 'purple', 'brown', 'cyan', |
|
31 |
'pink', 'grey'] |
|
32 |
markers = ['circle', 'dot', 'square', 'triangle', 'triangle_down', 'cross', 'plus', 'circle'] |
|
33 |
||
34 |
||
35 |
#------------------------------------------------------------------------------- |
|
36 |
# Debug Variable Graphic Viewer class |
|
37 |
#------------------------------------------------------------------------------- |
|
38 |
||
39 |
||
40 |
RANGE_VALUES = [str(25 * 2 ** i) for i in xrange(6)] |
|
41 |
||
42 |
[ID_GRAPHICVIEWER, ID_GRAPHICVIEWERCANVAS, |
|
43 |
ID_GRAPHICVIEWERCANVASRANGE, ID_GRAPHICVIEWERCANVASPOSITION, |
|
44 |
ID_GRAPHICVIEWERRESETBUTTON, ID_GRAPHICVIEWERCURRENTBUTTON, |
|
45 |
ID_GRAPHICVIEWERSTATICTEXT1, ID_GRAPHICVIEWERSTATICTEXT2, |
|
46 |
] = [wx.NewId() for _init_ctrls in range(8)] |
|
47 |
||
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
48 |
class GraphicViewer(EditorPanel, DebugViewer): |
301 | 49 |
|
50 |
def _init_coll_MainGridSizer_Items(self, parent): |
|
51 |
# generated method, don't edit |
|
52 |
parent.AddWindow(self.Canvas, 0, border=0, flag=wx.GROW) |
|
53 |
parent.AddSizer(self.RangeSizer, 0, border=0, flag=wx.GROW) |
|
54 |
||
55 |
def _init_coll_MainGridSizer_Growables(self, parent): |
|
56 |
# generated method, don't edit |
|
57 |
parent.AddGrowableCol(0) |
|
58 |
parent.AddGrowableRow(0) |
|
59 |
||
60 |
def _init_coll_RangeSizer_Items(self, parent): |
|
61 |
# generated method, don't edit |
|
391 | 62 |
parent.AddWindow(self.staticbox1, 0, border=5, flag=wx.ALIGN_CENTER_VERTICAL|wx.ALL) |
301 | 63 |
parent.AddWindow(self.CanvasRange, 0, border=5, flag=wx.ALL) |
391 | 64 |
parent.AddWindow(self.staticText2, 0, border=5, flag=wx.ALIGN_CENTER_VERTICAL|wx.ALL) |
301 | 65 |
parent.AddWindow(self.CanvasPosition, 0, border=5, flag=wx.GROW|wx.ALL) |
66 |
parent.AddWindow(self.ResetButton, 0, border=5, flag=wx.ALL) |
|
67 |
parent.AddWindow(self.CurrentButton, 0, border=5, flag=wx.ALL) |
|
68 |
||
69 |
def _init_coll_RangeSizer_Growables(self, parent): |
|
70 |
# generated method, don't edit |
|
71 |
parent.AddGrowableCol(3) |
|
72 |
parent.AddGrowableRow(0) |
|
73 |
||
74 |
def _init_sizers(self): |
|
75 |
# generated method, don't edit |
|
76 |
self.MainGridSizer = wx.FlexGridSizer(cols=1, hgap=0, rows=2, vgap=0) |
|
77 |
self.RangeSizer = wx.FlexGridSizer(cols=6, hgap=0, rows=1, vgap=0) |
|
78 |
||
79 |
self._init_coll_MainGridSizer_Items(self.MainGridSizer) |
|
80 |
self._init_coll_MainGridSizer_Growables(self.MainGridSizer) |
|
81 |
self._init_coll_RangeSizer_Items(self.RangeSizer) |
|
82 |
self._init_coll_RangeSizer_Growables(self.RangeSizer) |
|
83 |
||
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
84 |
self.Editor.SetSizer(self.MainGridSizer) |
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
85 |
|
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
86 |
def _init_Editor(self, prnt): |
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
87 |
self.Editor = wx.Panel(prnt, ID_GRAPHICVIEWER, wx.DefaultPosition, |
301 | 88 |
wx.DefaultSize, 0) |
89 |
||
90 |
self.Canvas = plot.PlotCanvas(id=ID_GRAPHICVIEWERCANVAS, |
|
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
91 |
name='Canvas', parent=self.Editor, pos=wx.Point(0, 0), |
301 | 92 |
size=wx.Size(0, 0), style=0) |
93 |
def _axisInterval(spec, lower, upper): |
|
94 |
if spec == 'border': |
|
95 |
if lower == upper: |
|
96 |
return lower - 0.5, upper + 0.5 |
|
97 |
else: |
|
98 |
border = (upper - lower) * 0.05 |
|
99 |
return lower - border, upper + border |
|
100 |
else: |
|
101 |
return plot.PlotCanvas._axisInterval(self.Canvas, spec, lower, upper) |
|
102 |
self.Canvas._axisInterval = _axisInterval |
|
103 |
self.Canvas.SetYSpec('border') |
|
104 |
||
105 |
self.staticbox1 = wx.StaticText(id=ID_GRAPHICVIEWERSTATICTEXT1, |
|
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
106 |
label=_('Range:'), name='staticText1', parent=self.Editor, |
391 | 107 |
pos=wx.Point(0, 0), size=wx.DefaultSize, style=0) |
301 | 108 |
|
109 |
self.CanvasRange = wx.ComboBox(id=ID_GRAPHICVIEWERCANVASRANGE, |
|
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
110 |
name='CanvasRange', parent=self.Editor, pos=wx.Point(0, 0), |
313 | 111 |
size=wx.Size(100, 28), choices=RANGE_VALUES, style=0) |
301 | 112 |
self.CanvasRange.SetStringSelection("25") |
113 |
self.Bind(wx.EVT_COMBOBOX, self.OnRangeChanged, id=ID_GRAPHICVIEWERCANVASRANGE) |
|
114 |
self.Bind(wx.EVT_TEXT_ENTER, self.OnRangeChanged, id=ID_GRAPHICVIEWERCANVASRANGE) |
|
115 |
||
116 |
self.staticText2 = wx.StaticText(id=ID_GRAPHICVIEWERSTATICTEXT2, |
|
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
117 |
label=_('Position:'), name='staticText2', parent=self.Editor, |
391 | 118 |
pos=wx.Point(0, 0), size=wx.DefaultSize, style=0) |
301 | 119 |
|
120 |
self.CanvasPosition = wx.ScrollBar(id=ID_GRAPHICVIEWERCANVASPOSITION, |
|
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
121 |
name='Position', parent=self.Editor, pos=wx.Point(0, 0), |
301 | 122 |
size=wx.Size(0, 16), style=wx.SB_HORIZONTAL) |
123 |
self.CanvasPosition.SetScrollbar(0, 10, 100, 10) |
|
124 |
self.CanvasPosition.Bind(wx.EVT_SCROLL_THUMBTRACK, self.OnPositionChanging, |
|
125 |
id = ID_GRAPHICVIEWERCANVASPOSITION) |
|
126 |
self.CanvasPosition.Bind(wx.EVT_SCROLL_LINEUP, self.OnPositionChanging, |
|
127 |
id = ID_GRAPHICVIEWERCANVASPOSITION) |
|
128 |
self.CanvasPosition.Bind(wx.EVT_SCROLL_LINEDOWN, self.OnPositionChanging, |
|
129 |
id = ID_GRAPHICVIEWERCANVASPOSITION) |
|
130 |
self.CanvasPosition.Bind(wx.EVT_SCROLL_PAGEUP, self.OnPositionChanging, |
|
131 |
id = ID_GRAPHICVIEWERCANVASPOSITION) |
|
132 |
self.CanvasPosition.Bind(wx.EVT_SCROLL_PAGEDOWN, self.OnPositionChanging, |
|
133 |
id = ID_GRAPHICVIEWERCANVASPOSITION) |
|
134 |
||
135 |
self.ResetButton = wx.Button(id=ID_GRAPHICVIEWERRESETBUTTON, label='Reset', |
|
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
136 |
name='ResetButton', parent=self.Editor, pos=wx.Point(0, 0), |
301 | 137 |
size=wx.Size(72, 24), style=0) |
138 |
self.Bind(wx.EVT_BUTTON, self.OnResetButton, id=ID_GRAPHICVIEWERRESETBUTTON) |
|
139 |
||
140 |
self.CurrentButton = wx.Button(id=ID_GRAPHICVIEWERCURRENTBUTTON, label='Current', |
|
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
141 |
name='CurrentButton', parent=self.Editor, pos=wx.Point(0, 0), |
301 | 142 |
size=wx.Size(72, 24), style=0) |
143 |
self.Bind(wx.EVT_BUTTON, self.OnCurrentButton, id=ID_GRAPHICVIEWERCURRENTBUTTON) |
|
144 |
||
145 |
self._init_sizers() |
|
146 |
||
415
d3d8f8f0b678
controler (PLCControler) and debug data producer (PluginsRoot) are no longer the same thing
b.taylor@willowglen.ca
parents:
374
diff
changeset
|
147 |
def __init__(self, parent, window, producer, instancepath = ""): |
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
148 |
EditorPanel.__init__(self, parent, "", window, None) |
415
d3d8f8f0b678
controler (PLCControler) and debug data producer (PluginsRoot) are no longer the same thing
b.taylor@willowglen.ca
parents:
374
diff
changeset
|
149 |
DebugViewer.__init__(self, producer, True, False) |
361
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
150 |
|
301 | 151 |
self.InstancePath = instancepath |
152 |
||
153 |
self.Datas = [] |
|
154 |
self.CurrentValue = 0 |
|
155 |
self.CurrentRange = 25 |
|
156 |
||
361
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
157 |
self.AddDataConsumer(self.InstancePath.upper(), self) |
301 | 158 |
|
338 | 159 |
def __del__(self): |
361
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
160 |
DebugViewer.__del__(self) |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
161 |
self.RemoveDataConsumer(self) |
338 | 162 |
|
586
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
163 |
def GetTitle(self): |
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
164 |
if len(self.InstancePath) > 15: |
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
165 |
return "..." + self.InstancePath[-12:] |
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
166 |
return self.InstancePath |
9aa96a36cf33
Moving variable panel from bottom notebook panel to POU editor panel
laurent
parents:
504
diff
changeset
|
167 |
|
338 | 168 |
def ResetView(self): |
169 |
self.Datas = [] |
|
170 |
self.CurrentValue = 0 |
|
171 |
self.RefreshView() |
|
172 |
||
361
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
173 |
def RefreshNewData(self): |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
174 |
self.RefreshView(False) |
374 | 175 |
DebugViewer.RefreshNewData(self) |
361
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
176 |
|
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
177 |
def RefreshView(self, force=True): |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
178 |
self.Freeze() |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
179 |
if force or self.CurrentValue + self.CurrentRange == len(self.Datas) or self.CurrentValue + len(self.Datas) < self.CurrentRange: |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
180 |
var_name = self.InstancePath.split(".")[-1] |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
181 |
|
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
182 |
self.VariableGraphic = plot.PolyLine(self.Datas[self.CurrentValue:self.CurrentValue + self.CurrentRange], |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
183 |
legend=var_name, colour=colours[0]) |
391 | 184 |
self.GraphicsObject = plot.PlotGraphics([self.VariableGraphic], _("%s Graphics") % var_name, _("Tick"), _("Values")) |
361
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
185 |
datas_length = len(self.Datas) |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
186 |
if datas_length > 1: |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
187 |
start = self.Datas[self.CurrentValue][0] |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
188 |
if self.CurrentValue + self.CurrentRange > datas_length: |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
189 |
end = start + (self.Datas[datas_length - 1][0] - start) * self.CurrentRange / (datas_length - self.CurrentValue - 1) |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
190 |
else: |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
191 |
end = self.Datas[self.CurrentValue + self.CurrentRange - 1][0] |
301 | 192 |
else: |
361
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
193 |
start = 0. |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
194 |
end = 25. |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
195 |
self.Canvas.Draw(self.GraphicsObject, xAxis=(start, end)) |
301 | 196 |
self.RefreshScrollBar() |
361
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
197 |
self.Thaw() |
326 | 198 |
|
301 | 199 |
def GetInstancePath(self): |
200 |
return self.InstancePath |
|
201 |
||
326 | 202 |
def IsViewing(self, tagname): |
203 |
return self.InstancePath == tagname |
|
204 |
||
504 | 205 |
def NewValue(self, tick, value, forced=False): |
361
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
206 |
self.Datas.append((float(tick), {True:1., False:0.}.get(value, float(value)))) |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
207 |
if self.CurrentValue + self.CurrentRange == len(self.Datas) - 1: |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
208 |
self.CurrentValue += 1 |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
209 |
self.NewDataAvailable() |
62570186dad4
Adding support for synchronize refreshing with tick and limit it to a defined period
greg
parents:
344
diff
changeset
|
210 |
|
301 | 211 |
def RefreshScrollBar(self): |
212 |
self.CanvasPosition.SetScrollbar(self.CurrentValue, self.CurrentRange, len(self.Datas), self.CurrentRange) |
|
213 |
||
214 |
def OnRangeChanged(self, event): |
|
215 |
old_range = self.CurrentRange |
|
216 |
try: |
|
217 |
self.CurrentRange = int(self.CanvasRange.GetValue()) |
|
218 |
except ValueError, e: |
|
219 |
self.CanvasRange.SetValue(str(self.CurrentRange)) |
|
220 |
self.CurrentValue = max(0, min(self.CurrentValue + old_range - self.CurrentRange, |
|
221 |
len(self.Datas) - self.CurrentRange)) |
|
222 |
self.RefreshView() |
|
223 |
event.Skip() |
|
224 |
||
225 |
def OnPositionChanging(self, event): |
|
226 |
self.CurrentValue = event.GetPosition() |
|
227 |
self.RefreshView() |
|
228 |
event.Skip() |
|
229 |
||
230 |
def OnResetButton(self, event): |
|
338 | 231 |
self.ResetView() |
301 | 232 |
event.Skip() |
233 |
||
234 |
def OnCurrentButton(self, event): |
|
235 |
self.CurrentValue = max(0, len(self.Datas) - self.CurrentRange) |
|
236 |
self.RefreshView() |
|
237 |
event.Skip() |
|
238 |