using update aggregator to avoid event queue flooding in fast simulations, restructured copy to clipboard popup menu

This commit is contained in:
fros4943 2010-03-26 09:29:04 +00:00
parent c6c0b1b486
commit 9a32ec8a31

View file

@ -26,7 +26,7 @@
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE. * SUCH DAMAGE.
* *
* $Id: LogListener.java,v 1.26 2010/02/24 10:45:44 fros4943 Exp $ * $Id: LogListener.java,v 1.27 2010/03/26 09:29:04 fros4943 Exp $
*/ */
package se.sics.cooja.plugins; package se.sics.cooja.plugins;
@ -47,8 +47,9 @@ import java.awt.event.MouseEvent;
import java.io.File; import java.io.File;
import java.io.FileWriter; import java.io.FileWriter;
import java.io.PrintWriter; import java.io.PrintWriter;
import java.util.ArrayList;
import java.util.Collection; import java.util.Collection;
import java.util.LinkedList;
import java.util.List;
import java.util.Vector; import java.util.Vector;
import java.util.regex.PatternSyntaxException; import java.util.regex.PatternSyntaxException;
@ -58,6 +59,7 @@ import javax.swing.Box;
import javax.swing.BoxLayout; import javax.swing.BoxLayout;
import javax.swing.JFileChooser; import javax.swing.JFileChooser;
import javax.swing.JLabel; import javax.swing.JLabel;
import javax.swing.JMenu;
import javax.swing.JMenuItem; import javax.swing.JMenuItem;
import javax.swing.JOptionPane; import javax.swing.JOptionPane;
import javax.swing.JPanel; import javax.swing.JPanel;
@ -66,6 +68,7 @@ import javax.swing.JScrollPane;
import javax.swing.JTable; import javax.swing.JTable;
import javax.swing.JTextField; import javax.swing.JTextField;
import javax.swing.RowFilter; import javax.swing.RowFilter;
import javax.swing.SwingUtilities;
import javax.swing.table.AbstractTableModel; import javax.swing.table.AbstractTableModel;
import javax.swing.table.TableModel; import javax.swing.table.TableModel;
import javax.swing.table.TableRowSorter; import javax.swing.table.TableRowSorter;
@ -82,6 +85,7 @@ import se.sics.cooja.VisPlugin;
import se.sics.cooja.SimEventCentral.LogOutputEvent; import se.sics.cooja.SimEventCentral.LogOutputEvent;
import se.sics.cooja.SimEventCentral.LogOutputListener; import se.sics.cooja.SimEventCentral.LogOutputListener;
import se.sics.cooja.dialogs.TableColumnAdjuster; import se.sics.cooja.dialogs.TableColumnAdjuster;
import se.sics.cooja.dialogs.UpdateAggregator;
/** /**
* A simple mote log listener. * A simple mote log listener.
@ -108,7 +112,7 @@ public class LogListener extends VisPlugin {
private final JTable logTable; private final JTable logTable;
private TableRowSorter<TableModel> logFilter; private TableRowSorter<TableModel> logFilter;
private ArrayList<LogData> logs = new ArrayList<LogData>(); private LinkedList<LogData> logs = new LinkedList<LogData>();
private Simulation simulation; private Simulation simulation;
@ -119,6 +123,44 @@ public class LogListener extends VisPlugin {
private LogOutputListener logOutputListener; private LogOutputListener logOutputListener;
private static final int UPDATE_INTERVAL = 250;
private UpdateAggregator<LogData> logUpdateAggregator = new UpdateAggregator<LogData>(UPDATE_INTERVAL) {
private Runnable scroll = new Runnable() {
public void run() {
logTable.scrollRectToVisible(
new Rectangle(0, logTable.getHeight() - 2, 1, logTable.getHeight()));
}
};
protected void handle(List<LogData> ls) {
boolean isVisible = true;
if (logTable.getRowCount() > 0) {
Rectangle visible = logTable.getVisibleRect();
if (visible.y + visible.height < logTable.getHeight()) {
isVisible = false;
}
}
/* Add */
int index = logs.size();
logs.addAll(ls);
model.fireTableRowsInserted(index, logs.size()-1);
/* Remove old */
int removed = 0;
while (logs.size() > simulation.getEventCentral().getLogOutputBufferSize()) {
logs.removeFirst();
removed++;
}
if (removed > 0) {
model.fireTableRowsDeleted(0, removed-1);
}
if (isVisible) {
SwingUtilities.invokeLater(scroll);
}
}
};
/** /**
* @param simulation Simulation * @param simulation Simulation
* @param gui GUI * @param gui GUI
@ -198,14 +240,16 @@ public class LogListener extends VisPlugin {
logTable.setRowSorter(logFilter); logTable.setRowSorter(logFilter);
/* Automatically update column widths */ /* Automatically update column widths */
TableColumnAdjuster adjuster = new TableColumnAdjuster(logTable); final TableColumnAdjuster adjuster = new TableColumnAdjuster(logTable);
adjuster.setDynamicAdjustment(true);
adjuster.packColumns(); adjuster.packColumns();
/* Popup menu */ /* Popup menu */
JPopupMenu popupMenu = new JPopupMenu(); JPopupMenu popupMenu = new JPopupMenu();
popupMenu.add(new JMenuItem(copyAction)); JMenu copyClipboard = new JMenu("Copy to clipboard");
popupMenu.add(new JMenuItem(copyAllAction)); copyClipboard.add(new JMenuItem(copyAllAction));
copyClipboard.add(new JMenuItem(copyAllMessagesAction));
copyClipboard.add(new JMenuItem(copyAction));
popupMenu.add(copyClipboard);
popupMenu.add(new JMenuItem(clearAction)); popupMenu.add(new JMenuItem(clearAction));
popupMenu.addSeparator(); popupMenu.addSeparator();
popupMenu.add(new JMenuItem(saveAction)); popupMenu.add(new JMenuItem(saveAction));
@ -222,17 +266,25 @@ public class LogListener extends VisPlugin {
LogData data = new LogData(historyEv); LogData data = new LogData(historyEv);
logs.add(data); logs.add(data);
} }
final int index = logs.size()-1;
java.awt.EventQueue.invokeLater(new Runnable() { java.awt.EventQueue.invokeLater(new Runnable() {
public void run() { public void run() {
model.fireTableRowsInserted(0, index); model.fireTableDataChanged();
logTable.scrollRectToVisible( logTable.scrollRectToVisible(
new Rectangle(0, logTable.getHeight() - 2, 1, logTable.getHeight())); new Rectangle(0, logTable.getHeight() - 2, 1, logTable.getHeight()));
} }
}); });
} }
/* Column width adjustment */
java.awt.EventQueue.invokeLater(new Runnable() {
public void run() {
/* Make sure this happens *after* adding history */
adjuster.setDynamicAdjustment(true);
}
});
/* Start observing motes for new log output */ /* Start observing motes for new log output */
logUpdateAggregator.start();
simulation.getEventCentral().addLogOutputListener(logOutputListener = new LogOutputListener() { simulation.getEventCentral().addLogOutputListener(logOutputListener = new LogOutputListener() {
public void moteWasAdded(Mote mote) { public void moteWasAdded(Mote mote) {
/* Update title */ /* Update title */
@ -244,38 +296,9 @@ public class LogListener extends VisPlugin {
} }
public void newLogOutput(LogOutputEvent ev) { public void newLogOutput(LogOutputEvent ev) {
/* Display new log output */ /* Display new log output */
final LogData data = new LogData(ev); logUpdateAggregator.add(new LogData(ev));
java.awt.EventQueue.invokeLater(new Runnable() {
public void run() {
/* Autoscroll */
boolean isVisible = false;
int rowCount = logTable.getRowCount();
if (rowCount > 0) {
Rectangle visible = logTable.getVisibleRect();
isVisible = visible.y + visible.height >= logTable.getHeight();
}
int index = logs.size();
logs.add(data);
model.fireTableRowsInserted(index, index);
if (isVisible) {
logTable.scrollRectToVisible(
new Rectangle(0, logTable.getHeight() - 2, 1, logTable.getHeight()));
}
}
});
} }
public void removedLogOutput(final LogOutputEvent ev) { public void removedLogOutput(LogOutputEvent ev) {
java.awt.EventQueue.invokeLater(new Runnable() {
public void run() {
for (int i = 0, n = logs.size(); i < n; i++) {
if (logs.get(i).ev == ev) {
logs.remove(i);
model.fireTableRowsDeleted(i, i);
break;
}
}
}
});
} }
}); });
@ -313,6 +336,7 @@ public class LogListener extends VisPlugin {
public void closePlugin() { public void closePlugin() {
/* Stop observing motes */ /* Stop observing motes */
logUpdateAggregator.stop();
simulation.getEventCentral().removeLogOutputListener(logOutputListener); simulation.getEventCentral().removeLogOutputListener(logOutputListener);
} }
@ -497,7 +521,7 @@ public class LogListener extends VisPlugin {
} }
}; };
private Action copyAction = new AbstractAction("Copy selected") { private Action copyAction = new AbstractAction("Selected") {
private static final long serialVersionUID = -8433490108577001803L; private static final long serialVersionUID = -8433490108577001803L;
public void actionPerformed(ActionEvent e) { public void actionPerformed(ActionEvent e) {
@ -520,7 +544,7 @@ public class LogListener extends VisPlugin {
} }
}; };
private Action copyAllAction = new AbstractAction("Copy all") { private Action copyAllAction = new AbstractAction("All") {
private static final long serialVersionUID = -5038884975254178373L; private static final long serialVersionUID = -5038884975254178373L;
public void actionPerformed(ActionEvent e) { public void actionPerformed(ActionEvent e) {
@ -541,4 +565,21 @@ public class LogListener extends VisPlugin {
} }
}; };
private Action copyAllMessagesAction = new AbstractAction("All messages") {
private static final long serialVersionUID = -5038884975254178373L;
public void actionPerformed(ActionEvent e) {
Clipboard clipboard = Toolkit.getDefaultToolkit().getSystemClipboard();
StringBuilder sb = new StringBuilder();
for(LogData data : logs) {
sb.append(data.ev.getMessage());
sb.append("\n");
}
StringSelection stringSelection = new StringSelection(sb.toString());
clipboard.setContents(stringSelection, null);
}
};
} }