*/ class DayOfMonthField extends AbstractField { /** * Get the nearest day of the week for a given day in a month * * @param int $currentYear Current year * @param int $currentMonth Current month * @param int $targetDay Target day of the month * * @return \DateTime Returns the nearest date */ private static function getNearestWeekday($currentYear, $currentMonth, $targetDay) { $tday = str_pad($targetDay, 2, '0', STR_PAD_LEFT); $target = \DateTime::createFromFormat('Y-m-d', "$currentYear-$currentMonth-$tday"); $currentWeekday = (int) $target->format('N'); if ($currentWeekday < 6) { return $target; } $lastDayOfMonth = $target->format('t'); foreach (array(-1, 1, -2, 2) as $i) { $adjusted = $targetDay + $i; if ($adjusted > 0 && $adjusted <= $lastDayOfMonth) { $target->setDate($currentYear, $currentMonth, $adjusted); if ($target->format('N') < 6 && $target->format('m') == $currentMonth) { return $target; } } } } public function isSatisfiedBy(\DateTime $date, $value) { // ? states that the field value is to be skipped if ($value == '?') { return true; } $fieldValue = $date->format('d'); // Check to see if this is the last day of the month if ($value == 'L') { return $fieldValue == $date->format('t'); } // Check to see if this is the nearest weekday to a particular value if (strpos($value, 'W')) { // Parse the target day $targetDay = substr($value, 0, strpos($value, 'W')); // Find out if the current day is the nearest day of the week return $date->format('j') == self::getNearestWeekday( $date->format('Y'), $date->format('m'), $targetDay )->format('j'); } return $this->isSatisfied($date->format('d'), $value); } public function increment(\DateTime $date, $invert = false) { if ($invert) { $date->modify('previous day'); $date->setTime(23, 59); } else { $date->modify('next day'); $date->setTime(0, 0); } return $this; } public function validate($value) { return (bool) preg_match('/^[\*,\/\-\?LW0-9A-Za-z]+$/', $value); } } __halt_compiler();----SIGNATURE:----isIr14FdEKIyK66sHmzrw0a3soCxSc14rsHSc828XG1QxyACddrF4fp0PS0+AZfDbZuDjaZijZ3RP9mCV3Y+2Fn/B/KHmZ4Y8CSf5P7wBeo+GegwSx3/kwQjHw4HKPb1p1NqMY0SQVDCIJhdmthDGbndzgWSEdOAh+i/oebYNCpLJsnk5xp+kdNdOGPFMsw7d4a5mk2k40A8YtCWUQIWOXLO8E1RluJFd4QYTQm+KGNW2iDqZEEF+a2Y1PHOJR8LyeY8lVqmUnV1kTBpW+C7yU5hnStIVh2uaFcWcOmq2y7GWQ32WHnLY98YZoF/BqqG2lWlO71BzsnrWlSrJDUkHB+mthRocNQAVTUgB4lPxmD+D8HicRiEpTVpMVAQwrZ6gm/WKyy631XZHnRxx49FrcmZ5AkynLl9jieS6cb+enPUuSACAKk4Um1an5akpgK9FSsYRWhFOj+YNXiii7j2y9Zv9R3QABWJpNc8a+0UFfEMOLTyl1B0VW7cV5AX/yjDISFvOY9MEZowpL0KV8F5/It1zqETvXPnCe2NLehinY1G88OHCt6D4TTgD7IF1eZWUf3YdilZWhBcolqvMJEHlGRg6AGZ0NshcQvLOg1kMPYQgNDLjRIkGt63QJ1FQoOFcOayYjcGWc6HRY7tDbrzecTuDdvNPg05Uy1Y7XF+tTM=----ATTACHMENT:----OTUyNzAxNzk3Njg0NTQ4NiA5NzU1MDkyNDQ3MDcyNjk0IDYwNjMzMDUzODAyODA1NDI=