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

impl.org.controlsfx.spreadsheet.GridViewSkin Maven / Gradle / Ivy

/**
 * Copyright (c) 2013, 2021 ControlsFX
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *     * Redistributions of source code must retain the above copyright
 * notice, this list of conditions and the following disclaimer.
 *     * Redistributions in binary form must reproduce the above copyright
 * notice, this list of conditions and the following disclaimer in the
 * documentation and/or other materials provided with the distribution.
 *     * Neither the name of ControlsFX, any associated website, nor the
 * names of its contributors may be used to endorse or promote products
 * derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL CONTROLSFX BE LIABLE FOR ANY
 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package impl.org.controlsfx.spreadsheet;

import java.time.LocalDate;
import java.util.BitSet;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;

import javafx.beans.InvalidationListener;
import javafx.beans.Observable;
import javafx.beans.property.BooleanProperty;
import javafx.beans.property.SimpleBooleanProperty;
import javafx.collections.FXCollections;
import javafx.collections.ListChangeListener;
import javafx.collections.ObservableList;
import javafx.collections.ObservableMap;
import javafx.collections.ObservableSet;
import javafx.collections.SetChangeListener;
import javafx.event.EventHandler;
import javafx.geometry.HPos;
import javafx.geometry.VPos;
import javafx.scene.Node;
import javafx.scene.control.IndexedCell;
import javafx.scene.control.TableCell;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableColumnBase;
import javafx.scene.control.TableFocusModel;
import javafx.scene.control.TableRow;
import javafx.scene.control.TableView;
import javafx.scene.input.MouseEvent;
import javafx.scene.layout.Region;
import javafx.stage.Screen;
import javafx.util.Callback;

import org.controlsfx.control.spreadsheet.Grid;
import org.controlsfx.control.spreadsheet.SpreadsheetCell;
import org.controlsfx.control.spreadsheet.SpreadsheetColumn;
import org.controlsfx.control.spreadsheet.SpreadsheetView;

import javafx.application.Platform;
import javafx.event.Event;
import javafx.scene.control.ScrollBar;
import javafx.scene.control.skin.TableHeaderRow;
import javafx.scene.control.skin.TableViewSkinBase;
import javafx.scene.control.skin.VirtualFlow;

/**
 * This skin is actually the skin of the SpreadsheetGridView (tableView)
 * contained within the SpreadsheetView. The skin for the SpreadsheetView itself
 * currently resides inside the SpreadsheetView constructor!
 *
 * We need to extends directly from TableViewSkinBase in order to work-around
 * https://javafx-jira.kenai.com/browse/RT-34753 if we want to set a custom
 * TableViewBehavior.
 *
 */
public class GridViewSkin extends TableViewSkinBase,ObservableList,TableView>,TableRow>,TableColumn,?>> {
        
    /***************************************************************************
     * * STATIC FIELDS * *
     **************************************************************************/

    /** Default height of a row. */
    public static final double DEFAULT_CELL_HEIGHT = 24.0;

    // FIXME This should seriously be investigated ..
    private static final double DATE_CELL_MIN_WIDTH = 200 - Screen.getPrimary().getDpi();

    /**
     * When we add some tableCell to some topRow in order for them to be on top
     * in term of z-order. We may end up with the situation where the row that
     * put the cell is not in the ViewPort anymore. For example when a fixedRow
     * has taken over the real row when scrolling down. Then, the tableCell
     * added is still hanging out in the topRow. That tableCell has no clue that
     * its "creator" has been destroyed or re-used since that tableCell was not
     * technically belonging to its "creator". Therefore, we need to track those
     * cells in order to remove them each time.
     */
    final Map> deportedCells = new HashMap<>();
    /***************************************************************************
     * * PRIVATE FIELDS * *
     **************************************************************************/
    /**
     * When resizing, we save the height here in order to override default row
     * height. package protected.
     */
    ObservableMap rowHeightMap = FXCollections.observableHashMap();

    /** The editor. */
    private GridCellEditor gridCellEditor;

    protected final SpreadsheetHandle handle;
    protected SpreadsheetView spreadsheetView;
    protected VerticalHeader verticalHeader;
    protected HorizontalPicker horizontalPickers;
    
    /**
     * The currently fixedRow. This handles an Integer's set of rows being
     * fixed. NOT Fixable but truly fixed.
     */
    private ObservableSet currentlyFixedRow = FXCollections.observableSet(new HashSet());

    /**
     * A list of Integer with the current selected Rows. This is useful for
     * HorizontalHeader and VerticalHeader because they need to highlight when a
     * selection is made.
     */
    private final ObservableList selectedRows = FXCollections.observableArrayList();

    /**
     * A list of Integer with the current selected Columns. This is useful for
     * HorizontalHeader and VerticalHeader because they need to highlight when a
     * selection is made.
     */
    private final ObservableList selectedColumns = FXCollections.observableArrayList();

    /**
     * The total height of the currently fixedRows.
     */
    private double fixedRowHeight = 0;

    /**
     * These variable try to optimize the layout of the rows in order not to layout
     * every time every row.
     * 
     * So rowToLayout contains the rows that really needs layout(contain span or fixed).
     * 
     * And hBarValue is an indicator for the VirtualFlow. When the Hbar is touched, this BitSet
     * is set to false. And when a row is drawing, it flips its value in this BitSet. 
     * So that we know when scrolling up or down whether a row has taken into account
     * that the HBar was moved (otherwise, blank area may appear).
     */
    BitSet hBarValue;
    BitSet rowToLayout;
    
    /**
     * This rectangle will be used for drawing a border around the selection.
     */
    RectangleSelection rectangleSelection;
    
    /**
     * This is the current width used by the currently fixed column on the left. 
     */
    double fixedColumnWidth;
    
    /**
     * When we try to select cells after a setGrid, we end up with the cell
     * selected but no visual confirmation. In order to prevent that, we need to
     * warn the selectionModel when the layout is starting and then the
     * selectionModel will do the appropriate actions in order to force the
     * visual to come.
     */
    BooleanProperty lastRowLayout = new SimpleBooleanProperty(true);
    private GridViewBehavior behavior;
    
    /***************************************************************************
     * * CONSTRUCTOR * *
     **************************************************************************/
    public GridViewSkin(final SpreadsheetHandle handle) {
        super(handle.getGridView());
        behavior = new GridViewBehavior(handle.getGridView());
        
        this.handle = handle;
        this.spreadsheetView = handle.getView();
        gridCellEditor = new GridCellEditor(handle);
        TableView> tableView = handle.getGridView();

        //Set a new row factory, useful when handling row height.
        tableView.setRowFactory(new Callback>, TableRow>>() {
            @Override
            public TableRow> call(TableView> p) {
                return new GridRow(handle);
            }
        });
        getVirtualFlow().setCellFactory(param -> createCell());

        tableView.getStyleClass().add("cell-spreadsheet"); //$NON-NLS-1$

        getCurrentlyFixedRow().addListener(currentlyFixedRowListener);
        spreadsheetView.getFixedRows().addListener(fixedRowsListener);
        spreadsheetView.getFixedColumns().addListener(fixedColumnsListener);

        init();
        /**
         * When we are changing the grid we re-instantiate the rowToLayout because
         * spans and fixedRow may have changed.
         */
        handle.getView().gridProperty().addListener(rowToLayoutListener);
        handle.getView().hiddenRowsProperty().addListener(rowToLayoutListener);
        handle.getView().hiddenColumnsProperty().addListener(rowToLayoutListener);
        
        hBarValue = new BitSet(getItemCount());
        rowToLayout = initRowToLayoutBitSet();
        // Because fixedRow Listener is not reacting first time.
        computeFixedRowHeight();
        
        
        EventHandler ml = (MouseEvent event) -> {
            // RT-15127: cancel editing on scroll. This is a bit extreme
            // (we are cancelling editing on touching the scrollbars).
            // This can be improved at a later date.
            if (tableView.getEditingCell() != null) {
                tableView.edit(-1, null); 
            }
            
            // This ensures that the table maintains the focus, even when the vbar
            // and hbar controls inside the flow are clicked. Without this, the
            // focus border will not be shown when the user interacts with the
            // scrollbars, and more importantly, keyboard navigation won't be
            // available to the user.
            tableView.requestFocus();
        };
        
        getFlow().getVerticalBar().addEventFilter(MouseEvent.MOUSE_PRESSED, ml);
        getFlow().getHorizontalBar().addEventFilter(MouseEvent.MOUSE_PRESSED, ml);

        // init the behavior 'closures'
        GridViewBehavior behavior = getBehavior();
        behavior.setOnFocusPreviousRow(() -> onFocusAboveCell());
        behavior.setOnFocusNextRow(() -> onFocusBelowCell());
        behavior.setOnMoveToFirstCell(() -> onMoveToFirstCell());
        behavior.setOnMoveToLastCell(() -> onMoveToLastCell());
        behavior.setOnScrollPageDown(isFocusDriven -> onScrollPageDown(isFocusDriven));
        behavior.setOnScrollPageUp(isFocusDriven -> onScrollPageUp(isFocusDriven));
        behavior.setOnSelectPreviousRow(() -> onSelectAboveCell());
        behavior.setOnSelectNextRow(() -> onSelectBelowCell());
        behavior.setOnSelectLeftCell(() -> onSelectLeftCell());
        behavior.setOnSelectRightCell(() -> onSelectRightCell());
        behavior.setOnFocusLeftCell(() -> onFocusLeftCell());
        behavior.setOnFocusRightCell(() -> onFocusRightCell());

        this.registerChangeListener(tableView.fixedCellSizeProperty(), (var1x) -> {
            getFlow().setFixedCellSize(((TableView)this.getSkinnable()).getFixedCellSize());
        });
    }

    private TableRow> createCell() {
        TableRow> row = null;

        TableView> tableView = getSkinnable();
        row = tableView.getRowFactory().call(tableView);

        row.updateTableView(tableView);
        return row;
    }
    
    private InvalidationListener rowToLayoutListener = new InvalidationListener() {
        @Override
        public void invalidated(Observable observable) {
            rowToLayout = initRowToLayoutBitSet();
        }
    };
    /**
     * Compute the height of a particular row. If the row is in
     * {@link Grid#AUTOFIT}, {@link #DEFAULT_CELL_HEIGHT} is returned.
     *
     * @param row
     * @return
     */
    public double getRowHeight(int row) {
        if (row == -1) {
            return DEFAULT_CELL_HEIGHT;
        }
        Double rowHeightCache = rowHeightMap.get(spreadsheetView.getModelRow(row));
        if (rowHeightCache == null) {
            double rowHeight = handle.getView().getGrid().getRowHeight(spreadsheetView.getModelRow(row));
            return rowHeight == Grid.AUTOFIT ? DEFAULT_CELL_HEIGHT : rowHeight;
        } else {
            return rowHeightCache;
        }
    }

    public double getFixedRowHeight() {
        return fixedRowHeight;
    }

    /**
     * Contains the index of the sortedList.
     * @return 
     */
    public ObservableList getSelectedRows() {
        return selectedRows;
    }

    public ObservableList getSelectedColumns() {
        return selectedColumns;
    }

    public GridCellEditor getSpreadsheetCellEditorImpl() {
        return gridCellEditor;
    }

    /**
     * This return the GridRow which has the specified index if found. Otherwise
     * null is returned.
     *
     * @param index
     * @return
     */
    public GridRow getRowIndexed(int index) {
        List cells = getFlow().getCells();
        if (!cells.isEmpty()) {
            IndexedCell cell = cells.get(0);
            if (index >= cell.getIndex() && index - cell.getIndex() < cells.size()) {
                return (GridRow) cells.get(index - cell.getIndex());
            }
        }
        for (IndexedCell cell : getFlow().getFixedCells()) {
            if (cell.getIndex() == index) {
                return (GridRow) cell;
            }
        }
        return null;
    }
    
    /**
     * This return the first index displaying a cell in case of a rowSpan. If
     * the returned index is the same as given, it means the current cell is the
     * one showing. Otherwise, it means another cell above will be the one
     * drawn.
     *
     * @param cell
     * @param index
     * @return
     */
    public int getFirstRow(SpreadsheetCell cell, int index) {
        do {
            --index;
        } while (index >= 0
                && spreadsheetView.getItems().get(index).get(cell.getColumn()) == cell);

        return index + 1;
    }
    
    /**
     * This return the row at the specified index in the list. The index
     * specified HAS NOTHING to do with the index of the row.
     * @see #getRowIndexed(int) for a getting a row with its real index.
     * @param index
     * @return
     */
    public GridRow getRow(int index) {
        if (index < getFlow().getCells().size()) {
            return (GridRow) getFlow().getCells().get(index);
        }
        return null;
    }

    /**
     * Indicate whether or not the row at the specified index is currently being
     * displayed.
     * 
     * @param index
     * @return
     */
    public final boolean containsRow(int index) {
        /**
         * When scrolling with mouse wheel, some row are present but will not be
         * lay out. We used to consider row with children but that is not
         * accurate. Instead, we simply look if the row layoutY is greater than
         * 0. Since this method is used when checking the row before the current
         * one, it should be a good indicator.
         */
        for (Object obj : getFlow().getCells()) {
            if (((GridRow) obj).getIndex() == index && ((GridRow) obj).getLayoutY() >= 0) {
                return true;
            }
        }
        return false;
    }

    public int getCellsSize() {
        return getFlow().getCells().size();
    }

    public ScrollBar getHBar() {
        if (getFlow() != null) {
            return getFlow().getHorizontalBar();
        }
        return null;
    }

    public ScrollBar getVBar() {
        return getFlow().getVerticalBar();
    }

    /**
     * Will compute for every row the necessary height and fit the line.
     * This can degrade performance a lot so need to use it wisely. 
     * But I don't see other solutions right now.
     */
    public void resizeRowsToFitContent() {
        Grid grid = spreadsheetView.getGrid();
        int maxRows = handle.getView().getGrid().getRowCount();
        for (int row = 0; row < maxRows; row++) {
            if (grid.isRowResizable(row)) {
                resizeRowToFitContent(row);
            }
        }
    }
    
    /**
     * Will compute for the row the necessary height and fit the line.
     * This can degrade performance a lot so need to use it wisely. 
     * But I don't see other solutions right now.
     * @param modelRow
     */
    public void resizeRowToFitContent(int modelRow) {
        if(getSkinnable().getColumns().isEmpty()){
            return;
        }
        final TableColumn, ?> col = getSkinnable().getColumns().get(0);
        List items = handle.getGridView().getItems();
        if (items == null || items.isEmpty()) {
            return;
        }
        
        if (!spreadsheetView.getGrid().isRowResizable(modelRow)) {
            return;
        }
        Callback/* , TableCell> */ cellFactory = col.getCellFactory();
        if (cellFactory == null) {
            return;
        }

        CellView cell = (CellView) cellFactory.call(col);
        if (cell == null) {
            return;
        }

        // set this property to tell the TableCell we want to know its actual
        // preferred width, not the width of the associated TableColumnBase
        cell.getProperties().put("deferToParentPrefWidth", Boolean.TRUE); //$NON-NLS-1$
        
        // determine cell padding
        double padding = 5;

        Node n = cell.getSkin() == null ? null : cell.getSkin().getNode();
        if (n instanceof Region) {
            Region r = (Region) n;
            padding = r.snappedTopInset() + r.snappedBottomInset();
        }

        double maxHeight;
        maxHeight = 0;
        getChildren().add(cell);
        int columnSize = getSkinnable().getColumns().size();
        for (int viewColumn = 0; viewColumn < columnSize; ++viewColumn) {
            TableColumn column = getSkinnable().getColumns().get(viewColumn);
            cell.updateTableColumn(column);
            cell.updateTableView(handle.getGridView());
            cell.updateIndex(modelRow);
            SpreadsheetCell spc = cell.getItem();
            double width = column.getWidth();
            if (spc != null && spc.getColumn() == viewColumn && spc.getColumnSpan() > 1) {
                /**
                 * we need to span multiple columns, so we sum up the width of
                 * the additional columns, adding it to the width variable
                 */
                final int max = getSkinnable().getVisibleLeafColumns().size() - viewColumn;
                for (int i = 1, colSpan = spc.getColumnSpan(); i < colSpan && i < max; i++) {
                    double tempWidth = snapSize(getSkinnable().getVisibleLeafColumn(viewColumn + i).getWidth());
                    width += tempWidth;
                }
            }

            if (spc != null && spc.getColumn() == viewColumn && 
                    ((cell.getText() != null && !cell.getText().isEmpty()) || cell.getGraphic() != null)) {
                cell.setWrapText(true);
                cell.applyCss();
                maxHeight = Math.max(maxHeight, cell.prefHeight(width));
            }
        }
        getChildren().remove(cell);
        rowHeightMap.put(modelRow, maxHeight + padding);
        Event.fireEvent(spreadsheetView, new SpreadsheetView.RowHeightEvent(modelRow, maxHeight + padding));

        rectangleSelection.updateRectangle();
    }
    
    public void resizeRowsToMaximum() {
        //First we resize to fit.
        resizeRowsToFitContent();
        
        Grid grid = spreadsheetView.getGrid();
        
        //Then we take the maximum and apply it everywhere.
        double maxHeight = 0;
        for(int key:rowHeightMap.keySet()){
            maxHeight = Math.max(maxHeight, rowHeightMap.get(key));
        }
        
        rowHeightMap.clear();
        int maxRows = handle.getView().getGrid().getRows().size();
        for (int modelRow = 0; modelRow < maxRows; modelRow++) {
            if (grid.isRowResizable(modelRow)) {
                Event.fireEvent(spreadsheetView, new SpreadsheetView.RowHeightEvent(modelRow, maxHeight));
                rowHeightMap.put(modelRow, maxHeight);
            }
        }
        rectangleSelection.updateRectangle();
    }
    
    public void resizeRowsToDefault() {
        rowHeightMap.clear();
        Grid grid = spreadsheetView.getGrid();
        /**
         * When resizing to default, we need to go through the visible rows in
         * order to update them directly. Because if the rowHeightMap is empty,
         * the rows will not detect that maybe the height has changed.
         */
        for (GridRow row : (List) getFlow().getCells()) {
            if (grid.isRowResizable(spreadsheetView.getModelRow(row.getIndex()))) {
                double newHeight = row.computePrefHeight(-1);
                if (row.getPrefHeight() != newHeight) {
                    row.setRowHeight(newHeight);
                    row.requestLayout();
                }
            }
        }

        //Fixing https://bitbucket.org/controlsfx/controlsfx/issue/358/
        getFlow().layoutChildren();

        for (GridRow row : (List) getFlow().getCells()) {
            double height = getRowHeight(spreadsheetView.getModelRow(row.getIndex()));
            if (row.getHeight() != height) {
                if (grid.isRowResizable(spreadsheetView.getModelRow(row.getIndex()))) {
                    row.setRowHeight(height);
                }
            }
        }
        rectangleSelection.updateRectangle();
    }
    /**
     * We want to have extra space when displaying LocalDate because they will
     * use an editor that display a little icon on the right. Thus, that icon is
     * reducing the visibility of the date string.
     */
//    @Override
    public void resizeColumnToFitContent(TableColumn, ?> tc, int maxRows) {
        
        final TableColumn, ?> col = tc;
        List items = handle.getGridView().getItems();
        if (items == null || items.isEmpty()) {
            return;
        }

        Callback/* , TableCell> */ cellFactory = col.getCellFactory();
        if (cellFactory == null) {
            return;
        }

        TableCell, ?> cell = (TableCell, ?>) cellFactory
                .call(col);
        if (cell == null) {
            return;
        }

        //The current index of that column
        int indexColumn = handle.getGridView().getColumns().indexOf(tc);
        
        /**
         * This is to prevent resize of columns that have the same default width
         * at initialisation. If the "system" is calling this method, the
         * maxRows will be set at 30. When we set a prefWidth and it's equal to
         * the "default width", the system wants to resize the column. We must
         * prevent that, thus we check if the two conditions are met.
         */
        if(maxRows == 30 && handle.isColumnWidthSet(indexColumn)){
            return;
        }

        /**
         * We add a default 10 padding in order not to have a compacted column.
         * But when we have a filter, an extra space will be already added so no
         * need to force this padding.
         */
        double padding = spreadsheetView.getColumns().get(indexColumn).getFilter() != null ? 0 : 10;
        Node n = cell.getSkin() == null ? null : cell.getSkin().getNode();
        if (n instanceof Region) {
            Region r = (Region) n;
            padding = r.snappedLeftInset() + r.snappedRightInset();
        }

        
        // set this property to tell the TableCell we want to know its actual
        // preferred width, not the width of the associated TableColumnBase
        cell.getProperties().put("deferToParentPrefWidth", Boolean.TRUE); //$NON-NLS-1$
        
        ObservableList> gridRows = spreadsheetView.getGrid().getRows();//.get(row)
        
        /**
         * If maxRows is -1, we take all rows. If it's 30, it means it's coming
         * from TableColumnHeader during initialization, so we push it to 100.
         */
        int rows = maxRows == -1 ? items.size() : Math.min(items.size(), maxRows == 30 ? 100 : maxRows);
        double maxWidth = 0;
        boolean datePresent = false;
        cell.updateTableColumn(col);
        cell.updateTableView(handle.getGridView());
        /**
         * Sometime the skin is not set, and the width computed is zero which
         * destroy the grid... So in that case, we manually set the skin...
         */
        if (cell.getSkin() == null) {
            cell.setSkin(new CellViewSkin((CellView) cell));
        }

        for (int row = 0; row < rows; row++) {
            cell.updateIndex(row);
            
            if ((cell.getText() != null && !cell.getText().isEmpty()) || cell.getGraphic() != null) {
                getChildren().add(cell);

                if (((SpreadsheetCell) cell.getItem()).getItem() instanceof LocalDate) {
                    datePresent = true;
                }
                cell.applyCss();
                /**
                 * The cell will automatically add the filter width if
                 * necessary. The padding is also directly computed.
                 */
                double width = cell.prefWidth(-1);
               
                /**
                 * If the cell is spanning in column, we need to take the other
                 * columns into account in the calculation of the width. So we
                 * compute the width needed by the cell and we substract the
                 * other columns width.
                 *
                 * Also if the cell considered is not in the column, we still
                 * have to compute because a previous column may have based its
                 * calculation on the current width which will be modified.
                 */
                SpreadsheetCell spc = gridRows.get(row).get(indexColumn);
                if (spreadsheetView.getColumnSpan(spc) > 1) {
                    for (int i = spreadsheetView.getViewColumn(spc.getColumn()); i < spreadsheetView.getViewColumn(spc.getColumn()) + spreadsheetView.getColumnSpan(spc); ++i) {
                        if(i != indexColumn){
                            width -= spreadsheetView.getColumns().get(i).getWidth();
                        }
                    }
                }
                maxWidth = Math.max(maxWidth, width);
                getChildren().remove(cell);
            }
        }

        // dispose of the cell to prevent it retaining listeners (see RT-31015)
        cell.updateIndex(-1);

        // RT-23486
        double widthMax = maxWidth + padding;
        if (handle.getGridView().getColumnResizePolicy() == TableView.CONSTRAINED_RESIZE_POLICY) {
            widthMax = Math.max(widthMax, col.getWidth());
        }
        
        if (datePresent && widthMax < DATE_CELL_MIN_WIDTH) {
            widthMax = DATE_CELL_MIN_WIDTH;
        }

        /**
         * This method is called by the system at initialisation and later by
         * some methods that check wether the specified column is resizable. So
         * we do not check if the column is resizable because it will be checked
         * before. If we end up here, it either means the column is resizable,
         * OR this is the initialisation and we haven't set a specific width so
         * we just compute one time the correct width for that column, and once
         * set, it will not be called again.
         *
         * Also, if the prefWidth has already been set but the user resized the
         * column with his mouse, we must force the column to resize because
         * setting the prefWidth again will not trigger the listeners.
         */
        widthMax = snapSize(widthMax);
        if (col.getPrefWidth() == widthMax && col.getWidth() != widthMax) {
            //FIXME
//            col.impl_setWidth(widthMax);
        } else {
            col.setPrefWidth(widthMax);
        }
        
        rectangleSelection.updateRectangle();
    }

    /***************************************************************************
     * * PRIVATE/PROTECTED METHOD * *
     **************************************************************************/
    protected final void init() {
        rectangleSelection = new RectangleSelection(this, (TableViewSpanSelectionModel) handle.getGridView().getSelectionModel());
        getFlow().getVerticalBar().valueProperty().addListener(vbarValueListener);
        verticalHeader = new VerticalHeader(handle);
        getChildren().add(verticalHeader);

        ((HorizontalHeader) getTableHeaderRow()).init();
        verticalHeader.init(this, (HorizontalHeader) getTableHeaderRow());
        
        horizontalPickers = new HorizontalPicker((HorizontalHeader) getTableHeaderRow(), spreadsheetView);
        getChildren().add(horizontalPickers);
        getFlow().init(spreadsheetView);
        ((GridViewBehavior)getBehavior()).setGridViewSkin(this);
    }

    public GridViewBehavior getBehavior(){
        return behavior;
    }
    protected final ObservableSet getCurrentlyFixedRow() {
        return currentlyFixedRow;
    }
    
     public ObservableList, ?>> getColumns(){
        return handle.getGridView().getColumns();
    }

    /**
     * Used in the HorizontalColumnHeader when we need to resize in double
     * click.
     * 
     * @param tc
     * @param maxRows
     */
    public void resize(TableColumnBase tc, int maxRows) {
        if(tc.isResizable()){
            int columnIndex = getColumns().indexOf(tc);
            TableColumn tableColumn = getColumns().get(columnIndex);
            resizeColumnToFitContent(tableColumn, maxRows);
            Event.fireEvent(spreadsheetView, new SpreadsheetView.ColumnWidthEvent(columnIndex, tableColumn.getWidth()));
        }
    }

    @Override
    protected void layoutChildren(double x, double y, double w, final double h) {
        if (spreadsheetView == null) {
            return;
        }
        double verticalHeaderWidth = verticalHeader.computeHeaderWidth();
        double horizontalPickerHeight = spreadsheetView.getColumnPickers().isEmpty() ? 0: VerticalHeader.PICKER_SIZE;
        
        if (spreadsheetView.isShowRowHeader() || !spreadsheetView.getRowPickers().isEmpty()) {
            x += verticalHeaderWidth;
            w -= verticalHeaderWidth;
        } else {
            x = 0.0;
        }

        
        y += horizontalPickerHeight;
        super.layoutChildren(x, y, w, h-horizontalPickerHeight);

        final double baselineOffset = getSkinnable().getLayoutBounds().getHeight() / 2;
        double tableHeaderRowHeight = 0;

        if(!spreadsheetView.getColumnPickers().isEmpty()){
            layoutInArea(horizontalPickers, x, y - VerticalHeader.PICKER_SIZE, w, tableHeaderRowHeight, baselineOffset, HPos.CENTER, VPos.CENTER);
        }
        
        if (spreadsheetView.showColumnHeaderProperty().get()) {
            // position the table header
            tableHeaderRowHeight = getTableHeaderRow().prefHeight(-1);
            //For unknow reason, the height of the columnHeader is smaller when nothing is clicked in the grid..
            tableHeaderRowHeight = tableHeaderRowHeight < DEFAULT_CELL_HEIGHT ? DEFAULT_CELL_HEIGHT : tableHeaderRowHeight;
            layoutInArea(getTableHeaderRow(), x, y, w, tableHeaderRowHeight, baselineOffset, HPos.CENTER, VPos.CENTER);

            y += tableHeaderRowHeight;
        } else {
            // This is temporary handled in the HorizontalHeader with Css
            // FIXME tweak open in https://javafx-jira.kenai.com/browse/RT-32673
        }

        if (spreadsheetView.isShowRowHeader() || !spreadsheetView.getRowPickers().isEmpty()) {
            layoutInArea(verticalHeader, x - verticalHeaderWidth, y - tableHeaderRowHeight, w, h, baselineOffset,
                    HPos.CENTER, VPos.CENTER);
        }
    }

    @Override
    protected void onFocusAboveCell() {
        focusScroll();
    }

    @Override
    protected void onFocusBelowCell() {
        focusScroll();
    }

    private int getFixedRowSize() {
        int i = 0;
        for (Integer fixedRow : spreadsheetView.getFixedRows()) {
            if (!spreadsheetView.getHiddenRows().get(fixedRow)) {
                i++;
            }
        }
        return i;
    }
    
    void focusScroll() {
        final TableFocusModel fm = handle.getGridView().getFocusModel();
        if (fm == null) {
            return;
        }
        /**
         * ***************************************************************
         * MODIFIED
         ****************************************************************
         */
        final int row = fm.getFocusedIndex();
        // We try to make visible the rows that may be hidden by Fixed rows
//        if (!getFlow().getCells().isEmpty()
//                //FIXME
//                && getFlow().getCells().get(getFixedRowSize()).getIndex() > row
//                && !spreadsheetView.getFixedRows().contains(spreadsheetView.getModelRow(row))) {
//            getFlow().scrollTo(row);
//        } else {
            // FIXME flow.show() has been removed so ScrollTo is the only method left
            getFlow().scrollTo(row);
//        }
        scrollHorizontally();
        /**
         * ***************************************************************
         * END OF MODIFIED
         ****************************************************************
         */
    }
    
    @Override
    protected void onSelectAboveCell() {
        super.onSelectAboveCell();
        scrollHorizontally();
    }

    @Override
    protected void onSelectBelowCell() {
        super.onSelectBelowCell();
        scrollHorizontally();
    }

    @Override
    protected VirtualFlow>> createVirtualFlow() {
        return new GridVirtualFlow<>(this);
    }

    @Override
    protected TableHeaderRow createTableHeaderRow() {
        return new HorizontalHeader(this);
    }
    
    public HorizontalHeader getHorizontalHeader(){
        return (HorizontalHeader) getTableHeaderRow();
    }

//    BooleanProperty getTableMenuButtonVisibleProperty() {
//        return tableMenuButtonVisibleProperty();
//    }

    @Override
    public void scrollHorizontally(){
        super.scrollHorizontally();
    }
    
    @Override
    protected void scrollHorizontally(TableColumn, ?> col) {
        if (col == null || !col.isVisible()) {
            return;
        }
        /**
         * We modified this function so that we ensure that any selected cells
         * will not be below a fixed column. Because when there's some fixed
         * columns, the "left border" is not the table anymore, but the right
         * side of the last fixed columns.
         *
         * Moreover, we need to re-compute the fixedColumnWidth because the
         * layout of the rows hasn't been done yet and the value is not right.
         * So we might end up below a fixedColumns.
         */
        
        fixedColumnWidth = 0;
        final double pos = getFlow().getHorizontalBar().getValue();
        int index = getColumns().indexOf(col);
        double start = 0;// scrollX;

        for (int columnIndex = 0; columnIndex < index; ++columnIndex) {
            //Do not add the width of hidden column!
            if (!spreadsheetView.isColumnHidden(columnIndex)) {
                SpreadsheetColumn column = spreadsheetView.getColumns().get(columnIndex);
                if (column.isFixed()) {
                    fixedColumnWidth += column.getWidth();
                }
                start += column.getWidth();
            }
        }

        final double end = start + col.getWidth();

        // determine the visible width of the table
        final double headerWidth = handle.getView().getWidth() - snappedLeftInset() - snappedRightInset() - verticalHeader.getVerticalHeaderWidth();

        // determine by how much we need to translate the table to ensure that
        // the start position of this column lines up with the left edge of the
        // tableview, and also that the columns don't become detached from the
        // right edge of the table
        final double max = getFlow().getHorizontalBar().getMax();
        double newPos;

        /**
         * If the starting position of our column if inferior to the left egde
         * (of tableView or fixed columns), then we need to scroll.
         */
        if (start < pos + fixedColumnWidth && start >= 0 && start >= fixedColumnWidth) {
            newPos = start - fixedColumnWidth < 0 ? start : start - fixedColumnWidth;
            getFlow().getHorizontalBar().setValue(newPos);
        //If the starting point is not visible on the right.    
        } else if(start > pos + headerWidth){
            final double delta = start < 0 || end > headerWidth ? start - pos - fixedColumnWidth : 0;
            newPos = pos + delta > max ? max : pos + delta;
            getFlow().getHorizontalBar().setValue(newPos);
        }
        /**
         * In all other cases, it means the cell is visible so no scroll needed,
         * because otherwise we may end up with a continous scroll that always
         * place the selected cell in the center of the screen.
         */
    }

    private void verticalScroll() {
        verticalHeader.requestLayout();
    }

    GridVirtualFlow getFlow() {
        return (GridVirtualFlow) getVirtualFlow();
    }

    /**
     * Return a BitSet of the rows that needs layout all the time. This
     * includes any row containing a span, or a fixed row.
     * @return 
     */
    private BitSet initRowToLayoutBitSet() {
        int rowCount = getItemCount();
        BitSet bitSet = new BitSet(rowCount);
        for (int row = 0; row < rowCount; ++row) {
            if (spreadsheetView.getFixedRows().contains(spreadsheetView.getModelRow(row))) {
                bitSet.set(row);
                continue;
            }
            List myRow = handle.getGridView().getItems().get(row);
            for (SpreadsheetCell cell : myRow) {
                /**
                 * No matter what the sort will do, we want to be behave with
                 * caution here, and take the rowSpan even if the cell is
                 * splitted afterwards.
                 */
                if (spreadsheetView.getRowSpanFilter(cell) > 1 /*|| cell.getColumnSpan() >1*/) {
                    bitSet.set(row);
                    break;
                }
            }
        }
        return bitSet;
    }

    /**
     * When the vertical moves, we update the verticalHeader
     */
    private final InvalidationListener vbarValueListener = new InvalidationListener() {
        @Override
        public void invalidated(Observable valueModel) {
            verticalScroll();
        }
    };

    /**
     * We listen on the FixedRows in order to do the modification in the
     * VirtualFlow
     */
    private final ListChangeListener fixedRowsListener = new ListChangeListener() {
        @Override
        public void onChanged(Change c) {
            hBarValue.clear();
            while (c.next()) {
                if (c.wasPermutated()) {
                    for (Integer fixedRow : c.getList()) {
                        rowToLayout.set(spreadsheetView.getFilteredRow(fixedRow), true);
                    }
                } else {
                    for (Integer unfixedRow : c.getRemoved()) {
                        rowToLayout.set(spreadsheetView.getFilteredRow(unfixedRow), false);
                    //If the grid permits it, we check the spanning in order not
                        //to remove a row that might need layout.
                        if (spreadsheetView.getGrid().getRows().size() > unfixedRow) {
                            List myRow = spreadsheetView.getGrid().getRows().get(unfixedRow);
                            for (SpreadsheetCell cell : myRow) {
                                if (spreadsheetView.getRowSpanFilter(cell) > 1 /*|| spreadsheetView.getColumnSpan(cell) > 1*/) {
                                    rowToLayout.set(spreadsheetView.getFilteredRow(unfixedRow), true);
                                    break;
                                }
                            }
                        }
                    }

                    //We check for the newly fixedRow
                    for (Integer fixedRow : c.getAddedSubList()) {
                        rowToLayout.set(spreadsheetView.getFilteredRow(fixedRow), true);
                    }
                }
            }
            // requestLayout() not responding immediately..
            getFlow().requestLayout();
        }
    };

    /**
     * We listen on the currentlyFixedRow in order to do the modification in the
     * FixedRowHeight.
     */
    private final SetChangeListener currentlyFixedRowListener = new SetChangeListener() {
        @Override
        public void onChanged(javafx.collections.SetChangeListener.Change arg0) {
            computeFixedRowHeight();
        }
    };

    /**
     * We compute the total height of the fixedRows so that the selection can
     * use it without performance regression.
     */
    public void computeFixedRowHeight() {
        fixedRowHeight = 0;
        for (int i : getCurrentlyFixedRow()) {
            fixedRowHeight += getRowHeight(i);
        }
    }

    /**
     * We listen on the FixedColumns in order to do the modification in the
     * VirtualFlow.
     */
    private final ListChangeListener fixedColumnsListener = new ListChangeListener() {
        @Override
        public void onChanged(Change c) {
            hBarValue.clear();
            getFlow().requestLayout();
            // requestLayout() not responding immediately..
//            getFlow().layoutTotal();
        }
    };

//    @Override
//    protected TableSelectionModel> getSelectionModel() {
//        return getSkinnable().getSelectionModel();
//    }
//
//    @Override
//    protected TableFocusModel, TableColumn, ?>> getFocusModel() {
//        return getSkinnable().getFocusModel();
//    }
//
//    @Override
//    protected TablePositionBase, ?>> getFocusedCell() {
//        return getSkinnable().getFocusModel().getFocusedCell();
//    }
//
//    @Override
//    protected ObservableList, ?>> getVisibleLeafColumns() {
//        return getSkinnable().getVisibleLeafColumns();
//    }
//
//    @Override
//    protected int getVisibleLeafIndex(TableColumn, ?> tc) {
//        return getSkinnable().getVisibleLeafIndex(tc);
//    }
//
//    @Override
//    protected TableColumn, ?> getVisibleLeafColumn(int col) {
//        return getSkinnable().getVisibleLeafColumn(col);
//    }
//
//    @Override
//    protected ObservableList, ?>> getColumns() {
//        return getSkinnable().getColumns();
//    }
//
//    @Override
//    protected ObservableList, ?>> getSortOrder() {
//        return getSkinnable().getSortOrder();
//    }
//
//    @Override
//    protected ObjectProperty>> itemsProperty() {
//        return getSkinnable().itemsProperty();
//    }
//
//    @Override
//    protected ObjectProperty>, TableRow>>> rowFactoryProperty() {
//        return getSkinnable().rowFactoryProperty();
//    }
//
//    @Override
//    protected ObjectProperty placeholderProperty() {
//        return getSkinnable().placeholderProperty();
//    }
//
//    @Override
//    protected BooleanProperty tableMenuButtonVisibleProperty() {
//        return getSkinnable().tableMenuButtonVisibleProperty();
//    }
//
//    @Override
//    protected ObjectProperty> columnResizePolicyProperty() {
//        return (ObjectProperty>) (Object)getSkinnable().columnResizePolicyProperty();
//    }
//
//    @Override
//    protected boolean resizeColumn(TableColumn, ?> tc, double delta) {
//        getHorizontalHeader().getRootHeader().lastColumnResized = getColumns().indexOf(tc);
//        boolean returnedValue = getSkinnable().resizeColumn(tc, delta);
//        if(returnedValue){
                //FIXME Reactivate that
//            Event.fireEvent(spreadsheetView, new SpreadsheetView.ColumnWidthEvent(getColumns().indexOf(tc), tc.getWidth()));
//        }
//        return returnedValue;
//    }
//
//    @Override
//    protected void edit(int index, TableColumn, ?> column) {
//        getSkinnable().edit(index, column);
//    }
//
//    @Override
//    public TableRow> createCell() {
//        TableRow> cell;
//
//        if (getSkinnable().getRowFactory() != null) {
//            cell = getSkinnable().getRowFactory().call(getSkinnable());
//        } else {
//            cell = new TableRow<>();
//        }
//
//        cell.updateTableView(getSkinnable());
//        return cell;
//    }

    @Override
    public final int getItemCount() {
        return getSkinnable().getItems() == null ? 0 : getSkinnable().getItems().size();
    }

    /**
     * If the scene is not yet instantiated, we need to wait otherwise the
     * VirtualFlow will not shift the cells properly.
     *
     * @param value
     */
    public void setHbarValue(double value) {
        setHbarValue(value, 0);
    }

    public void setHbarValue(double value, int count) {
        if (count > 5) {
            return;
        }
        final int newCount = count + 1;
        if (getFlow().getScene() == null) {
            Platform.runLater(() -> {
                setHbarValue(value, newCount);
            });
            return;
        }
        getHBar().setValue(value);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy