2010-10-26 Francois Chouinard <fchouinard@gmail.com> Contribution for Bug309042
[deliverable/tracecompass.git] / org.eclipse.linuxtools.lttng.ui / src / org / eclipse / linuxtools / lttng / ui / views / timeframe / TimeFrameView.java
1 /*******************************************************************************
2 * Copyright (c) 2009 2010 Ericsson
3 *
4 * All rights reserved. This program and the accompanying materials are
5 * made available under the terms of the Eclipse Public License v1.0 which
6 * accompanies this distribution, and is available at
7 * http://www.eclipse.org/legal/epl-v10.html
8 *
9 * Contributors:
10 * Francois Chouinard - Initial API and implementation
11 *******************************************************************************/
12
13 package org.eclipse.linuxtools.lttng.ui.views.timeframe;
14
15 import org.eclipse.linuxtools.lttng.event.LttngEvent;
16 import org.eclipse.linuxtools.tmf.event.TmfTimeRange;
17 import org.eclipse.linuxtools.tmf.event.TmfTimestamp;
18 import org.eclipse.linuxtools.tmf.experiment.TmfExperiment;
19 import org.eclipse.linuxtools.tmf.signal.TmfExperimentSelectedSignal;
20 import org.eclipse.linuxtools.tmf.signal.TmfExperimentUpdatedSignal;
21 import org.eclipse.linuxtools.tmf.signal.TmfRangeSynchSignal;
22 import org.eclipse.linuxtools.tmf.signal.TmfSignalHandler;
23 import org.eclipse.linuxtools.tmf.signal.TmfTimeSynchSignal;
24 import org.eclipse.linuxtools.tmf.ui.views.TmfView;
25 import org.eclipse.swt.SWT;
26 import org.eclipse.swt.layout.GridData;
27 import org.eclipse.swt.layout.GridLayout;
28 import org.eclipse.swt.widgets.Composite;
29 import org.eclipse.swt.widgets.Event;
30 import org.eclipse.swt.widgets.Listener;
31 import org.eclipse.swt.widgets.Slider;
32
33 /**
34 * <b><u>TimeFrameView</u></b>
35 * <p>
36 * The TimeFrameView provides a set of spinners to monitor and set the start
37 * time, end time, the current time interval and current time of the trace
38 * set at the nanosecond level.
39 * <p>
40 * It ensures that the following relations are always true:
41 * <p>
42 * <li>[ startTime >= start time of the trace ]
43 * <li>[ endTime <= end time of the trace ]
44 * <li>[ startTime <= currentTime <= endTime ]
45 * <li>[ interval == (endTime - startTime) ]
46 * </li>
47 * <p>
48 * It provides a slider to rapidly set the current time within the time range
49 * (i.e. between startTime and endTime).
50 * <p>
51 * Finally, it allows modification of the time range and the current time. This
52 * triggers notifications to the other LTTng views.
53 * <p>
54 * FIXME: The slider is very jumpy due to the large number of async updates
55 * FIXME: Revisit the control flow between View, Spinners and Slider
56 */
57 public class TimeFrameView extends TmfView {
58
59 public static final String ID = "org.eclipse.linuxtools.lttng.ui.views.timeframe";
60
61 // ========================================================================
62 // TimeFrameView
63 // ========================================================================
64
65 // The event log timestamp characteristics
66 private TmfTimestamp fTraceStartTime = new TmfTimestamp();
67 private TmfTimestamp fTraceEndTime = new TmfTimestamp();
68
69 private TmfTimestamp fCurrentTime = new TmfTimestamp();
70
71 private TmfTimeRange fTraceTimeRange = new TmfTimeRange(fTraceStartTime, fTraceEndTime);
72 private TmfTimeRange fTraceSpan = new TmfTimeRange(fTraceStartTime, fTraceEndTime);
73 private byte fScale = 0;
74
75 // Labels
76 private static final String START_TIME_LABEL = "Window Start Time";
77 private static final String END_TIME_LABEL = "Window End Time";
78 private static final String TIME_RANGE_LABEL = "Window Range";
79 private static final String CURRENT_TIME_LABEL = "Current Time";
80
81 private static final int SLIDER_RANGE = 10000;
82
83 private SpinnerGroup fStartGroup;
84 private SpinnerGroup fEndGroup;
85 private SpinnerGroup fRangeGroup;
86 private SpinnerGroup fCurrentGroup;
87
88 // The slider
89 private Slider fSlider;
90
91 // The current experiment
92 TmfExperiment<LttngEvent> fExperiment = null;
93 // notify external listeners may not be needed if the update originated
94 // externally
95 private boolean fupdateExternalListeners = true;
96
97
98 /**
99 * Constructor
100 */
101 public TimeFrameView() {
102 super("TimeFrameView");
103 }
104
105 /* (non-Javadoc)
106 * @see org.eclipse.ui.part.WorkbenchPart#createPartControl(org.eclipse.swt.widgets.Composite)
107 */
108 @Override
109 public void createPartControl(Composite parent) {
110
111 // Set the view layout
112 GridLayout layout = new GridLayout(4, true);
113 parent.setLayout(layout);
114
115 fStartGroup = new SpinnerGroup(this, parent, START_TIME_LABEL, fTraceTimeRange, fTraceStartTime);
116 fEndGroup = new SpinnerGroup(this, parent, END_TIME_LABEL, fTraceTimeRange, fTraceEndTime);
117 fRangeGroup = new SpinnerGroup(this, parent, TIME_RANGE_LABEL, fTraceTimeRange, fTraceEndTime);
118 fCurrentGroup = new SpinnerGroup(this, parent, CURRENT_TIME_LABEL, fTraceTimeRange, fTraceStartTime);
119
120 // Create the slider
121 createSlider(parent);
122 }
123
124 /* (non-Javadoc)
125 * @see org.eclipse.ui.part.WorkbenchPart#setFocus()
126 */
127 @Override
128 public void setFocus() {
129 // TODO Auto-generated method stub
130 }
131
132 // ========================================================================
133 // Operators
134 // ========================================================================
135
136 /**
137 * One of the spinners has been updated. Synchronize the other widgets.
138 */
139 public void synchTimeFrameWidgets(SpinnerGroup trigger) {
140 boolean trangeUpdated = false;
141
142 // Collect the data
143 TmfTimestamp startTime = fStartGroup.getCurrentTime();
144 TmfTimestamp endTime = fEndGroup.getCurrentTime();
145 TmfTimestamp timeRange = fRangeGroup.getCurrentTime();
146 TmfTimestamp currentTime = fCurrentGroup.getCurrentTime();
147
148 // If startTime was set beyond endTime, adjust endTime and interval
149 if (trigger == fStartGroup) {
150 if (startTime.compareTo(endTime, false) > 0) {
151 endTime = startTime;
152 trangeUpdated = true;
153 }
154 }
155
156 // If endTime was set beyond startTime, adjust startTime and interval
157 if (trigger == fEndGroup) {
158 if (endTime.compareTo(startTime, false) < 0) {
159 startTime = endTime;
160 trangeUpdated = true;
161 }
162 }
163
164 // If timeRange was set, adjust endTime
165 if (trigger == fRangeGroup) {
166 long start = startTime.getValue();
167 long span = timeRange.getValue();
168 TmfTimestamp ts = new TmfTimestamp(start + span, startTime.getScale(), 0);
169 if (ts.compareTo(fTraceEndTime, false) > 0) {
170 ts = fTraceEndTime.synchronize(fTraceEndTime.getValue(), startTime.getScale());
171 }
172 endTime = ts;
173 trangeUpdated = true;
174 }
175
176 // Compute the new time range
177 TmfTimeRange subrange = new TmfTimeRange(startTime, endTime);
178 byte scale = startTime.getScale();
179 TmfTimestamp interval = new TmfTimestamp(startTime.getAdjustment(endTime, scale), scale, 0);
180
181 // Update the spinner groups
182 fStartGroup.setContent(fTraceTimeRange, startTime);
183 fEndGroup.setContent(fTraceTimeRange, endTime);
184 fRangeGroup.setContent(fTraceSpan, interval);
185 fCurrentGroup.setContent(subrange, currentTime);
186
187 updateSlider(subrange, currentTime);
188 // Notify other views, only if the update originated from this view
189 if (fupdateExternalListeners) {
190 if (!fCurrentTime.equals(currentTime)) {
191 fCurrentTime = currentTime;
192 broadcast(new TmfTimeSynchSignal(this, currentTime));
193 }
194
195 // Notify the views if the time range has been impacted
196 if (trangeUpdated) {
197 TmfTimeRange trange = new TmfTimeRange(startTime, endTime);
198 broadcast(new TmfRangeSynchSignal(this, trange, currentTime));
199 }
200 }
201 }
202
203 // ========================================================================
204 // Slider Handling
205 // ========================================================================
206
207 /**
208 * @param parent
209 */
210 private void createSlider(Composite parent) {
211 fSlider = new Slider(parent, SWT.SMOOTH | SWT.FILL);
212 fSlider.setMinimum(0);
213 fSlider.setMaximum(SLIDER_RANGE + fSlider.getThumb());
214 fSlider.setIncrement(SLIDER_RANGE / 100);
215 fSlider.setPageIncrement(SLIDER_RANGE / 10);
216 fSlider.setSelection(0);
217
218 GridData gridData = new GridData(SWT.LEFT, SWT.TOP, true, false);
219 gridData.horizontalAlignment = SWT.FILL;
220 gridData.horizontalSpan = 4;
221 fSlider.setLayoutData(gridData);
222
223 fSlider.addListener(SWT.Selection, new Listener() {
224 @Override
225 public void handleEvent(Event event) {
226 int ratio = fSlider.getSelection();
227 TmfTimestamp span = fCurrentGroup.getSpan();
228 long value = span.getValue() * ratio / SLIDER_RANGE;
229 TmfTimestamp start = fCurrentGroup.getStartTime();
230 TmfTimestamp current = new TmfTimestamp(start.getValue() + value, start.getScale(), 0);
231 fCurrentGroup.setValue(current);
232 }
233 });
234
235 }
236
237 /**
238 * @param range
239 * @param timestamp
240 */
241 private void updateSlider(TmfTimeRange range, TmfTimestamp timestamp) {
242
243 // Ignore update if disposed
244 if (fSlider.isDisposed()) return;
245
246 // Determine the new relative position
247 byte scale = range.getEndTime().getScale();
248 long total = range.getStartTime().getAdjustment(range.getEndTime(), scale);
249 long relative = range.getStartTime().getAdjustment(timestamp, scale);
250
251 // Set the slider value
252 final long position = (total > 0) ? (relative * SLIDER_RANGE / total) : 0;
253
254 // Update the slider on the UI thread
255 long current = fSlider.getSelection();
256 if (position != current) {
257 fSlider.getDisplay().asyncExec(new Runnable() {
258 @Override
259 public void run() {
260 fSlider.setSelection((int) position);
261 }
262 });
263 }
264 }
265
266 /* (non-Javadoc)
267 * @see java.lang.Object#toString()
268 */
269 @Override
270 public String toString() {
271 return "[TimeFrameView]";
272 }
273
274 // ========================================================================
275 // TMF Signal Handling
276 // ========================================================================
277
278 /**
279 * @param signal
280 */
281 @SuppressWarnings("unchecked")
282 @TmfSignalHandler
283 public void experimentSelected(TmfExperimentSelectedSignal<LttngEvent> signal) {
284
285 // Update the trace reference
286 fExperiment = (TmfExperiment<LttngEvent>) signal.getExperiment();
287
288 // Update the time frame
289 fTraceTimeRange = fExperiment.getTimeRange();
290 fTraceStartTime = fTraceTimeRange.getStartTime();
291 fTraceEndTime = fTraceTimeRange.getEndTime();
292 fScale = fTraceStartTime.getScale();
293
294 // Update the widgets
295 fStartGroup.setContent(fTraceTimeRange, fTraceStartTime);
296 fEndGroup.setContent(fTraceTimeRange, fTraceEndTime);
297 fCurrentGroup.setContent(fTraceTimeRange, fTraceStartTime);
298
299 fCurrentTime = fTraceStartTime;
300
301 TmfTimestamp delta = new TmfTimestamp(fTraceStartTime.getAdjustment(fTraceEndTime, fScale), fScale, 0);
302 fTraceSpan = new TmfTimeRange(new TmfTimestamp(0, fScale, 0), delta);
303 // fRangeGroup.setContent(fTraceSpan, delta);
304 TmfTimestamp start = new TmfTimestamp(1, (byte) -1, 0);
305 fRangeGroup.setContent(fTraceSpan, start);
306 }
307
308 /**
309 * @param signal
310 */
311 @TmfSignalHandler
312 public void experimentUpdated(TmfExperimentUpdatedSignal signal) {
313
314 // Update the time frame
315 // fTraceTimeRange = signal.getTrace().getTimeRange();
316 fTraceTimeRange = signal.getExperiment().getTimeRange();
317 fTraceStartTime = fTraceTimeRange.getStartTime();
318 fTraceEndTime = fTraceTimeRange.getEndTime();
319 fScale = fTraceStartTime.getScale();
320
321 // Update the widgets
322 fStartGroup.setContent(fTraceTimeRange, fStartGroup.getCurrentTime());
323 fEndGroup.setContent(fTraceTimeRange, fTraceEndTime);
324 fCurrentGroup.setContent(fTraceTimeRange, fCurrentGroup.getCurrentTime());
325
326 TmfTimestamp delta = new TmfTimestamp(fTraceStartTime.getAdjustment(fTraceEndTime, fScale), fScale, 0);
327 fTraceSpan = new TmfTimeRange(new TmfTimestamp(0, fScale, 0), delta);
328 fRangeGroup.setContent(fTraceSpan, delta);
329 }
330
331 /**
332 * @param signal
333 */
334 @TmfSignalHandler
335 public void currentTimeRangeUpdated(TmfRangeSynchSignal signal) {
336 if (signal.getSource() != this) {
337 // Update the time frame
338 TmfTimeRange selTimeRange = signal.getCurrentRange();
339 TmfTimestamp selStart = selTimeRange.getStartTime().synchronize(0,
340 fScale);
341 TmfTimestamp selEnd = selTimeRange.getEndTime().synchronize(0,
342 fScale);
343
344 fupdateExternalListeners = false;
345 // Update the widgets and prevent broadcast notifications to
346 // the views which have been notified already.
347 {
348 fStartGroup.setContent(fTraceTimeRange, selStart);
349 fEndGroup.setContent(fTraceTimeRange, selEnd);
350
351 TmfTimestamp delta = new TmfTimestamp(selStart.getAdjustment(
352 selEnd, fScale), fScale, 0);
353
354 fRangeGroup.setContent(fTraceSpan, delta);
355 }
356
357 // restore the external notification flag
358 fupdateExternalListeners = true;
359
360 }
361 }
362
363 /**
364 * @param signal
365 */
366 @TmfSignalHandler
367 public void currentTimeUpdated(TmfTimeSynchSignal signal) {
368 if (signal.getSource() != this) {
369 // prevent loop to external notifications
370 fupdateExternalListeners = false;
371 fCurrentTime = signal.getCurrentTime().synchronize(0, fStartGroup.getCurrentTime().getScale());
372 if (fStartGroup.getCurrentTime().compareTo(fCurrentTime, false) > 0) {
373 fStartGroup.setContent(new TmfTimeRange(fCurrentTime, fEndGroup.getCurrentTime()), fCurrentTime);
374 }
375 if (fEndGroup.getCurrentTime().compareTo(fCurrentTime, false) < 0) {
376 fEndGroup.setContent(new TmfTimeRange(fStartGroup.getCurrentTime(), fCurrentTime), fCurrentTime);
377 }
378 fCurrentGroup.setContent(null, fCurrentTime);
379 updateSlider(fCurrentGroup.getTimeRange(), fCurrentTime);
380
381 // Enable external notifications
382 fupdateExternalListeners = true;
383 }
384 }
385
386 }
This page took 0.039 seconds and 5 git commands to generate.