calendarimporter/php/ical/lib/vcalendar.class.php

2133 lines
87 KiB
PHP

<?php
/*********************************************************************************/
/**
*
* iCalcreator, a PHP rfc2445/rfc5545 solution.
*
* @copyright Copyright (c) 2007-2015 Kjell-Inge Gustafsson, kigkonsult, All rights reserved
* @link http://kigkonsult.se/iCalcreator/index.php
* @license http://kigkonsult.se/downloads/dl.php?f=LGPL
* @package iCalcreator
* @version 2.22
*/
/**
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
/*********************************************************************************/
/**
* vcalendar class
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.11 - 2015-03-31
*/
class vcalendar extends iCalBase {
/**
* @var array $calscale calendar property variable
* @var array $method calendar property variable
* @var array $prodid calendar property variable
* @var array $version calendar property variable
* @access private
*/
private $calscale;
private $method;
private $prodid;
private $version;
/**
* @var array $directory calendar config variable
* @var array $filename calendar config variable
* @var array $url calendar config variable
* @access private
*/
private $directory;
private $filename;
private $url;
/**
* redirect headers
*
* @var array $headers
* @access private
* @static
*/
private static $headers = array( 'Content-Encoding: gzip',
'Vary: *',
'Content-Length: %s',
'Content-Type: application/calendar+xml; charset=utf-8',
'Content-Type: text/calendar; charset=utf-8',
'Content-Disposition: attachment; filename="%s"',
'Content-Disposition: inline; filename="%s"',
'Cache-Control: max-age=10',
);
/**
* constructor for calendar object
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.15 - 2015-04-04
* @param array $config
* @uses vcalendar::_makeVersion()
* @uses vcalendar::$calscale
* @uses vcalendar::$method
* @uses vcalendar::_makeUnique_id()
* @uses vcalendar::$prodid
* @uses vcalendar::$xprop
* @uses vcalendar::$language
* @uses vcalendar::$directory
* @uses vcalendar::$filename
* @uses vcalendar::$url
* @uses vcalendar::$dtzid
* @uses vcalendar::setConfig()
* @uses vcalendar::$xcaldecl
* @uses vcalendar::$components
*/
function vcalendar ( $config = array()) {
$this->_makeVersion();
$this->calscale = null;
$this->method = null;
$this->_makeUnique_id();
$this->prodid = null;
$this->xprop = array();
$this->language = null;
$this->directory = '.';
$this->filename = null;
$this->url = null;
$this->dtzid = null;
/**
* language = <Text identifying a language, as defined in [RFC 1766]>
*/
if( defined( 'ICAL_LANG' ) && !isset( $config['language'] ))
$config['language'] = ICAL_LANG;
if( !isset( $config['allowEmpty'] )) $config['allowEmpty'] = TRUE;
if( !isset( $config['nl'] )) $config['nl'] = "\r\n";
if( !isset( $config['format'] )) $config['format'] = 'iCal';
if( !isset( $config['delimiter'] )) $config['delimiter'] = DIRECTORY_SEPARATOR;
$this->setConfig( $config );
$this->xcaldecl = array();
$this->components = array();
}
/**
* return iCalcreator version number
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.18.5 - 2013-08-29
* @uses ICALCREATOR_VERSION
* @return string
*/
public static function iCalcreatorVersion() {
return trim( substr( ICALCREATOR_VERSION, strpos( ICALCREATOR_VERSION, ' ' )));
}
/*********************************************************************************/
/**
* Property Name: CALSCALE
*/
/**
* creates formatted output for calendar property calscale
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.10.16 - 2011-10-28
* @uses vcalendar::$calscale
* @uses vcalendar::$format
* @uses vcalendar::$nl
* @return string
*/
function createCalscale() {
if( empty( $this->calscale )) return FALSE;
switch( $this->format ) {
case 'xcal':
return $this->nl.' calscale="'.$this->calscale.'"';
break;
default:
return 'CALSCALE:'.$this->calscale.$this->nl;
break;
}
}
/**
* set calendar property calscale
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.4.8 - 2008-10-21
* @param string $value
* @uses vcalendar::$calscale
* @return void
*/
function setCalscale( $value ) {
if( empty( $value )) return FALSE;
$this->calscale = $value;
}
/*********************************************************************************/
/**
* Property Name: METHOD
*/
/**
* creates formatted output for calendar property method
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.10.16 - 2011-10-28
* @uses vcalendar::$method
* @uses vcalendar::$format
* @uses vcalendar::$nl
* @return string
*/
function createMethod() {
if( empty( $this->method )) return FALSE;
switch( $this->format ) {
case 'xcal':
return $this->nl.' method="'.$this->method.'"';
break;
default:
return 'METHOD:'.$this->method.$this->nl;
break;
}
}
/**
* set calendar property method
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.4.8 - 2008-20-23
* @param string $value
* @uses vcalendar::$method
* @return bool
*/
function setMethod( $value ) {
if( empty( $value )) return FALSE;
$this->method = $value;
return TRUE;
}
/*********************************************************************************/
/**
* Property Name: PRODID
*
*/
/**
* creates formatted output for calendar property prodid
*
* @copyright copyright (c) 2007-2013 Kjell-Inge Gustafsson, kigkonsult, All rights reserved
* @license http://kigkonsult.se/downloads/dl.php?f=LGPL
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.11 - 2015-03-31
* @uses vcalendar::$prodid
* @uses vcalendar::_makeProdid()
* @uses vcalendar::$format
* @uses vcalendar::$nl
* @uses vcalendar::_createElement()
* @return string
*/
function createProdid() {
if( !isset( $this->prodid ))
$this->_makeProdid();
switch( $this->format ) {
case 'xcal':
return $this->nl.' prodid="'.$this->prodid.'"';
break;
default:
return $this->_createElement( 'PRODID', '', $this->prodid );
break;
}
}
/**
* make default value for calendar prodid, do NOT alter or remove this method or invoke of this method
*
* @copyright copyright (c) 2007-2013 Kjell-Inge Gustafsson, kigkonsult, All rights reserved
* @license http://kigkonsult.se/downloads/dl.php?f=LGPL
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.6.8 - 2009-12-30
* @uses vcalendar::$prodid
* @uses vcalendar::$unique_id
* @uses ICALCREATOR_VERSION
* @uses vcalendar::$language
* @return void
*/
function _makeProdid() {
$this->prodid = '-//'.$this->unique_id.'//NONSGML kigkonsult.se '.ICALCREATOR_VERSION.'//'.strtoupper( $this->language );
}
/**
* Conformance: The property MUST be specified once in an iCalendar object.
* Description: The vendor of the implementation SHOULD assure that this
* is a globally unique identifier; using some technique such as an FPI
* value, as defined in [ISO 9070].
*/
/**
* make default unique_id for calendar prodid
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 0.3.0 - 2006-08-10
* @uses vcalendar::$unique_id
* @return void
*/
function _makeUnique_id() {
$this->unique_id = ( isset( $_SERVER['SERVER_NAME'] )) ? gethostbyname( $_SERVER['SERVER_NAME'] ) : 'localhost';
}
/*********************************************************************************/
/**
* Property Name: VERSION
*
* Description: A value of "2.0" corresponds to this memo.
*/
/**
* creates formatted output for calendar property version
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.10.16 - 2011-10-28
* @uses vcalendar::$version
* @uses vcalendar::$format
* @uses vcalendar::$nl
* @return string
*/
function createVersion() {
if( empty( $this->version ))
$this->_makeVersion();
switch( $this->format ) {
case 'xcal':
return $this->nl.' version="'.$this->version.'"';
break;
default:
return 'VERSION:'.$this->version.$this->nl;
break;
}
}
/**
* set default calendar version
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 0.3.0 - 2006-08-10
* @uses vcalendar::$version
* @return void
*/
function _makeVersion() {
$this->version = '2.0';
}
/**
* set calendar version
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.4.8 - 2008-10-23
* @param string $value
* @uses vcalendar::$version
* @return void
*/
function setVersion( $value ) {
if( empty( $value )) return FALSE;
$this->version = $value;
return TRUE;
}
/*********************************************************************************/
/*********************************************************************************/
/**
* delete calendar property value
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.8.8 - 2011-03-15
* @param mixed $propName bool FALSE => X-property
* @param int $propix specific property in case of multiply occurences
* @uses vcalendar::$propdelix
* @uses vcalendar::$calscale
* @uses vcalendar::$method
* @uses vcalendar::$xprop
* @return bool, if successfull delete
*/
function deleteProperty( $propName=FALSE, $propix=FALSE ) {
$propName = ( $propName ) ? strtoupper( $propName ) : 'X-PROP';
if( !$propix )
$propix = ( isset( $this->propdelix[$propName] ) && ( 'X-PROP' != $propName )) ? $this->propdelix[$propName] + 2 : 1;
$this->propdelix[$propName] = --$propix;
$return = FALSE;
switch( $propName ) {
case 'CALSCALE':
if( isset( $this->calscale )) {
$this->calscale = null;
$return = TRUE;
}
break;
case 'METHOD':
if( isset( $this->method )) {
$this->method = null;
$return = TRUE;
}
break;
default:
$reduced = array();
if( $propName != 'X-PROP' ) {
if( !isset( $this->xprop[$propName] )) { unset( $this->propdelix[$propName] ); return FALSE; }
foreach( $this->xprop as $k => $a ) {
if(( $k != $propName ) && !empty( $a ))
$reduced[$k] = $a;
}
}
else {
if( count( $this->xprop ) <= $propix ) return FALSE;
$xpropno = 0;
foreach( $this->xprop as $xpropkey => $xpropvalue ) {
if( $propix != $xpropno )
$reduced[$xpropkey] = $xpropvalue;
$xpropno++;
}
}
$this->xprop = $reduced;
if( empty( $this->xprop )) {
unset( $this->propdelix[$propName] );
return FALSE;
}
return TRUE;
}
return $return;
}
/**
* get calendar property value/params
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.09 - 2015-03-29
* @param string $propName
* @param int $propix specific property in case of multiply occurences
* @param bool $inclParam
* @uses vcalendar::$propix
* @uses vcalendar::$components
* @uses calendarComponent::$objName
* @uses iCalUtilityFunctions::$vComps
* @uses iCalUtilityFunctions::$mProps1
* @uses calendarComponent::_getProperties()
* @uses calendarComponent::getProperty()
* @uses iCalUtilityFunctions::_geo2str2()
* @uses iCalUtilityFunctions::$geoLatFmt
* @uses iCalUtilityFunctions::$geoLongFmt
* @uses iCalUtilityFunctions::$fmt
* @uses vcalendar::$calscale
* @uses vcalendar::$method
* @uses vcalendar::$prodid
* @uses vcalendar::_makeProdid()
* @uses vcalendar::$version
* @uses vcalendar::$xprop
* @return mixed
*/
function getProperty( $propName=FALSE, $propix=FALSE, $inclParam=FALSE ) {
$propName = ( $propName ) ? strtoupper( $propName ) : 'X-PROP';
if( 'X-PROP' == $propName ) {
if( empty( $propix ))
$propix = ( isset( $this->propix[$propName] )) ? $this->propix[$propName] + 2 : 1;
$this->propix[$propName] = --$propix;
}
switch( $propName ) {
case 'ATTENDEE':
case 'CATEGORIES':
case 'CONTACT':
case 'DTSTART':
case 'GEOLOCATION':
case 'LOCATION':
case 'ORGANIZER':
case 'PRIORITY':
case 'RESOURCES':
case 'STATUS':
case 'SUMMARY':
case 'RECURRENCE-ID-UID':
case 'RELATED-TO':
case 'R-UID':
case 'UID':
case 'URL':
$output = array();
foreach ( $this->components as $cix => $component) {
if( !in_array( $component->objName, iCalUtilityFunctions::$vComps ))
continue;
if( in_array( $propName, iCalUtilityFunctions::$mProps1 )) {
$component->_getProperties( $propName, $output );
continue;
}
elseif(( 3 < strlen( $propName )) && ( 'UID' == substr( $propName, -3 ))) {
if( FALSE !== ( $content = $component->getProperty( 'RECURRENCE-ID' )))
$content = $component->getProperty( 'UID' );
}
elseif( 'GEOLOCATION' == $propName ) {
$content = ( FALSE === ( $loc = $component->getProperty( 'LOCATION' ))) ? '' : $loc.' ';
if( FALSE === ( $geo = $component->getProperty( 'GEO' )))
continue;
$content .= iCalUtilityFunctions::_geo2str2( $geo['latitude'], iCalUtilityFunctions::$geoLatFmt ).
iCalUtilityFunctions::_geo2str2( $geo['longitude'], iCalUtilityFunctions::$geoLongFmt ).'/';
}
elseif( FALSE === ( $content = $component->getProperty( $propName )))
continue;
if(( FALSE === $content ) || empty( $content ))
continue;
elseif( is_array( $content )) {
if( isset( $content['year'] )) {
$key = sprintf( iCalUtilityFunctions::$fmt['Ymd'], (int) $content['year'], (int) $content['month'], (int) $content['day'] );
if( !isset( $output[$key] ))
$output[$key] = 1;
else
$output[$key] += 1;
}
else {
foreach( $content as $partValue => $partCount ) {
if( !isset( $output[$partValue] ))
$output[$partValue] = $partCount;
else
$output[$partValue] += $partCount;
}
}
} // end elseif( is_array( $content )) {
elseif( !isset( $output[$content] ))
$output[$content] = 1;
else
$output[$content] += 1;
} // end foreach ( $this->components as $cix => $component)
if( !empty( $output ))
ksort( $output );
return $output;
break;
case 'CALSCALE':
return ( !empty( $this->calscale )) ? $this->calscale : FALSE;
break;
case 'METHOD':
return ( !empty( $this->method )) ? $this->method : FALSE;
break;
case 'PRODID':
if( empty( $this->prodid ))
$this->_makeProdid();
return $this->prodid;
break;
case 'VERSION':
return ( !empty( $this->version )) ? $this->version : FALSE;
break;
default:
if( $propName != 'X-PROP' ) {
if( !isset( $this->xprop[$propName] )) return FALSE;
return ( $inclParam ) ? array( $propName, $this->xprop[$propName] )
: array( $propName, $this->xprop[$propName]['value'] );
}
else {
if( empty( $this->xprop )) return FALSE;
$xpropno = 0;
foreach( $this->xprop as $xpropkey => $xpropvalue ) {
if( $propix == $xpropno )
return ( $inclParam ) ? array( $xpropkey, $this->xprop[$xpropkey] )
: array( $xpropkey, $this->xprop[$xpropkey]['value'] );
else
$xpropno++;
}
unset( $this->propix[$propName] );
return FALSE; // not found ??
}
}
return FALSE;
}
/**
* general vcalendar property setting
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.2.13 - 2007-11-04
* @param mixed $args variable number of function arguments,
* first argument is ALWAYS component name,
* second ALWAYS component value!
* @uses vcalendar::setCalscale()
* @uses vcalendar::setMethod()
* @uses vcalendar::setVersion()
* @uses vcalendar::setXprop()
* @return bool
*/
function setProperty () {
$numargs = func_num_args();
if( 1 > $numargs )
return FALSE;
$arglist = func_get_args();
$arglist[0] = strtoupper( $arglist[0] );
switch( $arglist[0] ) {
case 'CALSCALE':
return $this->setCalscale( $arglist[1] );
case 'METHOD':
return $this->setMethod( $arglist[1] );
case 'VERSION':
return $this->setVersion( $arglist[1] );
default:
if( !isset( $arglist[1] )) $arglist[1] = null;
if( !isset( $arglist[2] )) $arglist[2] = null;
return $this->setXprop( $arglist[0], $arglist[1], $arglist[2] );
}
return FALSE;
}
/*********************************************************************************/
/**
* get vcalendar config values or * calendar components
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.11.7 - 2012-01-12
* @param mixed $config
* @uses vcalendar::getConfig()
* @uses vcalendar::$allowEmpty
* @uses vcalendar::$components
* @uses calendarComponent::_getProperties()
* @uses calendarComponent::$objName
* @uses calendarComponent::getProperty()
* @uses calendarComponent::_getConfig()
* @uses vcalendar::$url
* @uses vcalendar::$delimiter
* @uses vcalendar::$directory
* @uses vcalendar::$filename
* @uses vcalendar::$format
* @uses vcalendar::$language
* @uses vcalendar::$nl
* @uses vcalendar::$dtzid
* @uses vcalendar::$unique_id
* @return value
*/
function getConfig( $config = FALSE ) {
if( !$config ) {
$return = array();
$return['ALLOWEMPTY'] = $this->getConfig( 'ALLOWEMPTY' );
$return['DELIMITER'] = $this->getConfig( 'DELIMITER' );
$return['DIRECTORY'] = $this->getConfig( 'DIRECTORY' );
$return['FILENAME'] = $this->getConfig( 'FILENAME' );
$return['DIRFILE'] = $this->getConfig( 'DIRFILE' );
$return['FILESIZE'] = $this->getConfig( 'FILESIZE' );
$return['FORMAT'] = $this->getConfig( 'FORMAT' );
if( FALSE !== ( $lang = $this->getConfig( 'LANGUAGE' )))
$return['LANGUAGE'] = $lang;
$return['NEWLINECHAR'] = $this->getConfig( 'NEWLINECHAR' );
$return['UNIQUE_ID'] = $this->getConfig( 'UNIQUE_ID' );
if( FALSE !== ( $url = $this->getConfig( 'URL' )))
$return['URL'] = $url;
$return['TZID'] = $this->getConfig( 'TZID' );
return $return;
}
switch( strtoupper( $config )) {
case 'ALLOWEMPTY':
return $this->allowEmpty;
break;
case 'COMPSINFO':
unset( $this->compix );
$info = array();
foreach( $this->components as $cix => $component ) {
if( empty( $component )) continue;
$info[$cix]['ordno'] = $cix + 1;
$info[$cix]['type'] = $component->objName;
$info[$cix]['uid'] = $component->getProperty( 'uid' );
$info[$cix]['props'] = $component->getConfig( 'propinfo' );
$info[$cix]['sub'] = $component->getConfig( 'compsinfo' );
}
return $info;
break;
case 'DELIMITER':
return $this->delimiter;
break;
case 'DIRECTORY':
if( empty( $this->directory ) && ( '0' != $this->directory ))
$this->directory = '.';
return $this->directory;
break;
case 'DIRFILE':
return $this->getConfig( 'directory' ).$this->getConfig( 'delimiter' ).$this->getConfig( 'filename' );
break;
case 'FILEINFO':
return array( $this->getConfig( 'directory' )
, $this->getConfig( 'filename' )
, $this->getConfig( 'filesize' ));
break;
case 'FILENAME':
if( empty( $this->filename ) && ( '0' != $this->filename )) {
if( 'xcal' == $this->format )
$this->filename = date( 'YmdHis' ).'.xml'; // recommended xcs.. .
else
$this->filename = date( 'YmdHis' ).'.ics';
}
return $this->filename;
break;
case 'FILESIZE':
$size = 0;
if( empty( $this->url )) {
$dirfile = $this->getConfig( 'dirfile' );
if( !is_file( $dirfile ) || ( FALSE === ( $size = filesize( $dirfile ))))
$size = 0;
clearstatcache();
}
return $size;
break;
case 'FORMAT':
return ( $this->format == 'xcal' ) ? 'xCal' : 'iCal';
break;
case 'LANGUAGE':
/* get language for calendar component as defined in [RFC 1766] */
return $this->language;
break;
case 'NL':
case 'NEWLINECHAR':
return $this->nl;
break;
case 'TZID':
return $this->dtzid;
break;
case 'UNIQUE_ID':
return $this->unique_id;
break;
case 'URL':
if( !empty( $this->url ))
return $this->url;
else
return FALSE;
break;
}
}
/**
* general vcalendar config setting
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.20.2 - 2014-05-09
* @param mixed $config
* @param string $value
* @uses vcalendar::setConfig()
* @uses vcalendar::$allowEmpty
* @uses vcalendar::$components
* @uses vcalendar::$url
* @uses vcalendar::$delimiter
* @uses vcalendar::$directory
* @uses vcalendar::$filename
* @uses vcalendar::$format
* @uses vcalendar::$language
* @uses vcalendar::$nl
* @uses vcalendar::$dtzid
* @uses vcalendar::$unique_id
* @uses vcalendar::_makeProdid()
* @uses vcalendar::$components
* @uses calendarComponent::setConfig()
* @uses calendarComponent::copy()
* @return void
*/
function setConfig( $config, $value = FALSE) {
if( is_array( $config )) {
$config = array_change_key_case( $config, CASE_UPPER );
if( isset( $config['DELIMITER'] )) {
if( FALSE === $this->setConfig( 'DELIMITER', $config['DELIMITER'] ))
return FALSE;
unset( $config['DELIMITER'] );
}
if( isset( $config['DIRECTORY'] )) {
if( FALSE === $this->setConfig( 'DIRECTORY', $config['DIRECTORY'] ))
return FALSE;
unset( $config['DIRECTORY'] );
}
foreach( $config as $cKey => $cValue ) {
if( FALSE === $this->setConfig( $cKey, $cValue ))
return FALSE;
}
return TRUE;
}
else
$res = FALSE;
$config = strtoupper( $config );
switch( $config ) {
case 'ALLOWEMPTY':
$this->allowEmpty = $value;
$subcfg = array( 'ALLOWEMPTY' => $value );
$res = TRUE;
break;
case 'DELIMITER':
$this->delimiter = $value;
return TRUE;
break;
case 'DIRECTORY':
if( FALSE === ( $value = realpath( rtrim( trim( $value ), $this->delimiter ))))
return FALSE;
else {
/* local directory */
$this->directory = $value;
$this->url = null;
return TRUE;
}
break;
case 'FILENAME':
$value = trim( $value );
$dirfile = $this->directory.$this->delimiter.$value;
if( file_exists( $dirfile )) {
/* local file exists */
if( is_readable( $dirfile ) || is_writable( $dirfile )) {
clearstatcache();
$this->filename = $value;
return TRUE;
}
else
return FALSE;
}
elseif( is_readable( $this->directory ) || is_writable( $this->directory )) {
/* read- or writable directory */
clearstatcache();
$this->filename = $value;
return TRUE;
}
else
return FALSE;
break;
case 'FORMAT':
$value = trim( strtolower( $value ));
if( 'xcal' == $value ) {
$this->format = 'xcal';
$this->attributeDelimiter = $this->nl;
$this->valueInit = null;
}
else {
$this->format = null;
$this->attributeDelimiter = ';';
$this->valueInit = ':';
}
$subcfg = array( 'FORMAT' => $value );
$res = TRUE;
break;
case 'LANGUAGE': // set language for calendar component as defined in [RFC 1766]
$value = trim( $value );
$this->language = $value;
$this->_makeProdid();
$subcfg = array( 'LANGUAGE' => $value );
$res = TRUE;
break;
case 'NL':
case 'NEWLINECHAR':
$this->nl = $value;
if( 'xcal' == $value ) {
$this->attributeDelimiter = $this->nl;
$this->valueInit = null;
}
else {
$this->attributeDelimiter = ';';
$this->valueInit = ':';
}
$subcfg = array( 'NL' => $value );
$res = TRUE;
break;
case 'TZID':
$this->dtzid = $value;
$subcfg = array( 'TZID' => $value );
$res = TRUE;
break;
case 'UNIQUE_ID':
$value = trim( $value );
$this->unique_id = $value;
$this->_makeProdid();
$subcfg = array( 'UNIQUE_ID' => $value );
$res = TRUE;
break;
case 'URL':
/* remote file - URL */
$value = str_replace( array( 'HTTP://', 'WEBCAL://', 'webcal://' ), 'http://', trim( $value ));
$value = str_replace( 'HTTPS://', 'https://', trim( $value ));
if(( 'http://' != substr( $value, 0, 7 )) && ( 'https://' != substr( $value, 0, 8 )))
return FALSE;
$this->directory = '.';
$this->url = $value;
if( '.ics' != strtolower( substr( $value, -4 )))
unset( $this->filename );
else
$this->filename = basename( $value );
return TRUE;
break;
default: // any unvalid config key.. .
return TRUE;
}
if( !$res ) return FALSE;
if( isset( $subcfg ) && !empty( $this->components )) {
foreach( $subcfg as $cfgkey => $cfgvalue ) {
foreach( $this->components as $cix => $component ) {
$res = $component->setConfig( $cfgkey, $cfgvalue, TRUE );
if( !$res )
break 2;
$this->components[$cix] = $component->copy(); // PHP4 compliant
}
}
}
return $res;
}
/*********************************************************************************/
/**
* add calendar component to container
*
* alias to setComponent
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 1.x.x - 2007-04-24
* @param object $component calendar component
* @uses vcalendar::setComponent()
* @return void
*/
function addComponent( $component ) {
$this->setComponent( $component );
}
/**
* delete calendar component from container
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.8.8 - 2011-03-15
* @param mixed $arg1 ordno / component type / component uid
* @param mixed $arg2 optional, ordno if arg1 = component type
* @uses vcalendar::$components
* @uses calendarComponent::$objName
* @uses calendarComponent::getProperty()
* @return void
*/
function deleteComponent( $arg1, $arg2=FALSE ) {
$argType = $index = null;
if ( ctype_digit( (string) $arg1 )) {
$argType = 'INDEX';
$index = (int) $arg1 - 1;
}
elseif(( strlen( $arg1 ) <= strlen( 'vfreebusy' )) && ( FALSE === strpos( $arg1, '@' ))) {
$argType = strtolower( $arg1 );
$index = ( !empty( $arg2 ) && ctype_digit( (string) $arg2 )) ? (( int ) $arg2 - 1 ) : 0;
}
$cix1dC = 0;
foreach ( $this->components as $cix => $component) {
if( empty( $component )) continue;
if(( 'INDEX' == $argType ) && ( $index == $cix )) {
unset( $this->components[$cix] );
return TRUE;
}
elseif( $argType == $component->objName ) {
if( $index == $cix1dC ) {
unset( $this->components[$cix] );
return TRUE;
}
$cix1dC++;
}
elseif( !$argType && ($arg1 == $component->getProperty( 'uid' ))) {
unset( $this->components[$cix] );
return TRUE;
}
}
return FALSE;
}
/**
* get calendar component from container
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.11 - 2015-03-21
* @param mixed $arg1 optional, ordno/component type/ component uid
* @param mixed $arg2 optional, ordno if arg1 = component type
* @uses vcalendar::$compix
* @uses vcalendar::$components
* @uses iCalUtilityFunctions::$dateProps
* @uses iCalUtilityFunctions::$otherProps
* @uses iCalUtilityFunctions::$mProps1
* @uses calendarComponent::copy()
* @uses calendarComponent::$objName
* @uses calendarComponent::_getProperties()
* @uses calendarComponent::getProperty()
* @uses iCalUtilityFunctions::$fmt
* @return object
*/
function getComponent( $arg1=FALSE, $arg2=FALSE ) {
$index = $argType = null;
if ( !$arg1 ) { // first or next in component chain
$argType = 'INDEX';
$index = $this->compix['INDEX'] = ( isset( $this->compix['INDEX'] )) ? $this->compix['INDEX'] + 1 : 1;
}
elseif( is_array( $arg1 )) { // array( *[propertyName => propertyValue] )
$arg2 = implode( '-', array_keys( $arg1 ));
$index = $this->compix[$arg2] = ( isset( $this->compix[$arg2] )) ? $this->compix[$arg2] + 1 : 1;
}
elseif ( ctype_digit( (string) $arg1 )) { // specific component in chain
$argType = 'INDEX';
$index = (int) $arg1;
unset( $this->compix );
}
elseif(( strlen( $arg1 ) <= strlen( 'vfreebusy' )) && ( FALSE === strpos( $arg1, '@' ))) { // object class name
unset( $this->compix['INDEX'] );
$argType = strtolower( $arg1 );
if( !$arg2 )
$index = $this->compix[$argType] = ( isset( $this->compix[$argType] )) ? $this->compix[$argType] + 1 : 1;
elseif( isset( $arg2 ) && ctype_digit( (string) $arg2 ))
$index = (int) $arg2;
}
elseif(( strlen( $arg1 ) > strlen( 'vfreebusy' )) && ( FALSE !== strpos( $arg1, '@' ))) { // UID as 1st argument
if( !$arg2 )
$index = $this->compix[$arg1] = ( isset( $this->compix[$arg1] )) ? $this->compix[$arg1] + 1 : 1;
elseif( isset( $arg2 ) && ctype_digit( (string) $arg2 ))
$index = (int) $arg2;
}
if( isset( $index ))
$index -= 1;
$ckeys = array_keys( $this->components );
if( !empty( $index) && ( $index > end( $ckeys )))
return FALSE;
$cix1gC = 0;
foreach ( $this->components as $cix => $component) {
if( empty( $component )) continue;
if(( 'INDEX' == $argType ) && ( $index == $cix ))
return $component->copy();
elseif( $argType == $component->objName ) {
if( $index == $cix1gC )
return $component->copy();
$cix1gC++;
}
elseif( is_array( $arg1 )) { // array( *[propertyName => propertyValue] )
$hit = array();
$arg1 = array_change_key_case( $arg1, CASE_UPPER );
foreach( $arg1 as $pName => $pValue ) {
if( !in_array( $pName, iCalUtilityFunctions::$dateProps ) && !in_array( $pName, iCalUtilityFunctions::$otherProps ))
continue;
if( in_array( $pName, iCalUtilityFunctions::$mProps1 )) { // multiple occurrence
$propValues = array();
$component->_getProperties( $pName, $propValues );
$propValues = array_keys( $propValues );
$hit[] = ( in_array( $pValue, $propValues )) ? TRUE : FALSE;
continue;
} // end if(.. .// multiple occurrence
if( FALSE === ( $value = $component->getProperty( $pName ))) { // single occurrence
$hit[] = FALSE; // missing property
continue;
}
if( 'SUMMARY' == $pName ) { // exists within (any case)
$hit[] = ( FALSE !== stripos( $value, $pValue )) ? TRUE : FALSE;
continue;
}
if( in_array( $pName, iCalUtilityFunctions::$dateProps )) {
$valuedate = sprintf( iCalUtilityFunctions::$fmt['Ymd'], (int) $value['year'], (int) $value['month'], (int) $value['day'] );
if( 8 < strlen( $pValue )) {
if( isset( $value['hour'] )) {
if( 'T' == substr( $pValue, 8, 1 ))
$pValue = str_replace( 'T', '', $pValue );
$valuedate .= sprintf( iCalUtilityFunctions::$fmt['His'], (int) $value['hour'], (int) $value['min'], (int) $value['sec'] );
}
else
$pValue = substr( $pValue, 0, 8 );
}
$hit[] = ( $pValue == $valuedate ) ? TRUE : FALSE;
continue;
}
elseif( !is_array( $value ))
$value = array( $value );
foreach( $value as $part ) {
$part = ( FALSE !== strpos( $part, ',' )) ? explode( ',', $part ) : array( $part );
foreach( $part as $subPart ) {
if( $pValue == $subPart ) {
$hit[] = TRUE;
continue 3;
}
}
} // end foreach( $value as $part )
$hit[] = FALSE; // no hit in property
} // end foreach( $arg1 as $pName => $pValue )
if( in_array( TRUE, $hit )) {
if( $index == $cix1gC )
return $component->copy();
$cix1gC++;
}
} // end elseif( is_array( $arg1 )) { // array( *[propertyName => propertyValue] )
elseif( !$argType && ($arg1 == $component->getProperty( 'uid' ))) { // UID
if( $index == $cix1gC )
return $component->copy();
$cix1gC++;
}
} // end foreach ( $this->components.. .
/* not found.. . */
unset( $this->compix );
return FALSE;
}
/**
* create new calendar component, already included within calendar
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.20.4 - 2014-08-24
* @param string $compType component type
* @uses vcalendar::$components
* @return object
*/
function & newComponent( $compType ) {
$config = $this->getConfig();
$keys = array_keys( $this->components );
$ix = ( empty( $keys )) ? 0 : end( $keys) + 1;
switch( strtoupper( $compType )) {
case 'EVENT':
case 'VEVENT':
$this->components[$ix] = new vevent( $config );
break;
case 'TODO':
case 'VTODO':
$this->components[$ix] = new vtodo( $config );
break;
case 'JOURNAL':
case 'VJOURNAL':
$this->components[$ix] = new vjournal( $config );
break;
case 'FREEBUSY':
case 'VFREEBUSY':
$this->components[$ix] = new vfreebusy( $config );
break;
case 'TIMEZONE':
case 'VTIMEZONE':
array_unshift( $this->components, new vtimezone( $config ));
$ix = 0;
break;
default:
return FALSE;
}
return $this->components[$ix];
}
/**
* select components from calendar on date or selectOption basis
*
* Ensure DTSTART is set for every component.
* No date controls occurs.
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.11 - 2015-03-31
* @param mixed $startY optional, (int) start Year, default current Year
* ALT. (obj) start date (datetime)
* ALT. array selecOptions ( *[ <propName> => <uniqueValue> ] )
* @param mixed $startM optional, (int) start Month, default current Month
* ALT. (obj) end date (datetime)
* @param int $startD optional, start Day, default current Day
* @param int $endY optional, end Year, default $startY
* @param int $endM optional, end Month, default $startM
* @param int $endD optional, end Day, default $startD
* @param mixed $cType optional, calendar component type(-s), default FALSE=all else string/array type(-s)
* @param bool $flat optional, FALSE (default) => output : array[Year][Month][Day][]
* TRUE => output : array[] (ignores split)
* @param bool $any optional, TRUE (default) - select component(-s) that occurs within period
* FALSE - only component(-s) that starts within period
* @param bool $split optional, TRUE (default) - one component copy every DAY it occurs during the
* period (implies flat=FALSE)
* FALSE - one occurance of component only in output array
* @uses vcalendar::$components
* @uses vcalendar::selectComponents2()
* @uses iCalUtilityFunctions::$vComps
* @uses calendarComponent::$objName
* @uses calendarComponent::getProperty()
* @uses iCaldateTime::factory()
* @uses iCaldateTime::getTimezoneName()
* @uses iCaldateTime::getTime()
* @uses iCalUtilityFunctions::$fmt
* @uses iCaldateTime::$SCbools
* @uses iCaldateTime::format()
* @uses iCalUtilityFunctions::_strDate2arr()
* @uses iCalUtilityFunctions::_recur2date()
* @uses iCalUtilityFunctions::_inScope()
* @uses calendarComponent::copy()
* @uses calendarComponent::setProperty()
* @uses iCalUtilityFunctions::$fmt
* @uses calendarComponent::deleteProperty()
* @uses iCalUtilityFunctions::_setSortArgs()
* @uses iCalUtilityFunctions::_cmpfcn()
* @return array or FALSE
*/
function selectComponents( $startY=FALSE, $startM=FALSE, $startD=FALSE, $endY=FALSE, $endM=FALSE, $endD=FALSE, $cType=FALSE, $flat=FALSE, $any=TRUE, $split=TRUE ) {
/* check if empty calendar */
if( 0 >= count( $this->components )) return FALSE;
if( is_array( $startY ))
return $this->selectComponents2( $startY );
/* check default dates */
if( is_a( $startY, 'DateTime' ) && is_a( $startM, 'DateTime' )) {
$endY = $startM->format( 'Y' );
$endM = $startM->format( 'm' );
$endD = $startM->format( 'd' );
$startD = $startY->format( 'd' );
$startM = $startY->format( 'm' );
$startY = $startY->format( 'Y' );
}
else {
if( ! $startY ) $startY = date( 'Y' );
if( ! $startM ) $startM = date( 'm' );
if( ! $startD ) $startD = date( 'd' );
if( ! $endY ) $endY = $startY;
if( ! $endM ) $endM = $startM;
if( ! $endD ) $endD = $startD;
}
// echo "selectComp args={$startY}-{$startM}-{$startD} - {$endY}-{$endM}-{$endD}<br>\n"; $tcnt = 0;// test ###
/* check component types */
if( empty( $cType ))
$cType = iCalUtilityFunctions::$vComps;
else {
if( ! is_array( $cType ))
$cType = array( $cType );
$cType = array_map( 'strtolower', $cType );
foreach( $cType as $cix => $theType ) {
if( !in_array( $theType, iCalUtilityFunctions::$vComps ))
$cType[$cix] = 'vevent';
}
$cType = array_unique( $cType );
}
if(( FALSE === $flat ) && ( FALSE === $any )) // invalid combination
$split = FALSE;
if(( TRUE === $flat ) && ( TRUE === $split )) // invalid combination
$split = FALSE;
/* iterate components */
$result = array();
$this->sort( 'UID' );
$compUIDcmp = null;
$exdatelist = $recurridList = array();
$intervalP1D = new DateInterval( 'P1D' );
foreach ( $this->components as $cix => $component ) {
if( empty( $component )) continue;
/* deselect unvalid type components */
if( !in_array( $component->objName, $cType ))
continue;
unset( $compStart, $compEnd );
/* select start from dtstart or due if dtstart is missing */
$prop = $component->getProperty( 'dtstart', FALSE, TRUE );
if( empty( $prop ) && ( $component->objName == 'vtodo' ) && ( FALSE === ( $prop = $component->getProperty( 'due', FALSE, TRUE ))))
continue;
if( empty( $prop ))
continue;
/* get UID */
$compUID = $component->getProperty( 'UID' );
if( $compUIDcmp != $compUID ) {
$compUIDcmp = $compUID;
$exdatelist = $recurridList = array();
}
$recurrid = FALSE;
// file_put_contents( '/opt/work/iCal/iCalcreator/iCalcreator-2.20.x/log/log.txt', "#$cix".PHP_EOL.var_export( $component, TRUE ).PHP_EOL.PHP_EOL, FILE_APPEND ); // test ###
$compStart = iCaldateTime::factory( $prop['value'], $prop['params'], $prop['value'] );
$dtstartTz = $compStart->getTimezoneName();
if( isset( $prop['params']['VALUE'] ) && ( 'DATE' == $prop['params']['VALUE'] ))
$compStartHis = '';
else {
$his = $compStart->getTime();
$compStartHis = sprintf( iCalUtilityFunctions::$fmt['His'], (int) $his[0], (int) $his[1], (int) $his[2] );
}
/* get end date from dtend/due/duration properties */
if( FALSE !== ( $prop = $component->getProperty( 'dtend', FALSE, TRUE ))) {
$compEnd = iCaldateTime::factory( $prop['value'], $prop['params'], $prop['value'], $dtstartTz );
$compEnd->SCbools[ 'dtendExist'] = TRUE;
}
if( empty( $prop ) && ( $component->objName == 'vtodo' ) && ( FALSE !== ( $prop = $component->getProperty( 'due', FALSE, TRUE )))) {
$compEnd = iCaldateTime::factory( $prop['value'], $prop['params'], $prop['value'], $dtstartTz );
$compEnd->SCbools[ 'dueExist'] = TRUE;
}
if( empty( $prop ) && ( FALSE !== ( $prop = $component->getProperty( 'duration', FALSE, TRUE, TRUE )))) { // in dtend (array) format
$compEnd = iCaldateTime::factory( $prop['value'], $prop['params'], $prop['value'], $dtstartTz );
$compEnd->SCbools[ 'durationExist'] = TRUE;
}
if( ! empty( $prop ) && ! isset( $prop['value']['hour'] )) {
/* a DTEND without time part denotes an end of an event that actually ends the day before,
for an all-day event DTSTART=20071201 DTEND=20071202, taking place 20071201!!! */
$compEnd->SCbools[ 'endAllDayEvent'] = TRUE;
$compEnd->modify( '-1 day' );
$compEnd->setTime( 23, 59, 59 );
}
unset( $prop );
if( empty( $compEnd )) {
$compDuration = FALSE;
$compEnd = clone $compStart;
$compEnd->setTime( 23, 59, 59 ); // 23:59:59 the same day as start
}
else {
if( $compEnd->format( 'Ymd' ) < $compStart->format( 'Ymd' )) { // MUST be after start date!!
$compEnd = clone $compStart;
$compEnd->setTime( 23, 59, 59 ); // 23:59:59 the same day as start or ???
}
$compDuration = $compStart->diff( $compEnd ); // DateInterval
}
/* check recurrence-id (note, a missing sequence is the same as sequence=0 so don't test for sequence), to alter when hit dtstart/recurlist */
if( FALSE !== ( $prop = $component->getProperty( 'recurrence-id', FALSE, TRUE ))) {
$recurrid = iCaldateTime::factory( $prop['value'], $prop['params'], $prop['value'], $dtstartTz );
$rangeSet = ( isset( $prop['params']['RANGE'] ) && ( 'THISANDFUTURE' == $prop['params']['RANGE'] )) ? TRUE : FALSE;
$recurridList[$recurrid->key] = array( clone $compStart, clone $compEnd, $compDuration, $rangeSet ); // change recur this day to new YmdHis/duration/range
// echo "adding comp no:$cix with date=".$compStart->format(iCalUtilityFunctions::$fmt['YmdHis2e'])." to recurridList id={$recurrid->key}, newDate={$compStart->key}<br>\n"; // test ###
unset( $prop );
continue; // ignore any other props in the component
} // end recurrence-id/sequence test
// else echo "comp no:$cix with date=".$compStart->format().", NO recurrence-id<br>\n"; // test ###
ksort( $recurridList, SORT_STRING );
// echo 'recurridList='.implode(', ', array_keys( $recurridList ))."<br>\n"; // test ###
$fcnStart = clone $compStart;
$fcnStart->setDate((int) $startY, (int) $startM, (int) $startD );
$fcnStart->setTime( 0, 0, 0 );
$fcnEnd = clone $compEnd;
$fcnEnd->setDate((int) $endY, (int) $endM, (int) $endD );
$fcnEnd->setTime( 23, 59, 59 );
// echo 'compStart='.$compStart->format().', compEnd'.$compEnd->format(); if($compDuration)echo ', interval='.$compDuration->format( iCalUtilityFunctions::$fmt['durDHis'] ); echo "<br>\n"; $tcnt = 0;// test ###
/* *************************************************************
make a list of optional exclude dates for component occurence from exrule and exdate
*********************************************************** */
$workStart = clone $compStart;
$workStart->sub( $compDuration ? $compDuration : $intervalP1D );
$workEnd = clone $fcnEnd;
$workEnd->add( $compDuration ? $compDuration : $intervalP1D );
while( FALSE !== ( $prop = $component->getProperty( 'EXRULE' ))) {
$exdatelist2 = array();
if( isset( $prop['UNTIL']['hour'] )) { // convert until date to dtstart timezone
$until = iCaldateTime::factory( $prop['UNTIL'], array( 'TZID' => 'UTC' ), null, $dtstartTz );
$until = $until->format();
iCalUtilityFunctions::_strDate2arr( $until );
$prop['UNTIL'] = $until;
}
iCalUtilityFunctions::_recur2date( $exdatelist2, $prop, $compStart, $workStart, $workEnd );
foreach( $exdatelist2 as $k => $v )
$exdatelist[$k.$compStartHis] = $v; // point out exact every excluded ocurrence (incl. opt. His)
unset( $until, $exdatelist2 );
}
while( FALSE !== ( $prop = $component->getProperty( 'EXDATE', FALSE, TRUE ))) { // - start check exdate
foreach( $prop['value'] as $exdate ) {
$exdate = iCaldateTime::factory( $exdate, $prop['params'], $exdate, $dtstartTz );
$exdatelist[$exdate->key] = TRUE;
} // end - foreach( $exdate as $exdate )
} // end - check exdate
unset( $prop, $exdate );
// echo 'exdatelist=' .implode(', ', array_keys( $exdatelist )) ."<br>\n"; // test ###
/* *************************************************************
select only components within.. .
*********************************************************** */
$xRecurrence = 1;
if(( ! $any && iCalUtilityFunctions::_inScope( $compStart, $fcnStart, $compStart, $fcnEnd, $compStart->dateFormat )) ||
( $any && iCalUtilityFunctions::_inScope( $fcnEnd, $compStart, $fcnStart, $compEnd, $compStart->dateFormat ))) {
/* add the selected component (WITHIN valid dates) to output array */
if( $flat ) { // any=true/false, ignores split
if( !$recurrid )
$result[$compUID] = $component->copy(); // copy original to output (but not anyone with recurrence-id)
}
elseif( $split ) { // split the original component
// echo 'split org.:'.$compStart->format().' < '.$fcnStart->format( 'Ymd His e' )."<br>\n"; // test ###
if( $compStart->format( iCalUtilityFunctions::$fmt['YmdHis2'] ) < $fcnStart->format( iCalUtilityFunctions::$fmt['YmdHis2'] ))
$rstart = clone $fcnStart;
else
$rstart = clone $compStart;
if( $compEnd->format( iCalUtilityFunctions::$fmt['YmdHis2'] ) > $fcnEnd->format( iCalUtilityFunctions::$fmt['YmdHis2'] ))
$rend = clone $fcnEnd;
else
$rend = clone $compEnd;
// echo "going to test comp no:$cix, rstart=".$rstart->format( iCalUtilityFunctions::$fmt['YmdHis2e'] )." (key={$rstart->key}), end=".$rend->format( iCalUtilityFunctions::$fmt['YmdHis2e'] )."<br>\n"; // test ###
if( ! isset( $exdatelist[$rstart->key] )) { // not excluded in exrule/exdate
if( isset( $recurridList[$rstart->key] )) { // change start day to new YmdHis/duration
$k = $rstart->key;
// echo "recurridList HIT, key={$k}, recur Date=".$recurridList[$k][0]->key."<br>\n"; // test ###
$rstart = clone $recurridList[$k][0];
$startHis = $rstart->getTime();
$rend = clone $rstart;
if( FALSE !== $recurridList[$k][2] )
$rend->add( $recurridList[$k][2] );
elseif( FALSE !== $compDuration )
$rend->add( $compDuration );
$endHis = $rend->getTime();
unset( $recurridList[$k] );
}
else {
$startHis = $compStart->getTime();
$endHis = $compEnd->getTime();
}
// echo "_____testing comp no:$cix, rstart=".$rstart->format( iCalUtilityFunctions::$fmt['YmdHis2e'] )." (key={$rstart->key}), end=".$rend->format( iCalUtilityFunctions::$fmt['YmdHis2e'] )."<br>\n"; // test ###
$cnt = 0; // exclude any recurrence START date, found in exdatelist or recurridList but accept the reccurence-id comp itself
$occurenceDays = 1 + (int) $rstart->diff( $rend )->format( '%a' ); // count the days (incl start day)
while( $rstart->format( iCalUtilityFunctions::$fmt['Ymd2'] ) <= $rend->format( iCalUtilityFunctions::$fmt['Ymd2'] )) {
$cnt += 1;
if( 1 < $occurenceDays )
$component->setProperty( 'X-OCCURENCE', sprintf( iCalUtilityFunctions::$fmt['dayOfDays'], $cnt, $occurenceDays ));
if( 1 < $cnt )
$rstart->setTime( 0, 0, 0 );
else {
$rstart->setTime( $startHis[0], $startHis[1], $startHis[2] );
$exdatelist[$rstart->key] = $compDuration; // make sure to exclude start day from the recurrence pattern
}
$component->setProperty( 'X-CURRENT-DTSTART', $rstart->format( $compStart->dateFormat ));
if( FALSE !== $compDuration ) {
$propName = ( isset( $compEnd->SCbools[ 'dueExist'] )) ? 'X-CURRENT-DUE' : 'X-CURRENT-DTEND';
if( $cnt < $occurenceDays )
$rstart->setTime( 23, 59, 59 );
else
$rstart->setTime( $endHis[0], $endHis[1], $endHis[2] );
$component->setProperty( $propName, $rstart->format( $compEnd->dateFormat ));
}
$result[(int)$rstart->format( 'Y' )][(int)$rstart->format( 'm' )][(int)$rstart->format( 'd' )][$compUID] = $component->copy(); // copy to output
$rstart->add( $intervalP1D );
} // end while(( $rstart->format( 'Ymd' ) < $rend->format( 'Ymd' ))
unset( $cnt, $occurenceDays );
} // end if( ! isset( $exdatelist[$rstart->key] ))
// else echo "skip no:$cix with date=".$compStart->format()."<br>\n"; // test ###
unset( $rstart, $rend );
} // end elseif( $split ) - else use component date
else { // !$flat && !$split, i.e. no flat array and DTSTART within period
$tstart = ( isset( $recurridList[$compStart->key] )) ? clone $recurridList[$k][0] : clone $compStart;
// echo "going to test comp no:$cix with checkDate={$compStart->key} with recurridList=".implode(',',array_keys($recurridList)); // test ###
if( ! $any || ! isset( $exdatelist[$tstart->key] )) { // exclude any recurrence date, found in exdatelist
// echo " and copied to output<br>\n"; // test ###
$result[(int)$tstart->format( 'Y' )][(int)$tstart->format( 'm' )][(int)$tstart->format( 'd' )][$compUID] = $component->copy(); // copy to output
}
unset( $tstart );
}
} // end (dt)start within the period OR occurs within the period
/* *************************************************************
if 'any' components, check components with reccurrence rules, removing all excluding dates
*********************************************************** */
if( TRUE === $any ) {
$recurlist = array();
/* make a list of optional repeating dates for component occurence, rrule, rdate */
while( FALSE !== ( $prop = $component->getProperty( 'RRULE' ))) { // get all rrule dates (multiple values allowed)
$recurlist2 = array();
if( isset( $prop['UNTIL']['hour'] )) { // convert $rrule['UNTIL'] to the same timezone as DTSTART !!
$until = iCaldateTime::factory( $prop['UNTIL'], array( 'TZID' => 'UTC' ), null, $dtstartTz );
$until = $until->format();
iCalUtilityFunctions::_strDate2arr( $until );
$prop['UNTIL'] = $until;
}
iCalUtilityFunctions::_recur2date( $recurlist2, $prop, $compStart, $workStart, $workEnd );
foreach( $recurlist2 as $recurkey => $recurvalue ) { // recurkey=Ymd
$recurkey .= $compStartHis; // add opt His
if( ! isset( $exdatelist[$recurkey] ))
$recurlist[$recurkey] = $compDuration; // DateInterval or FALSE
}
unset( $prop, $until, $recurlist2 );
}
$workStart = clone $fcnStart;
$workStart->sub( $compDuration ? $compDuration : $intervalP1D );
$format = $compStart->dateFormat;
while( FALSE !== ( $prop = $component->getProperty( 'RDATE', FALSE, TRUE ))) {
$rdateFmt = ( isset( $prop['params']['VALUE'] )) ? $prop['params']['VALUE'] : 'DATETIME';
$params = $prop['params'];
$prop = $prop['value'];
foreach( $prop as $theRdate ) {
if( 'PERIOD' == $rdateFmt ) { // all days within PERIOD
$rdate = iCaldateTime::factory( $theRdate[0], $params, $theRdate[0], $dtstartTz );
if( ! iCalUtilityFunctions::_inScope( $rdate, $workStart, $rdate, $fcnEnd, $format ) || isset( $exdatelist[$rdate->key] ))
continue;
if( isset( $theRdate[1]['year'] )) // date-date period end
$recurlist[$rdate->key] = $rdate->diff( iCaldateTime::factory( $theRdate[1], $params, $theRdate[1], $dtstartTz ));
else // period duration
$recurlist[$rdate->key] = new DateInterval( iCalUtilityFunctions::_duration2str( $theRdate[1] ));
} // end if( 'PERIOD' == $rdateFmt )
elseif( 'DATE' == $rdateFmt ) { // single recurrence, date
$rdate = iCaldateTime::factory( $theRdate, array_merge( $params, array( 'TZID' => $dtstartTz )), null, $dtstartTz );
if( iCalUtilityFunctions::_inScope( $rdate, $workStart, $rdate, $fcnEnd, $format ) && ! isset( $exdatelist[$rdate->key] ))
$recurlist[$rdate->key.$compStartHis] = $compDuration; // set start date for recurrence + DateInterval/FALSE (+opt His)
} // end DATE
else { // start DATETIME
$rdate = iCaldateTime::factory( $theRdate, $params, $theRdate, $dtstartTz );
if( iCalUtilityFunctions::_inScope( $rdate, $workStart, $rdate, $fcnEnd, $format ) && ! isset( $exdatelist[$rdate->key] ))
$recurlist[$rdate->key] = $compDuration; // set start datetime for recurrence DateInterval/FALSE
} // end DATETIME
} // end foreach( $prop as $theRdate )
} // end while( FALSE !== ( $prop = $component->getProperty( 'rdate', FALSE, TRUE )))
unset( $prop, $workStart, $format, $theRdate, $rdate, $rend );
foreach( $recurridList as $rKey => $rVal ) { // check for recurrence-id, i.e. alter recur Ymd[His] and duration
if( isset( $recurlist[$rKey] )) {
unset( $recurlist[$rKey] );
$recurlist[$rVal[0]->key] = ( FALSE !== $rVal[2] ) ? $rVal[2] : $compDuration;
// echo "alter recurfrom {$rKey} to {$rVal[0]->key} ";if(FALSE!==$dur)echo " ({$dur->format( '%a days, %h-%i-%s' )})";echo "<br>\n"; // test ###
}
}
ksort( $recurlist, SORT_STRING );
// echo 'recurlist=' .implode(', ', array_keys( $recurlist )) ."<br>\n"; // test ###
// echo 'recurridList=' .implode(', ', array_keys( $recurridList )) ."<br>\n"; // test ###
/* *************************************************************
output all remaining components in recurlist
*********************************************************** */
if( 0 < count( $recurlist )) {
$component2 = $component->copy();
$compUID = $component2->getProperty( 'UID' );
$workStart = clone $fcnStart;
$workStart->sub( $compDuration ? $compDuration : $intervalP1D );
$YmdOld = null;
foreach( $recurlist as $recurkey => $durvalue ) {
if( $YmdOld == substr( $recurkey, 0, 8 )) // skip overlapping recur the same day, i.e. RDATE before RRULE
continue;
$YmdOld = substr( $recurkey, 0, 8 );
$rstart = clone $compStart;
$rstart->setDate((int) substr( $recurkey, 0, 4 ), (int) substr( $recurkey, 4, 2 ), (int) substr( $recurkey, 6, 2 ));
$rstart->setTime((int) substr( $recurkey, 8, 2 ), (int) substr( $recurkey, 10, 2 ), (int) substr( $recurkey, 12, 2 ));
// echo "recur start=".$rstart->format( iCalUtilityFunctions::$fmt['YmdHis2e'] )."<br>\n"; // test ###;
/* add recurring components within valid dates to output array, only start date set */
if( $flat ) {
if( !isset( $result[$compUID] )) // only one comp
$result[$compUID] = $component2->copy(); // copy to output
}
/* add recurring components within valid dates to output array, split for each day */
elseif( $split ) {
$rend = clone $rstart;
if( FALSE !== $durvalue )
$rend->add( $durvalue );
if( $rend->format( iCalUtilityFunctions::$fmt['Ymd2'] ) > $fcnEnd->format( iCalUtilityFunctions::$fmt['Ymd2'] ))
$rend = clone $fcnEnd;
// echo "recur 1={$recurkey}, start=".$rstart->format( iCalUtilityFunctions::$fmt['YmdHis2e'] ).", end=".$rend->format( iCalUtilityFunctions::$fmt['YmdHis2e'] );if($durvalue) echo ", duration=".$durvalue->format( iCalUtilityFunctions::$fmt['durDHis'] );echo "<br>\n"; // test ###
$xRecurrence += 1;
$cnt = 0;
$occurenceDays = 1 + (int) $rstart->diff( $rend )->format( '%a' ); // count the days (incl start day)
while( $rstart->format( iCalUtilityFunctions::$fmt['Ymd2'] ) <= $rend->format( iCalUtilityFunctions::$fmt['Ymd2'] )) { // iterate.. .
$cnt += 1;
if( $rstart->format( iCalUtilityFunctions::$fmt['Ymd2'] ) < $fcnStart->format( iCalUtilityFunctions::$fmt['Ymd2'] )) { // date before dtstart
// echo "recur 3, start=".$rstart->format( 'Y-m-d H:i:s' )." &gt;= fcnStart=".$fcnStart->format( 'Y-m-d H:i:s' )."<br>\n"; // test ###
$rstart->add( $intervalP1D );
$rstart->setTime( 0, 0, 0 );
continue;
}
elseif( 2 == $cnt )
$rstart->setTime( 0, 0, 0 );
$component2->setProperty( 'X-RECURRENCE', $xRecurrence );
if( 1 < $occurenceDays )
$component2->setProperty( 'X-OCCURENCE', sprintf( iCalUtilityFunctions::$fmt['dayOfDays'], $cnt, $occurenceDays ));
else
$component2->deleteProperty( 'X-OCCURENCE' );
$component2->setProperty( 'X-CURRENT-DTSTART', $rstart->format( $compStart->dateFormat ));
$propName = ( isset( $compEnd->SCbools[ 'dueExist'] )) ? 'X-CURRENT-DUE' : 'X-CURRENT-DTEND';
if( FALSE !== $durvalue ) {
if( $cnt < $occurenceDays )
$rstart->setTime( 23, 59, 59 );
else {
$His = $rend->getTime(); // set end time
$rstart->setTime( $His[0], $His[1], $His[2] );
}
$component2->setProperty( $propName, $rstart->format( $compEnd->dateFormat ));
// echo "checking date, (day {$cnt} of {$occurenceDays}), _end_=".$rstart->format( 'Y-m-d H:i:s e' )."<br>"; // test ###;
}
else
$component2->deleteProperty( $propName );
$result[(int)$rstart->format( 'Y' )][(int)$rstart->format( 'm' )][(int)$rstart->format( 'd' )][$compUID] = $component2->copy(); // copy to output
$rstart->add( $intervalP1D );
} // end while( $rstart->format( 'Ymd' ) <= $rend->format( 'Ymd' ))
unset( $rstart, $rend );
} // end elseif( $split )
elseif( $rstart->format( iCalUtilityFunctions::$fmt['Ymd2'] ) >= $fcnStart->format( iCalUtilityFunctions::$fmt['Ymd2'] )) {
$xRecurrence += 1; // date within period //* flat=FALSE && split=FALSE => one comp every recur startdate *//
$component2->setProperty( 'X-RECURRENCE', $xRecurrence );
$component2->setProperty( 'X-CURRENT-DTSTART', $rstart->format( $compStart->dateFormat ));
$propName = ( isset( $compEnd->SCbools[ 'dueExist'] )) ? 'X-CURRENT-DUE' : 'X-CURRENT-DTEND';
if( FALSE !== $durvalue ) {
$rstart->add( $durvalue );
$component2->setProperty( $propName, $rstart->format( $compEnd->dateFormat ));
}
else
$component2->deleteProperty( $propName );
$result[(int)$rstart->format( 'Y' )][(int)$rstart->format( 'm' )][(int)$rstart->format( 'd' )][$compUID] = $component2->copy(); // copy to output
} // end elseif( $rstart >= $fcnStart )
unset( $rstart );
} // end foreach( $recurlist as $recurkey => $durvalue )
unset( $component2, $xRecurrence, $compUID, $workStart, $rstart );
} // end if( 0 < count( $recurlist ))
} // end if( TRUE === $any )
unset( $component );
} // end foreach ( $this->components as $cix => $component )
unset( $recurrid, $recurridList, $fcnStart, $fcnEnd, $compStart, $compEnd, $exdatelist, $recurlist ); // clean up
if( 0 >= count( $result ))
return FALSE;
elseif( !$flat ) {
foreach( $result as $y => $yeararr ) {
foreach( $yeararr as $m => $montharr ) {
foreach( $montharr as $d => $dayarr ) {
if( empty( $result[$y][$m][$d] ))
unset( $result[$y][$m][$d] );
else {
$result[$y][$m][$d] = array_values( $dayarr ); // skip tricky UID-index
if( 1 < count( $result[$y][$m][$d] )) {
foreach( $result[$y][$m][$d] as & $c ) // sort
iCalUtilityFunctions::_setSortArgs( $c );
usort( $result[$y][$m][$d], array( 'iCalUtilityFunctions', '_cmpfcn' ));
}
}
} // end foreach( $montharr as $d => $dayarr )
if( empty( $result[$y][$m] ))
unset( $result[$y][$m] );
else
ksort( $result[$y][$m] );
} // end foreach( $yeararr as $m => $montharr )
if( empty( $result[$y] ))
unset( $result[$y] );
else
ksort( $result[$y] );
}// end foreach( $result as $y => $yeararr )
if( empty( $result ))
unset( $result );
else
ksort( $result );
} // end elseif( !$flat )
if( 0 >= count( $result ))
return FALSE;
return $result;
}
/**
* select components from calendar on based on specific property value(-s)
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.11 - 2015-03-21
* @param array $selectOptions (string) key => (mixed) value, (key=propertyName)
* @uses vcalendar::$components
* @uses calendarComponent::$objName
* @uses iCalUtilityFunctions::$vComps
* @uses calendarComponent::getProperty()
* @uses iCalUtilityFunctions::$otherProps
* @uses calendarComponent::copy()
* @uses iCalUtilityFunctions::$mProps1
* @uses calendarComponent::_getProperties()
* @return array
*/
function selectComponents2( $selectOptions ) {
// $output = array();
$selectOptions = array_change_key_case( $selectOptions, CASE_UPPER );
foreach( $this->components as $cix => $component3 ) {
if( !in_array( $component3->objName, iCalUtilityFunctions::$vComps ))
continue;
$uid = $component3->getProperty( 'UID' );
foreach( $selectOptions as $propName => $pvalue ) {
if( !in_array( $propName, iCalUtilityFunctions::$otherProps ))
continue;
if( !is_array( $pvalue ))
$pvalue = array( $pvalue );
if(( 'UID' == $propName ) && in_array( $uid, $pvalue )) {
$output[$uid][] = $component3->copy();
continue;
}
elseif( in_array( $propName, iCalUtilityFunctions::$mProps1 )) {
$propValues = array();
$component3->_getProperties( $propName, $propValues );
$propValues = array_keys( $propValues );
foreach( $pvalue as $theValue ) {
if( in_array( $theValue, $propValues )) { // && !isset( $output[$uid] )) {
$output[$uid][] = $component3->copy();
break;
}
}
continue;
} // end elseif( // multiple occurrence?
elseif( FALSE === ( $d = $component3->getProperty( $propName ))) // single occurrence
continue;
if( is_array( $d )) {
foreach( $d as $part ) {
if( in_array( $part, $pvalue ) && !isset( $output[$uid] ))
$output[$uid][] = $component3->copy();
}
}
elseif(( 'SUMMARY' == $propName ) && !isset( $output[$uid] )) {
foreach( $pvalue as $pval ) {
if( FALSE !== stripos( $d, $pval )) {
$output[$uid][] = $component3->copy();
break;
}
}
}
elseif( in_array( $d, $pvalue ) && !isset( $output[$uid] ))
$output[$uid][] = $component3->copy();
} // end foreach( $selectOptions as $propName => $pvalue ) {
} // end foreach( $this->components as $cix => $component3 ) {
if( !empty( $output )) {
ksort( $output ); // uid order
$output2 = array();
foreach( $output as $uid => $components ) {
foreach( $components as $component )
$output2[] = $component;
}
$output = $output2;
}
return $output;
}
/**
* replace calendar component in vcalendar
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.11 - 2015-03-21
* @param object $component calendar component
* @uses calendarComponent::$objName
* @uses iCalUtilityFunctions::$vComps
* @uses vcalendar::setComponent()
* @uses calendarComponent::getProperty()
* @return bool
*/
function replaceComponent( $component ) {
if( in_array( $component->objName, iCalUtilityFunctions::$vComps ))
return $this->setComponent( $component, $component->getProperty( 'UID' ));
if(( 'vtimezone' != $component->objName ) || ( FALSE === ( $tzid = $component->getProperty( 'TZID' ))))
return FALSE;
foreach( $this->components as $cix => $comp ) {
if( 'vtimezone' != $component->objName )
continue;
if( $tzid == $comp->getComponent( 'TZID' )) {
unset( $component->propix, $component->compix );
$this->components[$cix] = $component;
return TRUE;
}
}
return FALSE;
}
/**
* add calendar component to calendar
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.11 - 2015-03-21
* @param object $component calendar component
* @param mixed $arg1 optional, ordno/component type/ component uid
* @param mixed $arg2 optional, ordno if arg1 = component type
* @uses calendarComponent::setConfig()
* @uses vcalendar::getConfig()
* @uses calendarComponent::$objName
* @uses calendarComponent::getProperty()
* @uses vcalendar::$components
* @uses iCalUtilityFunctions::$mComps
* @uses calendarComponent::copy()
* @return bool
*/
function setComponent( $component, $arg1=FALSE, $arg2=FALSE ) {
$component->setConfig( $this->getConfig(), FALSE, TRUE );
if( !in_array( $component->objName, array( 'valarm', 'vtimezone' ))) {
/* make sure dtstamp and uid is set */
$dummy1 = $component->getProperty( 'dtstamp' );
$dummy2 = $component->getProperty( 'uid' );
}
unset( $component->propix, $component->compix );
if( !$arg1 ) { // plain insert, last in chain
$this->components[] = $component->copy();
return TRUE;
}
$argType = $index = null;
if ( ctype_digit( (string) $arg1 )) { // index insert/replace
$argType = 'INDEX';
$index = (int) $arg1 - 1;
}
elseif( in_array( strtolower( $arg1 ), iCalUtilityFunctions::$mComps )) {
$argType = strtolower( $arg1 );
$index = ( ctype_digit( (string) $arg2 )) ? ((int) $arg2) - 1 : 0;
}
// else if arg1 is set, arg1 must be an UID
$cix1sC = 0;
foreach ( $this->components as $cix => $component2) {
if( empty( $component2 )) continue;
if(( 'INDEX' == $argType ) && ( $index == $cix )) { // index insert/replace
$this->components[$cix] = $component->copy();
return TRUE;
}
elseif( $argType == $component2->objName ) { // component Type index insert/replace
if( $index == $cix1sC ) {
$this->components[$cix] = $component->copy();
return TRUE;
}
$cix1sC++;
}
elseif( !$argType && ( $arg1 == $component2->getProperty( 'uid' ))) { // UID insert/replace
$this->components[$cix] = $component->copy();
return TRUE;
}
}
/* arg1=index and not found.. . insert at index .. .*/
if( 'INDEX' == $argType ) {
$this->components[$index] = $component->copy();
ksort( $this->components, SORT_NUMERIC );
}
else /* not found.. . insert last in chain anyway .. .*/
$this->components[] = $component->copy();
return TRUE;
}
/**
* sort iCal compoments
*
* ascending sort on properties (if exist) x-current-dtstart, dtstart,
* x-current-dtend, dtend, x-current-due, due, duration, created, dtstamp, uid if called without arguments,
* otherwise sorting on specific (argument) property values
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.11 - 2015-03-21
* @param string $sortArg
* @uses vcalendar::$components
* @uses iCalUtilityFunctions::$otherProps
* @uses iCalUtilityFunctions::_setSortArgs()
* @uses iCalUtilityFunctions::_cmpfcn()
* @return void
*/
function sort( $sortArg=FALSE ) {
if( ! is_array( $this->components ) || ( 2 > count( $this->components )))
return;
if( $sortArg ) {
$sortArg = strtoupper( $sortArg );
if( ! in_array( $sortArg, iCalUtilityFunctions::$otherProps ) && ( 'DTSTAMP' != $sortArg ))
$sortArg = FALSE;
}
foreach( $this->components as & $c )
iCalUtilityFunctions::_setSortArgs( $c, $sortArg );
usort( $this->components, array( 'iCalUtilityFunctions', '_cmpfcn' ));
}
/**
* parse iCal text/file into vcalendar, components, properties and parameters
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.12 - 2014-03-10
* @param mixed $unparsedtext strict rfc2445 formatted, single property string or array of property strings
* @param resource $context PHP resource context
* @uses iCalUtilityFunctions::convEolChar()
* @uses vcalendar::getConfig()
* @uses vcalendar::$components
* @uses calendarComponent::copy()
* @uses vevent::construct()
* @uses vfreebusy::construct()
* @uses vjournal::construct()
* @uses vtodo::construct()
* @uses vtimezone::construct()
* @uses vcalendar::$unparsed
* @uses iCalUtilityFunctions::_splitContent()
* @uses iCalUtilityFunctions::_strunrep()
* @uses vcalendar::setProperty()
* @uses calendarComponent::$unparsed
* @uses calendarComponent::parse()
* @return bool FALSE if error occurs during parsing
*/
function parse( $unparsedtext=FALSE, $context=null ) {
$nl = $this->getConfig( 'nl' );
if(( FALSE === $unparsedtext ) || empty( $unparsedtext )) {
/* directory+filename is set previously via setConfig url or directory+filename */
if( FALSE === ( $file = $this->getConfig( 'url' ))) {
if( FALSE === ( $file = $this->getConfig( 'dirfile' )))
return FALSE; /* err 1 */
if( ! is_file( $file ))
return FALSE; /* err 2 */
if( ! is_readable( $file ))
return FALSE; /* err 3 */
}
/* READ FILE */
if( ! empty( $context ) && filter_var( $file, FILTER_VALIDATE_URL ) &&
( FALSE === ( $rows = file_get_contents( $file, FALSE, $context ))))
return FALSE; /* err 6 */
elseif( FALSE === ( $rows = file_get_contents( $file )))
return FALSE; /* err 5 */
} // end if(( FALSE === $unparsedtext ) || empty( $unparsedtext ))
elseif( is_array( $unparsedtext ))
$rows = implode( '\n'.$nl, $unparsedtext );
else
$rows = & $unparsedtext;
/* fix line folding */
$rows = iCalUtilityFunctions::convEolChar( $rows, $nl );
/* skip leading (empty/invalid) lines (and remove leading BOM chars etc) */
foreach( $rows as $lix => $line ) {
if( FALSE !== stripos( $line, 'BEGIN:VCALENDAR' )) {
$rows[$lix] = 'BEGIN:VCALENDAR';
break;
}
unset( $rows[$lix] );
}
$rcnt = count( $rows );
if( 3 > $rcnt ) /* err 10 */
return FALSE;
/* skip trailing empty lines and ensure an end row */
$lix = array_keys( $rows );
$lix = end( $lix );
while( 3 < $lix ) {
$tst = trim( $rows[$lix] );
if(( '\n' == $tst ) || empty( $tst )) {
unset( $rows[$lix] );
$lix--;
continue;
}
if( FALSE === stripos( $rows[$lix], 'END:VCALENDAR' ))
$rows[] = 'END:VCALENDAR';
else
$rows[$lix] = 'END:VCALENDAR';
break;
}
$comp = & $this;
$calsync = $compsync = 0;
/* identify components and update unparsed data within component */
$config = $this->getConfig();
$endtxt = array( 'END:VE', 'END:VF', 'END:VJ', 'END:VT' );
foreach( $rows as $lix => $line ) {
if( 'BEGIN:VCALENDAR' == strtoupper( substr( $line, 0, 15 ))) {
$calsync++;
continue;
}
elseif( 'END:VCALENDAR' == strtoupper( substr( $line, 0, 13 ))) {
if( 0 < $compsync )
$this->components[] = $comp->copy();
$compsync--;
$calsync--;
break;
}
elseif( 1 != $calsync )
return FALSE; /* err 20 */
elseif( in_array( strtoupper( substr( $line, 0, 6 )), $endtxt )) {
$this->components[] = $comp->copy();
$compsync--;
continue;
}
if( 'BEGIN:VEVENT' == strtoupper( substr( $line, 0, 12 ))) {
$comp = new vevent( $config );
$compsync++;
}
elseif( 'BEGIN:VFREEBUSY' == strtoupper( substr( $line, 0, 15 ))) {
$comp = new vfreebusy( $config );
$compsync++;
}
elseif( 'BEGIN:VJOURNAL' == strtoupper( substr( $line, 0, 14 ))) {
$comp = new vjournal( $config );
$compsync++;
}
elseif( 'BEGIN:VTODO' == strtoupper( substr( $line, 0, 11 ))) {
$comp = new vtodo( $config );
$compsync++;
}
elseif( 'BEGIN:VTIMEZONE' == strtoupper( substr( $line, 0, 15 ))) {
$comp = new vtimezone( $config );
$compsync++;
}
else { /* update component with unparsed data */
$comp->unparsed[] = $line;
}
} // end foreach( $rows as $line )
unset( $config, $endtxt );
/* parse data for calendar (this) object */
if( isset( $this->unparsed ) && is_array( $this->unparsed ) && ( 0 < count( $this->unparsed ))) {
/* concatenate property values spread over several lines */
$propnames = array( 'calscale','method','prodid','version','x-' );
$proprows = array();
for( $i = 0; $i < count( $this->unparsed ); $i++ ) { // concatenate lines
$line = rtrim( $this->unparsed[$i], $nl );
while( isset( $this->unparsed[$i+1] ) && !empty( $this->unparsed[$i+1] ) && ( ' ' == $this->unparsed[$i+1]{0} ))
$line .= rtrim( substr( $this->unparsed[++$i], 1 ), $nl );
$proprows[] = $line;
}
foreach( $proprows as $line ) {
if( '\n' == substr( $line, -2 ))
$line = substr( $line, 0, -2 );
/* get property name */
$propname = '';
$cix = 0;
while( FALSE !== ( $char = substr( $line, $cix, 1 ))) {
if( in_array( $char, array( ':', ';' )))
break;
else
$propname .= $char;
$cix++;
}
/* skip non standard property names */
if(( 'x-' != strtolower( substr( $propname, 0, 2 ))) && !in_array( strtolower( $propname ), $propnames ))
continue;
/* ignore version/prodid properties */
if( in_array( strtolower( $propname ), array( 'version', 'prodid' )))
continue;
/* rest of the line is opt.params and value */
$line = substr( $line, $cix);
/* separate attributes from value */
iCalUtilityFunctions::_splitContent( $line, $propAttr );
/* update Property */
if( FALSE !== strpos( $line, ',' )) {
$content = array( 0 => '' );
$cix = $lix = 0;
while( FALSE !== substr( $line, $lix, 1 )) {
if(( 0 < $lix ) && ( ',' == $line[$lix] ) && ( "\\" != $line[( $lix - 1 )])) {
$cix++;
$content[$cix] = '';
}
else
$content[$cix] .= $line[$lix];
$lix++;
}
if( 1 < count( $content )) {
foreach( $content as $cix => $contentPart )
$content[$cix] = iCalUtilityFunctions::_strunrep( $contentPart );
$this->setProperty( $propname, $content, $propAttr );
continue;
}
else
$line = reset( $content );
$line = iCalUtilityFunctions::_strunrep( $line );
}
$this->setProperty( $propname, rtrim( $line, "\x00..\x1F" ), $propAttr );
} // end - foreach( $this->unparsed.. .
} // end - if( is_array( $this->unparsed.. .
unset( $unparsedtext, $rows, $this->unparsed, $proprows );
/* parse Components */
if( is_array( $this->components ) && ( 0 < count( $this->components ))) {
$ckeys = array_keys( $this->components );
foreach( $ckeys as $ckey ) {
if( !empty( $this->components[$ckey] ) && !empty( $this->components[$ckey]->unparsed )) {
$this->components[$ckey]->parse();
}
}
}
else
return FALSE; /* err 91 or something.. . */
return TRUE;
}
/*********************************************************************************/
/**
* creates formatted output for calendar object instance
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.07 - 2015-03-31
* @uses vcalendar::$format
* @uses vcalendar::$nl
* @uses vcalendar::createVersion()
* @uses vcalendar::createProdid()
* @uses vcalendar::createCalscale()
* @uses vcalendar::createMethod()
* @uses vcalendar::createXprop()
* @uses vcalendar::$components
* @uses calendarComponent::setConfig()
* @uses vcalendar::getConfig()
* @uses calendarComponent::createComponent()
* @uses vcalendar::$xcaldecl
* @return string
*/
function createCalendar() {
parent::_createFormat();
$calendarInit = $calendarxCaldecl = $calendarStart = $calendar = '';
switch( $this->format ) {
case 'xcal':
$calendarInit = '<?xml version="1.0" encoding="UTF-8"?>'.$this->nl.
'<!DOCTYPE vcalendar PUBLIC "-//IETF//DTD XCAL/iCalendar XML//EN"'.$this->nl.
'"http://www.ietf.org/internet-drafts/draft-ietf-calsch-many-xcal-01.txt"';
$calendarStart = '>'.$this->nl.'<vcalendar';
break;
default:
$calendarStart = 'BEGIN:VCALENDAR'.$this->nl;
break;
}
$calendarStart .= $this->createVersion();
$calendarStart .= $this->createProdid();
$calendarStart .= $this->createCalscale();
$calendarStart .= $this->createMethod();
if( 'xcal' == $this->format )
$calendarStart .= '>'.$this->nl;
$calendar .= $this->createXprop();
foreach( $this->components as $component ) {
if( empty( $component )) continue;
$component->setConfig( $this->getConfig(), FALSE, TRUE );
$calendar .= $component->createComponent( $this->xcaldecl );
}
if(( 'xcal' == $this->format ) && ( 0 < count( $this->xcaldecl ))) { // xCal only
$calendarInit .= ' [';
$old_xcaldecl = array();
foreach( $this->xcaldecl as $declix => $declPart ) {
if(( 0 < count( $old_xcaldecl)) &&
isset( $declPart['uri'] ) && isset( $declPart['external'] ) &&
isset( $old_xcaldecl['uri'] ) && isset( $old_xcaldecl['external'] ) &&
( in_array( $declPart['uri'], $old_xcaldecl['uri'] )) &&
( in_array( $declPart['external'], $old_xcaldecl['external'] )))
continue; // no duplicate uri and ext. references
if(( 0 < count( $old_xcaldecl)) &&
!isset( $declPart['uri'] ) && !isset( $declPart['uri'] ) &&
isset( $declPart['ref'] ) && isset( $old_xcaldecl['ref'] ) &&
( in_array( $declPart['ref'], $old_xcaldecl['ref'] )))
continue; // no duplicate element declarations
$calendarxCaldecl .= $this->nl.'<!';
foreach( $declPart as $declKey => $declValue ) {
switch( $declKey ) { // index
case 'xmldecl': // no 1
$calendarxCaldecl .= $declValue.' ';
break;
case 'uri': // no 2
$calendarxCaldecl .= $declValue.' ';
$old_xcaldecl['uri'][] = $declValue;
break;
case 'ref': // no 3
$calendarxCaldecl .= $declValue.' ';
$old_xcaldecl['ref'][] = $declValue;
break;
case 'external': // no 4
$calendarxCaldecl .= '"'.$declValue.'" ';
$old_xcaldecl['external'][] = $declValue;
break;
case 'type': // no 5
$calendarxCaldecl .= $declValue.' ';
break;
case 'type2': // no 6
$calendarxCaldecl .= $declValue;
break;
}
}
$calendarxCaldecl .= '>';
}
$calendarxCaldecl .= $this->nl.']';
} // end if(( 'xcal'...
switch( $this->format ) {
case 'xcal':
$calendar .= '</vcalendar>'.$this->nl;
break;
default:
$calendar .= 'END:VCALENDAR'.$this->nl;
break;
}
return $calendarInit.$calendarxCaldecl.$calendarStart.$calendar;
}
/**
* a HTTP redirect header is sent with created, updated and/or parsed calendar
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.5 - 2015-03-29
* @param bool $utf8Encode
* @param bool $gzip
* @param bool $cdType TRUE : Content-Disposition: attachment... (default), FALSE : ...inline...
* @uses vcalendar::getConfig()
* @uses vcalendar::createCalendar()
* @uses vcalendar::$headers
* @uses vcalendar::$format
* @return bool TRUE on success, FALSE on error
*/
function returnCalendar( $utf8Encode=FALSE, $gzip=FALSE, $cdType=TRUE ) {
$filename = $this->getConfig( 'filename' );
$output = $this->createCalendar();
if( $utf8Encode )
$output = utf8_encode( $output );
$fsize = null;
if( $gzip ) {
$output = gzencode( $output, 9 );
$fsize = strlen( $output );
header( self::$headers[0] );
header( self::$headers[1] );
}
else {
if( FALSE !== ( $temp = tempnam( sys_get_temp_dir(), 'iCr' ))) {
if( FALSE !== file_put_contents( $temp, $output ))
$fsize = @filesize( $temp );
unlink( $temp );
}
}
if( ! empty( $fsize ))
header( sprintf( self::$headers[2], $fsize ));
if( 'xcal' == $this->format )
header( self::$headers[3] );
else
header( self::$headers[4] );
$cdType = ( $cdType ) ? 5 : 6;
header( sprintf( self::$headers[$cdType], $filename ));
header( self::$headers[7] );
echo $output;
return TRUE;
}
/**
* save content in a file
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.5 - 2015-03-29
* @uses vcalendar::createComponent()
* @uses vcalendar::getConfig()
* @return bool TRUE on success, FALSE on error
*/
function saveCalendar() {
$output = $this->createCalendar();
if( FALSE === ( $dirfile = $this->getConfig( 'url' )))
$dirfile = $this->getConfig( 'dirfile' );
return ( FALSE === file_put_contents( $dirfile, $output, LOCK_EX )) ? FALSE : TRUE;
}
/**
* if recent version of calendar file exists (default one hour), an HTTP redirect header is sent
* else FALSE is returned
*
* @author Kjell-Inge Gustafsson, kigkonsult <ical@kigkonsult.se>
* @since 2.21.5 - 2015-03-29
* @param int $timeout default 3600 sec
* @param bool $cdType TRUE : Content-Disposition: attachment... (default), FALSE : ...inline...
* @uses vcalendar::getConfig()
* @uses vcalendar::$headers
* @uses vcalendar::$format
* @return bool TRUE on success, FALSE on error
*/
function useCachedCalendar( $timeout=3600, $cdType=TRUE ) {
if( FALSE === ( $dirfile = $this->getConfig( 'url' )))
$dirfile = $this->getConfig( 'dirfile' );
if( ! is_file( $dirfile ) || ! is_readable( $dirfile ))
return FALSE;
if( time() - filemtime( $dirfile ) > $timeout )
return FALSE;
clearstatcache();
$dirfile = $this->getConfig( 'dirfile' );
$fsize = @filesize( $dirfile );
$filename = $this->getConfig( 'filename' );
if( 'xcal' == $this->format )
header( self::$headers[3] );
else
header( self::$headers[4] );
if( ! empty( $fsize ))
header( sprintf( self::$headers[2], $fsize ));
$cdType = ( $cdType ) ? 5 : 6;
header( sprintf( self::$headers[$cdType], $filename ));
header( self::$headers[7] );
if( FALSE === ( $fp = @fopen( $dirfile, 'r' )))
return FALSE;
fpassthru( $fp );
fclose( $fp );
return TRUE;
}
}