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

net.fortuna.ical4j.model.Component Maven / Gradle / Ivy

There is a newer version: 4.0.8
Show newest version
/**
 * Copyright (c) 2012, Ben Fortuna
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 *  o Redistributions of source code must retain the above copyright
 * notice, this list of conditions and the following disclaimer.
 *
 *  o Redistributions in binary form must reproduce the above copyright
 * notice, this list of conditions and the following disclaimer in the
 * documentation and/or other materials provided with the distribution.
 *
 *  o Neither the name of Ben Fortuna nor the names of any other contributors
 * may be used to endorse or promote products derived from this software
 * without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package net.fortuna.ical4j.model;

import net.fortuna.ical4j.model.parameter.Value;
import net.fortuna.ical4j.model.property.*;
import net.fortuna.ical4j.util.Strings;
import net.fortuna.ical4j.validate.ValidationException;
import net.fortuna.ical4j.validate.ValidationResult;
import org.apache.commons.lang3.builder.EqualsBuilder;
import org.apache.commons.lang3.builder.HashCodeBuilder;

import java.io.Serializable;
import java.time.temporal.Temporal;
import java.time.temporal.TemporalAmount;
import java.util.*;
import java.util.stream.Collectors;

import static net.fortuna.ical4j.model.Property.UID;

/**
 * $Id$ [Apr 5, 2004]
 * 

* Defines an iCalendar component. Subclasses of this class provide additional validation and typed values for specific * iCalendar components. * * @author Ben Fortuna */ public abstract class Component extends Content implements Prototype, Serializable, PropertyContainer, FluentComponent, Comparable { private static final long serialVersionUID = 4943193483665822201L; /** * Component start token. */ public static final String BEGIN = "BEGIN"; /** * Component end token. */ public static final String END = "END"; /** * Component token. */ public static final String PARTICIPANT = "PARTICIPANT"; /** * Component token. */ public static final String VLOCATION = "VLOCATION"; /** * Component token. */ public static final String VRESOURCE = "VRESOURCE"; /** * Component token. */ public static final String VEVENT = "VEVENT"; /** * Component token. */ public static final String VTODO = "VTODO"; /** * Component token. */ public static final String VJOURNAL = "VJOURNAL"; /** * Component token. */ public static final String VFREEBUSY = "VFREEBUSY"; /** * Component token. */ public static final String VTIMEZONE = "VTIMEZONE"; /** * Component token. */ public static final String VALARM = "VALARM"; /** * Component token. */ public static final String VAVAILABILITY = "VAVAILABILITY"; /** * Component token. */ public static final String VVENUE = "VVENUE"; /** * Component token. */ public static final String AVAILABLE = "AVAILABLE"; /** * Prefix for non-standard components. */ public static final String EXPERIMENTAL_PREFIX = "X-"; private final String name; protected PropertyList properties; protected ComponentList components; /** * Constructs a new component containing no properties. * * @param s a component name */ protected Component(final String s) { this(s, new PropertyList(), new ComponentList<>()); } protected Component(final String s, final PropertyList p) { this(s, p, new ComponentList<>()); } /** * Constructor made protected to enforce the use of ComponentFactory for component instantiation. * * @param s component name * @param p a list of properties */ protected Component(final String s, final PropertyList p, ComponentList c) { this.name = s; this.properties = p; this.components = c; } /** * {@inheritDoc} */ @Override public String toString() { return BEGIN + ':' + name + Strings.LINE_SEPARATOR + properties + components + END + ':' + name + Strings.LINE_SEPARATOR; } /** * @return Returns the name. */ @Override public final String getName() { return name; } @Override public String getValue() { return properties.toString(); } public List getProperties() { return PropertyContainer.super.getProperties(); } @Override public Component getFluentTarget() { return this; } /** * @return Returns the underlying property list. */ @Override public final PropertyList getPropertyList() { return properties; } @Override public void setPropertyList(PropertyList properties) { this.properties = properties; } /** * Returns the UID property of this component if available. * @return a Uid instance, or null if no UID property exists */ public Optional getUid() { return getProperty(UID); } /** * Perform validation on a component and its properties. * * @throws ValidationException where the component is not in a valid state */ public ValidationResult validate() throws ValidationException { return validate(true); } /** * Perform validation on a component. * * @param recurse indicates whether to validate the component's properties * @throws ValidationException where the component is not in a valid state */ public abstract ValidationResult validate(final boolean recurse) throws ValidationException; /** * Invoke validation on the component properties in its current state. * * @throws ValidationException where any of the component properties is not in a valid state */ protected ValidationResult validateProperties() throws ValidationException { var result = new ValidationResult(); for (final var property : getProperties()) { result = result.merge(property.validate()); } return result; } /** * {@inheritDoc} */ @Override public boolean equals(final Object arg0) { if (arg0 instanceof Component) { final var c = (Component) arg0; return new EqualsBuilder().append(getName(), c.getName()) .append(getProperties(), c.getProperties()).isEquals(); } return super.equals(arg0); } /** * {@inheritDoc} */ @Override public int hashCode() { return new HashCodeBuilder().append(getName()).append(getProperties()) .toHashCode(); } /** * Returns a new component factory used to create deep copies. * @return a component factory instance */ protected abstract ComponentFactory newFactory(); /** * Create a (deep) copy of this component. * @return the component copy */ public Component copy() { return newFactory().createComponent(new PropertyList(getProperties().parallelStream() .map(Prototype::copy).collect(Collectors.toList()))); } /** * Calculates the recurrence set for this component using the specified period. * The recurrence set is derived from a combination of the component start date, * recurrence rules and dates, and exception rules and dates. Note that component * transparency and anniversary-style dates do not affect the resulting * intersection. * *

If an explicit DURATION is not specified, the effective duration of each * returned period is derived from the DTSTART and DTEND or DUE properties. * If the component has no DURATION, DTEND or DUE, the effective duration is set * to PT0S

* * NOTE: As a component may be defined in terms of floating date-time values (i.e. without a specific * timezone), when calculating a recurrence set we must explicitly provide an applicable timezone * for calculations. * * @param period a range that defines the boundary for calculations * @return a list of periods representing component occurrences within the specified boundary */ public final Set> calculateRecurrenceSet(final Period period) { final Set> recurrenceSet = new TreeSet<>(); final Optional> start = getProperty(Property.DTSTART); Optional> end = getProperty(Property.DTEND); if (end.isEmpty()) { end = getProperty(Property.DUE); } Optional duration = getProperty(Property.DURATION); // if no start date specified return empty list.. if (start.isEmpty()) { return Collections.emptySet(); } // if an explicit event duration is not specified, derive a value for recurring // periods from the end date.. TemporalAmount rDuration; // if no end or duration specified, end date equals start date.. if (end.isEmpty() && duration.isEmpty()) { rDuration = java.time.Duration.ZERO; } else if (duration.isEmpty()) { rDuration = TemporalAmountAdapter.between(start.get().getDate(), end.get().getDate()).getDuration(); } else { rDuration = duration.get().getDuration(); } // add recurrence dates.. List rDates = getProperties(Property.RDATE); for (var p : rDates) { Optional value = p.getParameter(Parameter.VALUE); if (value.equals(Optional.of(Value.PERIOD))) { recurrenceSet.addAll(((RDate) p).getPeriods().orElse(Collections.emptySet()).parallelStream() .filter(period::intersects).collect(Collectors.toList())); } else { recurrenceSet.addAll(((DateListProperty) p).getDates().parallelStream() .filter(period::includes).map(d -> new Period<>(d, rDuration)).collect(Collectors.toList())); } } // allow for recurrence rules that start prior to the specified period // but still intersect with it.. final var startMinusDuration = period.getStart().minus(rDuration); final T seed = start.get().getDate(); // add recurrence rules.. List rRules = getProperties(Property.RRULE); if (!rRules.isEmpty()) { recurrenceSet.addAll(rRules.stream().map(r -> ((RRule) r).getRecur().getDates(seed, startMinusDuration, period.getEnd())).flatMap(List::stream) .map(rruleDate -> new Period<>(rruleDate, rDuration)).collect(Collectors.toList())); } else { // add initial instance if intersection with the specified period.. Period startPeriod; if (end.isPresent()) { startPeriod = new Period<>(seed, end.get().getDate()); } else { /* * PeS: Anniversary type has no DTEND nor DUR, define DUR * locally, otherwise we get NPE */ startPeriod = duration.map(value -> new Period<>(seed, value.getDuration())).orElseGet( () -> new Period<>(seed, new Duration(rDuration).getDuration())); } if (period.intersects(startPeriod)) { recurrenceSet.add(startPeriod); } } // subtract exception dates.. List exDateProps = getProperties(Property.EXDATE); List exDates = exDateProps.stream().map(p -> ((DateListProperty) p).getDates()) .flatMap(List::stream).collect(Collectors.toList()); recurrenceSet.removeIf(recurrence -> exDates.contains(recurrence.getStart())); // subtract exception rules.. List exRules = getProperties(Property.EXRULE); List exRuleDates = exRules.stream().map(e -> ((ExRule) e).getRecur().getDates(seed, period)).flatMap(List::stream).collect(Collectors.toList()); recurrenceSet.removeIf(recurrence -> exRuleDates.contains(recurrence.getStart())); // set a link to the origin recurrenceSet.forEach( p -> p.setComponent(this)); return recurrenceSet; } @Override public int compareTo(Component o) { if (this.equals(o)) { return 0; } return Comparator.comparing(Component::getName) .thenComparing(Component::getPropertyList) .compare(this, o); } }