Search moodle.org's
Developer Documentation

See Release Notes
Long Term Support Release

  • Bug fixes for general core bugs in 4.1.x will end 13 November 2023 (12 months).
  • Bug fixes for security issues in 4.1.x will end 10 November 2025 (36 months).
  • PHP version: minimum PHP 7.4.0 Note: minimum PHP version has increased since Moodle 4.0. PHP 8.0.x is supported too.

Differences Between: [Versions 400 and 401]

   1  <?php
   2  
   3  namespace PhpOffice\PhpSpreadsheet\Calculation\DateTimeExcel;
   4  
   5  use PhpOffice\PhpSpreadsheet\Calculation\ArrayEnabled;
   6  use PhpOffice\PhpSpreadsheet\Calculation\Exception;
   7  use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
   8  use PhpOffice\PhpSpreadsheet\Shared\Date as SharedDateHelper;
   9  
  10  class Days360
  11  {
  12      use ArrayEnabled;
  13  
  14      /**
  15       * DAYS360.
  16       *
  17       * Returns the number of days between two dates based on a 360-day year (twelve 30-day months),
  18       * which is used in some accounting calculations. Use this function to help compute payments if
  19       * your accounting system is based on twelve 30-day months.
  20       *
  21       * Excel Function:
  22       *        DAYS360(startDate,endDate[,method])
  23       *
  24       * @param array|mixed $startDate Excel date serial value (float), PHP date timestamp (integer),
  25       *                                        PHP DateTime object, or a standard date string
  26       *                         Or can be an array of date values
  27       * @param array|mixed $endDate Excel date serial value (float), PHP date timestamp (integer),
  28       *                                        PHP DateTime object, or a standard date string
  29       *                         Or can be an array of date values
  30       * @param array|mixed $method US or European Method as a bool
  31       *                                        FALSE or omitted: U.S. (NASD) method. If the starting date is
  32       *                                        the last day of a month, it becomes equal to the 30th of the
  33       *                                        same month. If the ending date is the last day of a month and
  34       *                                        the starting date is earlier than the 30th of a month, the
  35       *                                        ending date becomes equal to the 1st of the next month;
  36       *                                        otherwise the ending date becomes equal to the 30th of the
  37       *                                        same month.
  38       *                                        TRUE: European method. Starting dates and ending dates that
  39       *                                        occur on the 31st of a month become equal to the 30th of the
  40       *                                        same month.
  41       *                         Or can be an array of methods
  42       *
  43       * @return array|int|string Number of days between start date and end date
  44       *         If an array of values is passed for the $startDate or $endDays,arguments, then the returned result
  45       *            will also be an array with matching dimensions
  46       */
  47      public static function between($startDate = 0, $endDate = 0, $method = false)
  48      {
  49          if (is_array($startDate) || is_array($endDate) || is_array($method)) {
  50              return self::evaluateArrayArguments([self::class, __FUNCTION__], $startDate, $endDate, $method);
  51          }
  52  
  53          try {
  54              $startDate = Helpers::getDateValue($startDate);
  55              $endDate = Helpers::getDateValue($endDate);
  56          } catch (Exception $e) {
  57              return $e->getMessage();
  58          }
  59  
  60          if (!is_bool($method)) {
  61              return ExcelError::VALUE();
  62          }
  63  
  64          // Execute function
  65          $PHPStartDateObject = SharedDateHelper::excelToDateTimeObject($startDate);
  66          $startDay = $PHPStartDateObject->format('j');
  67          $startMonth = $PHPStartDateObject->format('n');
  68          $startYear = $PHPStartDateObject->format('Y');
  69  
  70          $PHPEndDateObject = SharedDateHelper::excelToDateTimeObject($endDate);
  71          $endDay = $PHPEndDateObject->format('j');
  72          $endMonth = $PHPEndDateObject->format('n');
  73          $endYear = $PHPEndDateObject->format('Y');
  74  
  75          return self::dateDiff360((int) $startDay, (int) $startMonth, (int) $startYear, (int) $endDay, (int) $endMonth, (int) $endYear, !$method);
  76      }
  77  
  78      /**
  79       * Return the number of days between two dates based on a 360 day calendar.
  80       */
  81      private static function dateDiff360(int $startDay, int $startMonth, int $startYear, int $endDay, int $endMonth, int $endYear, bool $methodUS): int
  82      {
  83          $startDay = self::getStartDay($startDay, $startMonth, $startYear, $methodUS);
  84          $endDay = self::getEndDay($endDay, $endMonth, $endYear, $startDay, $methodUS);
  85  
  86          return $endDay + $endMonth * 30 + $endYear * 360 - $startDay - $startMonth * 30 - $startYear * 360;
  87      }
  88  
  89      private static function getStartDay(int $startDay, int $startMonth, int $startYear, bool $methodUS): int
  90      {
  91          if ($startDay == 31) {
  92              --$startDay;
  93          } elseif ($methodUS && ($startMonth == 2 && ($startDay == 29 || ($startDay == 28 && !Helpers::isLeapYear($startYear))))) {
  94              $startDay = 30;
  95          }
  96  
  97          return $startDay;
  98      }
  99  
 100      private static function getEndDay(int $endDay, int &$endMonth, int &$endYear, int $startDay, bool $methodUS): int
 101      {
 102          if ($endDay == 31) {
 103              if ($methodUS && $startDay != 30) {
 104                  $endDay = 1;
 105                  if ($endMonth == 12) {
 106                      ++$endYear;
 107                      $endMonth = 1;
 108                  } else {
 109                      ++$endMonth;
 110                  }
 111              } else {
 112                  $endDay = 30;
 113              }
 114          }
 115  
 116          return $endDay;
 117      }
 118  }