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

org.netbeans.swing.etable.ETableColumnModel Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package org.netbeans.swing.etable;

import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.Iterator;
import java.util.List;
import java.util.Properties;
import java.util.Vector;
import javax.swing.event.TableColumnModelEvent;
import javax.swing.table.DefaultTableColumnModel;
import javax.swing.table.TableColumn;
import org.netbeans.swing.etable.ETable.RowMapping;

/**
 * Column model implementation for ETable.
 * 
 * @author David Strupl
 */
public class ETableColumnModel extends DefaultTableColumnModel {

    /** Used as a key or part of a key by the persistence mechanism. */
    private static final String NUMBER_OF_COLUMNS = "ColumnsNumber";
    
    /** Used as a key or part of a key by the persistence mechanism. */
    private static final String NUMBER_OF_HIDDEN_COLUMNS = "HiddenColumnsNumber";
    
    /** Used as a key or part of a key by the persistence mechanism. */
    private static final String PROP_HIDDEN_PREFIX = "Hidden";
    private static final String PROP_HIDDEN_POSITION_PREFIX = "HiddenPosition";
    
    /**
     * List<ETableColumn>: holds list of sorted columns in this model.
     * If the list is empty if no sorting is applied.
     */
    protected transient List sortedColumns = new ArrayList();
    
    /**
     * List<ETableColumn>: holds list of columns that were hidden by the
     * user. The columns contained here are not contained in the inherited
     * tableColumns list.
     */
    protected List hiddenColumns = new ArrayList();
    
    List hiddenColumnsPosition = new ArrayList();
    
    /**
     * Allows the user to customize the list of the visible columns using
     * a hierarchy in a special TableColumnSelector.
     */
    private TableColumnSelector.TreeNode columnHierarchyRoot;
    
    /** Creates a new instance of ETableColumnModel */
    public ETableColumnModel() {
        super();
    }
    
    /**
     * Method allowing to read stored values.
     * The stored values should be only those that the user has customized,
     * it does not make sense to store the values that were set using 
     * the initialization code because the initialization code can be run
     * in the same way after restart.
     */
    public void readSettings(Properties p, String propertyPrefix, ETable table) {
        tableColumns = new Vector();
        sortedColumns = new ArrayList();
        String s = p.getProperty(propertyPrefix + NUMBER_OF_COLUMNS);
        int numColumns = Integer.parseInt(s);
        for (int i = 0; i < numColumns; i++) {
            ETableColumn etc = (ETableColumn)table.createColumn(i);
            etc.readSettings(p, i, propertyPrefix);
            addColumn(etc);
            if (etc.getComparator() != null) {
                int j = 0;
                for ( ; j < sortedColumns.size(); j++) {
                    ETableColumn setc = (ETableColumn)sortedColumns.get(j);
                    if (setc.getSortRank() > etc.getSortRank()) {
                        break;
                    }
                }
                sortedColumns.add(j, etc);
            }
        }
        hiddenColumns = new ArrayList();
        String sh = p.getProperty(propertyPrefix + NUMBER_OF_HIDDEN_COLUMNS);
        int numHiddenColumns = Integer.parseInt(sh);
        for (int i = 0; i < numHiddenColumns; i++) {
            ETableColumn etc = (ETableColumn)table.createColumn(0);
            etc.readSettings(p, i, propertyPrefix + PROP_HIDDEN_PREFIX);
            hiddenColumns.add(etc);
        }
        hiddenColumnsPosition = new ArrayList();
        for (int i = 0; i < numHiddenColumns; i++) {
            String myPrefix = propertyPrefix + PROP_HIDDEN_POSITION_PREFIX + ETableColumn.PROP_PREFIX + Integer.toString(i);
            String posStr = p.getProperty(myPrefix);
            int pos = Integer.parseInt(posStr);
            hiddenColumnsPosition.add(pos);
        }
    }

    /**
     * Method allowing to store customization values.
     * The stored values should be only those that the user has customized,
     * it does not make sense to store the values that were set using 
     * the initialization code because the initialization code can be run
     * in the same way after restart.
     */
    public void writeSettings(Properties p, String propertyPrefix) {
        int i = 0;
        int numColumns = tableColumns.size();
        p.setProperty(propertyPrefix + NUMBER_OF_COLUMNS, Integer.toString(numColumns));
        for (Object obj : tableColumns) {
            if (obj instanceof ETableColumn) {
                ETableColumn etc = (ETableColumn) obj;
                etc.writeSettings(p, i++, propertyPrefix);
            }
        }
        i = 0;
        int numHiddenColumns = hiddenColumns.size();
        p.setProperty(propertyPrefix + NUMBER_OF_HIDDEN_COLUMNS, Integer.toString(numHiddenColumns));
        for (Object obj : hiddenColumns) {
            if (obj instanceof ETableColumn) {
                ETableColumn etc = (ETableColumn) obj;
                etc.writeSettings(p, i++, propertyPrefix + PROP_HIDDEN_PREFIX);
            }
        }
        for (i = 0; i < numHiddenColumns; i++) {
            int pos = hiddenColumnsPosition.get(i);
            String myPrefix = propertyPrefix + PROP_HIDDEN_POSITION_PREFIX + ETableColumn.PROP_PREFIX + Integer.toString(i);
            p.setProperty(myPrefix, Integer.toString(pos));
        }
    }

    /**
     * Provides a comparator for sorting the rows of the table. The comparator
     * operates over ETable.RowMappings objects.
     * @return The comparator for sorting the rows of the table.
     */
    public Comparator getComparator() {
        if (sortedColumns.isEmpty()) {
            return new ETable.OriginalRowComparator();
        }
        return new CompoundComparator();
    }

    /** This method marks this column as sorted. Value 0 of the parameter rank
     * means that this column is not sorted. 

* This method does not refresh the table UI. * Use {@link ETable#setColumnSorted(int, boolean, int)} instead when the UI * should be changed accordingly. * @param etc column in ETable column model * @param ascending true means ascending * @param newRank value 1 means that this is the most important sorted * column, number 2 means second etc. * @since 1.3 */ @SuppressWarnings("deprecation") public void setColumnSorted(ETableColumn etc, boolean ascending, int newRank) { if (! etc.isSortingAllowed()) { return; } // TODO: check the implementation! boolean wasSorted = sortedColumns.contains(etc); if (wasSorted) { etc.setAscending(ascending); etc.setSortRank(newRank); sortedColumns.remove(etc); } else { etc.setSorted(newRank, ascending); } if (newRank > 0) { sortedColumns.add(newRank-1, etc); } } /** * */ @SuppressWarnings("deprecation") void toggleSortedColumn(ETableColumn etc, boolean cleanAll) { if (! etc.isSortingAllowed()) { return; } boolean wasSorted = sortedColumns.contains(etc); if (cleanAll) { clearSortedColumns(etc); } if (wasSorted) { if (etc.isAscending()) { etc.setAscending(false); } else { sortedColumns.remove(etc); etc.setSorted(0, false); } updateRanks(); } else { etc.setSorted(sortedColumns.size()+1, true); sortedColumns.add(etc); } } @Override public void removeColumn(TableColumn column) { removeColumn(column, true); } private void removeColumn(TableColumn column, boolean doShift) { if (sortedColumns.remove(column)) { int i = 1; for (TableColumn sc : sortedColumns) { if (sc instanceof ETableColumn) { ETableColumn etc = (ETableColumn) sc; etc.setSorted(i, etc.isAscending()); } i++; } } if (removeHiddenColumn(column, doShift) < 0) { int columnIndex = tableColumns.indexOf(column); super.removeColumn(column); if (doShift) { int n = hiddenColumnsPosition.size(); for (int i = 0; i < n; i++) { if (hiddenColumnsPosition.get(i) <= columnIndex) { columnIndex++; } } for (int i = 0; i < n; i++) { int index = hiddenColumnsPosition.get(i); if (index > columnIndex) { hiddenColumnsPosition.set(i, --index); } } } } } /** @return the column position of the hidden column or -1 if the column is not hidden */ private int removeHiddenColumn(TableColumn column, boolean doShift) { int hiddenIndex = -1; for (int i = 0; i < hiddenColumns.size(); i++) { if (column.equals(hiddenColumns.get(i))) { hiddenIndex = i; break; } } if (hiddenIndex >= 0) { hiddenColumns.remove(hiddenIndex); int hi = hiddenColumnsPosition.remove(hiddenIndex); if (doShift) { int n = hiddenColumnsPosition.size(); for (int i = 0; i < n; i++) { int index = hiddenColumnsPosition.get(i); if (index > hi) { hiddenColumnsPosition.set(i, --index); } } } return hi; } else { return -1; } } /** * Makes the given column hidden or visible according to the parameter * hidden. * @param column The table column to change the visibility. * @param hidden true to make the column hidden, false to make it visible. */ public void setColumnHidden(TableColumn column, boolean hidden) { if (hidden) { if (! hiddenColumns.contains(column)) { int index = tableColumns.indexOf(column); if (index >= 0) { removeColumn(column, false); hiddenColumns.add(column); for (Integer pos : hiddenColumnsPosition) { if (pos <= index) index++; } while (hiddenColumnsPosition.contains(index)) index++; hiddenColumnsPosition.add(index); } } } else { if (! tableColumns.contains(column)) { int index = removeHiddenColumn(column, false); if (index >= 0) { int i = index; for (Integer pos : hiddenColumnsPosition) { if (pos < index) i--; } index = Math.min(i, tableColumns.size()); addColumn(column, index); } } } } /** Copy of addColumn(TableColumn) with an index specifying where to add the new column */ private void addColumn(TableColumn aColumn, int index) { if (aColumn == null) { throw new IllegalArgumentException("Object is null"); } tableColumns.insertElementAt(aColumn, index); aColumn.addPropertyChangeListener(this); //invalidateWidthCache(); totalColumnWidth = -1; // Post columnAdded event notification fireColumnAdded(new TableColumnModelEvent(this, 0, index)); } @Override public void moveColumn(int ci1, int ci2) { super.moveColumn(ci1, ci2); // hidden positions between ci1 and ci2 need to be adjusted: int n = hiddenColumns.size(); // Shift ci1 and ci2 by hidden columns: for (int i = 0; i < n; i++) { int index = hiddenColumnsPosition.get(i); if (ci1 >= index) ci1++; if (ci2 >= index) ci2++; } if (ci1 < ci2) { for (int i = 0; i < n; i++) { int index = hiddenColumnsPosition.get(i); if (ci1 < index && index <= ci2) { hiddenColumnsPosition.set(i, --index); } } } if (ci2 < ci1) { for (int i = 0; i < n; i++) { int index = hiddenColumnsPosition.get(i); if (ci2 <= index && index < ci1) { hiddenColumnsPosition.set(i, ++index); } } } } /** Get all columns, including hidden ones. */ List getAllColumns() { List columns = Collections.list(getColumns()); int n = hiddenColumns.size(); for (int i = 0; i < n; i++) { int index = hiddenColumnsPosition.get(i); index = Math.min(index, columns.size()); columns.add(index, hiddenColumns.get(i)); } return columns; } /** * Test if the column is hidden or visible. * @param tc The table column to test * @return true when the column is hidden, false when it's visible. */ public boolean isColumnHidden(TableColumn tc) { return hiddenColumns.contains(tc); } /** * Makes the whole table unsorted. */ @SuppressWarnings("deprecation") public void clearSortedColumns() { for (Object o : sortedColumns) { if (o instanceof ETableColumn) { ETableColumn etc = (ETableColumn)o; etc.setSorted(0, false); } } sortedColumns = new ArrayList(); } /** * Makes the whole table unsorted except for one column. */ @SuppressWarnings("deprecation") void clearSortedColumns(TableColumn notThisOne) { boolean wasSorted = sortedColumns.contains(notThisOne); for (Object o : sortedColumns) { if ((o instanceof ETableColumn) && (o != notThisOne)) { ETableColumn etc = (ETableColumn)o; etc.setSorted(0, false); } } sortedColumns = new ArrayList(); if (wasSorted) { sortedColumns.add(notThisOne); } } /** * Reasigns sorting ranks to ETableColumns contained in sortedColumns list. */ private void updateRanks() { int i = 1; for (Object o : sortedColumns) { if (o instanceof ETableColumn) { ETableColumn etc = (ETableColumn)o; if (etc.isSorted()) { etc.setSortRank(i); } } } } List getSortedColumns () { return sortedColumns; } /** * Removes all columns, including hidden ones. * @since 1.12 */ public void clean() { List allColumns = new ArrayList(tableColumns.size() + hiddenColumns.size()); allColumns.addAll(tableColumns); allColumns.addAll(hiddenColumns); for (TableColumn tc : allColumns) { removeColumn(tc); } } /** * Comparator that delegates to individual comparators supplied by * ETableColumns. It uses only the columns contained in the sortedColumns * list. */ private class CompoundComparator implements Comparator { private Comparator original; public CompoundComparator() { original = new ETable.OriginalRowComparator(); } @Override public int compare(RowMapping o1, RowMapping o2) { for (Object o : sortedColumns) { if (o instanceof ETableColumn) { ETableColumn etc = (ETableColumn)o; Comparator c = etc.getComparator(); if (c != null) { int res = c.compare(o1, o2); if (res != 0) { return res; } } } } return original.compare(o1, o2); } } /** * Allows the user to customize the list of the visible columns using * a hierarchy in a special TableColumnSelector. */ public TableColumnSelector.TreeNode getColumnHierarchyRoot() { return columnHierarchyRoot; } /** * Allows the user to customize the list of the visible columns using * a hierarchy in a special TableColumnSelector. */ public void setColumnHierarchyRoot(TableColumnSelector.TreeNode columnHierarchyRoot) { this.columnHierarchyRoot = columnHierarchyRoot; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy