/* * Copyright (c) 2009 The Jackson Laboratory * * This software was developed by Gary Churchill's Lab at The Jackson * Laboratory (see http://research.jax.org/faculty/churchill). * * This is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This software is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this software. If not, see <http://www.gnu.org/licenses/>. */ package org.jax.qtl.scan.gui; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import javax.swing.AbstractAction; import javax.swing.SwingUtilities; import org.jax.qtl.QTL; import org.jax.qtl.cross.Cross; import org.jax.qtl.project.QtlDataModel; import org.jax.qtl.project.QtlProject; import org.jax.qtl.project.QtlProjectManager; import org.jax.qtl.project.gui.QtlProjectTree; import org.jax.qtl.scan.ScanCommandBuilder; import org.jax.qtl.scan.ScanType; import org.jax.r.RCommand; import org.jax.r.jriutilities.RInterface; import org.jax.r.jriutilities.RInterfaceFactory; import org.jax.util.project.ProjectChangeListener; import org.jax.util.project.ProjectManager; /** * Action for showing the "scan one" command dialog. * @author <A HREF="mailto:keith.sheppard@jax.org">Keith Sheppard</A> */ public class ScanOneAction extends AbstractAction implements ProjectChangeListener { /** * every {@link java.io.Serializable} is supposed to have one of these */ private static final long serialVersionUID = -9028294159002679385L; /** * the selected cross object for this scan action */ private final Cross selectedCross; /** * action listener that responds to an approval of the scan action */ private final ActionListener scanApprovedListener = new ActionListener() { public void actionPerformed(ActionEvent e) { ScanDialog sourceDialog = (ScanDialog)e.getSource(); final ScanCommandBuilder scanCommandBuilder = sourceDialog.getScanCommand(); Thread evaluateCommandThread = new Thread() { @Override public void run() { ScanOneAction.this.evaluateScanCommand( scanCommandBuilder); } }; evaluateCommandThread.start(); } }; /** * Constructor for creating a scanone action without any specific cross. */ public ScanOneAction() { this(null); } /** * Create a scanone action for the given cross. * @param selectedCross * the cross */ public ScanOneAction(Cross selectedCross) { super("Run One QTL Genome Scan ..."); this.selectedCross = selectedCross; if(selectedCross == null) { // add a listener to the project so that we know when to refresh // our updated state QtlProjectManager projectManager = QtlProjectManager.getInstance(); projectManager.addProjectChangeListener(this); this.projectChangeOccurred(projectManager); } } /** * Evaluate the given scan command * @param scanCommandBuilder * the scan command to evaluate */ private void evaluateScanCommand(final ScanCommandBuilder scanCommandBuilder) { // run the scan RInterface rInterface = RInterfaceFactory.getRInterfaceInstance(); rInterface.insertComment( "running scanone on cross: " + scanCommandBuilder.getCross().getAccessorExpressionString()); rInterface.evaluateCommandNoReturn( scanCommandBuilder.getCommandWithoutPermutations()); RCommand permutationsCommand = scanCommandBuilder.getCommandWithPermutations(); if(permutationsCommand != null) { rInterface.insertComment( "running scanone permutations (this can take a while)"); rInterface.evaluateCommandNoReturn(permutationsCommand); } // add a record of which phenotype(s) we scanned RCommand phenotypeCommand = scanCommandBuilder.getPhenotypeAttributeCommand(); if(phenotypeCommand != null) { rInterface.insertComment( "adding an attribute so we know which phenotype(s) " + "this scan came from"); rInterface.evaluateCommandNoReturn(phenotypeCommand); } rInterface.flushCommands(); QtlProjectManager.getInstance().notifyActiveProjectModified(); } /** * {@inheritDoc} */ public void actionPerformed(ActionEvent e) { SwingUtilities.invokeLater( new Runnable() { public void run() { QtlProject activeProject = QtlProjectManager.getInstance().getActiveProject(); QtlDataModel dataModel = activeProject.getDataModel(); Cross[] crosses = dataModel.getCrosses(); Cross selectedCross = ScanOneAction.this.selectedCross; if(selectedCross == null) { QtlProjectTree projectTree = QTL.getInstance().getProjectTree(); selectedCross = projectTree.getSelectedCross(); } ScanDialog scanDialog = new ScanDialog( QTL.getInstance().getApplicationFrame(), ScanType.SCANONE, crosses, selectedCross); scanDialog.addActionListener( ScanOneAction.this.scanApprovedListener); scanDialog.setVisible( true); } }); } /** * {@inheritDoc} */ public void projectChangeOccurred(ProjectManager projectManager) { // we shouldn't enable this unless we have some active crosses QtlProjectManager qtlProjectManager = (QtlProjectManager)projectManager; QtlDataModel dataModel = qtlProjectManager.getActiveProject().getDataModel(); final boolean anyCrosses = !dataModel.getCrossMap().isEmpty(); SwingUtilities.invokeLater(new Runnable() { public void run() { ScanOneAction.this.setEnabled(anyCrosses); } }); } }