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

org.dominokit.domino.ui.datepicker.DatePicker Maven / Gradle / Ivy

There is a newer version: 1.0.139
Show newest version
package org.dominokit.domino.ui.datepicker;

import org.gwtproject.editor.client.TakesValue;
import elemental2.core.JsDate;
import elemental2.dom.HTMLDivElement;
import org.dominokit.domino.ui.button.Button;
import org.dominokit.domino.ui.forms.Select;
import org.dominokit.domino.ui.forms.SelectOption;
import org.dominokit.domino.ui.grid.Column;
import org.dominokit.domino.ui.grid.Row;
import org.dominokit.domino.ui.icons.Icon;
import org.dominokit.domino.ui.icons.Icons;
import org.dominokit.domino.ui.modals.ModalDialog;
import org.dominokit.domino.ui.pickers.PickerHandler;
import org.dominokit.domino.ui.style.ColorScheme;
import org.dominokit.domino.ui.style.Elevation;
import org.dominokit.domino.ui.style.Styles;
import org.dominokit.domino.ui.utils.BaseDominoElement;
import org.dominokit.domino.ui.utils.DominoElement;
import org.dominokit.domino.ui.utils.HasValue;
import org.gwtproject.i18n.shared.cldr.impl.DateTimeFormatInfo_factory;
import org.gwtproject.i18n.shared.DateTimeFormat;
import org.gwtproject.i18n.shared.cldr.DateTimeFormatInfo;

import java.util.ArrayList;
import java.util.Date;
import java.util.List;

import static java.util.Objects.nonNull;
import static org.dominokit.domino.ui.style.Unit.px;
import static org.jboss.elemento.Elements.div;

public class DatePicker extends BaseDominoElement implements HasValue,
        DatePickerMonth.InternalHandler, TakesValue {

    private final JsDate jsDate;
    private DominoElement element = DominoElement.of(div()
            .css(DatePickerStyles.CALENDAR))
            .elevate(Elevation.LEVEL_1);
    private DominoElement headerPanel = DominoElement.of(div().css(DatePickerStyles.DATE_PANEL));
    private DominoElement selectorsPanel = DominoElement.of(div().css(DatePickerStyles.SELECTOR_CONTAINER));
    private DominoElement footerPanel = DominoElement.of(div().css(DatePickerStyles.CAL_FOOTER));

    private DominoElement dayName = DominoElement.of(div().css(DatePickerStyles.DAY_NAME));
    private DominoElement monthName = DominoElement.of(div().css(DatePickerStyles.MONTH_NAME));
    private DominoElement dateNumber = DominoElement.of(div().css(DatePickerStyles.DAY_NUMBER));
    private DominoElement yearNumber = DominoElement.of(div().css(DatePickerStyles.YEAR_NUMBER));
    private Icon navigateBefore;
    private Icon navigateNext;

    private Select yearSelect;
    private Select monthSelect;
    private Button todayButton;
    private Button clearButton;
    private Button closeButton;

    private ColorScheme colorScheme = ColorScheme.LIGHT_BLUE;

    private final DatePickerMonth datePickerMonth;
    private DatePickerElement selectedPickerElement;

    private List closeHandlers = new ArrayList<>();
    private List clearHandlers = new ArrayList<>();
    private BackgroundHandler backgroundHandler = (oldBackground, newBackground) -> {
    };

    private JsDate minDate;
    private JsDate maxDate;

    private List dateSelectionHandlers = new ArrayList<>();
    private List dateDayClickedHandlers = new ArrayList<>();


    public DatePicker(Date date, DateTimeFormatInfo dateTimeFormatInfo) {

        this.jsDate = new JsDate((double) date.getTime());

        this.minDate = new JsDate((double) date.getTime());
        minDate.setFullYear(minDate.getFullYear() - 50);

        this.maxDate = new JsDate((double) date.getTime());
        maxDate.setFullYear(maxDate.getFullYear() + 50);

        datePickerMonth = DatePickerMonth.create(this.jsDate, dateTimeFormatInfo, this);
        build();
    }

    public DatePicker(Date date, DateTimeFormatInfo dateTimeFormatInfo, Date minDate, Date maxDate) {
        this.jsDate = new JsDate((double) date.getTime());
        this.minDate = new JsDate((double) minDate.getTime());
        this.maxDate = new JsDate((double) maxDate.getTime());
        datePickerMonth = DatePickerMonth.create(this.jsDate, dateTimeFormatInfo, this);
        build();
    }

    private DatePicker(JsDate date, DateTimeFormatInfo dateTimeFormatInfo, JsDate minDate, JsDate maxDate) {
        this.jsDate = date;
        this.minDate = minDate;
        this.maxDate = maxDate;
        datePickerMonth = DatePickerMonth.create(this.jsDate, dateTimeFormatInfo, this);
        build();
    }

    private void build() {

        element.appendChild(headerPanel);
        headerPanel.style().add(colorScheme.color().getBackground());
        dayName.style().add(colorScheme.darker_2().getBackground());
        headerPanel.appendChild(dayName);
        headerPanel.appendChild(monthName);
        headerPanel.appendChild(dateNumber);
        headerPanel.appendChild(yearNumber);

        element.appendChild(selectorsPanel);
        initSelectors();
        element.appendChild(datePickerMonth);
        element.appendChild(footerPanel);
        initFooter();

        datePickerMonth.init();
    }


    private void initFooter() {

        clearButton = Button.create("CLEAR").setColor(colorScheme.color());
        clearButton.style().add(DatePickerStyles.CLEAR_BUTTON);

        clearButton.addClickListener(evt -> {
            clearHandlers.forEach(PickerHandler::handle);
        });

        todayButton = Button.create("TODAY").setColor(colorScheme.color());
        todayButton.addClickListener(evt -> setDate(new Date()));

        closeButton = Button.create("CLOSE").setColor(colorScheme.color());
        closeButton.style().add(DatePickerStyles.CLOSE_BUTTON);

        closeButton.addClickListener(evt -> {
            closeHandlers.forEach(PickerHandler::handle);
        });

        footerPanel.appendChild(clearButton.linkify());
        footerPanel.appendChild(todayButton.linkify());
        footerPanel.appendChild(closeButton.linkify());

    }

    private void initSelectors() {

        int year = jsDate.getFullYear();
        yearSelect = Select.create()
                .css(DatePickerStyles.SELECTOR);

        yearSelect.getLeftAddonContainer().remove();
        yearSelect.getMandatoryAddOn().css(Styles.m_l_40);
        yearSelect.setPopupWidth(150);

        for (int i = minDate.getFullYear(); i <= maxDate.getFullYear(); i++) {
            SelectOption yearOption = SelectOption.create(i, i + "");
            yearSelect.appendChild(yearOption);

            if (i == year)
                yearSelect.select(yearOption);
        }
        yearSelect.addSelectionHandler(option -> {
            int selectedYear = option.getValue();
            jsDate.setYear(selectedYear);
            setDate(jsDate);
        });

        int month = jsDate.getMonth();
        monthSelect = Select.create()
                .css(DatePickerStyles.SELECTOR);
        monthSelect.getLeftAddonContainer().remove();
        monthSelect.getMandatoryAddOn().css(Styles.m_l_40);
        monthSelect.setPopupWidth(150);

        String[] months = getDateTimeFormatInfo().monthsShort();
        for (int i = 0; i < months.length; i++) {
            SelectOption monthOption = SelectOption.create(i, months[i]);
            monthSelect.appendChild(monthOption);
            if (i == month)
                monthSelect.select(monthOption);
        }
        monthSelect.addSelectionHandler(option -> {
            int selectedMonth = option.getValue();
            jsDate.setDate(DatePickerUtil.getValidMonthDate(jsDate.getFullYear(), selectedMonth, jsDate.getDate()));
            jsDate.setMonth(selectedMonth);
            setDate(jsDate);
        });

        Column yearColumn = Column.span5()
                .condenced()
                .appendChild(yearSelect.element());

        Column monthColumn = Column.span5()
                .condenced()
                .appendChild(monthSelect.element());

        Column backColumn = Column.span1()
                .condenced();

        Column forwardColumn = Column.span1()
                .condenced();

        Row row = Row.create()
                .setGap(px.of(5))
                .styler(style -> style
                        .add(DatePickerStyles.SELECTOR_ROW));
        navigateBefore = Icons.ALL.navigate_before()
                .clickable()
                .styler(style -> style.add(Styles.m_r_5))
                .addClickListener(evt -> {
                    JsDate jsDate = getJsDate();
                    int currentMonth = jsDate.getMonth();
                    if (currentMonth == 0) {
                        jsDate.setYear(jsDate.getFullYear() - 1);
                        jsDate.setMonth(11);
                    } else {
                        jsDate.setMonth(currentMonth - 1);
                    }

                    yearSelect.setValue(jsDate.getFullYear(), true);
                    monthSelect.selectAt(jsDate.getMonth(), true);
                    setDate(jsDate);
                });

        navigateNext = Icons.ALL.navigate_next()
                .clickable()
                .styler(style -> style.add(Styles.m_l_5))
                .addClickListener(evt -> {
                    JsDate jsDate = getJsDate();
                    int currentMonth = jsDate.getMonth();
                    if (currentMonth == 11) {
                        jsDate.setYear(jsDate.getFullYear() + 1);
                        jsDate.setMonth(0);
                    } else {
                        jsDate.setMonth(currentMonth + 1);
                    }

                    yearSelect.setValue(jsDate.getFullYear(), true);
                    monthSelect.selectAt(jsDate.getMonth(), true);
                    setDate(jsDate);
                });


        selectorsPanel.appendChild(row
                .appendChild(backColumn.appendChild(navigateBefore))
                .appendChild(yearColumn)
                .appendChild(monthColumn)
                .appendChild(forwardColumn.appendChild(navigateNext)));
    }


    public static DatePicker create() {
        DateTimeFormatInfo dateTimeFormatInfo = DateTimeFormatInfo_factory.create();

        return new DatePicker(new Date(), dateTimeFormatInfo);
    }

    public static DatePicker create(Date date) {
        DateTimeFormatInfo dateTimeFormatInfo = DateTimeFormatInfo_factory.create();
        return new DatePicker(date, dateTimeFormatInfo);
    }

    public static DatePicker create(Date date, Date minDate, Date maxDate) {
        DateTimeFormatInfo dateTimeFormatInfo = DateTimeFormatInfo_factory.create();
        return new DatePicker(date, dateTimeFormatInfo, minDate, maxDate);
    }

    public static DatePicker create(Date date, DateTimeFormatInfo dateTimeFormatInfo) {
        return new DatePicker(date, dateTimeFormatInfo);
    }

    public static DatePicker create(Date date, DateTimeFormatInfo dateTimeFormatInfo, Date minDate, Date maxDate) {
        return new DatePicker(date, dateTimeFormatInfo, minDate, maxDate);
    }

    @Override
    public HTMLDivElement element() {
        return element.element();
    }

    @Override
    public DatePicker value(Date value) {
        setValue(value);
        return this;
    }

    @Override
    public Date getValue() {
        return datePickerMonth.getValue();
    }

    @Override
    public void setValue(Date value) {
        datePickerMonth.value(value);
    }

    public DatePicker setDate(Date date) {
        this.value(date);
        return this;
    }

    public Date getDate() {
        return this.getValue();
    }

    public DatePicker setDate(JsDate jsDate) {
        this.value(new Date(new Double(jsDate.getTime()).longValue()));
        return this;
    }

    public JsDate getJsDate() {
        return new JsDate((double) getValue().getTime());
    }

    public DatePicker addDateSelectionHandler(DateSelectionHandler dateSelectionHandler) {
        this.dateSelectionHandlers.add(dateSelectionHandler);
        return this;
    }

    public DatePicker removeDateSelectionHandler(DateSelectionHandler dateSelectionHandler) {
        this.dateSelectionHandlers.remove(dateSelectionHandler);
        return this;
    }

    public List getDateSelectionHandlers() {
        return this.dateSelectionHandlers;
    }

    public DatePicker clearDaySelectionHandlers() {
        this.dateSelectionHandlers.clear();
        return this;
    }


    public DatePicker addDateDayClickHandler(DateDayClickedHandler dateDayClickedHandler) {
        this.dateDayClickedHandlers.add(dateDayClickedHandler);
        return this;
    }

    public DatePicker removeDateDayClickedHandler(DateDayClickedHandler dateClickedHandler) {
        this.dateDayClickedHandlers.remove(dateClickedHandler);
        return this;
    }

    public List getDateDayClickedHandlers() {
        return this.dateDayClickedHandlers;
    }

    public DatePicker clearDateDayClickedHandlers() {
        this.dateDayClickedHandlers.clear();
        return this;
    }

    public DatePicker setDateTimeFormatInfo(DateTimeFormatInfo dateTimeFormatInfo) {
        this.datePickerMonth.setDateTimeFormatInfo(dateTimeFormatInfo);
        updatePicker();
        return this;
    }

    public DateTimeFormatInfo getDateTimeFormatInfo() {
        return datePickerMonth.getDateTimeFormatInfo();
    }

    public DatePicker showBorder() {
        element.style().setBorder("1px solid " + colorScheme.color().getHex());
        return this;
    }

    public DatePicker setColorScheme(ColorScheme colorScheme) {
        backgroundHandler.onBackgroundChanged(getColorScheme(), colorScheme);
        this.headerPanel.style().remove(this.colorScheme.color().getBackground());
        this.dayName.style().remove(this.colorScheme.darker_2().getBackground());
        this.colorScheme = colorScheme;
        this.headerPanel.style().add(this.colorScheme.color().getBackground());
        this.dayName.style().add(this.colorScheme.darker_2().getBackground());
        this.datePickerMonth.setBackground(colorScheme.color());
        this.todayButton.setColor(colorScheme.color());
        this.closeButton.setColor(colorScheme.color());
        this.clearButton.setColor(colorScheme.color());
        return this;
    }

    public ColorScheme getColorScheme() {
        return colorScheme;
    }

    @Override
    public void onDaySelected(DatePickerElement datePickerElement) {
        this.selectedPickerElement = datePickerElement;
        this.jsDate.setTime(datePickerElement.getDate().getTime());
        updatePicker();
        publish();
    }

    @Override
    public void onDayClicked(DatePickerElement datePickerElement) {
        for (int i = 0; i < dateDayClickedHandlers.size(); i++) {
            dateDayClickedHandlers.get(i).onDateDayClicked(getDate(), getDateTimeFormatInfo());
        }
    }

    private void publish() {
        for (int i = 0; i < dateSelectionHandlers.size(); i++) {
            dateSelectionHandlers.get(i).onDateSelected(getDate(), getDateTimeFormatInfo());
        }
    }

    private void updatePicker() {
        int dayNameIndex = this.selectedPickerElement.getWeekDay() + getDateTimeFormatInfo().firstDayOfTheWeek();
        if (dayNameIndex > 6) {
            dayNameIndex = this.selectedPickerElement.getWeekDay() + getDateTimeFormatInfo().firstDayOfTheWeek() - 7;
        }
        this.dayName.setTextContent(getDateTimeFormatInfo().weekdaysFull()[dayNameIndex]);
        this.monthName.setTextContent(getDateTimeFormatInfo().monthsFull()[this.selectedPickerElement.getMonth()]);
        this.dateNumber.setTextContent(this.selectedPickerElement.getDay() + "");
        this.yearNumber.setTextContent(this.selectedPickerElement.getYear() + "");
        this.monthSelect.selectAt(this.selectedPickerElement.getMonth(), true);
        this.yearSelect.setValue(this.selectedPickerElement.getYear(), true);
    }

    public DatePicker showHeaderPanel() {
        headerPanel.show();
        return this;
    }

    public DatePicker hideHeaderPanel() {
        headerPanel.hide();
        return this;
    }

    public DatePicker showTodayButton() {
        this.todayButton.show();
        return this;
    }

    public DatePicker hideTodayButton() {
        this.todayButton.hide();
        return this;
    }

    public DatePicker showClearButton() {
        this.clearButton.show();
        return this;
    }

    public DatePicker hideClearButton() {
        this.clearButton.hide();
        return this;
    }


    public DatePicker showCloseButton() {
        this.closeButton.show();
        return this;
    }

    public DatePicker hideCloseButton() {
        this.closeButton.hide();
        return this;
    }

    public DatePicker addCloseHandler(PickerHandler closeHandler) {
        this.closeHandlers.add(closeHandler);
        return this;
    }

    public DatePicker removeCloseHandler(PickerHandler closeHandler) {
        this.closeHandlers.remove(closeHandler);
        return this;
    }

    public List getCloseHandlers() {
        return this.closeHandlers;
    }

    public DatePicker addClearHandler(PickerHandler clearHandler) {
        this.clearHandlers.add(clearHandler);
        return this;
    }

    public DatePicker removeClearHandler(PickerHandler clearHandler) {
        this.clearHandlers.remove(clearHandler);
        return this;
    }

    public List getClearHandlers() {
        return this.clearHandlers;
    }

    public DatePicker todayButtonText(String text) {
        this.todayButton.setContent(text);
        this.todayButton.element().title = text;
        return this;
    }

    public DatePicker clearButtonText(String text) {
        this.clearButton.setContent(text);
        this.clearButton.element().title = text;
        return this;
    }

    public DatePicker closeButtonText(String text) {
        this.closeButton.setContent(text);
        this.closeButton.element().title = text;
        return this;
    }

    public DatePicker fixedWidth() {
        element.style().setWidth(px.of(300), true);
        return this;
    }

    public DatePicker fixedWidth(String width) {
        element.style().setWidth(width, true);
        return this;
    }

    public DominoElement getHeaderPanel() {
        return DominoElement.of(headerPanel);
    }

    public DominoElement getSelectorsPanel() {
        return DominoElement.of(selectorsPanel);
    }

    public DominoElement getFooterPanel() {
        return DominoElement.of(footerPanel);
    }

    public DominoElement getDayNamePanel() {
        return DominoElement.of(dayName);
    }

    public DominoElement getMonthNamePanel() {
        return DominoElement.of(monthName);
    }

    public DominoElement getDateNumberPanel() {
        return DominoElement.of(dateNumber);
    }

    public DominoElement getYearNumberPanel() {
        return DominoElement.of(yearNumber);
    }

    public Icon getNavigateBefore() {
        return navigateBefore;
    }

    public Icon getNavigateNext() {
        return navigateNext;
    }

    public Button getTodayButton() {
        return todayButton;
    }

    public Button getClearButton() {
        return clearButton;
    }

    public Button getCloseButton() {
        return closeButton;
    }

    DatePicker setBackgroundHandler(BackgroundHandler backgroundHandler) {
        if (nonNull(backgroundHandler)) {
            this.backgroundHandler = backgroundHandler;
        }
        return this;
    }

    public ModalDialog createModal(String title) {
        return ModalDialog.createPickerModal(title, this.element());
    }

    @FunctionalInterface
    interface BackgroundHandler {
        void onBackgroundChanged(ColorScheme oldColorScheme, ColorScheme newColorScheme);
    }

    @FunctionalInterface
    public interface DateSelectionHandler {
        void onDateSelected(Date date, DateTimeFormatInfo dateTimeFormatInfo);
    }

    @FunctionalInterface
    public interface DateDayClickedHandler {
        void onDateDayClicked(Date date, DateTimeFormatInfo dateTimeFormatInfo);
    }

    public static class Formatter extends DateTimeFormat {

        protected Formatter(String pattern) {
            super(pattern);
        }

        protected Formatter(String pattern, DateTimeFormatInfo dtfi) {
            super(pattern, dtfi);
        }

        public static DateTimeFormat getFormat(String pattern, DateTimeFormatInfo dateTimeFormatInfo) {
            return DateTimeFormat.getFormat(pattern, dateTimeFormatInfo);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy