/******************************************************************************* * Copyright (c) 2015 MINRES Technologies GmbH and others. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * MINRES Technologies GmbH - initial API and implementation *******************************************************************************/ package com.minres.scviewer.e4.application.parts; import java.beans.PropertyChangeEvent; import java.beans.PropertyChangeListener; import java.io.File; import java.io.IOException; import java.util.ArrayList; import java.util.HashMap; import java.util.LinkedList; import java.util.List; import java.util.Map; import java.util.regex.Matcher; import java.util.regex.Pattern; import javax.annotation.PostConstruct; import javax.inject.Inject; import javax.inject.Named; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.core.runtime.IStatus; import org.eclipse.core.runtime.Status; import org.eclipse.core.runtime.SubMonitor; import org.eclipse.core.runtime.jobs.IJobChangeEvent; import org.eclipse.core.runtime.jobs.Job; import org.eclipse.core.runtime.jobs.JobChangeAdapter; import org.eclipse.core.runtime.preferences.IEclipsePreferences; import org.eclipse.core.runtime.preferences.IEclipsePreferences.IPreferenceChangeListener; import org.eclipse.core.runtime.preferences.IEclipsePreferences.PreferenceChangeEvent; import org.eclipse.e4.core.di.annotations.Optional; import org.eclipse.e4.core.di.extensions.Preference; import org.eclipse.e4.core.services.events.IEventBroker; import org.eclipse.e4.ui.di.Focus; import org.eclipse.e4.ui.di.PersistState; import org.eclipse.e4.ui.di.UIEventTopic; import org.eclipse.e4.ui.model.application.ui.basic.MPart; import org.eclipse.e4.ui.services.EMenuService; import org.eclipse.e4.ui.workbench.modeling.EPartService; import org.eclipse.e4.ui.workbench.modeling.ESelectionService; import org.eclipse.jface.dialogs.MessageDialog; import org.eclipse.jface.resource.StringConverter; import org.eclipse.jface.viewers.ISelection; import org.eclipse.jface.viewers.ISelectionChangedListener; import org.eclipse.jface.viewers.IStructuredSelection; import org.eclipse.jface.viewers.SelectionChangedEvent; import org.eclipse.swt.graphics.RGB; import org.eclipse.swt.widgets.Composite; import org.eclipse.swt.widgets.Display; import com.minres.scviewer.database.ITx; import com.minres.scviewer.database.IWaveform; import com.minres.scviewer.database.IWaveformDb; import com.minres.scviewer.database.IWaveformDbFactory; import com.minres.scviewer.database.IWaveformEvent; import com.minres.scviewer.database.swt.WaveformViewerFactory; import com.minres.scviewer.database.ui.GotoDirection; import com.minres.scviewer.database.ui.ICursor; import com.minres.scviewer.database.ui.IWaveformViewer; import com.minres.scviewer.database.ui.TrackEntry; import com.minres.scviewer.database.ui.WaveformColors; import com.minres.scviewer.e4.application.internal.status.WaveStatusBarControl; import com.minres.scviewer.e4.application.internal.util.FileMonitor; import com.minres.scviewer.e4.application.internal.util.IFileChangeListener; import com.minres.scviewer.e4.application.internal.util.IModificationChecker; import com.minres.scviewer.e4.application.preferences.DefaultValuesInitializer; import com.minres.scviewer.e4.application.preferences.PreferenceConstants; @SuppressWarnings("restriction") public class WaveformViewerPart implements IFileChangeListener, IPreferenceChangeListener { public static final String ACTIVE_WAVEFORMVIEW="Active_Waveform_View"; public static final String ADD_WAVEFORM="AddWaveform"; protected static final String DATABASE_FILE = "DATABASE_FILE"; protected static final String SHOWN_WAVEFORM = "SHOWN_WAVEFORM"; protected static final String SHOWN_CURSOR = "SHOWN_CURSOR"; protected static final String ZOOM_LEVEL = "ZOOM_LEVEL"; protected static final long FILE_CHECK_INTERVAL=60000; private String[] zoomLevel; public static final String ID = "com.minres.scviewer.ui.TxEditorPart"; //$NON-NLS-1$ public static final String WAVE_ACTION_ID = "com.minres.scviewer.ui.action.AddToWave"; WaveformViewerFactory factory = new WaveformViewerFactory(); private IWaveformViewer waveformPane; @Inject private IEventBroker eventBroker; @Inject EMenuService menuService; @Inject ESelectionService selectionService; @Inject EPartService ePartService; @Inject @Preference(nodePath = PreferenceConstants.PREFERENCES_SCOPE) IEclipsePreferences prefs; private IWaveformDb database; private boolean checkForUpdates; private MPart myPart; private Composite myParent; ArrayList filesToLoad; Map persistedState; FileMonitor fileMonitor = new FileMonitor(); IModificationChecker fileChecker; @PostConstruct public void createComposite(MPart part, Composite parent, IWaveformDbFactory dbFactory) { myPart=part; myParent=parent; database=dbFactory.getDatabase(); database.addPropertyChangeListener(new PropertyChangeListener() { @Override public void propertyChange(PropertyChangeEvent evt) { if("WAVEFORMS".equals(evt.getPropertyName())) { myParent.getDisplay().syncExec(new Runnable() { @Override public void run() { waveformPane.setMaxTime(database.getMaxTime()); } }); } } }); waveformPane = factory.createPanel(parent); waveformPane.setMaxTime(0); waveformPane.addPropertyChangeListener(IWaveformViewer.CURSOR_PROPERTY, new PropertyChangeListener() { @Override public void propertyChange(PropertyChangeEvent evt) { Long time = (Long) evt.getNewValue(); eventBroker.post(WaveStatusBarControl.CURSOR_TIME, waveformPane.getScaledTime(time)); long marker=waveformPane.getSelectedMarkerTime(); eventBroker.post(WaveStatusBarControl.MARKER_DIFF, waveformPane.getScaledTime(time-marker)); } }); waveformPane.addPropertyChangeListener(IWaveformViewer.MARKER_PROPERTY, new PropertyChangeListener() { @Override public void propertyChange(PropertyChangeEvent evt) { Long time = (Long) evt.getNewValue(); eventBroker.post(WaveStatusBarControl.MARKER_TIME, waveformPane.getScaledTime(time)); long cursor=waveformPane.getCursorTime(); eventBroker.post(WaveStatusBarControl.MARKER_DIFF, waveformPane.getScaledTime(cursor-time)); } }); waveformPane.addSelectionChangedListener(new ISelectionChangedListener() { @Override public void selectionChanged(SelectionChangedEvent event) { if(event.getSelection() instanceof IStructuredSelection) selectionService.setSelection(event.getSelection()); } }); zoomLevel=waveformPane.getZoomLevels(); setupColors(); checkForUpdates=prefs.getBoolean(PreferenceConstants.DATABASE_RELOAD, true); filesToLoad=new ArrayList(); persistedState = part.getPersistedState(); Integer files = persistedState.containsKey(DATABASE_FILE+"S")?Integer.parseInt(persistedState.get(DATABASE_FILE+"S")):0; for(int i=0; i0) loadDatabase(persistedState); eventBroker.post(WaveStatusBarControl.ZOOM_LEVEL, zoomLevel[waveformPane.getZoomLevel()]); menuService.registerContextMenu(waveformPane.getNameControl(), "com.minres.scviewer.e4.application.popupmenu.namecontext"); menuService.registerContextMenu(waveformPane.getValueControl(), "com.minres.scviewer.e4.application.popupmenu.namecontext"); menuService.registerContextMenu(waveformPane.getWaveformControl(), "com.minres.scviewer.e4.application.popupmenu.wavecontext"); ePartService.addPartListener(new PartListener(){ @Override public void partActivated(MPart part) { if (part == myPart) { if(fileChecker!=null) fileChecker.check(); updateAll(); } } }); prefs.addPreferenceChangeListener(this); } @Override public void preferenceChange(PreferenceChangeEvent event) { if(PreferenceConstants.DATABASE_RELOAD.equals(event.getKey())){ checkForUpdates = (Boolean) event.getNewValue(); fileChecker=null; if(checkForUpdates) fileChecker=fileMonitor.addFileChangeListener(WaveformViewerPart.this,filesToLoad, FILE_CHECK_INTERVAL); else fileMonitor.removeFileChangeListener(this); } else { setupColors(); } } protected void setupColors() { DefaultValuesInitializer initializer = new DefaultValuesInitializer(); HashMap colorPref=new HashMap<>(); for (WaveformColors c : WaveformColors.values()) { String prefValue=prefs.get(c.name() + "_COLOR", StringConverter.asString(initializer.colors[c.ordinal()].getRGB())); RGB rgb = StringConverter.asRGB(prefValue); colorPref.put(c, rgb); } waveformPane.setColors(colorPref); } protected void loadDatabase(final Map state) { fileMonitor.removeFileChangeListener(this); Job job = new Job(" My Job") { @Override protected IStatus run( IProgressMonitor monitor) { // convert to SubMonitor and set total number of work units SubMonitor subMonitor = SubMonitor.convert(monitor, filesToLoad.size()); subMonitor.setTaskName("Loading database"); try { for(File file: filesToLoad){ //TimeUnit.SECONDS.sleep(2); database.load(file); database.addPropertyChangeListener(waveformPane); subMonitor.worked(1); if(monitor.isCanceled()) return Status.CANCEL_STATUS; } // sleep a second } catch (Exception e) { database=null; e.printStackTrace(); return Status.CANCEL_STATUS; } subMonitor.done(); monitor.done(); return Status.OK_STATUS; } }; job.addJobChangeListener(new JobChangeAdapter(){ @Override public void done(IJobChangeEvent event) { if(event.getResult()==Status.OK_STATUS) myParent.getDisplay().asyncExec(new Runnable() { @Override public void run() { waveformPane.setMaxTime(database.getMaxTime()); if(state!=null) restoreWaveformViewerState(state); fileChecker=null; if(checkForUpdates) fileChecker=fileMonitor.addFileChangeListener( WaveformViewerPart.this,filesToLoad, FILE_CHECK_INTERVAL); } }); } }); job.schedule(0); } @Override public void fileChanged(List file) { final Display display = myParent.getDisplay(); display.asyncExec(new Runnable() { @Override public void run() { if(MessageDialog.openQuestion(display.getActiveShell(), "Database re-load", "Would you like to reload the database?")){ Map state = new HashMap<>(); saveWaveformViewerState(state); waveformPane.getStreamList().clear(); database.clear(); if(filesToLoad.size()>0) loadDatabase(state); } } }); } @Inject @Optional public void setPartInput( @Named( "input" ) Object partInput ) { if(partInput instanceof File){ filesToLoad=new ArrayList(); File file = (File) partInput; if(file.exists()){ filesToLoad.add(file); try { String ext = getFileExtension(file.getName()); if("vcd".equals(ext.toLowerCase())){ if(askIfToLoad(new File(renameFileExtension(file.getCanonicalPath(), "txdb")))){ filesToLoad.add(new File(renameFileExtension(file.getCanonicalPath(), "txdb"))); }else if(askIfToLoad(new File(renameFileExtension(file.getCanonicalPath(), "txlog")))){ filesToLoad.add(new File(renameFileExtension(file.getCanonicalPath(), "txlog"))); } } else if("txdb".equals(ext.toLowerCase()) || "txlog".equals(ext.toLowerCase())){ if(askIfToLoad(new File(renameFileExtension(file.getCanonicalPath(), "vcd")))){ filesToLoad.add(new File(renameFileExtension(file.getCanonicalPath(), "vcd"))); } } } catch (IOException e) { // silently ignore any error } } if(filesToLoad.size()>0) loadDatabase(persistedState); } } @Focus public void setFocus() { myParent.setFocus(); } @PersistState public void saveState(MPart part) { // save changes Map persistedState = part.getPersistedState(); persistedState.put(DATABASE_FILE+"S", Integer.toString(filesToLoad.size())); Integer index=0; for(File file:filesToLoad){ persistedState.put(DATABASE_FILE+index, file.getAbsolutePath()); index++; } saveWaveformViewerState(persistedState); } protected void saveWaveformViewerState(Map persistedState) { Integer index; persistedState.put(SHOWN_WAVEFORM+"S", Integer.toString(waveformPane.getStreamList().size())); index=0; for(TrackEntry trackEntry:waveformPane.getStreamList()){ persistedState.put(SHOWN_WAVEFORM+index, trackEntry.waveform.getFullName()); index++; } List cursors = waveformPane.getCursorList(); persistedState.put(SHOWN_CURSOR+"S", Integer.toString(cursors.size())); index=0; for(ICursor cursor:cursors){ persistedState.put(SHOWN_CURSOR+index, Long.toString(cursor.getTime())); index++; } persistedState.put(ZOOM_LEVEL, Integer.toString(waveformPane.getZoomLevel())); } protected void restoreWaveformViewerState(Map state) { updateAll(); Integer waves = state.containsKey(SHOWN_WAVEFORM+"S")?Integer.parseInt(state.get(SHOWN_WAVEFORM+"S")):0; List res = new LinkedList<>(); for(int i=0; i waveform = database.getStreamByName(state.get(SHOWN_WAVEFORM+i)); if(waveform!=null) res.add(new TrackEntry(waveform)); } if(res.size()>0) waveformPane.getStreamList().addAll(res); Integer cursorLength = state.containsKey(SHOWN_CURSOR+"S")?Integer.parseInt(state.get(SHOWN_CURSOR+"S")):0; List cursors = waveformPane.getCursorList(); if(cursorLength==cursors.size()){ for(int i=0; i) addStreamToList((IWaveform) el, false); } } protected boolean askIfToLoad(File txFile) { if(txFile.exists() && MessageDialog.openQuestion(myParent.getDisplay().getActiveShell(), "Database open", "Would you like to open the adjacent database "+txFile.getName()+" as well?")){ return true; } return false; } protected static String renameFileExtension(String source, String newExt) { String target; String currentExt = getFileExtension(source); if (currentExt.equals("")){ target=source+"."+newExt; } else { target=source.replaceFirst(Pattern.quote("."+currentExt)+"$", Matcher.quoteReplacement("."+newExt)); } return target; } protected static String getFileExtension(String f) { String ext = ""; int i = f.lastIndexOf('.'); if (i > 0 && i < f.length() - 1) { ext = f.substring(i + 1); } return ext; } public IWaveformDb getModel() { return database; } public IWaveformDb getDatabase() { return database; } public void addStreamToList(IWaveform obj, boolean insert){ addStreamsToList(new IWaveform[]{obj}, insert); } public void addStreamsToList(IWaveform[] iWaveforms, boolean insert){ List streams= new LinkedList<>(); for(IWaveform stream:iWaveforms) streams.add(new TrackEntry(stream)); IStructuredSelection selection = (IStructuredSelection) waveformPane.getSelection(); if(selection.size()==0){ waveformPane.getStreamList().addAll(streams); }else { Object first=selection.getFirstElement(); IWaveform stream = (first instanceof ITx)?((ITx)first).getStream():(IWaveform)first; TrackEntry trackEntry=waveformPane.getEntryForStream(stream); int index = waveformPane.getStreamList().indexOf(trackEntry); if(!insert) index++; waveformPane.getStreamList().addAll(index, streams); } } public void removeStreamFromList(IWaveform stream){ TrackEntry trackEntry=waveformPane.getEntryForStream(stream); waveformPane.getStreamList().remove(trackEntry); } public void removeStreamsFromList(IWaveform[] iWaveforms){ for(IWaveform stream:iWaveforms) removeStreamFromList(stream); } public void moveSelected(int i) { waveformPane.moveSelected(i); } public void moveSelection(GotoDirection direction) { waveformPane.moveSelection(direction); } public void moveCursor(GotoDirection direction) { waveformPane.moveCursor(direction); } public void setZoomLevel(Integer level) { if(level<0) level=0; if(level>zoomLevel.length-1) level=zoomLevel.length-1; waveformPane.setZoomLevel(level); updateAll(); } public void setZoomFit() { waveformPane.setZoomLevel(6); updateAll(); } public int getZoomLevel() { return waveformPane.getZoomLevel(); } public ISelection getSelection() { return waveformPane.getSelection(); } public void setSelection(IStructuredSelection structuredSelection) { waveformPane.setSelection(structuredSelection, true); } public String getScaledTime(Long time) { return waveformPane.getScaledTime(time); } }