General improvements:
[deliverable/tracecompass.git] / org.eclipse.linuxtools.lttng.ui / src / org / eclipse / linuxtools / lttng / ui / views / timeframe / TimeFrameView.java
1 /*******************************************************************************
2 * Copyright (c) 2009 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.tmf.event.TmfTimeRange;
16 import org.eclipse.linuxtools.tmf.event.TmfTimestamp;
17 import org.eclipse.linuxtools.tmf.signal.TmfSignalHandler;
18 import org.eclipse.linuxtools.tmf.signal.TmfTimeSynchSignal;
19 import org.eclipse.linuxtools.tmf.trace.TmfExperiment;
20 import org.eclipse.linuxtools.tmf.trace.TmfExperimentSelectedSignal;
21 import org.eclipse.linuxtools.tmf.trace.TmfExperimentUpdatedSignal;
22 import org.eclipse.linuxtools.tmf.ui.views.TmfView;
23 import org.eclipse.swt.SWT;
24 import org.eclipse.swt.layout.GridData;
25 import org.eclipse.swt.layout.GridLayout;
26 import org.eclipse.swt.widgets.Composite;
27 import org.eclipse.swt.widgets.Event;
28 import org.eclipse.swt.widgets.Listener;
29 import org.eclipse.swt.widgets.Slider;
30
31 /**
32 * <b><u>TimeFrameView</u></b>
33 * <p>
34 * The TimeFrameView provides a set of spinners to monitor and set the start
35 * time, end time, the current time interval and current time of the trace
36 * set at the nanosecond level.
37 * <p>
38 * It ensures that the following relations are always true:
39 * <p>
40 * <li>[ startTime >= start time of the trace ]
41 * <li>[ endTime <= end time of the trace ]
42 * <li>[ startTime <= currentTime <= endTime ]
43 * <li>[ interval == (endTime - startTime) ]
44 * </li>
45 * <p>
46 * It provides a slider to rapidly set the current time within the time range
47 * (i.e. between startTime and endTime).
48 * <p>
49 * Finally, it allows modification of the time range and the current time. This
50 * triggers notifications to the other LTTng views.
51 * <p>
52 * FIXME: The slider is very jumpy due to the large number of async updates
53 * FIXME: Revisit the control flow between View, Spinners and Slider
54 */
55 public class TimeFrameView extends TmfView {
56
57 public static final String ID = "org.eclipse.linuxtools.lttng.ui.views.timeframe";
58
59 // ========================================================================
60 // TimeFrameView
61 // ========================================================================
62
63 // The event log timestamp characteristics
64 private TmfTimestamp fTraceStartTime = new TmfTimestamp();
65 private TmfTimestamp fTraceEndTime = new TmfTimestamp();
66
67 private TmfTimestamp fCurrentTime = new TmfTimestamp();
68
69 private TmfTimeRange fTraceTimeRange = new TmfTimeRange(fTraceStartTime, fTraceEndTime);
70 private TmfTimeRange fTraceSpan = new TmfTimeRange(fTraceStartTime, fTraceEndTime);
71 private byte fScale = 0;
72
73 // Labels
74 private static final String START_TIME_LABEL = "Start Time";
75 private static final String END_TIME_LABEL = "End Time";
76 private static final String TIME_RANGE_LABEL = "Interval";
77 private static final String CURRENT_TIME_LABEL = "Current Time";
78
79 private static final int SLIDER_RANGE = 10000;
80
81 private SpinnerGroup fStartGroup;
82 private SpinnerGroup fEndGroup;
83 private SpinnerGroup fRangeGroup;
84 private SpinnerGroup fCurrentGroup;
85
86 // The slider
87 private Slider fSlider;
88
89 // The current experiment
90 TmfExperiment fExperiment = null;
91
92 /**
93 * Constructor
94 */
95 public TimeFrameView() {
96 super();
97 }
98
99 /* (non-Javadoc)
100 * @see org.eclipse.ui.part.WorkbenchPart#createPartControl(org.eclipse.swt.widgets.Composite)
101 */
102 @Override
103 public void createPartControl(Composite parent) {
104
105 // Set the view layout
106 GridLayout layout = new GridLayout(4, true);
107 parent.setLayout(layout);
108
109 fStartGroup = new SpinnerGroup(this, parent, START_TIME_LABEL, fTraceTimeRange, fTraceStartTime);
110 fEndGroup = new SpinnerGroup(this, parent, END_TIME_LABEL, fTraceTimeRange, fTraceEndTime);
111 fRangeGroup = new SpinnerGroup(this, parent, TIME_RANGE_LABEL, fTraceTimeRange, fTraceEndTime);
112 fCurrentGroup = new SpinnerGroup(this, parent, CURRENT_TIME_LABEL, fTraceTimeRange, fTraceStartTime);
113
114 // Create the slider
115 createSlider(parent);
116 }
117
118 /* (non-Javadoc)
119 * @see org.eclipse.ui.part.WorkbenchPart#setFocus()
120 */
121 @Override
122 public void setFocus() {
123 // TODO Auto-generated method stub
124 }
125
126 // ========================================================================
127 // Operators
128 // ========================================================================
129
130 /**
131 * One of the spinners has been updated. Synchronize the other widgets.
132 */
133 public void synchTimeFrameWidgets(SpinnerGroup trigger) {
134
135 // Collect the data
136 TmfTimestamp startTime = fStartGroup.getCurrentTime();
137 TmfTimestamp endTime = fEndGroup.getCurrentTime();
138 TmfTimestamp timeRange = fRangeGroup.getCurrentTime();
139 TmfTimestamp currentTime = fCurrentGroup.getCurrentTime();
140
141 // If startTime was set beyond endTime, adjust endTime and interval
142 if (trigger == fStartGroup) {
143 if (startTime.compareTo(endTime, false) > 0) {
144 endTime = startTime;
145 }
146 }
147
148 // If endTime was set beyond startTime, adjust startTime and interval
149 if (trigger == fEndGroup) {
150 if (endTime.compareTo(startTime, false) < 0) {
151 startTime = endTime;
152 }
153 }
154
155 // If timeRange was set, adjust endTime
156 if (trigger == fRangeGroup) {
157 long start = startTime.getValue();
158 long span = timeRange.getValue();
159 TmfTimestamp ts = new TmfTimestamp(start + span, startTime.getScale(), 0);
160 if (ts.compareTo(fTraceEndTime, false) > 0) {
161 ts = fTraceEndTime.synchronize(fTraceEndTime.getValue(), startTime.getScale());
162 }
163 endTime = ts;
164 }
165
166 // Compute the new time range
167 TmfTimeRange subrange = new TmfTimeRange(startTime, endTime);
168 TmfTimestamp interval = new TmfTimestamp(startTime.getAdjustment(endTime), startTime.getScale(), 0);
169
170 // Update the spinner groups
171 fStartGroup.setContent(fTraceTimeRange, startTime);
172 fEndGroup.setContent(fTraceTimeRange, endTime);
173 fRangeGroup.setContent(fTraceSpan, interval);
174 fCurrentGroup.setContent(subrange, currentTime);
175
176 updateSlider(subrange, currentTime);
177
178 // Notify other views
179 if (!fCurrentTime.equals(currentTime)) {
180 fCurrentTime = currentTime;
181 broadcastSignal(new TmfTimeSynchSignal(this, currentTime));
182 }
183 }
184
185 // ========================================================================
186 // Slider Handling
187 // ========================================================================
188
189 /**
190 * @param parent
191 */
192 private void createSlider(Composite parent) {
193 fSlider = new Slider(parent, SWT.SMOOTH | SWT.FILL);
194 fSlider.setMinimum(0);
195 fSlider.setMaximum(SLIDER_RANGE + fSlider.getThumb());
196 fSlider.setIncrement(SLIDER_RANGE / 100);
197 fSlider.setPageIncrement(SLIDER_RANGE / 10);
198 fSlider.setSelection(0);
199
200 GridData gridData = new GridData(SWT.LEFT, SWT.TOP, true, false);
201 gridData.horizontalAlignment = SWT.FILL;
202 gridData.horizontalSpan = 4;
203 fSlider.setLayoutData(gridData);
204
205 fSlider.addListener(SWT.Selection, new Listener() {
206 public void handleEvent(Event event) {
207 int ratio = fSlider.getSelection();
208 TmfTimestamp span = fCurrentGroup.getSpan();
209 long value = span.getValue() * ratio / SLIDER_RANGE;
210 TmfTimestamp start = fCurrentGroup.getStartTime();
211 TmfTimestamp current = new TmfTimestamp(start.getValue() + value, start.getScale(), 0);
212 fCurrentGroup.setValue(current);
213 }
214 });
215
216 }
217
218 /**
219 * @param range
220 * @param timestamp
221 */
222 private void updateSlider(TmfTimeRange range, TmfTimestamp timestamp) {
223
224 // Determine the new relative position
225 long total = range.getStartTime().getAdjustment(range.getEndTime());
226 long relative = range.getStartTime().getAdjustment(timestamp);
227
228 // Set the slider value
229 final long position = (total > 0) ? (relative * SLIDER_RANGE / total) : 0;
230
231 // Update the slider on the UI thread
232 long current = fSlider.getSelection();
233 if (position != current) {
234 fSlider.getDisplay().asyncExec(new Runnable() {
235 public void run() {
236 fSlider.setSelection((int) position);
237 }
238 });
239 }
240 }
241
242 /* (non-Javadoc)
243 * @see java.lang.Object#toString()
244 */
245 @Override
246 public String toString() {
247 return "[TimeFrameView]";
248 }
249
250 // ========================================================================
251 // TMF Signal Handling
252 // ========================================================================
253
254 /**
255 * @param signal
256 */
257 @TmfSignalHandler
258 public void experimentSelected(TmfExperimentSelectedSignal signal) {
259
260 // Update the trace reference
261 fExperiment = signal.getExperiment();
262
263 // Update the time frame
264 fTraceTimeRange = fExperiment.getTimeRange();
265 fTraceStartTime = fTraceTimeRange.getStartTime();
266 fTraceEndTime = fTraceTimeRange.getEndTime();
267 fScale = fTraceStartTime.getScale();
268
269 // Update the widgets
270 fStartGroup.setContent(fTraceTimeRange, fTraceStartTime);
271 fEndGroup.setContent(fTraceTimeRange, fTraceEndTime);
272 fCurrentGroup.setContent(fTraceTimeRange, fTraceStartTime);
273
274 fCurrentTime = fTraceStartTime;
275
276 TmfTimestamp delta = new TmfTimestamp(fTraceStartTime.getAdjustment(fTraceEndTime), fScale, 0);
277 fTraceSpan = new TmfTimeRange(new TmfTimestamp(0, fScale, 0), delta);
278 fRangeGroup.setContent(fTraceSpan, delta);
279 }
280
281 /**
282 * @param signal
283 */
284 @TmfSignalHandler
285 public void experimentUpdated(TmfExperimentUpdatedSignal signal) {
286
287 // Update the time frame
288 fTraceTimeRange = signal.getTrace().getTimeRange();
289 fTraceStartTime = fTraceTimeRange.getStartTime();
290 fTraceEndTime = fTraceTimeRange.getEndTime();
291 fScale = fTraceStartTime.getScale();
292
293 // Update the widgets
294 fStartGroup.setContent(fTraceTimeRange, fStartGroup.getCurrentTime());
295 fEndGroup.setContent(fTraceTimeRange, fTraceEndTime);
296 fCurrentGroup.setContent(fTraceTimeRange, fCurrentGroup.getCurrentTime());
297
298 TmfTimestamp delta = new TmfTimestamp(fTraceStartTime.getAdjustment(fTraceEndTime), fScale, 0);
299 fTraceSpan = new TmfTimeRange(new TmfTimestamp(0, fScale, 0), delta);
300 fRangeGroup.setContent(fTraceSpan, delta);
301 }
302
303 /**
304 * @param signal
305 */
306 @TmfSignalHandler
307 public void currentTimeUpdated(TmfTimeSynchSignal signal) {
308 if (signal.getSource() != this) {
309 fCurrentTime = signal.getCurrentTime().synchronize(0, fStartGroup.getCurrentTime().getScale());
310 if (fStartGroup.getCurrentTime().compareTo(fCurrentTime, false) > 0) {
311 fStartGroup.setContent(new TmfTimeRange(fCurrentTime, fEndGroup.getCurrentTime()), fCurrentTime);
312 }
313 if (fEndGroup.getCurrentTime().compareTo(fCurrentTime, false) < 0) {
314 fEndGroup.setContent(new TmfTimeRange(fStartGroup.getCurrentTime(), fCurrentTime), fCurrentTime);
315 }
316 fCurrentGroup.setContent(null, fCurrentTime);
317 updateSlider(fCurrentGroup.getTimeRange(), fCurrentTime);
318 }
319 }
320
321 }
This page took 0.037253 seconds and 5 git commands to generate.