All Downloads are FREE. Search and download functionalities are using the official Maven repository.

edu.cmu.tetradapp.editor.TabularDataTransferHandler Maven / Gradle / Ivy

There is a newer version: 7.6.6
Show newest version
///////////////////////////////////////////////////////////////////////////////
// For information as to what this class does, see the Javadoc, below.       //
// Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006,       //
// 2007, 2008, 2009, 2010, 2014, 2015, 2022 by Peter Spirtes, Richard        //
// Scheines, Joseph Ramsey, and Clark Glymour.                               //
//                                                                           //
// This program 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 2 of the License, or         //
// (at your option) any later version.                                       //
//                                                                           //
// This program 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 program; if not, write to the Free Software               //
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA //
///////////////////////////////////////////////////////////////////////////////

package edu.cmu.tetradapp.editor;

import edu.cmu.tetrad.data.DataSet;
import edu.cmu.tetrad.data.RegexTokenizer;
import edu.cmu.tetrad.graph.Node;
import edu.cmu.tetrad.util.JOptionUtils;

import javax.swing.*;
import java.awt.datatransfer.DataFlavor;
import java.awt.datatransfer.StringSelection;
import java.awt.datatransfer.Transferable;
import java.awt.datatransfer.UnsupportedFlavorException;
import java.io.BufferedReader;
import java.io.CharArrayReader;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.StringTokenizer;
import java.util.regex.Pattern;

/**
 * Implements basic cut and paste operations for DataDisplay.
 */
class TabularDataTransferHandler extends TransferHandler {

    public int getSourceActions(JComponent c) {
        return TransferHandler.COPY_OR_MOVE;
    }

    /**
     * Create a Transferable to use as the source for a data transfer.
     *
     * @param c The component holding the data to be transfered.  This argument is provided to enable sharing of
     *          TransferHandlers by multiple components.
     * @return The representation of the data to be transfered.
     */
    protected Transferable createTransferable(JComponent c) {
        if (c instanceof TabularDataJTable) {
            TabularDataJTable tabularData = (TabularDataJTable) c;
            DataSet dataSet = tabularData.getDataSet();

            int[] rows;
            int[] cols;

            if (!tabularData.getRowSelectionAllowed() &&
                    !tabularData.getColumnSelectionAllowed()) {
                return null;
            }

            // Column selection.
            if (!tabularData.getRowSelectionAllowed()) {
                int rowCount = tabularData.getDataSet().getNumRows();
                rows = new int[rowCount + 1];

                // Need to include the variable names.
                for (int i = 0; i < rowCount + 1; i++) {
                    rows[i] = i + getNumLeadingRows() - 1;
                }
            } else {
                int[] _rows = tabularData.getSelectedRows();

                if (Arrays.binarySearch(_rows, 1) == -1) {
                    rows = new int[_rows.length + 1];
                    rows[0] = 1;
                    System.arraycopy(_rows, 0, rows, 1, _rows.length);
                } else {
                    rows = _rows;
                }
            }

            // Row selection.
            if (!tabularData.getColumnSelectionAllowed()) {
                int colCount = tabularData.getDataSet().getNumColumns();
                cols = new int[colCount];

                for (int j = 0; j < colCount; j++) {
                    cols[j] = j + getNumLeadingCols();
                }
            } else {
                cols = tabularData.getSelectedColumns();
            }

            if (cols == null || rows.length == 0 || cols.length == 0) {
                return null;
            }

            StringBuilder buf = new StringBuilder();

            for (int displayRow : rows) {
                if (displayRow == 0) {
                    continue;
                }

                for (int displayCol : cols) {
                    if (displayCol == 0) {
                        continue;
                    }

                    String name = (String) (tabularData.getValueAt(1, displayCol));

                    if (name == null) {
                        continue;
                    }

                    if (displayRow == 1) {
                        String s = (String) tabularData.getValueAt(1, displayCol);

                        if (s.trim().equals("")) {
                            s = "C" + (displayCol - 1);
                        }

                        String val = s;

                        buf.append(val).append("\t");
                    } else {
                        int dataRow = displayRow - getNumLeadingRows();
                        int dataCol = displayCol - getNumLeadingCols();

                        if (dataCol < 0) {
                            continue;
                        }

                        if (dataCol < dataSet.getNumColumns()) {
                            if (dataRow < dataSet.getNumRows()) {
                                Object datumObj = dataSet.getObject(dataRow, dataCol);
                                String datumString = "";

                                if (datumObj != null) {
                                    if (datumObj instanceof Number) {
                                        datumString = datumObj.toString();
                                    } else if (datumObj instanceof String) {

                                        // Let's quote all Strings...
                                        datumString = "\"" + datumObj + "\"";

                                    } else {
                                        throw new IllegalArgumentException();
                                    }
                                }

                                buf.append(datumString).append("\t");
                            } else {
                                buf.append("\t");
                            }
                        }
                    }
                }

                // we want a newline at the end of each line and not a tab
                if (buf.length() - 1 > 0) {
                    buf.deleteCharAt(buf.length() - 1).append("\n");
                }
            }

            // remove the last newline
            if (buf.length() - 1 > 0) {
                buf.deleteCharAt(buf.length() - 1);
            }

            return new StringSelection(buf.toString());
        }

        return null;
    }

    public boolean importData(JComponent c, Transferable t) {
        if (c instanceof TabularDataJTable) {
            try {
                TabularDataJTable tabularData = (TabularDataJTable) c;
                String s = (String) t.getTransferData(DataFlavor.stringFlavor);

                int startRow = tabularData.getSelectedRow();
                int startCol = tabularData.getSelectedColumn();

                if (startRow == 0) {
                    startRow = 1;
                }

                if (startCol < getNumLeadingCols()) {
                    startCol = getNumLeadingCols();
                }

                if (!checkRanges(s, startCol, tabularData)) {
                    return false;
                }

                boolean shouldAsk = false;
                boolean shiftDown = true;

                BufferedReader preReader = new BufferedReader(
                        new CharArrayReader(s.toCharArray()));

                String preLine = preReader.readLine();
                StringTokenizer preTokenizer =
                        new StringTokenizer(preLine, "\t");
                int numTokens = preTokenizer.countTokens();

                for (int col = startCol; col < startCol + numTokens; col++) {
                    Object value = tabularData.getValueAt(startRow, col);
                    if (!"".equals(value) && !(null == value)) {
                        shouldAsk = true;
                    }

                    if (startRow - getNumLeadingRows() >= tabularData.getDataSet().getNumRows() ||
                            startCol - getNumLeadingCols() >= tabularData.getDataSet().getNumColumns()) {
                        shouldAsk = false;
                        shiftDown = false;
                    }
                }

                if (shouldAsk) {
                    String[] choices = {
                            "Shift corresponding cells down to make room",
                            "Replace corresponding cells"};

                    Object choice = JOptionPane.showInputDialog(
                            JOptionUtils.centeringComp(),
                            "How should the clipboard contents be pasted?",
                            "Paste Contents", JOptionPane.INFORMATION_MESSAGE,
                            null, choices, choices[0]);

                    // Null means the user cancelled the input.
                    if (choice == null) {
                        return false;
                    }

                    shiftDown = choice.equals(choices[0]);
                }

                doPaste(s, startRow, startCol, shiftDown, tabularData);
            } catch (UnsupportedFlavorException | IOException e) {
                e.printStackTrace();
            }

            return true;
        }

        return false;
    }

    private boolean checkRanges(String s, int startCol,
                                TabularDataJTable tabularData) {
        RegexTokenizer lines = new RegexTokenizer(s, Pattern.compile("\n"), '"');
        lines.nextToken();

        while (lines.hasMoreTokens()) {
            String line = lines.nextToken();
            RegexTokenizer tokens = new RegexTokenizer(line, Pattern.compile("\t"), '"');
            int col = startCol;

            while (tokens.hasMoreTokens()) {
                String token = tokens.nextToken();

                if (!tabularData.checkValueAt(token, col)) {
                    int dataCol = col - getNumLeadingCols();

                    JOptionPane.showMessageDialog(JOptionUtils.centeringComp(),
                            "" +
                                    "This paste cannot be completed, since the variable in " +
                                    "
column " + dataCol + " cannot accept the value '" + token + "'." + ""); return false; } col++; } } return true; } private void doPaste(String s, int startRow, int startCol, boolean shiftDown, TabularDataJTable tabularData) { startRow -= getNumLeadingRows(); startCol -= getNumLeadingCols(); if (startRow < 0) startRow = 0; if (startCol < 0) startCol = 0; // Determine the number of rows and columns in the string s. int pasteRows = 0; int pasteCols = -1; RegexTokenizer lines = new RegexTokenizer(s, Pattern.compile("\n"), '"'); lines.setQuoteSensitive(false); // Read the variable names. String line = lines.nextToken(); RegexTokenizer _names = new RegexTokenizer(line, Pattern.compile("\t"), '"'); List varNames = new ArrayList<>(); while (_names.hasMoreTokens()) { varNames.add(_names.nextToken()); } System.out.println("varnames = " + varNames); // Scan the rest of the data to determine dimensions. while (lines.hasMoreTokens()) { line = lines.nextToken(); if (line.length() == 0) continue; System.out.println("line = " + line); pasteRows++; RegexTokenizer numbers = new RegexTokenizer(line, Pattern.compile("\t"), '"'); int _cols = 0; while (numbers.hasMoreTokens()) { numbers.nextToken(); _cols++; } if (pasteCols == -1) { pasteCols = _cols; } // } else if (pasteCols != _cols) { //// throw new IllegalArgumentException("Number of tokens per row not uniform."); // } } if (varNames.size() != pasteCols) { throw new IllegalArgumentException("Number of variable names must " + "match the number of columns."); } // Resize the dataset if necessary to accomodate the new data. DataSet dataSet = tabularData.getDataSet(); int originalCols = dataSet.getNumColumns(); // Make the dataset big enough, making sure not to use the parsed // variable names to create new columns. dataSet.ensureColumns(startCol + pasteCols, varNames); if (shiftDown) { dataSet.ensureRows(startRow + 2 * pasteRows); } else { dataSet.ensureRows(startRow + pasteRows); } int newCols = dataSet.getNumColumns(); // Use variable names from the paste where possible, without changing // any existing variable names. If necessary, append numbers. for (int j = originalCols; j < newCols; j++) { Node node = dataSet.getVariable(j); int index = (j - (originalCols - 1)) + ((originalCols - 1) - startCol); if (index < 0) { continue; } String name = varNames.get(index); if (dataSet.getVariable(name) == null) { node.setName(name); } else { int i = 0; String _name; do { _name = name + "_" + (++i); } while (dataSet.getVariable(_name) != null); node.setName(_name); } } // Copy existing data down, if requested. if (shiftDown) { for (int i = pasteRows - 1; i >= 0; i--) { for (int j = 0; j < pasteCols; j++) { int oldRow = startRow + i; int newRow = oldRow + pasteRows; int col = startCol + j; int numRows = dataSet.getNumRows(); if (newRow < numRows) { Object value = tabularData.getValueAt( oldRow + getNumLeadingRows(), col + getNumLeadingCols()); tabularData.setValueAt(value, newRow + getNumLeadingRows(), col + getNumLeadingCols()); } } } } lines = new RegexTokenizer(s, Pattern.compile("\n"), '"'); lines.setQuoteSensitive(false); lines.nextToken(); for (int i = 0; i < pasteRows; i++) { line = lines.nextToken(); if (line.length() == 0) continue; RegexTokenizer tokens = new RegexTokenizer(line, Pattern.compile("\t"), '"'); for (int j = 0; j < pasteCols; j++) { int row = startRow + i; int col = startCol + j; String token = tokens.nextToken(); tabularData.setValueAt(token, row + getNumLeadingRows(), col + getNumLeadingCols()); } } TabularDataTable tableModel = (TabularDataTable) tabularData.getModel(); tableModel.fireTableDataChanged(); } public void exportDone(JComponent source, Transferable data, int action) { if (action == TransferHandler.MOVE && source instanceof TabularDataJTable) { TabularDataJTable tableTabular = (TabularDataJTable) source; tableTabular.deleteSelected(); } } private int getNumLeadingCols() { /* The number of initial "special" columns not used to display the data set. */ return 1; } private int getNumLeadingRows() { /* The number of initial "special" rows not used to display the data set. */ return 2; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy