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

org.jdesktop.swingx.calendar.AbstractDateSelectionModel Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2006 Sun Microsystems, Inc., 4150 Network Circle,
 * Santa Clara, California 95054, U.S.A. All rights reserved.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 * 
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 * 
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 */
package org.jdesktop.swingx.calendar;

import java.util.Calendar;
import java.util.Collections;
import java.util.Date;
import java.util.List;
import java.util.Locale;
import java.util.SortedSet;
import java.util.TimeZone;
import java.util.TreeSet;

import org.jdesktop.swingx.event.DateSelectionEvent;
import org.jdesktop.swingx.event.DateSelectionListener;
import org.jdesktop.swingx.event.EventListenerMap;
import org.jdesktop.swingx.event.DateSelectionEvent.EventType;

/**
 * Abstract base implementation of DateSelectionModel. 
 * Implements notification, Calendar related properties and lower/upper bounds.
 * 
 * @author Jeanette Winzenburg
 */
public abstract class AbstractDateSelectionModel implements DateSelectionModel {
	
    /** empty dates */
    public static final SortedSet EMPTY_DATES = Collections.unmodifiableSortedSet(new TreeSet());
    
    /**
     * the Listeners collection
     */
    protected EventListenerMap listenerMap;
    
    /** TODO doc */
    protected boolean adjusting;
    /** java.util.Calendar */
    protected Calendar calendar;
    /** upper bound for the model */
    protected Date upperBound;
    /** lower bound for the model */
    protected Date lowerBound;

    /** 
     * the locale used by the calendar. 

* NOTE: need to keep separately as a Calendar has no getter. */ protected Locale locale; /** * Instantiates a DateSelectionModel with default locale. */ public AbstractDateSelectionModel() { this(null); } /** * Instantiates a DateSelectionModel with the given locale. If the locale is * null, the Locale's default is used. * * PENDING JW: fall back to JComponent.getDefaultLocale instead? We use this * with components anyway? * * @param locale the Locale to use with this model, defaults to Locale.default() * if null. */ public AbstractDateSelectionModel(Locale locale) { this.listenerMap = new EventListenerMap(); setLocale(locale); } /** * {@inheritDoc} */ @Override public Calendar getCalendar() { return (Calendar) calendar.clone(); } /** * {@inheritDoc} */ @Override public int getFirstDayOfWeek() { return calendar.getFirstDayOfWeek(); } /** * {@inheritDoc} */ @Override public void setFirstDayOfWeek(final int firstDayOfWeek) { if (firstDayOfWeek == getFirstDayOfWeek()) return; calendar.setFirstDayOfWeek(firstDayOfWeek); fireValueChanged(EventType.CALENDAR_CHANGED); } /** * {@inheritDoc} */ @Override public int getMinimalDaysInFirstWeek() { return calendar.getMinimalDaysInFirstWeek(); } /** * {@inheritDoc} */ @Override public void setMinimalDaysInFirstWeek(int minimalDays) { if (minimalDays == getMinimalDaysInFirstWeek()) return; calendar.setMinimalDaysInFirstWeek(minimalDays); fireValueChanged(EventType.CALENDAR_CHANGED); } /** * {@inheritDoc} */ @Override public TimeZone getTimeZone() { return calendar.getTimeZone(); } /** * {@inheritDoc} */ @Override public void setTimeZone(TimeZone timeZone) { if (getTimeZone().equals(timeZone)) return; TimeZone oldTimeZone = getTimeZone(); calendar.setTimeZone(timeZone); adjustDatesToTimeZone(oldTimeZone); fireValueChanged(EventType.CALENDAR_CHANGED); } /** * Adjusts all stored dates to a new time zone. * This method is called after the change had been made.

* * This implementation resets all dates to null, clears everything. * Subclasses may override to really map to the new time zone. * * @param oldTimeZone the old time zone * */ protected void adjustDatesToTimeZone(TimeZone oldTimeZone) { clearSelection(); setLowerBound(null); setUpperBound(null); setUnselectableDates(EMPTY_DATES); } /** * {@inheritDoc} */ @Override public Locale getLocale() { return locale; } /** * {@inheritDoc} */ @Override public void setLocale(Locale locale) { if (locale == null) { locale = Locale.getDefault(); } if (locale.equals(getLocale())) return; this.locale = locale; if (calendar != null) { calendar = Calendar.getInstance(calendar.getTimeZone(), locale); } else { calendar = Calendar.getInstance(locale); } fireValueChanged(EventType.CALENDAR_CHANGED); } //------------------- utility methods /** * Returns the start of the day of the given date in this model's calendar. * NOTE: the calendar is changed by this operation. * * @param date the Date to get the start for. * @return the Date representing the start of the day of the input date. */ protected Date startOfDay(Date date) { return CalendarUtils.startOfDay(calendar, date); } /** * Returns the end of the day of the given date in this model's calendar. * NOTE: the calendar is changed by this operation. * * @param date the Date to get the start for. * @return the Date representing the end of the day of the input date. */ protected Date endOfDay(Date date) { return CalendarUtils.endOfDay(calendar, date); } /** * Returns a boolean indicating whether the given dates are on the same day in * the coordinates of the model's calendar. * * @param selected one of the dates to check, must not be null. * @param compare the other of the dates to check, must not be null. * @return true if both dates represent the same day in this model's calendar. */ protected boolean isSameDay(Date selected, Date compare) { return startOfDay(selected).equals(startOfDay(compare)); } //------------------- bounds /** * {@inheritDoc} */ @Override public Date getUpperBound() { return upperBound; } /** * {@inheritDoc} */ @Override public void setUpperBound(Date upperBound) { if (upperBound != null) { upperBound = getNormalizedDate(upperBound); } if (CalendarUtils.areEqual(upperBound, getUpperBound())) return; this.upperBound = upperBound; if (this.upperBound != null && !isSelectionEmpty()) { long justAboveUpperBoundMs = this.upperBound.getTime() + 1; removeSelectionInterval(new Date(justAboveUpperBoundMs), getLastSelectionDate()); } fireValueChanged(EventType.UPPER_BOUND_CHANGED); } /** * {@inheritDoc} */ @Override public Date getLowerBound() { return lowerBound; } /** * {@inheritDoc} */ @Override public void setLowerBound(Date lowerBound) { if (lowerBound != null) { lowerBound = getNormalizedDate(lowerBound); } if (CalendarUtils.areEqual(lowerBound, getLowerBound())) return; this.lowerBound = lowerBound; if (this.lowerBound != null && !isSelectionEmpty()) { // Remove anything below the lower bound long justBelowLowerBoundMs = this.lowerBound.getTime() - 1; removeSelectionInterval(getFirstSelectionDate(), new Date( justBelowLowerBoundMs)); } fireValueChanged(EventType.LOWER_BOUND_CHANGED); } /** * {@inheritDoc} */ @Override public boolean isAdjusting() { return adjusting; } /** * {@inheritDoc} */ @Override public void setAdjusting(boolean adjusting) { if (adjusting == isAdjusting()) return; this.adjusting = adjusting; fireValueChanged(adjusting ? EventType.ADJUSTING_STARTED : EventType.ADJUSTING_STOPPED); } //----------------- notification /** * {@inheritDoc} */ @Override public void addDateSelectionListener(DateSelectionListener l) { listenerMap.add(DateSelectionListener.class, l); } /** * {@inheritDoc} */ @Override public void removeDateSelectionListener(DateSelectionListener l) { listenerMap.remove(DateSelectionListener.class, l); } /** * get all registered listener to this model. * @return a List of listeners */ public List getDateSelectionListeners() { return listenerMap.getListeners(DateSelectionListener.class); } /** * Notifies all listeners that have registered interest for notification on value change. * @param eventType DateSelectionEvent type */ protected void fireValueChanged(DateSelectionEvent.EventType eventType) { List listeners = getDateSelectionListeners(); DateSelectionEvent e = null; for (DateSelectionListener listener : listeners) { if (e == null) { e = new DateSelectionEvent(this, eventType, isAdjusting()); } listener.valueChanged(e); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy