1 /*******************************************************************************
2 * Copyright (c) 2009, 2010 Ericsson
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
10 * Francois Chouinard - Initial API and implementation
11 *******************************************************************************/
13 package org
.eclipse
.linuxtools
.tmf
.ui
.views
;
15 import org
.eclipse
.linuxtools
.tmf
.event
.TmfEvent
;
16 import org
.eclipse
.linuxtools
.tmf
.event
.TmfTimestamp
;
17 import org
.eclipse
.linuxtools
.tmf
.experiment
.TmfExperiment
;
18 import org
.eclipse
.linuxtools
.tmf
.request
.TmfDataRequest
;
19 import org
.eclipse
.linuxtools
.tmf
.signal
.TmfExperimentSelectedSignal
;
20 import org
.eclipse
.linuxtools
.tmf
.signal
.TmfExperimentUpdatedSignal
;
21 import org
.eclipse
.linuxtools
.tmf
.signal
.TmfSignalHandler
;
22 import org
.eclipse
.linuxtools
.tmf
.signal
.TmfTimeSynchSignal
;
23 import org
.eclipse
.swt
.SWT
;
24 import org
.eclipse
.swt
.events
.SelectionAdapter
;
25 import org
.eclipse
.swt
.events
.SelectionEvent
;
26 import org
.eclipse
.swt
.layout
.GridData
;
27 import org
.eclipse
.swt
.widgets
.Composite
;
28 import org
.eclipse
.swt
.widgets
.Event
;
29 import org
.eclipse
.swt
.widgets
.Listener
;
30 import org
.eclipse
.swt
.widgets
.Table
;
31 import org
.eclipse
.swt
.widgets
.TableColumn
;
32 import org
.eclipse
.swt
.widgets
.TableItem
;
35 * <b><u>TmfEventsView</u></b>
38 * TODO: Implement me. Please.
39 * TODO: Handle column selection, sort, ... generically (nothing less...)
40 * TODO: Implement hide/display columns
42 public class TmfEventsView
extends TmfView
{
44 public static final String ID
= "org.eclipse.linuxtools.tmf.ui.views.events";
46 private TmfExperiment
<TmfEvent
> fExperiment
;
47 private String fTitlePrefix
;
49 // ------------------------------------------------------------------------
51 // ------------------------------------------------------------------------
56 private final String TIMESTAMP_COLUMN
= "Timestamp";
57 private final String SOURCE_COLUMN
= "Source";
58 private final String TYPE_COLUMN
= "Type";
59 private final String REFERENCE_COLUMN
= "File";
60 private final String CONTENT_COLUMN
= "Content";
61 private final String
[] columnProperties
= new String
[] {
70 private class ColumnData
{
71 public final String header
;
72 public final int width
;
73 public final int alignment
;
75 public ColumnData(String h
, int w
, int a
) {
82 private ColumnData
[] columnData
= new ColumnData
[] {
83 new ColumnData(columnProperties
[0], 100, SWT
.LEFT
),
84 new ColumnData(columnProperties
[1], 100, SWT
.LEFT
),
85 new ColumnData(columnProperties
[2], 100, SWT
.LEFT
),
86 new ColumnData(columnProperties
[3], 100, SWT
.LEFT
),
87 new ColumnData(columnProperties
[4], 100, SWT
.LEFT
)
90 // ------------------------------------------------------------------------
92 // ------------------------------------------------------------------------
94 private static final int DEFAULT_CACHE_SIZE
= 1000;
95 private final int fCacheSize
;
96 private TmfEvent
[] cache
= new TmfEvent
[1];
97 private int cacheStartIndex
= 0;
98 private int cacheEndIndex
= 0;
100 // ------------------------------------------------------------------------
102 // ------------------------------------------------------------------------
104 public TmfEventsView(int cacheSize
) {
105 super("TmfEventsView");
106 fCacheSize
= cacheSize
;
109 public TmfEventsView() {
110 this(DEFAULT_CACHE_SIZE
);
113 // ------------------------------------------------------------------------
115 // ------------------------------------------------------------------------
117 @SuppressWarnings("unchecked")
119 public void createPartControl(Composite parent
) {
121 // Create a virtual table
122 // TODO: change SINGLE to MULTI line selection and adjust the selection listener
123 final int style
= SWT
.SINGLE
| SWT
.H_SCROLL
| SWT
.V_SCROLL
| SWT
.BORDER
| SWT
.VIRTUAL
;
124 fTable
= new Table(parent
, style
);
126 // Set the table layout
127 GridData layoutData
= new GridData(SWT
.FILL
, SWT
.FILL
, true, true);
128 fTable
.setLayoutData(layoutData
);
130 // Some cosmetic enhancements
131 fTable
.setHeaderVisible(true);
132 fTable
.setLinesVisible(true);
135 createColumnHeaders(fTable
);
137 // Handle the table item requests
138 fTable
.addSelectionListener(new SelectionAdapter() {
141 public void widgetSelected(SelectionEvent e
) {
142 TmfTimestamp ts
= (TmfTimestamp
) fTable
.getSelection()[0].getData();
143 broadcast(new TmfTimeSynchSignal(fTable
, ts
));
147 // Handle the table item requests
148 fTable
.addListener(SWT
.SetData
, new Listener() {
150 public void handleEvent(Event event
) {
152 TableItem item
= (TableItem
) event
.item
;
153 final int index
= fTable
.indexOf(item
);
155 // Note: this works because handleEvent() is called once for each row, in sequence
156 if ((index
>= cacheStartIndex
) && (index
< cacheEndIndex
)) {
157 int i
= index
- cacheStartIndex
;
158 item
.setText(extractItemFields(cache
[i
]));
159 item
.setData(new TmfTimestamp(cache
[i
].getTimestamp()));
163 TmfDataRequest
<TmfEvent
> request
= new TmfDataRequest
<TmfEvent
>(TmfEvent
.class, index
, fCacheSize
) {
165 public void handleData() {
166 TmfEvent
[] tmpEvent
= getData();
167 if ((tmpEvent
!= null) && (tmpEvent
.length
> 0)) {
169 cacheStartIndex
= index
;
170 cacheEndIndex
= index
+ tmpEvent
.length
;
174 fExperiment
.sendRequest(request
);
176 request
.waitForCompletion();
177 } catch (InterruptedException e
) {
181 if (cache
[0] != null && cacheStartIndex
== index
) {
182 item
.setText(extractItemFields(cache
[0]));
183 item
.setData(new TmfTimestamp(cache
[0].getTimestamp()));
189 fTable
.setItemCount(0);
190 fTitlePrefix
= getTitle();
192 // If an experiment is already selected, update the table
193 fExperiment
= (TmfExperiment
<TmfEvent
>) TmfExperiment
.getCurrentExperiment();
194 if (fExperiment
!= null) {
195 experimentSelected(new TmfExperimentSelectedSignal
<TmfEvent
>(fTable
, fExperiment
));
202 * FIXME: Add support for column selection
204 protected void createColumnHeaders(Table table
) {
205 for (int i
= 0; i
< columnData
.length
; i
++) {
206 TableColumn column
= new TableColumn(table
, columnData
[i
].alignment
, i
);
207 column
.setText(columnData
[i
].header
);
208 column
.setWidth(columnData
[i
].width
);
216 * FIXME: Add support for column selection
218 protected String
[] extractItemFields(TmfEvent event
) {
219 String
[] fields
= new String
[0];
221 fields
= new String
[] {
222 new Long(event
.getTimestamp().getValue()).toString(),
223 event
.getSource().getSourceId().toString(),
224 event
.getType().getTypeId().toString(),
225 event
.getReference().getReference().toString(),
226 event
.getContent().toString()
233 * @see org.eclipse.ui.part.WorkbenchPart#setFocus()
236 public void setFocus() {
240 * @see java.lang.Object#toString()
243 public String
toString() {
244 return "[TmfEventsView]";
247 // ------------------------------------------------------------------------
249 // ------------------------------------------------------------------------
251 @SuppressWarnings("unchecked")
253 public void experimentSelected(TmfExperimentSelectedSignal
<TmfEvent
> signal
) {
254 // Update the trace reference
255 fExperiment
= (TmfExperiment
<TmfEvent
>) signal
.getExperiment();
256 setPartName(fTitlePrefix
+ " - " + fExperiment
.getName());
258 // Perform the updates on the UI thread
259 fTable
.getDisplay().asyncExec(new Runnable() {
261 // TODO: Potentially long operation. Add some feedback for the user
262 fTable
.setSelection(0);
264 cacheStartIndex
= cacheEndIndex
= 0; // Clear the cache
265 fTable
.setItemCount((int) fExperiment
.getNbEvents());
271 public void experimentUpdated(TmfExperimentUpdatedSignal signal
) {
272 // Perform the refresh on the UI thread
273 fTable
.getDisplay().asyncExec(new Runnable() {
275 if (!fTable
.isDisposed() && fExperiment
!= null) {
276 fTable
.setItemCount((int) fExperiment
.getNbEvents());
283 // public void currentTimeUpdated(TmfTimeSynchSignal signal) {
284 // if (signal.getSource() != fTable && fExperiment != null) {
285 // final int index = (int) fExperiment.getRank(signal.getCurrentTime());
286 // // Perform the updates on the UI thread
287 // fTable.getDisplay().asyncExec(new Runnable() {
288 // public void run() {
289 // fTable.setSelection(index);
290 // // The timestamp might not correspond to an actual event
291 // // and the selection will point to the next experiment event.
292 // // But we would like to display both the event before and
293 // // after the selected timestamp.
294 // // This works fine by default except when the selected event
295 // // is the top displayed event. The following ensures that we
296 // // always see both events.
297 // if ((index > 0) && (index == fTable.getTopIndex())) {
298 // fTable.setTopIndex(index - 1);