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

com.dua3.utility.fx.controls.OptionsPane Maven / Gradle / Ivy

There is a newer version: 15.0.2
Show newest version
package com.dua3.utility.fx.controls;

import org.jspecify.annotations.Nullable;
import com.dua3.utility.options.Arguments;
import com.dua3.utility.options.ChoiceOption;
import com.dua3.utility.options.Flag;
import com.dua3.utility.options.Option;
import com.dua3.utility.options.SimpleOption;
import javafx.beans.property.Property;
import javafx.beans.property.ReadOnlyBooleanProperty;
import javafx.beans.property.ReadOnlyStringProperty;
import javafx.beans.property.SimpleObjectProperty;
import javafx.geometry.Insets;
import javafx.scene.Node;
import javafx.scene.control.CheckBox;
import javafx.scene.control.Label;
import javafx.scene.layout.GridPane;
import javafx.util.StringConverter;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;

import java.util.ArrayDeque;
import java.util.Collection;
import java.util.Deque;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.function.Function;
import java.util.function.Supplier;
import java.util.stream.Stream;

/**
 * OptionsPane is a custom JavaFX GridPane used as a control element for managing
 * a collection of options represented by instances of the {@link Option} class.
 * It implements the {@link InputControl} interface which allows it to handle input
 * and provide output in the form of {@link Arguments}.
 */
public class OptionsPane extends GridPane implements InputControl {

    /**
     * Logger
     */
    protected static final Logger LOG = LogManager.getLogger(OptionsPane.class);
    private static final Insets INSETS = new Insets(2);
    private final InputControl.State<@Nullable Arguments> state;
    private final Supplier>> options;
    private final Supplier<@Nullable Arguments> dflt;
    private final Map, InputControl> items = new LinkedHashMap<>();

    /**
     * Create new OptionsPane.
     *
     * @param optionSet     the available options
     * @param currentValues the current values
     * @see Option
     * @see Arguments
     */
    public OptionsPane(Collection> optionSet, Arguments currentValues) {
        this(() -> optionSet, () -> currentValues);
    }

    /**
     * Constructs a new OptionsPane with the given suppliers for options and default arguments.
     *
     * @param options A supplier providing a collection of options.
     * @param dflt    A supplier providing the default arguments.
     * @see Option
     * @see Arguments
     */
    public OptionsPane(Supplier>> options, Supplier dflt) {
        this.options = options;
        this.dflt = dflt;
        Property<@Nullable Arguments> value = new SimpleObjectProperty<>();
        this.state = new State<>(value, dflt);
    }

    @Override
    public Node node() {
        return this;
    }

    @SuppressWarnings({"unchecked", "rawtypes"})
    @Override
    public Arguments get() {
        Deque> entries = new ArrayDeque<>();
        for (var entry : items.entrySet()) {
            Option option = entry.getKey();
            Object value = entry.getValue().valueProperty().getValue();
            if (value != null) {
                entries.add(Arguments.createEntry(option, value));
            }
        }
        return Arguments.of(entries.toArray(Arguments.Entry[]::new));
    }

    @SuppressWarnings({"rawtypes", "unchecked"})
    @Override
    public void set(@Nullable Arguments arg) {
        for (var item : items.entrySet()) {
            Option option = item.getKey();
            InputControl control = item.getValue();
            Stream> stream = Objects.requireNonNullElseGet(arg, Arguments::empty).stream(option);
            Optional value = stream.filter(list -> !list.isEmpty())
                    .reduce((first, second) -> second)
                    .map(list -> list.get(list.size() - 1));
            control.set(value.orElse(null));
        }
    }

    @Override
    public void init() {
        getChildren().clear();

        Collection> optionSet = options.get();
        Arguments values = dflt.get();

        int row = 0;
        for (Option option : optionSet) {
            Label label = new Label(option.displayName());

            var control = createControl(Objects.requireNonNullElseGet(values, Arguments::empty), option);
            items.put(option, control);

            addToGrid(label, 0, row);
            addToGrid(control.node(), 1, row);

            row++;
        }
    }

    @SuppressWarnings("unchecked")
    private  InputControl createControl(Arguments values, Option option) {
        if (option instanceof ChoiceOption co) {
            return new ChoiceInputControl<>(co, supplyDefault(co, values));
        } else if (option instanceof Flag f) {
            CheckBox checkBox = new CheckBox(f.displayName());
            return (InputControl) new SimpleInputControl<>(checkBox, checkBox.selectedProperty(), supplyDefault(f, values), nopValidator());
        } else if (option instanceof SimpleOption so) {
            StringConverter converter = new StringConverter<>() {
                @Override
                public String toString(T v) {
                    return option.format(v);
                }

                @Override
                public T fromString(String s) {
                    return option.map(s);
                }
            };
            return InputControl.stringInput(supplyDefault(so, values), nopValidator(), converter);
        }

        throw new UnsupportedOperationException("unsupported input type: " + option.getClass().getName());
    }

    @SuppressWarnings({"unchecked", "rawtypes"})
    private static  T getValue(Option option, Arguments values) {
        if (option instanceof Flag flag) {
            return (T) (Object) values.isSet(flag);
        }
        if (option instanceof SimpleOption so) {
            return (T) values.get(so).orElse(so.getDefault());
        }
        if (option instanceof ChoiceOption co) {
            return (T) values.get(co).orElse(co.getDefault());
        }
        throw new IllegalArgumentException("Unknown option type: " + option);
    }

    private static  Function> nopValidator() {
        return s -> Optional.empty();
    }

    private static  Supplier supplyDefault(Option option, Arguments values) {
        return () -> getValue(option, values);
    }

    private void addToGrid(@Nullable Node node, int c, int r) {
        if (node != null) {
            add(node, c, r);
            setMargin(node, INSETS);
        }
    }

    @Override
    public void reset() {
        items.forEach((item, control) -> control.reset());
    }

    @Override
    public Property<@Nullable Arguments> valueProperty() {
        return state.valueProperty();
    }

    @Override
    public ReadOnlyBooleanProperty validProperty() {
        return state.validProperty();
    }

    @Override
    public ReadOnlyStringProperty errorProperty() {
        return state.errorProperty();
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy