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

griffon.javafx.scene.control.DefaultTableFormat Maven / Gradle / Ivy

/*
 * SPDX-License-Identifier: Apache-2.0
 *
 * Copyright 2008-2018 the original author or authors.
 *
 * Licensed 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 griffon.javafx.scene.control;

import javafx.beans.value.ObservableValue;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.LinkedHashMap;
import java.util.Map;

import static griffon.util.GriffonClassUtils.requireState;
import static griffon.util.GriffonNameUtils.getNaturalName;
import static griffon.util.GriffonNameUtils.requireNonBlank;
import static java.lang.System.arraycopy;
import static java.util.Objects.requireNonNull;

/**
 * @author Andres Almiray
 * @since 2.11.0
 */
public class DefaultTableFormat implements TableViewFormat {
    public static class Column {
        private final String name;
        private final String title;
        private final Double size;
        private final TableCellFactory tableCellFactory;

        public Column(@Nonnull String name) {
            this(name, getNaturalName(name), null, null);
        }

        public Column(@Nonnull String name, @Nonnull String title) {
            this(name, title, null, null);
        }

        public Column(@Nonnull String name, @Nonnull Double size) {
            this(name, getNaturalName(name), size, null);
        }

        public Column(@Nonnull String name, @Nonnull TableCellFactory tableCellFactory) {
            this(name, getNaturalName(name), null, tableCellFactory);
        }

        public Column(@Nonnull String name, @Nonnull String title, @Nonnull Double size) {
            this(name, title, size, null);
        }

        public Column(@Nonnull String name, @Nonnull String title, @Nonnull TableCellFactory tableCellFactory) {
            this(name, title, null, tableCellFactory);
        }

        public Column(@Nonnull String name, @Nonnull Double size, @Nonnull TableCellFactory tableCellFactory) {
            this(name, getNaturalName(name), size, tableCellFactory);
        }

        public Column(@Nonnull String name, @Nonnull String title, @Nullable Double size, @Nonnull TableCellFactory tableCellFactory) {
            this.name = requireNonBlank(name, "Argument 'name' must not be blank");
            this.title = requireNonBlank(title, "Argument 'title' must not be blank");
            if (size != null) {
                requireState(size > 0, "Argument 'size' must be greater than 0.0d");
                requireState(size <= 1, "Argument 'size' must be less than or equal to 1.0d");
            }
            this.size = size;
            this.tableCellFactory = tableCellFactory;
        }

        @Nonnull
        public String getName() {
            return name;
        }

        @Nonnull
        public String getTitle() {
            return title;
        }

        @Nullable
        public Double getSize() {
            return size;
        }

        @Nullable
        public TableCellFactory getTableCellFactory() {
            return tableCellFactory;
        }
    }

    protected final Column[] columns;
    private final Map, Map> observableMetadata = new LinkedHashMap<>();

    public DefaultTableFormat(@Nonnull String... names) {
        requireNonNull(names, "Argument 'names' must not be null");
        requireState(names.length > 0, "Column size must be greater than zero");
        this.columns = new Column[names.length];
        for (int i = 0; i < names.length; i++) {
            this.columns[i] = new Column(names[i]);
        }
    }

    public DefaultTableFormat(@Nonnull Column... columns) {
        requireNonNull(columns, "Argument 'columns' must not be null");
        requireState(columns.length > 0, "Column size must be greater than zero");
        this.columns = new Column[columns.length];
        arraycopy(columns, 0, this.columns, 0, columns.length);
    }

    @Override
    public int getColumnCount() {
        return columns.length;
    }

    @Nonnull
    @Override
    public String getColumnName(int index) {
        return columns[index].getTitle();
    }

    @Nullable
    @Override
    public Double getColumnSize(int index) {
        return columns[index].getSize();
    }

    @Nonnull
    @Override
    public ObservableValue getObservableValue(@Nonnull E instance, int index) {
        final String columnName = columns[index].getName();
        Class klass = instance.getClass();
        Map metadata = observableMetadata.computeIfAbsent(klass, this::harvestMetadata);

        try {
            Method method = metadata.get(columnName);
            if (method == null) {
                throw new IllegalStateException("Could not find a method in " + klass +
                    " returning " + ObservableValue.class.getSimpleName() + " associated with column " + columnName);
            }
            return (ObservableValue) method.invoke(instance);
        } catch (IllegalAccessException e) {
            throw new IllegalStateException(e);
        } catch (InvocationTargetException e) {
            throw new IllegalStateException(e.getTargetException());
        }
    }

    @Nullable
    @Override
    public TableCellFactory getTableCellFactory(int index) {
        return columns[index].getTableCellFactory();
    }

    @Nonnull
    private Map harvestMetadata(@Nonnull Class klass) {
        Map map = new LinkedHashMap<>();

        for (Method method : klass.getMethods()) {
            if (ObservableValue.class.isAssignableFrom(method.getReturnType()) &&
                method.getParameterCount() == 0) {
                Arrays.stream(columns)
                    .map(Column::getName)
                    .filter(name -> method.getName().startsWith(name))
                    .findFirst()
                    .ifPresent(name -> map.put(name, method));
            }
        }

        return map;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy