Commit | Line | Data |
---|---|---|
12c155f5 | 1 | /******************************************************************************* |
58ffe079 | 2 | * Copyright (c) 2010, 2015 Ericsson, École Polytechnique de Montréal |
ce2388e0 | 3 | * |
12c155f5 FC |
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 | |
ce2388e0 | 8 | * |
12c155f5 FC |
9 | * Contributors: |
10 | * Francois Chouinard - Initial API and implementation | |
b544077e | 11 | * Bernd Hufmann - Added supplementary files handling |
a72a6830 PT |
12 | * Geneviève Bastien - Moved supplementary files handling to parent class, |
13 | * added code to copy trace | |
14 | * Patrick Tasse - Close editors to release resources | |
4962833a JCK |
15 | * Jean-Christian Kouame - added trace properties to be shown into |
16 | * the properties view | |
8f5221c2 | 17 | * Geneviève Bastien - Moved trace type related methods to parent class |
12c155f5 FC |
18 | *******************************************************************************/ |
19 | ||
2bdf0193 | 20 | package org.eclipse.tracecompass.tmf.ui.project.model; |
12c155f5 | 21 | |
5a5c2fc7 | 22 | import java.util.Arrays; |
b75a5a1b | 23 | import java.util.Date; |
12c155f5 | 24 | import java.util.HashMap; |
6e651d8b | 25 | import java.util.LinkedList; |
c068a752 | 26 | import java.util.List; |
12c155f5 FC |
27 | import java.util.Map; |
28 | ||
b75a5a1b PT |
29 | import org.eclipse.core.filesystem.EFS; |
30 | import org.eclipse.core.filesystem.IFileInfo; | |
c4c81d91 | 31 | import org.eclipse.core.resources.IFile; |
5e4bf87d | 32 | import org.eclipse.core.resources.IFolder; |
12c155f5 FC |
33 | import org.eclipse.core.resources.IResource; |
34 | import org.eclipse.core.runtime.CoreException; | |
35 | import org.eclipse.core.runtime.IConfigurationElement; | |
c068a752 | 36 | import org.eclipse.core.runtime.IPath; |
6e651d8b | 37 | import org.eclipse.core.runtime.IProgressMonitor; |
12c155f5 | 38 | import org.eclipse.core.runtime.Platform; |
13f2a21a | 39 | import org.eclipse.core.runtime.URIUtil; |
b75a5a1b | 40 | import org.eclipse.osgi.util.NLS; |
30c7c7d5 | 41 | import org.eclipse.swt.widgets.Display; |
2bdf0193 | 42 | import org.eclipse.tracecompass.internal.tmf.ui.Activator; |
58ffe079 | 43 | import org.eclipse.tracecompass.internal.tmf.ui.editors.ITmfEventsEditorConstants; |
2bdf0193 AM |
44 | import org.eclipse.tracecompass.tmf.core.TmfCommonConstants; |
45 | import org.eclipse.tracecompass.tmf.core.event.ITmfEvent; | |
46 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomTxtEvent; | |
47 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomTxtTrace; | |
48 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomTxtTraceDefinition; | |
49 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomXmlEvent; | |
50 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomXmlTrace; | |
51 | import org.eclipse.tracecompass.tmf.core.parsers.custom.CustomXmlTraceDefinition; | |
234838b2 | 52 | import org.eclipse.tracecompass.tmf.core.project.model.ITmfPropertiesProvider; |
2bdf0193 AM |
53 | import org.eclipse.tracecompass.tmf.core.project.model.TmfTraceType; |
54 | import org.eclipse.tracecompass.tmf.core.project.model.TraceTypeHelper; | |
55 | import org.eclipse.tracecompass.tmf.core.synchronization.TimestampTransformFactory; | |
56 | import org.eclipse.tracecompass.tmf.core.timestamp.TmfTimestampFormat; | |
57 | import org.eclipse.tracecompass.tmf.core.trace.ITmfTrace; | |
2bdf0193 AM |
58 | import org.eclipse.tracecompass.tmf.core.trace.TmfTraceManager; |
59 | import org.eclipse.tracecompass.tmf.ui.editors.TmfEventsEditor; | |
60 | import org.eclipse.tracecompass.tmf.ui.properties.ReadOnlyTextPropertyDescriptor; | |
12c155f5 FC |
61 | import org.eclipse.ui.IActionFilter; |
62 | import org.eclipse.ui.views.properties.IPropertyDescriptor; | |
63 | import org.eclipse.ui.views.properties.IPropertySource2; | |
12c155f5 | 64 | |
b75a5a1b PT |
65 | import com.ibm.icu.text.DateFormat; |
66 | import com.ibm.icu.text.NumberFormat; | |
67 | ||
12c155f5 | 68 | /** |
d04ec5a7 MK |
69 | * Implementation of trace model element representing a trace. It provides |
70 | * methods to instantiate <code>ITmfTrace</code> and <code>ITmfEvent</code> as | |
71 | * well as editor ID from the trace type extension definition. | |
abbdd66a | 72 | * |
b544077e BH |
73 | * @version 1.0 |
74 | * @author Francois Chouinard | |
12c155f5 | 75 | */ |
16036bc2 | 76 | public class TmfTraceElement extends TmfCommonProjectElement implements IActionFilter, IPropertySource2 { |
12c155f5 FC |
77 | |
78 | // ------------------------------------------------------------------------ | |
79 | // Constants | |
80 | // ------------------------------------------------------------------------ | |
81 | ||
12c155f5 | 82 | // Other attributes |
b544077e BH |
83 | /** |
84 | * Bundle attribute name | |
85 | */ | |
12c155f5 | 86 | public static final String BUNDLE = "bundle"; //$NON-NLS-1$ |
b544077e BH |
87 | /** |
88 | * IsLinked attribute name. | |
89 | */ | |
12c155f5 FC |
90 | public static final String IS_LINKED = "isLinked"; //$NON-NLS-1$ |
91 | ||
92 | // Property View stuff | |
067cd9de BH |
93 | private static final String RESOURCE_PROPERTIES_CATEGORY = Messages.TmfTraceElement_ResourceProperties; |
94 | private static final String NAME = Messages.TmfTraceElement_Name; | |
95 | private static final String PATH = Messages.TmfTraceElement_Path; | |
96 | private static final String LOCATION = Messages.TmfTraceElement_Location; | |
97 | private static final String TRACE_TYPE = Messages.TmfTraceElement_EventType; | |
98 | private static final String IS_LINKED_PROPERTY = Messages.TmfTraceElement_IsLinked; | |
99 | private static final String SOURCE_LOCATION = Messages.TmfTraceElement_SourceLocation; | |
100 | private static final String TIME_OFFSET = Messages.TmfTraceElement_TimeOffset; | |
101 | private static final String LAST_MODIFIED = Messages.TmfTraceElement_LastModified; | |
102 | private static final String SIZE = Messages.TmfTraceElement_Size; | |
103 | private static final String TRACE_PROPERTIES_CATEGORY = Messages.TmfTraceElement_TraceProperties; | |
104 | ||
105 | private static final ReadOnlyTextPropertyDescriptor NAME_DESCRIPTOR = new ReadOnlyTextPropertyDescriptor(NAME, NAME); | |
106 | private static final ReadOnlyTextPropertyDescriptor PATH_DESCRIPTOR = new ReadOnlyTextPropertyDescriptor(PATH, PATH); | |
107 | private static final ReadOnlyTextPropertyDescriptor LOCATION_DESCRIPTOR = new ReadOnlyTextPropertyDescriptor(LOCATION, LOCATION); | |
108 | private static final ReadOnlyTextPropertyDescriptor TYPE_DESCRIPTOR = new ReadOnlyTextPropertyDescriptor(TRACE_TYPE, TRACE_TYPE); | |
109 | private static final ReadOnlyTextPropertyDescriptor IS_LINKED_DESCRIPTOR = new ReadOnlyTextPropertyDescriptor(IS_LINKED_PROPERTY, IS_LINKED_PROPERTY); | |
110 | private static final ReadOnlyTextPropertyDescriptor SOURCE_LOCATION_DESCRIPTOR = new ReadOnlyTextPropertyDescriptor(SOURCE_LOCATION, SOURCE_LOCATION); | |
111 | private static final ReadOnlyTextPropertyDescriptor TIME_OFFSET_DESCRIPTOR = new ReadOnlyTextPropertyDescriptor(TIME_OFFSET, TIME_OFFSET); | |
112 | private static final ReadOnlyTextPropertyDescriptor LAST_MODIFIED_DESCRIPTOR = new ReadOnlyTextPropertyDescriptor(LAST_MODIFIED, LAST_MODIFIED); | |
113 | private static final ReadOnlyTextPropertyDescriptor SIZE_DESCRIPTOR = new ReadOnlyTextPropertyDescriptor(SIZE, SIZE); | |
114 | ||
115 | private static final IPropertyDescriptor[] sfDescriptors = { NAME_DESCRIPTOR, PATH_DESCRIPTOR, LOCATION_DESCRIPTOR, | |
116 | TYPE_DESCRIPTOR, IS_LINKED_DESCRIPTOR, SOURCE_LOCATION_DESCRIPTOR, | |
117 | TIME_OFFSET_DESCRIPTOR, LAST_MODIFIED_DESCRIPTOR, SIZE_DESCRIPTOR }; | |
12c155f5 FC |
118 | |
119 | static { | |
067cd9de BH |
120 | NAME_DESCRIPTOR.setCategory(RESOURCE_PROPERTIES_CATEGORY); |
121 | PATH_DESCRIPTOR.setCategory(RESOURCE_PROPERTIES_CATEGORY); | |
122 | LOCATION_DESCRIPTOR.setCategory(RESOURCE_PROPERTIES_CATEGORY); | |
123 | TYPE_DESCRIPTOR.setCategory(RESOURCE_PROPERTIES_CATEGORY); | |
124 | IS_LINKED_DESCRIPTOR.setCategory(RESOURCE_PROPERTIES_CATEGORY); | |
125 | SOURCE_LOCATION_DESCRIPTOR.setCategory(RESOURCE_PROPERTIES_CATEGORY); | |
126 | TIME_OFFSET_DESCRIPTOR.setCategory(RESOURCE_PROPERTIES_CATEGORY); | |
127 | LAST_MODIFIED_DESCRIPTOR.setCategory(RESOURCE_PROPERTIES_CATEGORY); | |
128 | SIZE_DESCRIPTOR.setCategory(RESOURCE_PROPERTIES_CATEGORY); | |
12c155f5 | 129 | } |
6256d8ad | 130 | |
6b44794a MK |
131 | private static final TmfTimestampFormat OFFSET_FORMAT = new TmfTimestampFormat("T.SSS SSS SSS s"); //$NON-NLS-1$ |
132 | ||
b75a5a1b PT |
133 | private static final int FOLDER_MAX_COUNT = 1024; |
134 | ||
12c155f5 FC |
135 | // ------------------------------------------------------------------------ |
136 | // Static initialization | |
137 | // ------------------------------------------------------------------------ | |
138 | ||
d04ec5a7 MK |
139 | // The mapping of available trace type IDs to their corresponding |
140 | // configuration element | |
067cd9de BH |
141 | private static final Map<String, IConfigurationElement> TRACE_TYPE_ATTRIBUTES = new HashMap<>(); |
142 | private static final Map<String, IConfigurationElement> TRACE_TYPE_UI_ATTRIBUTES = new HashMap<>(); | |
143 | private static final Map<String, IConfigurationElement> TRACE_CATEGORIES = new HashMap<>(); | |
12c155f5 | 144 | |
b544077e | 145 | /** |
d04ec5a7 MK |
146 | * Initialize statically at startup by getting extensions from the platform |
147 | * extension registry. | |
b544077e | 148 | */ |
12c155f5 | 149 | public static void init() { |
a926c25c | 150 | /* Read the tmf.core "tracetype" extension point */ |
4bf17f4a | 151 | IConfigurationElement[] config = Platform.getExtensionRegistry().getConfigurationElementsFor(TmfTraceType.TMF_TRACE_TYPE_ID); |
12c155f5 | 152 | for (IConfigurationElement ce : config) { |
a926c25c AM |
153 | switch (ce.getName()) { |
154 | case TmfTraceType.TYPE_ELEM: | |
4bf17f4a | 155 | String traceTypeId = ce.getAttribute(TmfTraceType.ID_ATTR); |
067cd9de | 156 | TRACE_TYPE_ATTRIBUTES.put(traceTypeId, ce); |
a926c25c AM |
157 | break; |
158 | case TmfTraceType.CATEGORY_ELEM: | |
4bf17f4a | 159 | String categoryId = ce.getAttribute(TmfTraceType.ID_ATTR); |
067cd9de | 160 | TRACE_CATEGORIES.put(categoryId, ce); |
a926c25c AM |
161 | break; |
162 | default: | |
163 | } | |
164 | } | |
165 | ||
166 | /* | |
167 | * Read the corresponding tmf.ui "tracetypeui" extension point for this | |
168 | * trace type, if it exists. | |
169 | */ | |
170 | config = Platform.getExtensionRegistry().getConfigurationElementsFor(TmfTraceTypeUIUtils.TMF_TRACE_TYPE_UI_ID); | |
171 | for (IConfigurationElement ce : config) { | |
172 | String elemName = ce.getName(); | |
173 | if (TmfTraceTypeUIUtils.TYPE_ELEM.equals(elemName)) { | |
174 | String traceType = ce.getAttribute(TmfTraceTypeUIUtils.TRACETYPE_ATTR); | |
067cd9de | 175 | TRACE_TYPE_UI_ATTRIBUTES.put(traceType, ce); |
12c155f5 FC |
176 | } |
177 | } | |
178 | } | |
179 | ||
b75a5a1b PT |
180 | // ------------------------------------------------------------------------ |
181 | // Classes | |
182 | // ------------------------------------------------------------------------ | |
183 | ||
184 | private class FileInfo { | |
185 | long lastModified; | |
186 | long size; | |
187 | int count; | |
188 | } | |
189 | ||
190 | // ------------------------------------------------------------------------ | |
191 | // Attributes | |
192 | // ------------------------------------------------------------------------ | |
193 | ||
194 | private FileInfo fFileInfo; | |
195 | ||
12c155f5 FC |
196 | // ------------------------------------------------------------------------ |
197 | // Constructors | |
198 | // ------------------------------------------------------------------------ | |
b544077e | 199 | /** |
d04ec5a7 MK |
200 | * Constructor. Creates trace model element under the trace folder. |
201 | * | |
202 | * @param name | |
203 | * The name of trace | |
204 | * @param trace | |
205 | * The trace resource. | |
206 | * @param parent | |
207 | * The parent element (trace folder) | |
b544077e | 208 | */ |
12c155f5 | 209 | public TmfTraceElement(String name, IResource trace, TmfTraceFolder parent) { |
339d539c | 210 | super(name, trace, parent); |
12c155f5 | 211 | } |
d04ec5a7 | 212 | |
b544077e | 213 | /** |
d04ec5a7 MK |
214 | * Constructor. Creates trace model element under the experiment folder. |
215 | * | |
216 | * @param name | |
217 | * The name of trace | |
218 | * @param trace | |
219 | * The trace resource. | |
220 | * @param parent | |
221 | * The parent element (experiment folder) | |
b544077e | 222 | */ |
12c155f5 | 223 | public TmfTraceElement(String name, IResource trace, TmfExperimentElement parent) { |
12c155f5 | 224 | super(name, trace, parent); |
f537c959 PT |
225 | } |
226 | ||
12c155f5 FC |
227 | // ------------------------------------------------------------------------ |
228 | // Operations | |
229 | // ------------------------------------------------------------------------ | |
12c155f5 | 230 | |
b544077e | 231 | /** |
d04ec5a7 MK |
232 | * Instantiate a <code>ITmfTrace</code> object based on the trace type and |
233 | * the corresponding extension. | |
abbdd66a | 234 | * |
b544077e BH |
235 | * @return the <code>ITmfTrace</code> or <code>null</code> for an error |
236 | */ | |
8f5221c2 | 237 | @Override |
6256d8ad | 238 | public ITmfTrace instantiateTrace() { |
12c155f5 | 239 | try { |
e12ecd30 BH |
240 | |
241 | // make sure that supplementary folder exists | |
242 | refreshSupplementaryFolder(); | |
243 | ||
c9b31f60 BH |
244 | String traceTypeId = getTraceType(); |
245 | if (traceTypeId != null) { | |
246 | if (CustomTxtTrace.isCustomTraceTypeId(traceTypeId)) { | |
4bf17f4a | 247 | for (CustomTxtTraceDefinition def : CustomTxtTraceDefinition.loadAll()) { |
c9b31f60 BH |
248 | String id = CustomTxtTrace.buildTraceTypeId(def.categoryName, def.definitionName); |
249 | if (traceTypeId.equals(id)) { | |
4bf17f4a | 250 | return new CustomTxtTrace(def); |
251 | } | |
252 | } | |
253 | } | |
c9b31f60 | 254 | if (CustomXmlTrace.isCustomTraceTypeId(traceTypeId)) { |
4bf17f4a | 255 | for (CustomXmlTraceDefinition def : CustomXmlTraceDefinition.loadAll()) { |
c9b31f60 BH |
256 | String id = CustomXmlTrace.buildTraceTypeId(def.categoryName, def.definitionName); |
257 | if (traceTypeId.equals(id)) { | |
4bf17f4a | 258 | return new CustomXmlTrace(def); |
259 | } | |
260 | } | |
261 | } | |
067cd9de | 262 | IConfigurationElement ce = TRACE_TYPE_ATTRIBUTES.get(traceTypeId); |
d04ec5a7 MK |
263 | if (ce == null) { |
264 | return null; | |
265 | } | |
6256d8ad | 266 | ITmfTrace trace = (ITmfTrace) ce.createExecutableExtension(TmfTraceType.TRACE_TYPE_ATTR); |
12c155f5 FC |
267 | return trace; |
268 | } | |
269 | } catch (CoreException e) { | |
8fd82db5 | 270 | Activator.getDefault().logError("Error instantiating ITmfTrace object for trace " + getName(), e); //$NON-NLS-1$ |
12c155f5 FC |
271 | } |
272 | return null; | |
273 | } | |
274 | ||
b544077e | 275 | /** |
d04ec5a7 MK |
276 | * Instantiate a <code>ITmfEvent</code> object based on the trace type and |
277 | * the corresponding extension. | |
abbdd66a | 278 | * |
b544077e BH |
279 | * @return the <code>ITmfEvent</code> or <code>null</code> for an error |
280 | */ | |
ce2388e0 | 281 | public ITmfEvent instantiateEvent() { |
12c155f5 | 282 | try { |
c9b31f60 BH |
283 | String traceTypeId = getTraceType(); |
284 | if (traceTypeId != null) { | |
285 | if (CustomTxtTrace.isCustomTraceTypeId(traceTypeId)) { | |
4bf17f4a | 286 | for (CustomTxtTraceDefinition def : CustomTxtTraceDefinition.loadAll()) { |
c9b31f60 BH |
287 | String id = CustomTxtTrace.buildTraceTypeId(def.categoryName, def.definitionName); |
288 | if (traceTypeId.equals(id)) { | |
4bf17f4a | 289 | return new CustomTxtEvent(def); |
290 | } | |
291 | } | |
292 | } | |
c9b31f60 | 293 | if (CustomXmlTrace.isCustomTraceTypeId(traceTypeId)) { |
4bf17f4a | 294 | for (CustomXmlTraceDefinition def : CustomXmlTraceDefinition.loadAll()) { |
c9b31f60 BH |
295 | String id = CustomXmlTrace.buildTraceTypeId(def.categoryName, def.definitionName); |
296 | if (traceTypeId.equals(id)) { | |
4bf17f4a | 297 | return new CustomXmlEvent(def); |
298 | } | |
299 | } | |
300 | } | |
067cd9de | 301 | IConfigurationElement ce = TRACE_TYPE_ATTRIBUTES.get(traceTypeId); |
d04ec5a7 MK |
302 | if (ce == null) { |
303 | return null; | |
304 | } | |
ce2388e0 | 305 | ITmfEvent event = (ITmfEvent) ce.createExecutableExtension(TmfTraceType.EVENT_TYPE_ATTR); |
12c155f5 FC |
306 | return event; |
307 | } | |
308 | } catch (CoreException e) { | |
8fd82db5 | 309 | Activator.getDefault().logError("Error instantiating ITmfEvent object for trace " + getName(), e); //$NON-NLS-1$ |
12c155f5 FC |
310 | } |
311 | return null; | |
312 | } | |
313 | ||
8f5221c2 | 314 | @Override |
12c155f5 | 315 | public String getEditorId() { |
c9b31f60 BH |
316 | String traceTypeId = getTraceType(); |
317 | if (traceTypeId != null) { | |
318 | if (CustomTxtTrace.isCustomTraceTypeId(traceTypeId) || CustomXmlTrace.isCustomTraceTypeId(traceTypeId)) { | |
4bf17f4a | 319 | return TmfEventsEditor.ID; |
320 | } | |
c9b31f60 | 321 | |
067cd9de | 322 | IConfigurationElement ce = TRACE_TYPE_UI_ATTRIBUTES.get(getTraceType()); |
a926c25c AM |
323 | if (ce == null) { |
324 | /* This trace type does not define UI attributes */ | |
325 | return null; | |
326 | } | |
327 | IConfigurationElement[] defaultEditorCE = ce.getChildren(TmfTraceTypeUIUtils.DEFAULT_EDITOR_ELEM); | |
12c155f5 | 328 | if (defaultEditorCE.length == 1) { |
4bf17f4a | 329 | return defaultEditorCE[0].getAttribute(TmfTraceType.ID_ATTR); |
12c155f5 FC |
330 | } |
331 | } | |
332 | return null; | |
333 | } | |
e12ecd30 | 334 | |
c4c81d91 | 335 | /** |
d04ec5a7 MK |
336 | * Returns the file resource used to store bookmarks after creating it if |
337 | * necessary. If the trace resource is a file, it is returned directly. If | |
338 | * the trace resource is a folder, a linked file is returned. The file will | |
339 | * be created if it does not exist. | |
340 | * | |
c4c81d91 | 341 | * @return the bookmarks file |
d04ec5a7 MK |
342 | * @throws CoreException |
343 | * if the bookmarks file cannot be created | |
c4c81d91 | 344 | */ |
8f5221c2 | 345 | @Override |
81fe3479 PT |
346 | public IFile createBookmarksFile() throws CoreException { |
347 | IFile file = getBookmarksFile(); | |
b3e4798c | 348 | if (getResource() instanceof IFolder) { |
58ffe079 | 349 | return createBookmarksFile(getProject().getTracesFolder().getResource(), ITmfEventsEditorConstants.TRACE_EDITOR_INPUT_TYPE); |
81fe3479 PT |
350 | } |
351 | return file; | |
352 | } | |
353 | ||
354 | /** | |
d04ec5a7 MK |
355 | * Returns the file resource used to store bookmarks. The file may not |
356 | * exist. | |
357 | * | |
81fe3479 | 358 | * @return the bookmarks file |
81fe3479 | 359 | */ |
8f5221c2 | 360 | @Override |
81fe3479 | 361 | public IFile getBookmarksFile() { |
c4c81d91 | 362 | IFile file = null; |
b3e4798c AM |
363 | IResource resource = getResource(); |
364 | if (resource instanceof IFile) { | |
365 | file = (IFile) resource; | |
366 | } else if (resource instanceof IFolder) { | |
367 | final IFolder folder = (IFolder) resource; | |
c4c81d91 | 368 | file = folder.getFile(getName() + '_'); |
c4c81d91 PT |
369 | } |
370 | return file; | |
371 | } | |
372 | ||
e12ecd30 | 373 | /** |
d04ec5a7 MK |
374 | * Returns the <code>TmfTraceElement</code> located under the |
375 | * <code>TmfTracesFolder</code>. | |
6256d8ad | 376 | * |
d04ec5a7 MK |
377 | * @return <code>this</code> if this element is under the |
378 | * <code>TmfTracesFolder</code> else the corresponding | |
379 | * <code>TmfTraceElement</code> if this element is under | |
e12ecd30 BH |
380 | * <code>TmfExperimentElement</code>. |
381 | */ | |
5e4bf87d | 382 | public TmfTraceElement getElementUnderTraceFolder() { |
e12ecd30 | 383 | |
d04ec5a7 MK |
384 | // If trace is under an experiment, return original trace from the |
385 | // traces folder | |
5e4bf87d BH |
386 | if (getParent() instanceof TmfExperimentElement) { |
387 | for (TmfTraceElement aTrace : getProject().getTracesFolder().getTraces()) { | |
339d539c | 388 | if (aTrace.getElementPath().equals(getElementPath())) { |
5e4bf87d BH |
389 | return aTrace; |
390 | } | |
391 | } | |
392 | } | |
393 | return this; | |
394 | } | |
6256d8ad | 395 | |
8f5221c2 GB |
396 | @Override |
397 | public String getTypeName() { | |
398 | return Messages.TmfTraceElement_TypeName; | |
399 | } | |
400 | ||
12c155f5 FC |
401 | // ------------------------------------------------------------------------ |
402 | // IActionFilter | |
403 | // ------------------------------------------------------------------------ | |
404 | ||
405 | @Override | |
406 | public boolean testAttribute(Object target, String name, String value) { | |
407 | if (name.equals(IS_LINKED)) { | |
408 | boolean isLinked = getResource().isLinked(); | |
409 | return Boolean.toString(isLinked).equals(value); | |
410 | } | |
411 | return false; | |
412 | } | |
413 | ||
12c155f5 FC |
414 | // ------------------------------------------------------------------------ |
415 | // IPropertySource2 | |
416 | // ------------------------------------------------------------------------ | |
417 | ||
418 | @Override | |
419 | public Object getEditableValue() { | |
420 | return null; | |
421 | } | |
422 | ||
4962833a JCK |
423 | /** |
424 | * Get the trace properties of this traceElement if the corresponding trace | |
425 | * is opened in an editor | |
426 | * | |
427 | * @return a map with the names and values of the trace properties | |
428 | * respectively as keys and values | |
429 | */ | |
430 | private Map<String, String> getTraceProperties() { | |
8d534d69 JCK |
431 | for (ITmfTrace openedTrace : TmfTraceManager.getInstance().getOpenedTraces()) { |
432 | for (ITmfTrace singleTrace : TmfTraceManager.getTraceSet(openedTrace)) { | |
6e74bac3 | 433 | if (getElementUnderTraceFolder().getResource().equals(singleTrace.getResource())) { |
234838b2 GB |
434 | if (singleTrace instanceof ITmfPropertiesProvider) { |
435 | ITmfPropertiesProvider traceProperties = (ITmfPropertiesProvider) singleTrace; | |
436 | return traceProperties.getProperties(); | |
8d534d69 | 437 | } |
4962833a JCK |
438 | } |
439 | } | |
440 | } | |
507b1336 | 441 | return new HashMap<>(); |
4962833a JCK |
442 | } |
443 | ||
12c155f5 FC |
444 | @Override |
445 | public IPropertyDescriptor[] getPropertyDescriptors() { | |
4962833a JCK |
446 | Map<String, String> traceProperties = getTraceProperties(); |
447 | if (!traceProperties.isEmpty()) { | |
448 | IPropertyDescriptor[] propertyDescriptorArray = new IPropertyDescriptor[traceProperties.size() + sfDescriptors.length]; | |
449 | int index = 0; | |
450 | for (Map.Entry<String, String> varName : traceProperties.entrySet()) { | |
451 | ReadOnlyTextPropertyDescriptor descriptor = new ReadOnlyTextPropertyDescriptor(this.getName() + "_" + varName.getKey(), varName.getKey()); //$NON-NLS-1$ | |
067cd9de | 452 | descriptor.setCategory(TRACE_PROPERTIES_CATEGORY); |
4962833a JCK |
453 | propertyDescriptorArray[index] = descriptor; |
454 | index++; | |
455 | } | |
456 | for (int i = 0; i < sfDescriptors.length; i++) { | |
457 | propertyDescriptorArray[index] = sfDescriptors[i]; | |
458 | index++; | |
459 | } | |
460 | return propertyDescriptorArray; | |
461 | } | |
77fdc5df | 462 | return Arrays.copyOf(sfDescriptors, sfDescriptors.length); |
12c155f5 FC |
463 | } |
464 | ||
465 | @Override | |
466 | public Object getPropertyValue(Object id) { | |
467 | ||
067cd9de | 468 | if (NAME.equals(id)) { |
12c155f5 | 469 | return getName(); |
ce2388e0 | 470 | } |
12c155f5 | 471 | |
067cd9de | 472 | if (PATH.equals(id)) { |
12c155f5 | 473 | return getPath().toString(); |
ce2388e0 | 474 | } |
12c155f5 | 475 | |
067cd9de | 476 | if (LOCATION.equals(id)) { |
13f2a21a | 477 | return URIUtil.toUnencodedString(getLocation()); |
ce2388e0 | 478 | } |
12c155f5 | 479 | |
067cd9de | 480 | if (IS_LINKED_PROPERTY.equals(id)) { |
933ff94f PT |
481 | return Boolean.valueOf(getResource().isLinked()).toString(); |
482 | } | |
483 | ||
067cd9de | 484 | if (SOURCE_LOCATION.equals(id)) { |
89730b51 PT |
485 | try { |
486 | String sourceLocation = getElementUnderTraceFolder().getResource().getPersistentProperty(TmfCommonConstants.SOURCE_LOCATION); | |
487 | if (sourceLocation != null) { | |
488 | return sourceLocation; | |
489 | } | |
490 | } catch (CoreException e) { | |
491 | } | |
492 | return ""; //$NON-NLS-1$ | |
ce2388e0 | 493 | } |
12c155f5 | 494 | |
067cd9de | 495 | if (LAST_MODIFIED.equals(id)) { |
b75a5a1b PT |
496 | FileInfo fileInfo = getFileInfo(); |
497 | if (fileInfo == null) { | |
498 | return ""; //$NON-NLS-1$ | |
499 | } | |
500 | long date = fileInfo.lastModified; | |
501 | DateFormat format = DateFormat.getDateTimeInstance(DateFormat.LONG, DateFormat.MEDIUM); | |
502 | return format.format(new Date(date)); | |
503 | } | |
504 | ||
067cd9de | 505 | if (SIZE.equals(id)) { |
b75a5a1b PT |
506 | FileInfo fileInfo = getFileInfo(); |
507 | if (fileInfo == null) { | |
508 | return ""; //$NON-NLS-1$ | |
509 | } | |
510 | if (getResource() instanceof IFolder) { | |
511 | if (fileInfo.count <= FOLDER_MAX_COUNT) { | |
512 | return NLS.bind(Messages.TmfTraceElement_FolderSizeString, | |
513 | NumberFormat.getInstance().format(fileInfo.size), fileInfo.count); | |
514 | } | |
515 | return NLS.bind(Messages.TmfTraceElement_FolderSizeOverflowString, | |
516 | NumberFormat.getInstance().format(fileInfo.size), FOLDER_MAX_COUNT); | |
517 | } | |
518 | return NLS.bind(Messages.TmfTraceElement_FileSizeString, NumberFormat.getInstance().format(fileInfo.size)); | |
519 | } | |
520 | ||
067cd9de | 521 | if (TRACE_TYPE.equals(id)) { |
8f5221c2 | 522 | if (getTraceType() != null) { |
4b3b667b | 523 | TraceTypeHelper helper = TmfTraceType.getTraceType(getTraceType()); |
2918b427 | 524 | if (helper != null) { |
2b0005f0 | 525 | return helper.getLabel(); |
2918b427 | 526 | } |
12c155f5 | 527 | } |
339d539c | 528 | return ""; //$NON-NLS-1$ |
12c155f5 FC |
529 | } |
530 | ||
067cd9de | 531 | if (TIME_OFFSET.equals(id)) { |
6b44794a MK |
532 | long offset = TimestampTransformFactory.getTimestampTransform(getElementUnderTraceFolder().getResource()).transform(0); |
533 | if (offset != 0) { | |
534 | return OFFSET_FORMAT.format(offset); | |
535 | } | |
536 | return ""; //$NON-NLS-1$ | |
537 | } | |
538 | ||
4962833a JCK |
539 | Map<String, String> traceProperties = getTraceProperties(); |
540 | if (id != null && !traceProperties.isEmpty()) { | |
541 | String key = (String) id; | |
e48bd720 | 542 | key = key.substring(this.getName().length() + 1); // remove name_ |
4962833a JCK |
543 | String value = traceProperties.get(key); |
544 | return value; | |
545 | } | |
546 | ||
12c155f5 FC |
547 | return null; |
548 | } | |
549 | ||
b75a5a1b PT |
550 | private FileInfo getFileInfo() { |
551 | /* FileInfo is needed for both 'last modified' and 'size' properties. | |
552 | * It is freshly computed for one, and reused for the other, then | |
553 | * cleared so that the information can be refreshed the next time. | |
554 | */ | |
555 | FileInfo fileInfo; | |
556 | if (fFileInfo == null) { | |
557 | try { | |
558 | fileInfo = computeFileInfo(new FileInfo(), getResource()); | |
559 | } catch (CoreException e) { | |
560 | return null; | |
561 | } | |
562 | fFileInfo = fileInfo; | |
563 | } else { | |
564 | fileInfo = fFileInfo; | |
565 | fFileInfo = null; | |
566 | } | |
567 | return fileInfo; | |
568 | } | |
569 | ||
570 | private FileInfo computeFileInfo(FileInfo fileInfo, IResource resource) throws CoreException { | |
571 | if (fileInfo == null || fileInfo.count > FOLDER_MAX_COUNT) { | |
572 | return fileInfo; | |
573 | } | |
574 | if (resource instanceof IFolder) { | |
575 | IFolder folder = (IFolder) resource; | |
576 | for (IResource member : folder.members()) { | |
577 | computeFileInfo(fileInfo, member); | |
578 | } | |
579 | return fileInfo; | |
580 | } | |
581 | IFileInfo info = EFS.getStore(resource.getLocationURI()).fetchInfo(); | |
582 | fileInfo.lastModified = Math.max(fileInfo.lastModified, info.getLastModified()); | |
583 | fileInfo.size += info.getLength(); | |
584 | fileInfo.count++; | |
585 | return fileInfo; | |
586 | } | |
587 | ||
12c155f5 FC |
588 | @Override |
589 | public void resetPropertyValue(Object id) { | |
590 | } | |
591 | ||
592 | @Override | |
593 | public void setPropertyValue(Object id, Object value) { | |
594 | } | |
595 | ||
596 | @Override | |
597 | public boolean isPropertyResettable(Object id) { | |
598 | return false; | |
599 | } | |
600 | ||
601 | @Override | |
602 | public boolean isPropertySet(Object id) { | |
603 | return false; | |
604 | } | |
605 | ||
beb19106 GB |
606 | /** |
607 | * Copy this trace in the trace folder. No other parameters are mentioned so | |
608 | * the trace is copied in this element's project trace folder | |
609 | * | |
f3cbd37d | 610 | * @param newName |
beb19106 GB |
611 | * The new trace name |
612 | * @return the new Resource object | |
beb19106 | 613 | */ |
f3cbd37d PT |
614 | public TmfTraceElement copy(String newName) { |
615 | TmfTraceFolder folder = (TmfTraceFolder) getParent(); | |
616 | IResource res = super.copy(newName, false); | |
617 | for (TmfTraceElement trace : folder.getTraces()) { | |
618 | if (trace.getResource().equals(res)) { | |
619 | return trace; | |
620 | } | |
621 | } | |
622 | return null; | |
beb19106 GB |
623 | } |
624 | ||
a72a6830 PT |
625 | /** |
626 | * Close opened editors associated with this trace. | |
a72a6830 | 627 | */ |
8f5221c2 | 628 | @Override |
a72a6830 | 629 | public void closeEditors() { |
8f5221c2 | 630 | super.closeEditors(); |
a72a6830 PT |
631 | |
632 | // Close experiments that contain the trace if open | |
633 | if (getParent() instanceof TmfTraceFolder) { | |
339d539c PT |
634 | TmfExperimentFolder experimentsFolder = getProject().getExperimentsFolder(); |
635 | for (TmfExperimentElement experiment : experimentsFolder.getExperiments()) { | |
636 | for (TmfTraceElement trace : experiment.getTraces()) { | |
637 | if (trace.getElementPath().equals(getElementPath())) { | |
638 | experiment.closeEditors(); | |
a72a6830 PT |
639 | break; |
640 | } | |
641 | } | |
642 | } | |
643 | } else if (getParent() instanceof TmfExperimentElement) { | |
644 | TmfExperimentElement experiment = (TmfExperimentElement) getParent(); | |
645 | experiment.closeEditors(); | |
646 | } | |
647 | } | |
8dcdf263 | 648 | |
6e651d8b MAL |
649 | /** |
650 | * Delete the trace resource, remove it from experiments and delete its | |
651 | * supplementary files | |
652 | * | |
653 | * @param progressMonitor | |
654 | * a progress monitor, or null if progress reporting is not | |
655 | * desired | |
656 | * | |
657 | * @throws CoreException | |
658 | * thrown when IResource.delete fails | |
6e651d8b MAL |
659 | */ |
660 | public void delete(IProgressMonitor progressMonitor) throws CoreException { | |
30c7c7d5 MAL |
661 | // Close editors in UI Thread |
662 | Display.getDefault().syncExec(new Runnable() { | |
663 | @Override | |
664 | public void run() { | |
665 | closeEditors(); | |
666 | } | |
667 | }); | |
6e651d8b | 668 | |
b3e4798c | 669 | IPath path = getResource().getLocation(); |
2bb6a6b8 BH |
670 | if (path != null) { |
671 | if (getParent() instanceof TmfTraceFolder) { | |
672 | TmfExperimentFolder experimentFolder = getProject().getExperimentsFolder(); | |
673 | ||
674 | // Propagate the removal to traces | |
339d539c PT |
675 | for (TmfExperimentElement experiment : experimentFolder.getExperiments()) { |
676 | List<TmfTraceElement> toRemove = new LinkedList<>(); | |
677 | for (TmfTraceElement trace : experiment.getTraces()) { | |
678 | if (trace.getElementPath().equals(getElementPath())) { | |
679 | toRemove.add(trace); | |
2bb6a6b8 BH |
680 | } |
681 | } | |
339d539c PT |
682 | for (TmfTraceElement child : toRemove) { |
683 | experiment.removeTrace(child); | |
6e651d8b MAL |
684 | } |
685 | } | |
6e651d8b | 686 | |
2bb6a6b8 BH |
687 | // Delete supplementary files |
688 | deleteSupplementaryFolder(); | |
689 | ||
690 | } else if (getParent() instanceof TmfExperimentElement) { | |
691 | TmfExperimentElement experimentElement = (TmfExperimentElement) getParent(); | |
692 | experimentElement.removeTrace(this); | |
693 | } | |
6e651d8b MAL |
694 | } |
695 | ||
696 | // Finally, delete the trace | |
b3e4798c | 697 | getResource().delete(true, progressMonitor); |
6e651d8b MAL |
698 | } |
699 | ||
12c155f5 | 700 | } |