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

org.dominokit.domino.ui.forms.AbstractSelect Maven / Gradle / Ivy

package org.dominokit.domino.ui.forms;

import elemental2.dom.*;
import org.dominokit.domino.ui.dropdown.DropDownMenu;
import org.dominokit.domino.ui.dropdown.DropDownPosition;
import org.dominokit.domino.ui.dropdown.DropdownAction;
import org.dominokit.domino.ui.dropdown.DropdownActionsGroup;
import org.dominokit.domino.ui.grid.flex.FlexItem;
import org.dominokit.domino.ui.icons.BaseIcon;
import org.dominokit.domino.ui.icons.Icons;
import org.dominokit.domino.ui.icons.MdiIcon;
import org.dominokit.domino.ui.style.Styles;
import org.dominokit.domino.ui.utils.DominoElement;

import java.util.ArrayList;
import java.util.Collection;
import java.util.LinkedList;
import java.util.List;
import java.util.function.Supplier;
import java.util.stream.Collectors;

import static elemental2.dom.DomGlobal.window;
import static java.util.Objects.isNull;
import static java.util.Objects.nonNull;
import static org.dominokit.domino.ui.style.Unit.px;
import static org.jboss.elemento.Elements.button;
import static org.jboss.elemento.Elements.span;

public abstract class AbstractSelect> extends AbstractValueBox {
    private static final String CLICK_EVENT = "click";

    private SelectOption noneOption = SelectOption.create(null, "none", "None");

    private DominoElement buttonElement;
    protected DominoElement buttonValueContainer = DominoElement.of(span().css("select-value", Styles.ellipsis_text));
    protected LinkedList> options = new LinkedList<>();
    private DropDownMenu optionsMenu;
    private List> selectionHandlers = new ArrayList<>();
    private Supplier> arrowIconSupplier = Icons.ALL::menu_down_mdi;
    private BaseIcon arrowIcon;

    private boolean searchable;
    private boolean clearable;
    private FlexItem arrowIconContainer;
    private int popupWidth = 0;
    private String dropDirection = "auto";

    public AbstractSelect() {
        super("button", "");
        optionsMenu = DropDownMenu.create(fieldContainer).styler(style1 -> style1.add("select-option-menu"));
        optionsMenu.setAppendTarget(DomGlobal.document.body);
        optionsMenu.setAppendStrategy(DropDownMenu.AppendStrategy.FIRST);
        optionsMenu.setPosition(new PopupPositionTopDown<>(this));
        optionsMenu.addOpenHandler(this::resumeFocusValidation);
        buttonElement.appendChild(buttonValueContainer);
        initListeners();
        dropdown();
        setSearchable(true);
        addChangeHandler(value -> {
            if (isNull(value)) {
                clear();
            }
        });
        css("d-select");
    }

    public AbstractSelect(String label) {
        this();
        setLabel(label);
    }

    public AbstractSelect(List> options) {
        this("", options);
    }

    public AbstractSelect(String label, List> options) {
        this(label);
        options.forEach(this::appendChild);
    }

    private void initListeners() {
        EventListener clickListener = evt -> {
            pauseFocusValidation();
            open();
            evt.stopPropagation();
        };
        if (nonNull(arrowIcon)) {
            arrowIcon.addClickListener(clickListener);
        }

        buttonElement.addEventListener(CLICK_EVENT, clickListener);
        getLabelElement().addEventListener(CLICK_EVENT, clickListener);
        buttonElement.addEventListener("focus", evt -> focus());
        buttonElement.addEventListener("blur", evt -> unfocus());
        optionsMenu.addCloseHandler(() -> {
            this.focus();
            validate();
        });
    }

    public void setArrowIconSupplier(Supplier> arrowIconSupplier) {
        if (nonNull(arrowIconSupplier)) {
            this.arrowIconSupplier = arrowIconSupplier;
        }
    }

    @Override
    public S clear() {
        unfloatLabel();
        getOptions().forEach(selectOption -> selectOption.deselect(true));
        buttonValueContainer.setTextContent("");
        if (isAutoValidation())
            validate();
        return (S) this;
    }

    public S open() {
        if (isEnabled() && !isReadOnly()) {
            DropDownMenu.closeAllMenus();
            doOpen();
        }
        return (S) this;
    }

    private void doOpen() {
        optionsMenu.open();
        optionsMenu.styler(style -> style.setWidth(getFieldInputContainer().getBoundingClientRect().width + "px"));
    }

    public void close() {
        optionsMenu.close();
    }

    public S divider() {
        optionsMenu.separator();
        return (S) this;
    }

    public S addGroup(SelectOptionGroup group) {
        DropdownActionsGroup> dropdownActionsGroup = DropdownActionsGroup.create(group.getTitleElement());
        for (SelectOption option : group.getOptions()) {
            addOptionToGroup(dropdownActionsGroup, option);
        }
        group.setAddOptionConsumer(selectOption -> {
            addOptionToGroup(dropdownActionsGroup, selectOption);
        });

        optionsMenu.addGroup(dropdownActionsGroup);
        return (S) this;
    }

    private void addOptionToGroup(DropdownActionsGroup> dropdownActionsGroup, SelectOption option) {
        dropdownActionsGroup.appendChild(asDropDownAction(option));
        options.add(option);
    }

    public S addOptions(List> options) {
        options.forEach(this::appendChild);
        return (S) this;
    }

    public S setPopupWidth(int width) {
        this.popupWidth = width;
        return (S) this;
    }

    public S appendChild(SelectOption option) {
        options.add(option);
        appendOptionValue(option);
        return (S) this;
    }

    public S insertFirst(SelectOption option) {
        options.add(0, option);
        insertFirstOptionValue(option);
        return (S) this;
    }

    private void doSelectOption(SelectOption option) {
        if (isEnabled()) {
            select(option);
            close();
        }
    }

    private void appendOptionValue(SelectOption option) {
        optionsMenu.appendChild(asDropDownAction(option));
    }

    private void insertFirstOptionValue(SelectOption option) {
        optionsMenu.insertFirst(asDropDownAction(option));
    }

    private DropdownAction> asDropDownAction(SelectOption option) {
        return DropdownAction.create(option, option.element())
                .setExcludeFromSearchResults(option.isExcludeFromSearchResults())
                .addSelectionHandler(value -> doSelectOption(option));
    }

    public S selectAt(int index) {
        return selectAt(index, false);
    }

    public S selectAt(int index, boolean silent) {
        if (index < options.size() && index >= 0)
            select(options.get(index), silent);
        return (S) this;
    }

    public SelectOption getOptionAt(int index) {
        if (index < options.size() && index >= 0)
            return options.get(index);
        return null;
    }

    public List> getOptions() {
        return options;
    }

    public S select(SelectOption option) {
        return select(option, false);
    }

    public abstract S select(SelectOption option, boolean silent);

    public boolean isSelected() {
        return !isEmpty();
    }

    protected void onSelection(SelectOption option) {
        for (SelectionHandler handler : selectionHandlers) {
            handler.onSelection(option);
        }
        for (ChangeHandler c : changeHandlers) {
            c.onValueChanged(getValue());
        }
    }

    public S addSelectionHandler(SelectionHandler selectionHandler) {
        selectionHandlers.add(selectionHandler);
        return (S) this;
    }

    @Override
    public S enable() {
        super.enable();
        buttonElement.enable();
        getLabelElement().enable();
        return (S) this;
    }

    @Override
    public S disable() {
        super.disable();
        buttonElement.disable();
        getLabelElement().disable();
        return (S) this;
    }

    @Override
    public boolean isEnabled() {
        return !buttonElement.hasAttribute("disabled");
    }

    public S dropup() {
        this.dropDirection = "up";
        return (S) this;
    }

    private void onDropup() {
        if (searchable) {
            optionsMenu.appendChild(optionsMenu.getSearchContainer());
            optionsMenu
                    .getSearchContainer()
                    .style()
                    .remove("pos-top")
                    .add("pos-bottom");
            optionsMenu
                    .style()
                    .remove("pos-top")
                    .add("pos-bottom");
        }
    }

    public S dropdown() {
        this.dropDirection = "down";
        return (S) this;
    }

    private void onDropdown() {
        if (searchable) {
            optionsMenu.insertFirst(optionsMenu.getSearchContainer());
            optionsMenu.getSearchContainer()
                    .style()
                    .remove("pos-bottom")
                    .add("pos-top");
            optionsMenu
                    .style()
                    .remove("pos-bottom")
                    .add("pos-top");
        }
    }

    private MdiIcon getDropdownIcon() {
        return Icons.ALL.menu_down_mdi();
    }

    private MdiIcon getDropupIcon() {
        return Icons.ALL.menu_up_mdi();
    }

    @Override
    public S value(T value) {
        return setValue(value, false);
    }

    public abstract S setValue(T value, boolean silent);

    public S removeSelectionHandler(SelectionHandler selectionHandler) {
        if (nonNull(selectionHandler))
            selectionHandlers.remove(selectionHandler);
        return (S) this;
    }

    public S removeOption(SelectOption option) {
        if (nonNull(option) && getOptions().contains(option)) {
            option.deselect(true);
            option.element().remove();
        }
        return (S) this;
    }

    public S removeOptions(Collection> options) {
        if (nonNull(options) && !options.isEmpty() && !this.options.isEmpty()) {
            options.forEach(this::removeOption);
        }
        return (S) this;
    }

    public S removeAllOptions() {
        options.clear();
        optionsMenu.clearActions();
        clear();
        if (isClearable()) {
            setClearable(true);
        }
        return (S) this;
    }

    @Override
    public S setReadOnly(boolean readOnly) {
        super.setReadOnly(readOnly);
        if (readOnly) {
            arrowIconContainer.hide();
            floatLabel();
        } else {
            arrowIconContainer.show();
            if (isEmpty()) {
                unfloatLabel();
            }
        }
        buttonElement.setReadOnly(readOnly);
        return (S) this;
    }

    @FunctionalInterface
    public interface SelectionHandler {
        void onSelection(SelectOption option);
    }

    public DominoElement getSelectButton() {
        return buttonElement;
    }

    public DominoElement getSelectLabel() {
        return getLabelElement();
    }

    @Override
    protected AutoValidator createAutoValidator(AutoValidate autoValidate) {
        return new SelectAutoValidator<>(this, autoValidate);
    }

    private void setAddon(DominoElement container, DominoElement oldAddon, Element addon) {
        if (nonNull(oldAddon)) {
            oldAddon.remove();
        }
        if (nonNull(addon)) {
            List oldClasses = new ArrayList<>(addon.classList.asList());
            for (String oldClass : oldClasses) {
                addon.classList.remove(oldClass);
            }
            oldClasses.add(0, "input-addon");
            for (String oldClass : oldClasses) {
                addon.classList.add(oldClass);
            }
            container.appendChild(addon);
        }
    }

    public List getValues() {
        return options.stream().map(SelectOption::getValue).collect(Collectors.toList());
    }

    public List getKeys() {
        return options.stream().map(SelectOption::getKey).collect(Collectors.toList());
    }

    public boolean containsKey(String key) {
        return getKeys().contains(key);
    }

    public boolean containsValue(V value) {
        return getValues().contains(value);
    }

    public S setSearchable(boolean searchable) {
        optionsMenu.setSearchable(searchable);
        this.searchable = searchable;
        return (S) this;
    }

    public boolean isSearchable() {
        return searchable;
    }

    public static void closeAllSelects() {
        DropDownMenu.closeAllMenus();
    }

    public S selectByKey(String key) {
        return selectByKey(key, false);
    }

    public S selectByKey(String key, boolean silent) {
        for (SelectOption option : getOptions()) {
            if (option.getKey().equals(key)) {
                select(option, silent);
            }
        }
        return (S) this;
    }

    public S setClearable(boolean clearable) {
        this.clearable = clearable;
        if (clearable && !options.contains(noneOption)) {
            insertFirst(noneOption);
        } else {
            removeOption(noneOption);
        }
        return (S) this;
    }

    public boolean isClearable() {
        return clearable;
    }

    public S setClearableText(String clearableText) {
        noneOption.setDisplayValue(clearableText);
        return (S) this;
    }

    public String getClearableText() {
        return noneOption.getDisplayValue();
    }

    public String getDropDirection() {
        return dropDirection;
    }

    public DominoElement getButtonValueContainer() {
        return buttonValueContainer;
    }

    public S setDropPosition(DropDownPosition dropPosition) {
        optionsMenu.setPosition(dropPosition);
        return (S) this;
    }

    @Override
    protected HTMLElement createInputElement(String type) {
        buttonElement = DominoElement.of(button().attr("type", "button").css("select-button"));
        return buttonElement.element();
    }

    @Override
    protected FlexItem createMandatoryAddOn() {
        if (isNull(arrowIconSupplier)) {
            arrowIcon = Icons.ALL.menu_down_mdi()
                    .clickable();
        } else {
            arrowIcon = arrowIconSupplier.get()
                    .clickable();
        }
        arrowIconContainer = FlexItem.create().appendChild(arrowIcon);
        return arrowIconContainer;
    }

    @Override
    protected void clearValue() {

    }

    @Override
    protected void doSetValue(T value) {

    }

    public S setSearchFilter(DropDownMenu.SearchFilter searchFilter){
        this.optionsMenu.setSearchFilter(searchFilter);
        return (S) this;
    }

    public static class PopupPositionTopDown> implements DropDownPosition {

        private DropDownPositionUp up = new DropDownPositionUp();
        private DropDownPositionDown down = new DropDownPositionDown();

        private final AbstractSelect select;

        public PopupPositionTopDown(AbstractSelect select) {
            this.select = select;
        }

        @Override
        public void position(HTMLElement popup, HTMLElement target) {
            ClientRect targetRect = target.getBoundingClientRect();

            double distanceToMiddle = ((targetRect.top) - (targetRect.height / 2));
            double windowMiddle = DomGlobal.window.innerHeight / 2;
            double popupHeight = popup.getBoundingClientRect().height;
            double distanceToBottom = window.innerHeight - targetRect.top;
            double distanceToTop = (targetRect.top + targetRect.height);

            boolean hasSpaceBelow = distanceToBottom > popupHeight;
            boolean hasSpaceUp = distanceToTop > popupHeight;

            if (("up".equalsIgnoreCase(select.dropDirection) && hasSpaceUp) || ((distanceToMiddle >= windowMiddle) && !hasSpaceBelow)) {
                up.position(popup, target);
                select.onDropup();
                popup.setAttribute("popup-direction", "top");
            } else {
                down.position(popup, target);
                select.onDropdown();
                popup.setAttribute("popup-direction", "down");
            }

            popup.style.setProperty("width", select.popupWidth > 0 ? (select.popupWidth + "px") : (targetRect.width + "px"));
        }
    }

    public static class DropDownPositionUp implements DropDownPosition {
        @Override
        public void position(HTMLElement actionsMenu, HTMLElement target) {

            ClientRect targetRect = target.getBoundingClientRect();

            actionsMenu.style.setProperty("bottom", px.of(((window.innerHeight - targetRect.bottom) - window.pageYOffset)));
            actionsMenu.style.setProperty("left", px.of((targetRect.left + window.pageXOffset)));
            actionsMenu.style.removeProperty("top");
        }
    }

    public static class DropDownPositionDown implements DropDownPosition {
        @Override
        public void position(HTMLElement actionsMenu, HTMLElement target) {

            ClientRect targetRect = target.getBoundingClientRect();

            actionsMenu.style.setProperty("top", px.of((targetRect.top + window.pageYOffset)));
            actionsMenu.style.setProperty("left", px.of((targetRect.left + window.pageXOffset)));
            actionsMenu.style.removeProperty("bottom");
        }
    }

    private static class SelectAutoValidator> extends AutoValidator {

        private AbstractSelect select;
        private SelectionHandler selectionHandler;

        public SelectAutoValidator(AbstractSelect select, AutoValidate autoValidate) {
            super(autoValidate);
            this.select = select;
        }

        @Override
        public void attach() {
            selectionHandler = option -> autoValidate.apply();
            select.addSelectionHandler(selectionHandler);
        }

        @Override
        public void remove() {
            select.removeSelectionHandler(selectionHandler);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy