package net.sf.jabref; import java.awt.Component; import java.awt.Container; import java.awt.Dimension; import java.awt.Frame; import java.awt.GridBagConstraints; import java.awt.GridBagLayout; import java.awt.Insets; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.awt.event.ComponentAdapter; import java.awt.event.ComponentEvent; import java.awt.event.ComponentListener; import java.awt.event.ItemEvent; import java.awt.event.ItemListener; import java.awt.event.KeyEvent; import java.awt.event.MouseAdapter; import java.awt.event.MouseEvent; import java.awt.event.MouseListener; import java.awt.event.WindowAdapter; import java.awt.event.WindowEvent; import java.io.File; import java.io.FileFilter; import java.io.IOException; import java.util.*; import javax.swing.AbstractAction; import javax.swing.Action; import javax.swing.BorderFactory; import javax.swing.DefaultListCellRenderer; import javax.swing.JButton; import javax.swing.JCheckBox; import javax.swing.JComboBox; import javax.swing.JComponent; import javax.swing.JDialog; import javax.swing.JFileChooser; import javax.swing.JLabel; import javax.swing.JList; import javax.swing.JOptionPane; import javax.swing.JPanel; import javax.swing.JProgressBar; import javax.swing.JRootPane; import javax.swing.JScrollPane; import javax.swing.JTextField; import javax.swing.JTree; import javax.swing.KeyStroke; import javax.swing.event.ChangeEvent; import javax.swing.event.ChangeListener; import javax.swing.filechooser.FileSystemView; import javax.swing.tree.DefaultMutableTreeNode; import javax.swing.tree.DefaultTreeCellRenderer; import javax.swing.tree.DefaultTreeModel; import javax.swing.tree.TreeModel; import javax.swing.tree.TreeNode; import javax.swing.tree.TreePath; import com.jgoodies.forms.builder.ButtonBarBuilder; import net.sf.jabref.imports.EntryFromFileCreator; import net.sf.jabref.imports.EntryFromFileCreatorManager; import net.sf.jabref.imports.UnlinkedFilesCrawler; import net.sf.jabref.imports.UnlinkedPDFFileFilter; import net.sf.jabref.undo.CountingUndoManager; /** * GUI Dialog for the feature "Find unlinked files". * * @author Nosh&Dan * @version 25.11.2008 | 23:13:29 * */ public class FindUnlinkedFilesDialog extends JDialog { private static final long serialVersionUID = -5778378185253640030L; /** * Keys to be used for referencing this Action. */ public static final String ACTION_COMMAND = "findUnlinkedFiles"; public static final String ACTION_TITLE = "Find unlinked Files..."; public static final String ACTION_ICON = "toggleSearch"; public static final String ACTION_SHORT_DESCRIPTION = "Searches for unlinked PDF files on the file system"; private static final String GLOBAL_PREFS_WORKING_DIRECTORY_KEY = "findUnlinkedFilesWD"; private static final String GLOBAL_PREFS_DIALOG_SIZE_KEY = "findUnlinkedFilesDialogSize"; protected JabRefFrame frame; protected BibtexDatabase database; private EntryFromFileCreatorManager creatorManager; private UnlinkedFilesCrawler crawler; protected File lastSelectedDirectory; private TreeModel treeModel; /* PANELS */ protected JPanel panelDirectory; protected JPanel panelSearchArea; protected JPanel panelFiles; protected JPanel panelOptions; protected JPanel panelButtons; protected JPanel panelEntryTypesSelection; protected JPanel panelImportArea; protected JPanel mainArea; protected JButton buttonBrowse; protected JButton buttonScan; protected JButton buttonApply; protected JButton buttonClose; /* Options for the TreeView */ protected JButton buttonOptionSelectAll; protected JButton buttonOptionUnselectAll; protected JButton buttonOptionExpandAll; protected JButton buttonOptionCollapseAll; protected JCheckBox checkboxCreateKeywords; protected JTextField textfieldDirectoryPath; protected JLabel labelDirectoryDescription; protected JLabel labelFileTypesDescription; protected JLabel labelFilesDescription; protected JLabel labelEntryTypeDescription; protected JLabel labelSearchingDirectoryInfo; protected JLabel labelImportingInfo; protected JTree tree; protected JScrollPane scrollpaneTree; protected JComboBox comboBoxFileTypeSelection; protected JComboBox comboBoxEntryTypeSelection; private JProgressBar progressBarSearching; private JProgressBar progressBarImporting; private JFileChooser fileChooser; private MouseListener treeMouseListener; private Action actionSelectAll; private Action actionUnselectAll; private Action actionExpandTree; private Action actionCollapseTree; private ActionListener actionListenerImportEntrys; private ComponentListener dialogPositionListener; private int[] threadState = new int[] {1}; private boolean checkBoxWhyIsThereNoGetSelectedStupidSwing = false; /** * For Unit-testing only. <i>Don't remove!</i> <br> * Used via reflection in {@link DatabaseFileLookupTest} to construct this * class. */ @SuppressWarnings("unused") private FindUnlinkedFilesDialog() { } public FindUnlinkedFilesDialog(Frame owner, JabRefFrame frame, BasePanel panel) { super(owner, Globals.lang("Find unlinked files"), true); this.frame = frame; restoreSizeOfDialog(); database = panel.getDatabase(); creatorManager = new EntryFromFileCreatorManager(); crawler = new UnlinkedFilesCrawler(database); lastSelectedDirectory = loadLastSelectedDirectory(); initialize(); buttonApply.setEnabled(false); } /** * Close dialog when pressing escape */ protected JRootPane createRootPane() { ActionListener actionListener = new ActionListener() { public void actionPerformed(ActionEvent actionEvent) { setVisible(false); } }; JRootPane rootPane = new JRootPane(); KeyStroke stroke = KeyStroke.getKeyStroke(KeyEvent.VK_ESCAPE, 0); rootPane.registerKeyboardAction(actionListener, stroke, JComponent.WHEN_IN_FOCUSED_WINDOW); return rootPane; } /** * Stores the current size of this dialog persistently. */ private void storeSizeOfDialog() { Dimension dim = getSize(); String store = dim.width + ";" + dim.height; Globals.prefs.put(GLOBAL_PREFS_DIALOG_SIZE_KEY, store); } /** * Restores the location and size of this dialog from the persistent storage. */ private void restoreSizeOfDialog() { String store = Globals.prefs.get(GLOBAL_PREFS_DIALOG_SIZE_KEY); Dimension dimension = null; if (store != null) { try { String[] dim = store.split(";"); dimension = new Dimension(new Integer(dim[0]), new Integer(dim[1])); } catch (Exception e) { } } if (dimension != null) { setPreferredSize(dimension); } } /** * Initializes the components, the layout, the data structure and the * actions in this dialog. */ protected void initialize() { initializeActions(); initComponents(); createTree(); createFileTypesCombobox(); createEntryTypesCombobox(); initLayout(); setupActions(); pack(); } /** * Initializes action objects. <br> * Does not assign actions to components yet! */ protected void initializeActions() { actionSelectAll = new AbstractAction(Globals.lang("Select All")) { private static final long serialVersionUID = 9153126361571985383L; public void actionPerformed(ActionEvent e) { CheckableTreeNode rootNode = (CheckableTreeNode) tree.getModel().getRoot(); rootNode.setSelected(true); tree.invalidate(); tree.repaint(); } }; actionUnselectAll = new AbstractAction(Globals.lang("Unselect All")) { private static final long serialVersionUID = -9012913268030356450L; public void actionPerformed(ActionEvent e) { CheckableTreeNode rootNode = (CheckableTreeNode) tree.getModel().getRoot(); rootNode.setSelected(false); tree.invalidate(); tree.repaint(); } }; actionExpandTree = new AbstractAction(Globals.lang("Expand All")) { private static final long serialVersionUID = -4794088095435302433L; public void actionPerformed(ActionEvent e) { CheckableTreeNode rootNode = (CheckableTreeNode) tree.getModel().getRoot(); expandTree(tree, new TreePath(rootNode), true); } }; actionCollapseTree = new AbstractAction(Globals.lang("Collapse All")) { private static final long serialVersionUID = 6007246730370326456L; public void actionPerformed(ActionEvent e) { CheckableTreeNode rootNode = (CheckableTreeNode) tree.getModel().getRoot(); expandTree(tree, new TreePath(rootNode), false); } }; dialogPositionListener = new ComponentAdapter() { /* (non-Javadoc) * @see java.awt.event.ComponentAdapter#componentResized(java.awt.event.ComponentEvent) */ @Override public void componentResized(ComponentEvent e) { storeSizeOfDialog(); } /* (non-Javadoc) * @see java.awt.event.ComponentAdapter#componentMoved(java.awt.event.ComponentEvent) */ @Override public void componentMoved(ComponentEvent e) { storeSizeOfDialog(); } }; } /** * Stores the working directory path for this view in the global * preferences. * * @param A * directory that is used as the working directory in this view. */ private void storeLastSelectedDirectory(File lastSelectedDirectory) { this.lastSelectedDirectory = lastSelectedDirectory; if (lastSelectedDirectory != null) { Globals.prefs.put(GLOBAL_PREFS_WORKING_DIRECTORY_KEY, lastSelectedDirectory.getAbsolutePath()); } } /** * Loads the working directory path which is persistantly stored for this * view and returns it as a {@link File}-Object. <br> * <br> * If there is no working directory path stored, the general working * directory will be consulted. * * @return The persistently stored working directory path for this view. */ private File loadLastSelectedDirectory() { String workingDirectory = Globals.prefs.get(GLOBAL_PREFS_WORKING_DIRECTORY_KEY); if (workingDirectory == null) workingDirectory = Globals.prefs.get("workingDirectory"); lastSelectedDirectory = new File(workingDirectory); if (lastSelectedDirectory.isFile()) lastSelectedDirectory = lastSelectedDirectory.getParentFile(); return lastSelectedDirectory; } /** * Opens a {@link JFileChooser} and receives the user input as a * {@link File} object, which this method returns. <br> * <br> * The "Open file" dialog will start at the path that is set in the * "directory" textfield, or at the last stored path for this dialog, if the * textfield is empty. <br> * <br> * If the user cancels the "Open file" dialog, this method returns null. <br> * <br> * If the user has selected a valid directory in the "Open file" dialog, * this path will be stored persistently for this dialog, so that it can be * preset at the next time this dialog is opened. * * @return The selected directory from the user, or <code>null</code>, if * the user has aborted the selection. */ private File chooseDirectory() { if (fileChooser == null) { fileChooser = new JFileChooser(); fileChooser.setAutoscrolls(true); fileChooser.setDialogTitle(Globals.lang("Select Directory")); fileChooser.setApproveButtonText(Globals.lang("Choose Directory")); fileChooser.setApproveButtonToolTipText(Globals.lang("Use the selected directory to start with the search.")); fileChooser.setFileSelectionMode(JFileChooser.DIRECTORIES_ONLY); } String path = textfieldDirectoryPath.getText(); if (path == null || path.equals("")) fileChooser.setCurrentDirectory(lastSelectedDirectory); else fileChooser.setCurrentDirectory(new File(path)); int result = fileChooser.showOpenDialog(frame); if (result == JFileChooser.CANCEL_OPTION) { return null; } File selectedDirectory = fileChooser.getSelectedFile(); String filepath = ""; if (selectedDirectory != null) { filepath = selectedDirectory.getAbsolutePath(); } textfieldDirectoryPath.setText(filepath); return selectedDirectory; } /** * Disables or enables all visible Elements in this Dialog. <br> * <br> * This also removes the {@link MouseListener} from the Tree-View to prevent * it from receiving mouse events when in disabled-state. * * @param enable * <code>true</code> when the elements shall get enabled, * <code>false</code> when they shall get disabled. */ private void disOrEnableDialog(boolean enable) { if (enable) { tree.addMouseListener(treeMouseListener); } else { tree.removeMouseListener(treeMouseListener); } disOrEnableAllElements(FindUnlinkedFilesDialog.this, enable); } /** * Recursively disables or enables all swing and awt components in this * dialog, starting with but not including the container * <code>startContainer</code>. * * @param startContainer * The GUI Element to start with. * @param enable * <code>true</code>, if all elements will get enabled, * <code>false</code> if all elements will get disabled. */ private void disOrEnableAllElements(Container startContainer, boolean enable) { Component[] children = startContainer.getComponents(); for (int i = 0; i < children.length; i++) { Component child = children[i]; if (child instanceof Container) { disOrEnableAllElements((Container) child, enable); } child.setEnabled(enable); } } /** * Expands or collapses the specified tree according to the * <code>expand</code>-parameter. */ private void expandTree(JTree tree, TreePath parent, boolean expand) { TreeNode node = (TreeNode) parent.getLastPathComponent(); if (node.getChildCount() >= 0) { for (Enumeration e = node.children(); e.hasMoreElements();) { TreePath path = parent.pathByAddingChild(e.nextElement()); expandTree(tree, path, expand); } } if (expand) { tree.expandPath(parent); } else { tree.collapsePath(parent); } } /** * Starts the search of unlinked files according to the current dialog * state. <br> * <br> * This state is made of: <br> * <li>The value of the "directory"-input-textfield and <li>The file type * selection. <br> * The search will process in a seperate thread and the progress bar behind * the "search" button will be displayed. <br> * <br> * When the search has completed, the * {@link #searchFinishedHandler(CheckableTreeNode)} handler method is * invoked. */ protected void startSearch() { final File directory = new File(textfieldDirectoryPath.getText()); if (!directory.exists() || !directory.isDirectory()) { textfieldDirectoryPath.setText(""); return; } storeLastSelectedDirectory(directory); progressBarSearching.setMinimumSize(new Dimension(buttonScan.getSize().width, progressBarSearching.getMinimumSize().height)); progressBarSearching.setVisible(true); progressBarSearching.setString(""); labelSearchingDirectoryInfo.setVisible(true); buttonScan.setVisible(false); disOrEnableDialog(false); labelSearchingDirectoryInfo.setEnabled(true); final FileFilter selectedFileFilter = (FileFilter) comboBoxFileTypeSelection.getSelectedItem(); threadState = new int[] {1}; new Thread(new Runnable() { public void run() { UnlinkedPDFFileFilter ff = new UnlinkedPDFFileFilter(selectedFileFilter, database); CheckableTreeNode rootNode = crawler.searchDirectory(directory, ff, threadState, new ChangeListener() { int counter = 0; public void stateChanged(ChangeEvent e) { progressBarSearching.setString(++counter + " files found"); } }); searchFinishedHandler(rootNode); } }).start(); } /** * This will start the import of all file of all selected nodes in this * dialogs tree view. <br> * <br> * The import itself will run in a seperate thread, whilst this dialog will * be showing a progress bar, until the thread has finished its work. <br> * <br> * When the import has finished, the {@link #importFinishedHandler(List)} is * invoked. */ protected void startImport() { if (treeModel == null) { return; } setDefaultCloseOperation(DO_NOTHING_ON_CLOSE); CheckableTreeNode root = (CheckableTreeNode) treeModel.getRoot(); final List<File> fileList = getFileListFromNode(root); if (fileList == null || fileList.isEmpty()) { return; } progressBarImporting.setVisible(true); labelImportingInfo.setVisible(true); buttonApply.setVisible(false); buttonClose.setVisible(false); disOrEnableDialog(false); labelImportingInfo.setEnabled(true); progressBarImporting.setMinimum(0); progressBarImporting.setMaximum(fileList.size()); progressBarImporting.setValue(0); progressBarImporting.setString(""); final BibtexEntryType entryType = ((BibtexEntryTypeWrapper) comboBoxEntryTypeSelection.getSelectedItem()).entryType; threadState = new int[] {1}; new Thread(new Runnable() { public void run() { List<String> errors = new LinkedList<String>(); int count = creatorManager.addEntrysFromFiles(fileList, database, frame.basePanel(), entryType, checkBoxWhyIsThereNoGetSelectedStupidSwing, new ChangeListener() { int counter = 0; public void stateChanged(ChangeEvent e) { progressBarImporting.setValue(++counter); progressBarImporting.setString(counter + " of " + progressBarImporting.getMaximum()); } }, errors); importFinishedHandler(count, errors); } }).start(); } /** * * @param errors */ protected void importFinishedHandler(int count, List<String> errors) { if (errors != null && errors.size() > 0) { StringBuffer warningInfo = new StringBuffer(); warningInfo.append("The import finished with warnings:\n"); warningInfo.append("There "); warningInfo.append(errors.size() > 1 ? "were " : "was "); warningInfo.append(errors.size()); warningInfo.append(errors.size() > 1 ? " files" : " file"); warningInfo.append(errors.size() > 1 ? " which" : " that"); warningInfo.append(" could not be imported."); JOptionPane.showMessageDialog(this, warningInfo.toString(), Globals.lang("Warning"), JOptionPane.WARNING_MESSAGE); } progressBarImporting.setVisible(false); labelImportingInfo.setVisible(false); buttonApply.setVisible(true); buttonClose.setVisible(true); disOrEnableDialog(true); setDefaultCloseOperation(DISPOSE_ON_CLOSE); frame.basePanel().markBaseChanged(); } /** * Will be called from the Thread in which the "unlinked files search" is * processed. As the result of the search, the root node of the determined * file structure is passed. * * @param rootNode * The root of the file structure as the result of the search. */ protected void searchFinishedHandler(CheckableTreeNode rootNode) { treeModel = new DefaultTreeModel(rootNode); tree.setModel(treeModel); tree.setRootVisible(rootNode.getChildCount() > 0); tree.invalidate(); tree.repaint(); progressBarSearching.setVisible(false); labelSearchingDirectoryInfo.setVisible(false); buttonScan.setVisible(true); actionSelectAll.actionPerformed(null); disOrEnableDialog(true); buttonApply.setEnabled(true); } /** * Sets up the actions for the components. */ private void setupActions() { /** * Stores the selected Directory. */ buttonBrowse.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent e) { File selectedDirectory = chooseDirectory(); storeLastSelectedDirectory(selectedDirectory); } }); buttonScan.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent e) { startSearch(); } }); /** * Action for the button "Import...". <br> * <br> * Actions on this button will start the import of all file of all * selected nodes in this dialogs tree view. <br> */ actionListenerImportEntrys = new ActionListener() { public void actionPerformed(ActionEvent e) { startImport(); } }; buttonApply.addActionListener(actionListenerImportEntrys); buttonClose.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent e) { dispose(); } }); } /** * Creates a list of {@link File}s for all leaf nodes in the tree structure * <code>node</code>, which have been marked as <i>selected</i>. <br> * <br> * <code>Selected</code> nodes correspond to those entries in the tree, * whose checkbox is <code>checked</code>. * * SIDE EFFECT: The checked nodes are removed from the tree. * * @param node * The root node representing a tree structure. * @return A list of files of all checked leaf nodes. */ @SuppressWarnings("unchecked") private List<File> getFileListFromNode(CheckableTreeNode node) { List<File> filesList = new ArrayList<File>(); Enumeration<CheckableTreeNode> childs = node.depthFirstEnumeration(); ArrayList<CheckableTreeNode> nodesToRemove = new ArrayList<FindUnlinkedFilesDialog.CheckableTreeNode>(); while (childs.hasMoreElements()) { CheckableTreeNode child = childs.nextElement(); if (child.isLeaf() && child.getSelected()) { File nodeFile = ((FileNodeWrapper) child.getUserObject()).file; if (nodeFile != null && nodeFile.isFile()) { filesList.add(nodeFile); nodesToRemove.add(child); } } } // remove imported files from tree DefaultTreeModel model = (DefaultTreeModel) tree.getModel(); for (CheckableTreeNode nodeToRemove: nodesToRemove) { DefaultMutableTreeNode parent = (DefaultMutableTreeNode) nodeToRemove.getParent(); model.removeNodeFromParent(nodeToRemove); // remove empty parent node while (parent != null && parent.isLeaf()) { DefaultMutableTreeNode pp = (DefaultMutableTreeNode) parent.getParent(); if (pp != null) { model.removeNodeFromParent(parent); } parent = pp; } // TODO: update counter / see: getTreeCellRendererComponent for label generation } tree.invalidate(); tree.repaint(); return filesList; } /** * Initializes the visible components in this dialog. */ private void initComponents() { this.addComponentListener(dialogPositionListener); /* Interrupts the searchThread by setting the State-Array to 0 */ this.addWindowListener(new WindowAdapter() { @Override public void windowClosing(WindowEvent e) { threadState[0] = 0; } }); panelDirectory = new JPanel(); panelSearchArea = new JPanel(); panelFiles = new JPanel(); panelOptions = new JPanel(); panelEntryTypesSelection = new JPanel(); panelButtons = new JPanel(); panelImportArea = new JPanel(); mainArea = new JPanel(); buttonBrowse = new JButton(Globals.lang("Browse...")); buttonBrowse.setMnemonic('B'); buttonBrowse.setToolTipText(Globals.lang("Opens the file browser.")); buttonScan = new JButton(Globals.lang("Scan directory")); buttonScan.setMnemonic('S'); buttonScan.setToolTipText(Globals.lang("Searches the selected directory for unlinked files.")); buttonApply = new JButton(Globals.lang("Apply")); buttonApply.setMnemonic('I'); buttonApply.setToolTipText(Globals.lang("Starts the import of bibtex entries.")); buttonClose = new JButton(Globals.lang("Close")); buttonClose.setToolTipText(Globals.lang("Leave this dialog.")); buttonClose.setMnemonic('C'); /* Options for the TreeView */ buttonOptionSelectAll = new JButton(); buttonOptionSelectAll.setMnemonic('A'); buttonOptionSelectAll.setAction(actionSelectAll); buttonOptionUnselectAll = new JButton(); buttonOptionUnselectAll.setMnemonic('U'); buttonOptionUnselectAll.setAction(actionUnselectAll); buttonOptionExpandAll = new JButton(); buttonOptionExpandAll.setMnemonic('E'); buttonOptionExpandAll.setAction(actionExpandTree); buttonOptionCollapseAll = new JButton(); buttonOptionCollapseAll.setMnemonic('L'); buttonOptionCollapseAll.setAction(actionCollapseTree); checkboxCreateKeywords = new JCheckBox(Globals.lang("Create directory based keywords")); checkboxCreateKeywords.setToolTipText(Globals.lang("Creates keywords in created entrys with directory pathnames")); checkboxCreateKeywords.setSelected(checkBoxWhyIsThereNoGetSelectedStupidSwing); checkboxCreateKeywords.addItemListener(new ItemListener() { public void itemStateChanged(ItemEvent e) { checkBoxWhyIsThereNoGetSelectedStupidSwing = !checkBoxWhyIsThereNoGetSelectedStupidSwing; } }); textfieldDirectoryPath = new JTextField(); textfieldDirectoryPath.setText(lastSelectedDirectory == null ? "" : lastSelectedDirectory.getAbsolutePath()); labelDirectoryDescription = new JLabel(Globals.lang("Select a directory where the search shall start.")); labelFileTypesDescription = new JLabel(Globals.lang("Select file type:")); labelFilesDescription = new JLabel(Globals.lang("These files are not linked in the active database.")); labelEntryTypeDescription = new JLabel(Globals.lang("Entry type to be created:")); labelSearchingDirectoryInfo = new JLabel(Globals.lang("Searching file system...")); labelSearchingDirectoryInfo.setHorizontalAlignment(JTextField.CENTER); labelSearchingDirectoryInfo.setVisible(false); labelImportingInfo = new JLabel(Globals.lang("Importing into Database...")); labelImportingInfo.setHorizontalAlignment(JTextField.CENTER); labelImportingInfo.setVisible(false); tree = new JTree(); scrollpaneTree = new JScrollPane(tree); scrollpaneTree.setWheelScrollingEnabled(true); progressBarSearching = new JProgressBar(); progressBarSearching.setIndeterminate(true); progressBarSearching.setVisible(false); progressBarSearching.setStringPainted(true); progressBarImporting = new JProgressBar(); progressBarImporting.setIndeterminate(false); progressBarImporting.setVisible(false); progressBarImporting.setStringPainted(true); } /** * Initializes the layout for the visible components in this menu. A * {@link GridBagLayout} is used. */ private void initLayout() { GridBagLayout gbl = new GridBagLayout(); panelDirectory.setBorder(BorderFactory.createTitledBorder(BorderFactory.createEtchedBorder(), Globals.lang("Select directory"))); panelFiles.setBorder(BorderFactory.createTitledBorder(BorderFactory.createEtchedBorder(), Globals.lang("Select files"))); panelEntryTypesSelection.setBorder(BorderFactory.createTitledBorder(BorderFactory.createEtchedBorder(), Globals.lang("Bibtex entry creation"))); Insets basicInsets = new Insets(6, 6, 6, 6); Insets smallInsets = new Insets(3, 2, 3, 1); Insets noInsets = new Insets(0, 0, 0, 0); // x, y, w, h, wx,wy,ix,iy addComponent(gbl, panelSearchArea, buttonScan, GridBagConstraints.HORIZONTAL, GridBagConstraints.EAST, noInsets, 0, 1, 1, 1, 1, 1, 40, 10); addComponent(gbl, panelSearchArea, labelSearchingDirectoryInfo, GridBagConstraints.HORIZONTAL, GridBagConstraints.EAST, noInsets, 0, 2, 1, 1, 0, 0, 0, 0); addComponent(gbl, panelSearchArea, progressBarSearching, GridBagConstraints.HORIZONTAL, GridBagConstraints.EAST, noInsets, 0, 3, 1, 1, 0, 0, 0, 0); addComponent(gbl, panelDirectory, labelDirectoryDescription, null, GridBagConstraints.WEST, new Insets(6, 6, 0, 6), 0, 0, 3, 1, 0, 0, 0, 0); addComponent(gbl, panelDirectory, textfieldDirectoryPath, GridBagConstraints.HORIZONTAL, null, basicInsets, 0, 1, 2, 1, 1, 1, 0, 0); addComponent(gbl, panelDirectory, buttonBrowse, GridBagConstraints.HORIZONTAL, GridBagConstraints.EAST, basicInsets, 2, 1, 1, 1, 0, 0, 0, 0); addComponent(gbl, panelDirectory, labelFileTypesDescription, GridBagConstraints.NONE, GridBagConstraints.WEST, new Insets(18, 6, 18, 3), 0, 3, 1, 1, 0, 0, 0, 0); addComponent(gbl, panelDirectory, comboBoxFileTypeSelection, GridBagConstraints.HORIZONTAL, GridBagConstraints.WEST, new Insets(18, 3, 18, 6), 1, 3, 1, 1, 1, 0, 0, 0); addComponent(gbl, panelDirectory, panelSearchArea, GridBagConstraints.HORIZONTAL, GridBagConstraints.EAST, new Insets(18, 6, 18, 6), 2, 3, 1, 1, 0, 0, 0, 0); addComponent(gbl, panelFiles, labelFilesDescription, GridBagConstraints.HORIZONTAL, GridBagConstraints.WEST, new Insets(6, 6, 0, 6), 0, 0, 1, 1, 0, 0, 0, 0); addComponent(gbl, panelFiles, scrollpaneTree, GridBagConstraints.BOTH, GridBagConstraints.CENTER, basicInsets, 0, 1, 1, 1, 1, 1, 0, 0); addComponent(gbl, panelFiles, panelOptions, GridBagConstraints.NONE, GridBagConstraints.NORTHEAST, basicInsets, 1, 1, 1, 1, 0, 0, 0, 0); addComponent(gbl, panelOptions, buttonOptionSelectAll, GridBagConstraints.HORIZONTAL, GridBagConstraints.NORTH, noInsets, 0, 0, 1, 1, 1, 0, 0, 0); addComponent(gbl, panelOptions, buttonOptionUnselectAll, GridBagConstraints.HORIZONTAL, GridBagConstraints.NORTH, noInsets, 0, 1, 1, 1, 0, 0, 0, 0); addComponent(gbl, panelOptions, buttonOptionExpandAll, GridBagConstraints.HORIZONTAL, GridBagConstraints.NORTH, new Insets(6, 0, 0, 0), 0, 2, 1, 1, 0, 0, 0, 0); addComponent(gbl, panelOptions, buttonOptionCollapseAll, GridBagConstraints.HORIZONTAL, GridBagConstraints.NORTH, noInsets, 0, 3, 1, 1, 0, 0, 0, 0); addComponent(gbl, panelEntryTypesSelection, labelEntryTypeDescription, GridBagConstraints.NONE, GridBagConstraints.WEST, basicInsets, 0, 0, 1, 1, 0, 0, 0, 0); addComponent(gbl, panelEntryTypesSelection, comboBoxEntryTypeSelection, GridBagConstraints.NONE, GridBagConstraints.WEST, basicInsets, 1, 0, 1, 1, 1, 0, 0, 0); addComponent(gbl, panelEntryTypesSelection, checkboxCreateKeywords, GridBagConstraints.HORIZONTAL, GridBagConstraints.WEST, basicInsets, 0, 1, 2, 1, 0, 0, 0, 0); addComponent(gbl, panelImportArea, labelImportingInfo, GridBagConstraints.HORIZONTAL, GridBagConstraints.CENTER, new Insets(6, 6, 0, 6), 0, 1, 1, 1, 1, 0, 0, 0); addComponent(gbl, panelImportArea, progressBarImporting, GridBagConstraints.HORIZONTAL, GridBagConstraints.CENTER, new Insets(0, 6, 6, 6), 0, 2, 1, 1, 1, 0, 0, 0); addComponent(gbl, panelButtons, panelImportArea, GridBagConstraints.NONE, GridBagConstraints.EAST, smallInsets, 1, 0, 1, 1, 0, 0, 0, 0); addComponent(gbl, getContentPane(), panelDirectory, GridBagConstraints.HORIZONTAL, GridBagConstraints.CENTER, basicInsets, 0, 0, 1, 1, 0, 0, 0, 0); addComponent(gbl, getContentPane(), panelFiles, GridBagConstraints.BOTH, GridBagConstraints.NORTHWEST, new Insets(12, 6, 2, 2), 0, 1, 1, 1, 1, 1, 0, 0); addComponent(gbl, getContentPane(), panelEntryTypesSelection, GridBagConstraints.HORIZONTAL, GridBagConstraints.SOUTHWEST, new Insets(12, 6, 2, 2), 0, 2, 1, 1, 0, 0, 0, 0); addComponent(gbl, getContentPane(), panelButtons, GridBagConstraints.HORIZONTAL, GridBagConstraints.CENTER, new Insets(10, 6, 10, 6), 0, 3, 1, 1, 0, 0, 0, 0); ButtonBarBuilder bb = new ButtonBarBuilder(); bb.addGlue(); bb.addButton(buttonApply); bb.addButton(buttonClose); bb.addGlue(); bb.getPanel().setBorder(BorderFactory.createEmptyBorder(5,5,5,5)); panelImportArea.add(bb.getPanel(), GridBagConstraints.NONE); pack(); } /** * Adds a component to a container, using the specified gridbag-layout and * the supplied parameters. <br> * <br> * This method is simply used to ged rid of thousands of lines of code, * which inevitably rise when layouts such as the gridbag-layout is being * used. * * @param layout * The layout to be used. * @param container * The {@link Container}, to which the component will be added. * @param component * An AWT {@link Component}, that will be added to the container. * @param fill * A constant describing the fill behaviour (see * {@link GridBagConstraints}). Can be <code>null</code>, if no * filling wants to be specified. * @param anchor * A constant describing the anchor of the element in its parent * container (see {@link GridBagConstraints}). Can be * <code>null</code>, if no specification is needed. * @param gridX * The relative grid-X coordinate. * @param gridY * The relative grid-Y coordinate. * @param width * The relative width of the component. * @param height * The relative height of the component. * @param weightX * A value for the horizontal weight. * @param weightY * A value for the vertical weight. * @param insets * Insets of the component. Can be <code>null</code>. */ private final static void addComponent(GridBagLayout layout, Container container, Component component, Integer fill, Integer anchor, Insets insets, int gridX, int gridY, int width, int height, double weightX, double weightY, int ipadX, int ipadY) { container.setLayout(layout); GridBagConstraints constraints = new GridBagConstraints(); constraints.gridx = gridX; constraints.gridy = gridY; constraints.gridwidth = width; constraints.gridheight = height; constraints.weightx = weightX; constraints.weighty = weightY; constraints.ipadx = ipadX; constraints.ipady = ipadY; if (fill != null) constraints.fill = fill; if (insets != null) constraints.insets = insets; if (anchor != null) constraints.anchor = anchor; layout.setConstraints(component, constraints); container.add(component); } /** * Creates the tree view, that holds the data structure. <br> * <br> * Initially, the root node is <b>not</b> visible, so that the tree appears empty at the beginning. */ private void createTree() { /** * Mouse listener to listen for mouse events on the tree. <br> * This will mark the selected tree entry as "selected" or "unselected", * which will cause this nodes checkbox to appear as either "checked" or * "unchecked". */ treeMouseListener = new MouseAdapter() { public void mousePressed(MouseEvent e) { int x = e.getX(); int y = e.getY(); int row = tree.getRowForLocation(x, y); TreePath path = tree.getPathForRow(row); if (path != null) { CheckableTreeNode node = (CheckableTreeNode) path.getLastPathComponent(); if (e.getClickCount() == 2) { Object userObject = node.getUserObject(); if (userObject instanceof FileNodeWrapper && node.isLeaf()) { FileNodeWrapper fnw = (FileNodeWrapper) userObject; try { Util.openExternalViewer(JabRef.jrf.basePanel().metaData(), fnw.file.getAbsolutePath(), "pdf"); } catch (IOException e1) { Globals.logger("Error opening file"); } } } else { node.check(); tree.invalidate(); tree.repaint(); } } } }; CheckableTreeNode startNode = new CheckableTreeNode("ROOT"); DefaultTreeModel model = new DefaultTreeModel(startNode); tree.setModel(model); tree.setRootVisible(false); DefaultTreeCellRenderer renderer = new CheckboxTreeCellRenderer(); tree.setCellRenderer(renderer); tree.addMouseListener(treeMouseListener); } /** * Initialises the combobox that contains the available file types which * bibtex entries can be created of. */ private void createFileTypesCombobox() { List<FileFilter> fileFilterList = creatorManager.getFileFilterList(); Vector<FileFilter> vector = new Vector<FileFilter>(); for (FileFilter fileFilter : fileFilterList) { vector.add(fileFilter); } comboBoxFileTypeSelection = new JComboBox(vector); comboBoxFileTypeSelection.setRenderer(new DefaultListCellRenderer() { private static final long serialVersionUID = 8503499454763947465L; /* (non-Javadoc) * @see javax.swing.DefaultListCellRenderer#getListCellRendererComponent(javax.swing.JList, java.lang.Object, int, boolean, boolean) */ @Override public Component getListCellRendererComponent(JList list, Object value, int index, boolean isSelected, boolean cellHasFocus) { JLabel label = (JLabel) super.getListCellRendererComponent(list, value, index, isSelected, cellHasFocus); if (value instanceof EntryFromFileCreator) { EntryFromFileCreator creator = (EntryFromFileCreator) value; if (creator.getExternalFileType() != null) { label.setIcon(creator.getExternalFileType().getIcon()); } } return label; } }); } /** * Creates the ComboBox-View vor the Listbox that holds the Bibtex entry * types. */ private void createEntryTypesCombobox() { TreeMap<String, BibtexEntryType> entryTypes = BibtexEntryType.ALL_TYPES; Iterator<BibtexEntryType> iterator = entryTypes.values().iterator(); Vector<BibtexEntryTypeWrapper> list = new Vector<BibtexEntryTypeWrapper>(); list.add(new BibtexEntryTypeWrapper(null)); while(iterator.hasNext()) { list.add(new BibtexEntryTypeWrapper(iterator.next())); } comboBoxEntryTypeSelection = new JComboBox(list); } /** * Wrapper for displaying the Type {@link BibtexEntryType} in a Combobox. * * @author Nosh&Dan * @version 12.11.2008 | 01:02:30 * */ private static class BibtexEntryTypeWrapper { final BibtexEntryType entryType; BibtexEntryTypeWrapper(BibtexEntryType bibtexType) { this.entryType = bibtexType; } /* (non-Javadoc) * @see java.lang.Object#toString() */ @Override public String toString() { if (entryType == null) return Globals.lang("<No selection>"); return entryType.getName(); } } public static class CheckableTreeNode extends DefaultMutableTreeNode { private static final long serialVersionUID = -1131241777605570737L; private boolean isSelected; private JCheckBox checkbox; public CheckableTreeNode(Object userObject) { super(userObject); checkbox = new JCheckBox(); } /** * @return the checkbox */ public JCheckBox getCheckbox() { return checkbox; } /** * @param checkbox * the checkbox to set */ public void setCheckbox(JCheckBox checkbox) { this.checkbox = checkbox; } public void check() { setSelected(!isSelected); } @SuppressWarnings("unchecked") public void setSelected(boolean bSelected) { isSelected = bSelected; Enumeration<CheckableTreeNode> children = this.children(); while (children.hasMoreElements()) { CheckableTreeNode child = children.nextElement(); child.setSelected(bSelected); } } public boolean getSelected() { return isSelected; } } private static class CheckboxTreeCellRenderer extends DefaultTreeCellRenderer { private static final long serialVersionUID = 3737245079578074387L; protected FileSystemView fsv = FileSystemView.getFileSystemView(); public Component getTreeCellRendererComponent(final JTree tree, Object value, boolean sel, boolean expanded, boolean leaf, int row, boolean hasFocus) { Component nodeComponent = super.getTreeCellRendererComponent(tree, value, sel, expanded, leaf, row, hasFocus); CheckableTreeNode node = (CheckableTreeNode) value; FileNodeWrapper userObject = (FileNodeWrapper) node.getUserObject(); JPanel newPanel = new JPanel(); JCheckBox checkbox = node.getCheckbox(); checkbox.setSelected(node.getSelected()); try { setIcon(fsv.getSystemIcon(userObject.file)); } catch (Exception e) {} newPanel.setBackground(nodeComponent.getBackground()); checkbox.setBackground(nodeComponent.getBackground()); GridBagLayout gbl = new GridBagLayout(); addComponent(gbl, newPanel, checkbox, null, null, null, 0, 0, 1, 1, 0, 0, 0, 0); addComponent(gbl, newPanel, nodeComponent, GridBagConstraints.HORIZONTAL, null, new Insets(1, 2, 0, 0), 1, 0, 1, 1, 1, 0, 0, 0); if (userObject.fileCount > 0) { JLabel label = new JLabel("(" + userObject.fileCount + " file" + (userObject.fileCount > 1 ? "s" : "") + ")"); addComponent(gbl, newPanel, label, null, null, new Insets(1, 2, 0, 0), 2, 0, 1, 1, 0, 0, 0, 0); } return newPanel; } } public static class FileNodeWrapper { public final File file; public final int fileCount; public FileNodeWrapper(File aFile) { this(aFile, 0); } /** * @param directory * @param length */ public FileNodeWrapper(File aDirectory, int fileCount) { this.file = aDirectory; this.fileCount = fileCount; } /* * (non-Javadoc) * * @see java.lang.Object#toString() */ @Override public String toString() { return file.getName(); } } }