package dr.app.tracer.traces; import dr.app.gui.chart.*; import dr.inference.trace.Trace; import dr.inference.trace.TraceDistribution; import dr.inference.trace.TraceFactory; import dr.inference.trace.TraceList; import jam.framework.Exportable; import javax.sound.sampled.AudioFormat; import javax.sound.sampled.AudioSystem; import javax.sound.sampled.LineUnavailableException; import javax.sound.sampled.SourceDataLine; import javax.swing.*; import java.awt.*; import java.awt.event.ActionEvent; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; /** * A panel that displays information about traces * * @author Andrew Rambaut * @author Alexei Drummond * @version $Id: RawTracePanel.java,v 1.2 2006/11/30 17:39:29 rambaut Exp $ */ public class RawTracePanel extends JPanel implements Exportable { public static int COLOUR_BY_TRACE = 0; public static int COLOUR_BY_FILE = 1; public static int COLOUR_BY_ALL = 2; private static final Color[] paints = new Color[]{ Color.BLACK, new Color(64, 35, 225), new Color(229, 35, 60), new Color(255, 174, 34), new Color(86, 255, 34), new Color(35, 141, 148), new Color(146, 35, 142), new Color(255, 90, 34), new Color(239, 255, 34), Color.DARK_GRAY }; private ChartSetupDialog chartSetupDialog = null; private JTraceChart traceChart = new JTraceChart(new LinearAxis(Axis.AT_ZERO, Axis.AT_DATA), new LinearAxis()); private JChartPanel chartPanel = new JChartPanel(traceChart, null, "", ""); private JCheckBox burninCheckBox = new JCheckBox("Show Burn-in"); private JCheckBox sampleCheckBox = new JCheckBox("Sample only"); private JCheckBox linePlotCheckBox = new JCheckBox("Draw line plot"); private JComboBox legendCombo = new JComboBox( new String[]{"None", "Top-Left", "Top", "Top-Right", "Left", "Right", "Bottom-Left", "Bottom", "Bottom-Right"} ); private JComboBox colourByCombo = new JComboBox( new String[]{"Trace", "Trace File", "All"} ); private JLabel messageLabel = new JLabel("No data loaded"); private JButton listenButton = new JButton("Listen"); private int colourBy = COLOUR_BY_TRACE; /** * Creates new RawTracePanel */ public RawTracePanel(final JFrame frame) { setOpaque(false); setMinimumSize(new Dimension(300, 150)); setLayout(new BorderLayout()); JToolBar toolBar = new JToolBar(); toolBar.setOpaque(false); toolBar.setLayout(new FlowLayout(FlowLayout.LEFT)); toolBar.setFloatable(false); JButton chartSetupButton = new JButton("Axes..."); chartSetupButton.putClientProperty( "Quaqua.Button.style", "placard" ); chartSetupButton.setFont(UIManager.getFont("SmallSystemFont")); toolBar.add(chartSetupButton); burninCheckBox.setSelected(true); burninCheckBox.setFont(UIManager.getFont("SmallSystemFont")); burninCheckBox.setOpaque(false); toolBar.add(burninCheckBox); sampleCheckBox.setSelected(false); sampleCheckBox.setFont(UIManager.getFont("SmallSystemFont")); sampleCheckBox.setOpaque(false); toolBar.add(sampleCheckBox); toolBar.add(new JToolBar.Separator(new Dimension(8, 8))); linePlotCheckBox.setSelected(true); linePlotCheckBox.setFont(UIManager.getFont("SmallSystemFont")); linePlotCheckBox.setOpaque(false); toolBar.add(linePlotCheckBox); toolBar.add(new JToolBar.Separator(new Dimension(8, 8))); JLabel label = new JLabel("Legend:"); label.setFont(UIManager.getFont("SmallSystemFont")); label.setLabelFor(legendCombo); toolBar.add(label); legendCombo.setFont(UIManager.getFont("SmallSystemFont")); legendCombo.setOpaque(false); toolBar.add(legendCombo); toolBar.add(new JToolBar.Separator(new Dimension(8, 8))); label = new JLabel("Colour by:"); label.setFont(UIManager.getFont("SmallSystemFont")); label.setLabelFor(colourByCombo); toolBar.add(label); colourByCombo.setFont(UIManager.getFont("SmallSystemFont")); colourByCombo.setOpaque(false); toolBar.add(colourByCombo); toolBar.add(listenButton); toolBar.add(new JToolBar.Separator(new Dimension(8, 8))); add(messageLabel, BorderLayout.NORTH); add(toolBar, BorderLayout.SOUTH); add(chartPanel, BorderLayout.CENTER); chartSetupButton.addActionListener( new java.awt.event.ActionListener() { public void actionPerformed(ActionEvent actionEvent) { if (chartSetupDialog == null) { chartSetupDialog = new ChartSetupDialog(frame, false, true, Axis.AT_ZERO, Axis.AT_MAJOR_TICK, Axis.AT_MAJOR_TICK, Axis.AT_MAJOR_TICK); } chartSetupDialog.showDialog(traceChart); validate(); repaint(); } } ); burninCheckBox.addActionListener( new java.awt.event.ActionListener() { public void actionPerformed(java.awt.event.ActionEvent ev) { setupTraces(); } } ); sampleCheckBox.addActionListener( new java.awt.event.ActionListener() { public void actionPerformed(java.awt.event.ActionEvent ev) { traceChart.setUseSample(sampleCheckBox.isSelected()); validate(); repaint(); } } ); linePlotCheckBox.addActionListener( new java.awt.event.ActionListener() { public void actionPerformed(java.awt.event.ActionEvent ev) { traceChart.setIsLinePlot(linePlotCheckBox.isSelected()); validate(); repaint(); } } ); legendCombo.addItemListener( new java.awt.event.ItemListener() { public void itemStateChanged(java.awt.event.ItemEvent ev) { switch (legendCombo.getSelectedIndex()) { case 0: break; case 1: traceChart.setLegendAlignment(SwingConstants.NORTH_WEST); break; case 2: traceChart.setLegendAlignment(SwingConstants.NORTH); break; case 3: traceChart.setLegendAlignment(SwingConstants.NORTH_EAST); break; case 4: traceChart.setLegendAlignment(SwingConstants.WEST); break; case 5: traceChart.setLegendAlignment(SwingConstants.EAST); break; case 6: traceChart.setLegendAlignment(SwingConstants.SOUTH_WEST); break; case 7: traceChart.setLegendAlignment(SwingConstants.SOUTH); break; case 8: traceChart.setLegendAlignment(SwingConstants.SOUTH_EAST); break; } traceChart.setShowLegend(legendCombo.getSelectedIndex() != 0); validate(); repaint(); } } ); colourByCombo.addItemListener( new java.awt.event.ItemListener() { public void itemStateChanged(java.awt.event.ItemEvent ev) { colourBy = colourByCombo.getSelectedIndex(); setupTraces(); validate(); repaint(); } } ); listenButton.addActionListener( new java.awt.event.ActionListener() { public void actionPerformed(java.awt.event.ActionEvent ev) { int n = 0; for (TraceList tl : traceLists) { for (String traceName : traceNames) { int traceIndex = tl.getTraceIndex(traceName); Trace trace = tl.getTrace(traceIndex); if (trace != null) { if (trace.getTraceType() == TraceFactory.TraceType.DOUBLE) { n++; } } } } Double[][] valueArrays = new Double[n][]; int k = 0; for (TraceList tl : traceLists) { for (String traceName : traceNames) { int traceIndex = tl.getTraceIndex(traceName); Trace trace = tl.getTrace(traceIndex); if (trace != null) { if (trace.getTraceType() == TraceFactory.TraceType.DOUBLE) { List values = tl.getValues(traceIndex); Double[] ar = new Double[values.size()]; values.toArray(ar); valueArrays[k] = ar; k++; } } } } toAudio(valueArrays); } } ); } private TraceList[] traceLists = null; private java.util.List<String> traceNames = null; public void setTraces(TraceList[] traceLists, java.util.List<String> traceNames) { this.traceLists = traceLists; this.traceNames = traceNames; setupTraces(); } private void setupTraces() { traceChart.removeAllTraces(); if (traceLists == null || traceNames == null || traceNames.size() == 0) { chartPanel.setXAxisTitle(""); chartPanel.setYAxisTitle(""); messageLabel.setText("No traces selected"); add(messageLabel, BorderLayout.NORTH); return; } remove(messageLabel); int i = 0; for (TraceList tl : traceLists) { int stateStart = tl.getBurnIn(); int stateStep = tl.getStepSize(); for (String traceName : traceNames) { int traceIndex = tl.getTraceIndex(traceName); String name = tl.getTraceName(traceIndex); if (traceLists.length > 1) { name = tl.getName() + " - " + name; } Trace trace = tl.getTrace(traceIndex); TraceDistribution td = tl.getDistributionStatistics(traceIndex); if (trace != null) { Map<Integer, String> categoryDataMap = new HashMap<Integer, String>(); List values = tl.getValues(traceIndex); List burninValues = null; if (burninCheckBox.isSelected() && tl.getBurninStateCount() > 0) { burninValues = tl.getBurninValues(traceIndex); } if (trace.getTraceType() == TraceFactory.TraceType.DOUBLE || trace.getTraceType() == TraceFactory.TraceType.INTEGER) { traceChart.setYAxis(trace.getTraceType() == TraceFactory.TraceType.INTEGER, new HashMap<Integer, String>()); traceChart.addTrace(name, stateStart, stateStep, values, burninValues, paints[i]); } else if (trace.getTraceType() == TraceFactory.TraceType.STRING) { List<Double> doubleData = new ArrayList<Double>(); for (int v = 0; v < values.size(); v++) { Integer index = td.getIndex(values.get(v).toString()); doubleData.add(v, index.doubleValue()); categoryDataMap.put(index, values.get(v).toString()); } List<Double> doubleBurninData = null; if (burninCheckBox.isSelected() && tl.getBurninStateCount() > 0) { doubleBurninData = new ArrayList<Double>(); categoryDataMap.clear(); for (int v = 0; v < burninValues.size(); v++) { Integer index = td.getIndex(burninValues.get(v).toString()); doubleBurninData.add(v, index.doubleValue()); categoryDataMap.put(index, burninValues.get(v).toString()); } } traceChart.setYAxis(false, categoryDataMap); traceChart.addTrace(name, stateStart, stateStep, doubleData, doubleBurninData, paints[i]); } else { throw new RuntimeException("Trace type is not recognized: " + trace.getTraceType()); } if (colourBy == COLOUR_BY_TRACE || colourBy == COLOUR_BY_ALL) { i++; } if (i == paints.length) i = 0; } } if (colourBy == COLOUR_BY_FILE) { i++; } else if (colourBy == COLOUR_BY_TRACE) { i = 0; } if (i == paints.length) i = 0; } chartPanel.setXAxisTitle("State"); if (traceLists.length == 1) { chartPanel.setYAxisTitle(traceLists[0].getName()); } else if (traceNames.size() == 1) { chartPanel.setYAxisTitle(traceNames.get(0)); } else { chartPanel.setYAxisTitle("Multiple Traces"); } validate(); repaint(); } public JComponent getExportableComponent() { return chartPanel; } public void toAudio(Double[][] values) { int volume = 128; int count = values[0].length; // float frequency = 44100; float frequency = 10000; float audioLength = 2; // 2 second clip byte[] buf; AudioFormat af; int repeats = (int) (audioLength * frequency / count); buf = new byte[values.length]; af = new AudioFormat(frequency, 8, values.length, true, false); double[] minValues = new double[values.length]; double[] ranges = new double[values.length]; for (int k = 0; k < values.length; k++) { double maxValue = -Double.MAX_VALUE; double minValue = Double.MAX_VALUE; for (int i = 0; i < values.length; i++) { if (values[k][i] > maxValue) { maxValue = values[k][i]; } if (values[k][i] < minValue) { minValue = values[k][i]; } } minValues[k] = minValue; ranges[k] = maxValue - minValue; } SourceDataLine sdl = null; try { sdl = AudioSystem.getSourceDataLine(af); sdl = AudioSystem.getSourceDataLine(af); sdl.open(af); sdl.start(); // for(int i=0; i < msecs*frequency/1000; i++){ for (int i = 0; i < values[0].length; i++) { for (int k = 0; k < values.length; k++) { double x = (values[k][i] - minValues[k]) / ranges[k]; buf[k] = (byte) (x * volume); } // if(addHarmonic) { // double angle2 = (i)/(frequency/hz)*2.0*Math.PI; // buf[1]=(byte)(Math.sin(2*angle2)*volume*0.6); // sdl.write(buf,0,2); // } else { for (int j = 0; j < repeats; j++) { sdl.write(buf, 0, values.length); } // } } sdl.drain(); sdl.stop(); sdl.close(); } catch (LineUnavailableException e) { e.printStackTrace(); } } public String toString() { if (traceChart.getPlotCount() == 0) { return "no plot available"; } StringBuffer buffer = new StringBuffer(); //Plot plot = densityChart.getPlot(0); ArrayList<ArrayList> traceStates = new ArrayList<ArrayList>(); ArrayList<ArrayList> traceValues = new ArrayList<ArrayList>(); int maxLength = 0; for (int i = 0; i < traceChart.getPlotCount(); i++) { Plot plot = traceChart.getPlot(i); if (i > 0) { buffer.append("\t"); } buffer.append("state"); buffer.append("\t"); buffer.append(plot.getName()); traceStates.add(i, new ArrayList(traceChart.getTraceStates(i))); traceValues.add(i, new ArrayList(traceChart.getTraceValues(i))); if (traceStates.get(i).size() > maxLength) { maxLength = traceStates.get(i).size(); } } buffer.append("\n"); for (int i = 0; i < maxLength; i++) { if (traceStates.get(0).size() > i) { buffer.append(traceStates.get(0).get(i)); buffer.append("\t"); buffer.append(String.valueOf(traceValues.get(0).get(i))); } else { buffer.append("\t"); } for (int j = 1; j < traceStates.size(); j++) { if (traceStates.get(j).size() > i) { buffer.append("\t"); buffer.append(traceStates.get(j).get(i)); buffer.append("\t"); buffer.append(String.valueOf(traceValues.get(j).get(i))); } else { buffer.append("\t\t"); } } buffer.append("\n"); } return buffer.toString(); } }