using update aggregator to avoid event queue flooding in fast simulations, restructured copy to clipboard popup menu
This commit is contained in:
parent
c6c0b1b486
commit
9a32ec8a31
1 changed files with 83 additions and 42 deletions
|
@ -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);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue