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

org.jdesktop.swingx.search.TableSearchable Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2007 Sun Microsystems, Inc., 4150 Network Circle,
 * Santa Clara, California 95054, U.S.A. All rights reserved.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 * 
 * This library 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
 * Lesser General Public License for more details.
 * 
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 */
package org.jdesktop.swingx.search;

import java.awt.Rectangle;
import java.util.logging.Logger;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import javax.swing.tree.TreePath;

import org.jdesktop.swingx.JXTable;
import org.jdesktop.swingx.JXTree;
import org.jdesktop.swingx.JXTreeTable;
import org.jdesktop.swingx.decorator.AbstractHighlighter;
import org.jdesktop.swingx.decorator.Highlighter;
import org.jdesktop.swingx.renderer.StringValue;

/**
 * An Searchable implementation for use in JXTable.
 * 
 * @author Jeanette Winzenburg
 * @author EUG https://github.com/homebeaver (use in JXTreeTable)
 */
public class TableSearchable extends AbstractSearchable {

    private static final Logger LOG = Logger.getLogger(TableSearchable.class.getName());

    /** The target JXTable. */
    protected JXTable table;

    /**
     * Instantiates a TableSearchable with the given table as target.
     * 
     * @param table the JXTable to search.
     */
    public TableSearchable(JXTable table) {
        this.table = table;
        if(this.table instanceof JXTreeTable tt) {
            LOG.config("searchable table:"+tt);
        }
    }

    /**
     * {@inheritDoc}
     * 

* * This implementation loops through the cells in a row to find a match. */ @Override protected void findMatchAndUpdateState(Pattern pattern, int startRow, boolean backwards) { SearchResult matchRow = null; if (backwards) { // CHECK: off-one end still needed? // Probably not - the findXX don't have side-effects any longer // hmmm... still needed: even without side-effects we need to // guarantee calling the notfound update at the very end of the loop. for (int r = startRow; r >= -1 && matchRow == null; r--) { matchRow = findMatchBackwardsInRow(pattern, r); updateState(matchRow); } } else { for (int r = startRow; r <= getSize() && matchRow == null; r++) { matchRow = findMatchForwardInRow(pattern, r); updateState(matchRow); } } // KEEP - JW: Needed to update if loop wasn't entered! // the alternative is to go one off in the loop. Hmm - which is // preferable? // updateState(matchRow); } /** * {@inheritDoc} *

* * Implemented to search for an extension in the cell given by row and foundColumn. */ @Override protected SearchResult findExtendedMatch(Pattern pattern, int row) { return findMatchAt(pattern, row, lastSearchResult.foundColumn); } /** * Searches forward through columns of the given row. Starts at * lastFoundColumn or first column if lastFoundColumn < 0. returns an * appropriate SearchResult if a matching cell is found in this row or null * if no match is found. A row index out off range results in a no-match. * * @param pattern Pattern that we will try to locate * @param row the row to search * @return an appropriate SearchResult if a matching cell is * found in this row or null if no match is found */ private SearchResult findMatchForwardInRow(Pattern pattern, int row) { int startColumn = (lastSearchResult.foundColumn < 0) ? 0 : lastSearchResult.foundColumn; if (isValidIndex(row)) { for (int column = startColumn; column < table.getColumnCount(); column++) { SearchResult result = findMatchAt(pattern, row, column); if (result != null) return result; } } return null; } /** * Searches forward through columns of the given row. Starts at * lastFoundColumn or first column if lastFoundColumn < 0. returns an * appropriate SearchResult if a matching cell is found in this row or null * if no match is found. A row index out off range results in a no-match. * * @param pattern Pattern that we will try to locate * @param row the row to search * @return an appropriate SearchResult if a matching cell is * found in this row or null if no match is found */ private SearchResult findMatchBackwardsInRow(Pattern pattern, int row) { int startColumn = (lastSearchResult.foundColumn < 0) ? table.getColumnCount() - 1 : lastSearchResult.foundColumn; if (isValidIndex(row)) { for (int column = startColumn; column >= 0; column--) { SearchResult result = findMatchAt(pattern, row, column); if (result != null) return result; } } return null; } /** * Matches the cell content at row/col against the given Pattern. Returns an * appropriate SearchResult if matching or null if no matching * * @param pattern Pattern that we will try to locate * @param row a valid row index in view coordinates * @param column a valid column index in view coordinates * @return an appropriate SearchResult if matching or null */ protected SearchResult findMatchAt(Pattern pattern, int row, int column) { String text; if(table instanceof JXTreeTable tt) { text = tt.getStringAt(row, column); Object renderer = tt.getCellRenderer(row, column); if(renderer instanceof JXTree xTree) { // for hierarchical column StringValue sv = (StringValue)xTree.getCellRenderer(); TreePath path = xTree.getPathForRow(row); text = sv.getString(path.getLastPathComponent()); LOG.fine("hierarchical column "+column + " row="+row + " text="+text); } } else { text = table.getStringAt(row, column); } if ((text != null) && (text.length() > 0)) { Matcher matcher = pattern.matcher(text); if (matcher.find()) { LOG.fine("matcher.find() is TRUE for column "+column+" row="+row); return createSearchResult(matcher, row, column); } } return null; } /** * * {@inheritDoc} *

* * Overridden to adjust the column index to -1. */ @Override protected int adjustStartPosition(int startIndex, boolean backwards) { lastSearchResult.foundColumn = -1; return super.adjustStartPosition(startIndex, backwards); } /** * {@inheritDoc} *

* * Overridden to loop through all columns in a row. */ @Override protected int moveStartPosition(int startRow, boolean backwards) { if (backwards) { lastSearchResult.foundColumn--; if (lastSearchResult.foundColumn < 0) { startRow--; } } else { lastSearchResult.foundColumn++; // check tt.getColumnCount() == table.getColumnCount() // if(table instanceof JXTreeTable tt) { // assert table.getColumnCount() == tt.getColumnCount(); // } if (lastSearchResult.foundColumn >= table.getColumnCount()) { lastSearchResult.foundColumn = -1; startRow++; } } return startRow; } /** * {@inheritDoc} *

* * Overridden to check the column index of last find. */ @Override protected boolean isEqualStartIndex(final int startIndex) { return super.isEqualStartIndex(startIndex) && isValidColumn(lastSearchResult.foundColumn); } /** * Checks if row is in range: 0 <= row < getRowCount(). * * @param column the column index to check in view coordinates. * @return true if the column is in range, false otherwise */ private boolean isValidColumn(int column) { return column >= 0 && column < table.getColumnCount(); } /** * {@inheritDoc} */ @Override protected int getSize() { return table.getRowCount(); } /** * {@inheritDoc} */ @Override public JXTable getTarget() { return table; } /** * Configures the match highlighter to the current match. Ensures that the * matched cell is visible, if there is a match. * * PRE: markByHighlighter * */ protected void moveMatchByHighlighter() { AbstractHighlighter searchHL = getConfiguredMatchHighlighter(); // no match if (!hasMatch()) { return; } else { ensureInsertedSearchHighlighters(searchHL); if(table instanceof JXTreeTable tt) { tt.scrollCellToVisible(lastSearchResult.foundRow, lastSearchResult.foundColumn); } else { table.scrollCellToVisible(lastSearchResult.foundRow, lastSearchResult.foundColumn); } } } /** * {@inheritDoc} *

* * Overridden to convert the column index in the table's view coordinate * system to model coordinate. *

* * PENDING JW: this is only necessary because the SearchPredicate wants its * highlight column in model coordinates. But code comments in the * SearchPredicate seem to indicate that we probably want to revise that * (legacy?). */ @Override protected int convertColumnIndexToModel(int viewColumn) { return getTarget().convertColumnIndexToModel(viewColumn); } /** * Moves the row selection to the matching cell and ensures its visibility, * if any. Does nothing if there is no match. * */ protected void moveMatchBySelection() { if (!hasMatch()) { return; } int row = lastSearchResult.foundRow; int column = lastSearchResult.foundColumn; if(table instanceof JXTreeTable tt) { tt.changeSelection(row, column, false, false); if (!tt.getAutoscrolls()) { // scrolling not handled by moving selection Rectangle cellRect = tt.getCellRect(row, column, true); if (cellRect != null) { tt.scrollRectToVisible(cellRect); } } } else { table.changeSelection(row, column, false, false); if (!table.getAutoscrolls()) { // scrolling not handled by moving selection Rectangle cellRect = table.getCellRect(row, column, true); if (cellRect != null) { table.scrollRectToVisible(cellRect); } } } } /** * {@inheritDoc} *

*/ @Override protected void moveMatchMarker() { if (markByHighlighter()) { moveMatchByHighlighter(); } else { // use selection moveMatchBySelection(); } } /** * {@inheritDoc} *

*/ @Override protected void removeHighlighter(Highlighter searchHighlighter) { // LOG.info("TODO "+searchHighlighter); if(table instanceof JXTreeTable tt) { Object renderer = tt.getTreeCellRenderer(); if(renderer instanceof JXTree xTree) { xTree.removeHighlighter(searchHighlighter); } tt.removeHighlighter(searchHighlighter); } else { table.removeHighlighter(searchHighlighter); } } /** * {@inheritDoc} *

*/ @Override protected Highlighter[] getHighlighters() { if(table instanceof JXTreeTable tt) { return tt.getHighlighters(); } else { return table.getHighlighters(); } } /** * {@inheritDoc} *

*/ @Override protected void addHighlighter(Highlighter highlighter) { // LOG.info("TODO "+highlighter); if(table instanceof JXTreeTable tt) { Object renderer = tt.getTreeCellRenderer(); if(renderer instanceof JXTree xTree) { xTree.addHighlighter(highlighter); } tt.addHighlighter(highlighter); } else { table.addHighlighter(highlighter); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy