app/Customize/Service/kintoneCalendarService.php line 922

Open in your IDE?
  1. <?php
  2. //kakunin コメント確認する
  3. /*
  4. ファイル名:kintoneCalendarService.php
  5. 処理概要:kintone予定アプリから情報を渡す
  6. 作成年月日:2023/08/18
  7. 作成者:a.tojo
  8. 修正年月日:2024/04/24 k.saito
  9. 修正内容:kintoneでキャンセルになった(非表示チェックがON)データは、抽出対象から外す
  10. 修正年月日:2024/07/23 k.saito
  11. 修正内容:エアコン2階下ろし工事のカレンダ表示ができない現象の調査で、ログ埋込
  12. */
  13. declare(strict_types=1); //ECCUBEでエラー出たら外す
  14. namespace Customize\Service;
  15. require_once(dirname(__FILE__4) . '/php/kintone/kntn_class_schedule.php');
  16. require_once(dirname(__FILE__4) . '/php/kintone/kntn_class_staff.php');
  17. require_once(dirname(__FILE__4) . '/php/common/error_notification.php');   // エラー通知クラス
  18. //PHPUnit用
  19. //namespace kintone\sinwasama;
  20. //require_once(dirname(__FILE__) . '/php/kintone/kntn_class_schedule.php');
  21. //require_once(dirname(__FILE__) . '/php/kintone/kntn_class_staff.php');
  22. use Carbon\Carbon;
  23. use Carbon\CarbonPeriod;
  24. use php\kintone\KntnSchedule;
  25. use php\kintone\KntnStaff;
  26. //use php\common\ErrNotifi;
  27. //use php\common\LogOut; 
  28. use \Datetime;
  29. use Eccube\Request\Context;
  30. use Customize\Service\getAreaForCalendarService//各エリアの変数が格納されている
  31. class kintoneCalendarService
  32. {
  33.     /**
  34.      * @var KntnSchedule
  35.      */
  36.     private $kntn;
  37.     /**
  38.      * @var sendErrorNotifiService
  39.      */
  40.     protected $sendErrorNotifi;
  41.     public function __construct(
  42.         sendErrorNotifiService $sendErrorNotifi,
  43.         getAreaForCalendarService $getAreaForCalendarService
  44.     ) {
  45.         $this->sendErrorNotifi $sendErrorNotifi;
  46.         $this->getAreaForCalendarService $getAreaForCalendarService;
  47.     }
  48.     /**
  49.      * カレンダーに設定する予約可能日の詳細を作成する
  50.      *
  51.      * @param Carbon $targetDate ターゲット日 本日
  52.      * @param array $reservationDetail 予約可能日の詳細。カレンダーで使用。商品カテゴリ対応のスタッフを各地域ごとに表示
  53.      * @param array $subStaffReservationDetail サブ担当者の予約可能日の詳細。カレンダーで使用。商品カテゴリ関係なく各地域対応スタッフを表示
  54.      * @param array $areaStaffFromCalender 各エリアの担当スタッフ
  55.      * @param Carbon $startMonth 本日の前月の中旬 ※作業開始日と作業終了日が前月〜今月を跨ぐ可能性があるため
  56.      * @param Carbon $endMonth 本日の来月末
  57.      * @param boolean $subStaffFlg エコキュート判断用。trueの場合は対象カテゴリがエコキュート
  58.      * @param array $subStaff エコキュート用、担当エリアのサブスタッフ
  59.      * 
  60.      * @return array $reservationDetail,$subStaffReservationDetail
  61.      */
  62.     public function
  63.     makeReservationDetail(Carbon $targetDate$area$categoryName, &$reservationDetail, &$subStaffReservationDetail$subStaffFlg)
  64.     {
  65.         log_debug('makereservationDetail start');
  66.         //dump($targetDate);
  67.         //初期化 start
  68.         $startMonth $targetDate->copy()->startOfMonth(); //>subMonth()->endOfMonth()->subDays(15);
  69.         $endMonth $targetDate->copy()->addMonth()->endOfMonth();
  70.         $startMonthString $startMonth->toDateString();
  71.         $endMonthString $endMonth->toDateString();
  72.         $subStaff = [];
  73.         $subStaffKey 'サブスタッフ';
  74.         $reservationDetail = [];
  75.         $staffFlg 'default';
  76.         //deleteTargetの値で削除対象の時間帯が変わる為
  77.         $am1 'AM1';
  78.         $am2 'AM2';
  79.         $pm1 'PM1';
  80.         $pm2 'PM2';
  81.         $timePeriodDelete = array(
  82.             'AM1' => array($am1),
  83.             'AM2' => array($am2),
  84.             'AM' => array($am1$am2),
  85.             'PM1' => array($pm1),
  86.             'PM2' => array($pm2),
  87.             'PM' => array($pm1$pm2),
  88.             'AM1~PM1' => array($am1$am2$pm1),
  89.             'AM2~PM2' => array($am2$pm1$pm2),
  90.             'AM2~PM1' => array($am2$pm1),
  91.             '終日' => array($am1$am2$pm1$pm2)
  92.         );
  93.         $normalTimezone $timePeriodDelete['終日'];
  94.         $ecocuteTimezone $timePeriodDelete['AM'];
  95.         //kintoneから商品カテゴリ対応のメインスタッフを取得(エコキュート含む)
  96.         $categoryStaff $this->getStaffScheduleRecord($categoryName$startMonthString$endMonthString);
  97.         dump($categoryStaff);
  98.         log_debug($categoryName 'に対応しているスタッフ');
  99.         log_debug(print_r($categoryStafftrue));
  100.         //各エリア対応のスタッフを取得する
  101.         $areaStaff $this->getAreaStaffRecord($categoryStaff$area);
  102.         dump($areaStaff);
  103.         //デバッグ
  104.         $areastaff_txt print_r($areaStafftrue);
  105.         log_debug('areastaff:'.$areastaff_txt);
  106.         //エコキュートの場合はサブ担当者の情報も取得するため
  107.         if ($subStaffFlg) {
  108.             log_info('エコキュートなのでサブ担当者を選択');
  109.             $createdStaff $categoryStaff;
  110.             //検索しやすくする為、キー値をスタッフコードにする
  111.             foreach ($createdStaff as $staffKey => $staffValue) {
  112.                 $staffCode $staffValue['ユーザー選択'][0]['code'];
  113.                 $createdStaff[$staffCode] = $staffValue;
  114.                 unset($createdStaff[$staffKey]);
  115.             }
  116.             foreach ($area as $targetArea) {
  117.                 $variable $targetArea['variable'];
  118.                 $areaName $targetArea['name'];
  119.                 $subStaff[$variable] = $this->getStaffScheduleRecord($areaName$startMonthString$endMonthString$subStaffFlg$createdStaff); //サブスタッフのレコードを取得
  120.             }
  121.         }
  122.         //カレンダー用の各エリアスタッフ $areaStaffFromCalender を作成する
  123.         $key '';
  124.         $value '';
  125.         $areaStaffFromCalender = [];
  126.         foreach ($areaStaff as $targetAreaKey => $targetArea) {
  127.             //商品カテゴリ担当&エリア担当のスタッフがいない可能性がある為
  128.             if ($areaStaff[$targetAreaKey]['対応スタッフ']) {
  129.                 foreach ($areaStaff[$targetAreaKey]['対応スタッフ'] as $targetAreaStaff) {
  130.                     $key $targetAreaStaff['id']; //レコードid
  131.                     $value $targetAreaStaff['name']; //名前
  132.                     $areaStaffFromCalender[$targetAreaKey]['対応スタッフ'][$key] = $value;
  133.                 }
  134.             } else {
  135.                 $areaStaffFromCalender[$targetAreaKey]['対応スタッフ'] = [];
  136.             }
  137.         };
  138.         dump($areaStaffFromCalender);
  139.         //デバッグ
  140.         $areaStaffFromCalender_txt print_r($areaStaffFromCalendertrue);
  141.         log_debug('areaStaffFromCalender:'.$areaStaffFromCalender_txt);
  142.         //各日付の予約状況を作成する
  143.         //初期化 各日付・時間帯を作成する
  144.         $period $startMonth->copy()->toPeriod($endMonth);
  145.         foreach ($period as $day) {
  146.             //dump($day);
  147.             $day $day->format('Y-m-d');
  148.             $reservationDetail[$day] = [];
  149.             foreach ($area as $targetarea) {
  150.                 $variable $targetarea['variable']; //anjou
  151.                 $reservationDetail[$day][$variable]['AM1'] = [];
  152.                 $reservationDetail[$day][$variable]['AM2'] = [];
  153.                 $reservationDetail[$day][$variable]['PM1'] = [];
  154.                 $reservationDetail[$day][$variable]['PM2'] = [];
  155.             }
  156.         }
  157.         //dump($areaStaffFromCalender);
  158.         foreach ($reservationDetail as $dayKey => $dayValue) {
  159.             foreach ($dayValue as $areaKey => $areaValue) {
  160.                 // reservationDetailに値を設定
  161.                 foreach ($normalTimezone as $timeZoneKey) {
  162.                     $reservationDetail[$dayKey][$areaKey][$timeZoneKey] = $areaStaffFromCalender[$areaKey]['対応スタッフ'];
  163.                 }
  164.                 if ($subStaffFlg) {
  165.                     //dump('エコキュートの場合');
  166.                     foreach ($ecocuteTimezone as $timeZoneKey) {
  167.                         $subStaffReservationDetail[$dayKey][$areaKey][$timeZoneKey] = $subStaff[$areaKey]['list'];
  168.                     }
  169.                 }
  170.             }
  171.         }
  172.         //デバッグ
  173.         $reservationDetail_txt print_r($reservationDetailtrue);
  174.         log_debug('削除前reservationDetail:'.$reservationDetail_txt);
  175.         //dump('削除前');
  176.         //dump($reservationDetail);
  177.         //dump($subStaffReservationDetail);
  178.         //dump($categoryStaff);
  179.         //dump($subStaff);
  180.         //予定がある担当スタッフを配列から削除
  181.         if ($subStaffFlg) {
  182.             $staffFlg 'main';
  183.         }
  184.         //dump($categoryStaff);
  185.         //予定がある担当スタッフを配列から削除
  186.         foreach ($categoryStaff as $staff) {
  187.             //dump($staff);
  188.             $targetStaffId $staff['レコードid'];
  189.             $targetStaff $staff['名前']; //削除対象スタッフ
  190.             $deleteArea $staff['担当エリアテーブル']; //削除対象エリア
  191.             //dump($staff);
  192.             //来月の予定がない場合がある為
  193.             if (is_array($staff['予定'])) {
  194.                 foreach ($staff['予定'] as $schedule) { //スタッフの予定を取得
  195.                     //dump($schedule);
  196.                     $deleteTarget $schedule['スケジュール選択']; //削除する時間帯。AM1、AM1~PM2など
  197.                     $deleteDayStart $schedule['作業開始日'];
  198.                     $deleteDayEnd $schedule['作業完了日'];
  199.                     if ($deleteDayStart === $deleteDayEnd) {
  200.                         log_debug('予定アプリレコード番号' $schedule['レコード番号'] . ':作業開始日と作業完了日が同じ');
  201.                         log_info('$reservationDetailから' $deleteDayStart '~' $deleteDayEnd 'までの' $targetStaff 'さんを削除開始。 削除スケジュール:' $deleteTarget);
  202.                         $deleteDay $deleteDayStart;
  203.                         //作業開始日が先月末の予定もあるため
  204.                         if (array_key_exists($deleteDay$reservationDetail)) {
  205.                             $reservationDetail[$deleteDayStart] = $this->deleteStaffFromScheduleArray($reservationDetail[$deleteDay], $timePeriodDelete$deleteDay$targetStaffId$targetStaff$deleteArea$deleteTarget$staffFlg);
  206.                         }
  207.                     } else {
  208.                         log_debug('予定アプリレコード番号' $schedule['レコード番号'] . ':作業開始日と作業完了日が違う');
  209.                         //削除する作業開始日から作業完了日までの日付の範囲を作成する
  210.                         log_debug('作業開始日' $deleteDayStart 'と作業終了日' $deleteDayEnd 'をcarbonに変換');
  211.                         $deleteDayStart Carbon::parse($deleteDayStart);
  212.                         $deleteDayEnd Carbon::parse($deleteDayEnd);
  213.                         //dump($deleteDayStart);
  214.                         //dump($deleteDayEnd);
  215.                         $deleteDayPeriod $deleteDayStart->copy()->toPeriod($deleteDayEnd);
  216.                         foreach ($deleteDayPeriod as $deleteDayCarbon) {
  217.                             $deleteDay $deleteDayCarbon->toDateString();
  218.                             //dump($deleteDay);
  219.                             //作業開始日が先月末の予定もあるため
  220.                             if (array_key_exists($deleteDay$reservationDetail)) {
  221.                                 $reservationDetail[$deleteDay] = $this->deleteStaffFromScheduleArray($reservationDetail[$deleteDay], $timePeriodDelete$deleteDay$targetStaffId$targetStaff$deleteArea$deleteTarget$staffFlg);
  222.                             }
  223.                         }
  224.                     }
  225.                 }
  226.             }
  227.         }
  228.         if ($subStaffFlg) {
  229.             //エコキュートの場合
  230.             //サブスタッフ
  231.             $staffFlg 'sub';
  232.             foreach ($subStaff as $areaKey => $areaValue) {
  233.                 foreach ($areaValue as $staffKey => $staff) {
  234.                     if ($staffKey === 'list') { //listキーは見ない
  235.                         continue;
  236.                     }
  237.                     $targetStaffId $staff['レコードid'];
  238.                     $targetStaff $staff['名前']; //削除対象スタッフ
  239.                     $deleteArea $staff['担当エリアテーブル']; //削除対象エリア
  240.                     //来月の予定がない場合がある為
  241.                     if (is_array($staff['予定'])) {
  242.                         foreach ($staff['予定'] as $schedule) { //スタッフの予定を取得
  243.                             //dump($schedule);
  244.                             $deleteDayStart $schedule['作業開始日'];
  245.                             $deleteDayEnd $schedule['作業完了日'];
  246.                             if ($deleteDayStart === $deleteDayEnd) {
  247.                                 log_debug('予定アプリレコード番号' $schedule['レコード番号'] . ':作業開始日と作業完了日が同じ' $staffFlg);
  248.                                 log_debug('$reservationDetailから' $deleteDayStart '~' $deleteDayEnd 'までの' $targetStaff 'さんを削除開始。 削除スケジュール:' $deleteTarget);
  249.                                 $deleteDay $deleteDayStart;
  250.                                 //作業開始日が先月末の予定もあるため
  251.                                 if (array_key_exists($deleteDay$subStaffReservationDetail)) {
  252.                                     $subStaffReservationDetail[$deleteDayStart] = $this->deleteStaffFromScheduleArray($subStaffReservationDetail[$deleteDay], $timePeriodDelete$deleteDay$targetStaffId$targetStaff$deleteArea$deleteTarget$staffFlg);
  253.                                 }
  254.                             } else {
  255.                                 log_debug('予定アプリレコード番号' $schedule['レコード番号'] . ':作業開始日と作業完了日が違う');
  256.                                 //削除する作業開始日から作業完了日までの日付の範囲を作成する
  257.                                 log_debug('作業開始日' $deleteDayStart 'と作業終了日' $deleteDayEnd 'をcarbonに変換');
  258.                                 $deleteDayStart Carbon::parse($deleteDayStart);
  259.                                 $deleteDayEnd Carbon::parse($deleteDayEnd);
  260.                                 //dump($deleteDayStart);
  261.                                 //dump($deleteDayEnd);
  262.                                 $deleteDayPeriod $deleteDayStart->copy()->toPeriod($deleteDayEnd);
  263.                                 foreach ($deleteDayPeriod as $deleteDayCarbon) {
  264.                                     $deleteDay $deleteDayCarbon->toDateString();
  265.                                     //dump($deleteDay);
  266.                                     //作業開始日が先月末の予定もあるため
  267.                                     if (array_key_exists($deleteDay$subStaffReservationDetail)) {
  268.                                         log_debug('サブ担当者削除開始日と終了日が違う');
  269.                                         log_debug($deleteDayStart);
  270.                                         log_debug($targetStaff);
  271.                                         $subStaffReservationDetail[$deleteDay] = $this->deleteStaffFromScheduleArray($subStaffReservationDetail[$deleteDay], $timePeriodDelete$deleteDay$targetStaffId$targetStaff$deleteArea$deleteTarget$staffFlg);
  272.                                     }
  273.                                 }
  274.                             }
  275.                         }
  276.                     }
  277.                 }
  278.             }
  279.         }
  280.         //デバッグ
  281.         $reservationDetail_txt print_r($reservationDetailtrue);
  282.         log_debug('削除後reservationDetail:'.$reservationDetail_txt);
  283.         //dump('削除後');
  284.         //dump($reservationDetail);
  285.         //dump($subStaffReservationDetail);
  286.         //debug 問題なければ消す
  287.         dump($categoryName 'に対応しているスタッフは(作業レベル');
  288.         dump($categoryStaff);
  289.         dump('エリアごとのスタッフ');
  290.         dump($areaStaff);
  291.         if ($subStaff) {
  292.             dump('エコキュートの場合');
  293.             dump($subStaff);
  294.         }
  295.         //debug
  296.         //問題なければ消す
  297.         //debug
  298.         log_debug('makereservationDetail end');
  299.     }
  300.     /** 
  301.      *kintone 予約アプリの情報を返す
  302.      *担当スタッフの空き状況を確認する為
  303.      */
  304.     public function getScheduleRecord($staffCode$startMonthString$endMonthString)
  305.     {
  306.         log_debug('getScheduleRecord start');
  307.         log_info($staffCode 'の' $startMonthString 'から' $endMonthString 'までの予定をkintoneから取得する');
  308.         $staff_record '予定アプリにデータなし';
  309.         //レコード取得
  310.         $result = array(); // 戻り値格納変数
  311.         // 抽出条件。カテゴリー & レコード番号で抽出
  312.         $query '(内容 not in ("削除") and 非表示チェック not in ("非表示") and 対応スタッフ in (' '"' $staffCode '")' ' and (作業開始日時 >= "' $startMonthString '" and 作業開始日時 <= ' '"' $endMonthString '" )) or (内容 not in ("削除") and 非表示チェック not in ("非表示") and 対応スタッフ in (' '"' $staffCode '")' 'and (作業完了日時 >= "' $startMonthString '" and 作業完了日時 <= ' '"' $endMonthString '" ))';
  313.         //dump($staffCode);
  314.         //dump($query);
  315.         // クエリ以外を指定
  316.         $non_query "order by 作業開始日時 asc";
  317.         // 取得するフィールドコードを指定。nullは全フィールドコード。
  318.         $fields = array('fields' => array('レコード番号''対応スタッフ''作業開始日時''作業完了日時''スケジュール選択''内容'));
  319.         //$fields = null;
  320.         $kntn = new KntnSchedule();
  321.         // $count > 0 の間、ループ。クエリー対象の全レコードのECコード取得
  322.         while (($count $kntn->getRecords($result$fields$query$non_query)) > 0) {
  323.             // 1000件単位で読み出し。各処理は、以下のようにレコード単位で取り出して処理したり、色々。
  324.             $staff_record = [];
  325.             foreach ($result['records'] as $record) {
  326.                 $startOfWork Carbon::parse($record['作業開始日時']['value'])->setTimezone('Asia/Tokyo');
  327.                 $endOfWork Carbon::parse($record['作業完了日時']['value'])->setTimezone('Asia/Tokyo');
  328.                 $data = array(
  329.                     'レコード番号' => $record['レコード番号']['value'],
  330.                     '対応スタッフ' => $record['対応スタッフ']['value'],
  331.                     '作業開始日時' => $startOfWork->format('Y-m-d H:i:s'),
  332.                     '作業完了日時' => $endOfWork->format('Y-m-d H:i:s'),
  333.                     '作業開始日' => $startOfWork->format('Y-m-d'),
  334.                     '作業完了日' => $endOfWork->format('Y-m-d'),
  335.                     'スケジュール選択' => $record['スケジュール選択']['value']
  336.                 );
  337.                 $staff_record[] = $data;
  338.             }
  339.         }
  340.         //dump($staff_record);
  341.         if (!$kntn->getSuccessFlg()) {
  342.             // 失敗
  343.             // エラーログ出力
  344.             dump('kntn error');
  345.             log_error('kntn getCalendarRecord fail');
  346.             log_error('kntn result' print_r($resulttrue));
  347.             return false;
  348.         }
  349.         log_debug('getScheduleRecord end');
  350.         //dump($staff_record);
  351.         return $staff_record;
  352.     }
  353.     //各地域の対応スタッフ名を返す
  354.     public function getAreaStaffRecord($CategoryStaff$area): array
  355.     {
  356.         log_debug('getAreaStaffRecord start');
  357.         $staff '';
  358.         $areaData = [];
  359.         //dump($area);
  360.         foreach ($area as $targetArea) {
  361.             $areaName $targetArea['name'];
  362.             $variable $targetArea['variable'];
  363.             $areaData[$variable] = [];
  364.             $areaCharge = [];
  365.             foreach ($CategoryStaff as $staff) {
  366.                 //dump($staff);
  367.                 if (in_array($areaName$staff['担当エリアテーブル'], true)) {
  368.                     $staffCode $staff['ユーザー選択'][0]['code'];
  369.                     $staffData = array('id' => $staff['レコードid'], 'name' => $staff['名前'], 'ユーザー選択コード' => $staffCode);
  370.                     $areaCharge[] = $staffData;
  371.                 }
  372.             }
  373.             $areaData[$variable] = array('variable' => $variable'対応エリア' => $areaName'対応スタッフ' => $areaCharge);
  374.         }
  375.         //log_debug(print_r($areaData, true));
  376.         log_debug('getAreaStaffRecord end');
  377.         return $areaData;
  378.     }
  379.     /** 
  380.      *kintone スタッフアプリの情報を返す
  381.      *担当エリア、設置出来る商品を取得するため
  382.      */
  383.     public function getStaffRecord($targetName$subStaffFlg)
  384.     {
  385.         log_debug('getStaffRecord start:');
  386.         log_info($targetName '対応のスタッフをkintoneから検索');
  387.         //レコード取得
  388.         $staffRecord = [];
  389.         $data = [];
  390.         $result = []; // 戻り値格納変数
  391.         //エリアに応じた変数を検索出来る配列を作成
  392.         //$getAreaForCalendarService = new getAreaForCalendarService(); //各エリアの変数が格納されたクラス
  393.         //$getAreaForCalendarService = $getAreaForCalendarService->getArea();
  394.         $getAreaForCalendarService $this->getAreaForCalendarService->getArea();
  395.         $searchAreaVariables = [];
  396.         foreach ($getAreaForCalendarService as $variable) {
  397.             $key $variable['name'];
  398.             $value $variable['variable'];
  399.             $searchAreaVariables[$key] = $value;
  400.         }
  401.         dump($searchAreaVariables);
  402.         $query '中分類 in ("' $targetName '")';
  403.         //エコキュートの場合はサブ担当者も決めないといけないため条件を変える
  404.         if ($subStaffFlg) {
  405.             //dump('サブ担当者を検索する');
  406.             $query '担当エリア in ("' $targetName '")';
  407.         }
  408.         $non_query "order by レコード番号 desc";
  409.         // 取得するフィールドコードを指定。nullは全フィールドコード。
  410.         $fields = array('fields' => array('$id''名前''ユーザー選択''スキル入力テーブル''担当エリアテーブル'));
  411.         $kntn = new KntnStaff();
  412.         // $count > 0 の間、ループ。クエリー対象の全レコードのECコード取得
  413.         while (($count $kntn->getRecords($result$fields$query$non_query)) > 0) {
  414.             //dump($result['records']);
  415.             // 1000件単位で読み出し。各処理は、以下のようにレコード単位で取り出して処理したり、色々。
  416.             foreach ($result['records'] as $record) {
  417.                 //テーブルフィールド用
  418.                 $table = [];
  419.                 $area = [];
  420.                 $skill = [];
  421.                 $user = [];
  422.                 foreach ($record['ユーザー選択']['value'] as $table) {
  423.                     $tabledata = array('code' => $table['code'], 'name' => $table['name']);
  424.                     $user[] = $tabledata;
  425.                 }
  426.                 foreach ($record['スキル入力テーブル']['value'] as $table) {
  427.                     $skill[] = $table['value']['中分類']['value'];
  428.                 }
  429.                 foreach ($record['担当エリアテーブル']['value'] as $table) {
  430.                     $value $table['value']['担当エリア']['value'];
  431.                     //kintoneスタッフアプリで対応エリア外の地域を設定する可能性がある為、設定した場合はスルーする
  432.                     if (array_key_exists($value$searchAreaVariables)) {
  433.                         $key $searchAreaVariables[$value]; //該当エリアの変数をキーにする。安城市の場合はanjouをキーにする
  434.                     } else {
  435.                         log_info($record['名前']['value'] . 'さんはスタッフアプリの担当エリアテーブルにECCUBEエリア外のエリア[' $value ']を設定しております。');
  436.                     }
  437.                     $area[$key] = $value;
  438.                 }
  439.                 $data = array(
  440.                     'レコードid' => $record['$id']['value'],
  441.                     '名前' => $record['名前']['value'],
  442.                     'ユーザー選択' => $user,
  443.                     'スキル入力テーブル' => $skill,
  444.                     '担当エリアテーブル' => $area
  445.                 );
  446.                 $staffRecord[] = $data;
  447.                 //dump($staffRecord);
  448.             }
  449.         }
  450.         if (!$kntn->getSuccessFlg()) {
  451.             // 失敗
  452.             // エラーログ出力
  453.             dump('kntn error');
  454.             log_error('kntn getCalendarRecord fail');
  455.             log_error('kntn result' print_r($resulttrue));
  456.             //LogOut::error(basename(__FILE__), __LINE__, "kntn getCalendarRecord", 'fail', dirname(__FILE__));
  457.             //LogOut::error(basename(__FILE__), __LINE__, "kntn result", print_r($result, true), dirname(__FILE__));
  458.             return false;
  459.         }
  460.         log_debug('getStaffRecord end');
  461.         return $staffRecord;
  462.     }
  463.     //商品カテゴリ対応スタッフの配列に予定アプリのスケジュールを追加して返す
  464.     public function getStaffScheduleRecord($targetName$startMonthString$endMonthString$subStaffFlg false, &$createdStaff null): array
  465.     {
  466.         log_debug('getStaffScheduleRecord start');
  467.         $CategoryScheduleStaff = [];
  468.         $CategoryStaff $this->getStaffRecord($targetName$subStaffFlg);
  469.         dump($CategoryStaff);
  470.         if ($subStaffFlg === false) {
  471.             log_debug('通常スタッフ');
  472.             foreach ($CategoryStaff as $staff) {
  473.                 $staffCode $staff['ユーザー選択'][0]['code'];
  474.                 $scheduleData $this->getScheduleRecord($staffCode$startMonthString$endMonthString);
  475.                 $staff['予定'] = $scheduleData;
  476.                 $CategoryScheduleStaff[] = $staff;
  477.             }
  478.         } else {
  479.             log_debug('サブスタッフ');
  480.             $CategoryScheduleStaff['list'] = [];
  481.             //スタッフのレコードが既に存在する場合は使い回す
  482.             foreach ($CategoryStaff as $staff) {
  483.                 //dump($staff);
  484.                 $staffCode $staff['ユーザー選択'][0]['code'];
  485.                 $recordId $staff['レコードid'];
  486.                 if (array_key_exists($staffCode$createdStaff)) {
  487.                     //dump('$createdStaffに' . $staffCode . 'あり');
  488.                     $staff['予定'] = $createdStaff[$staffCode]['予定'];
  489.                 } else {
  490.                     log_debug('$createdStaffに' $staffCode 'なし');
  491.                     $scheduleData $this->getScheduleRecord($staffCode$startMonthString$endMonthString);
  492.                     $staff['予定'] = $scheduleData;
  493.                     $createdStaff[$staffCode] = $staff;
  494.                 }
  495.                 $CategoryScheduleStaff[] = $staff;
  496.                 $CategoryScheduleStaff['list'][$recordId] = $staff['名前'];
  497.             }
  498.         }
  499.         //dump($CategoryScheduleStaff);
  500.         log_debug('getStaffScheduleRecord end');
  501.         return $CategoryScheduleStaff;
  502.     }
  503.     /**
  504.      * カレンダー生成用の配列からスタッフを削除する
  505.      *
  506.      * @param string $deleteDay 削除日
  507.      * @param array $reservationDetailDeleteDay カレンダー生成用の各スタッフの予定が格納された配列(削除日のみ)。
  508.      * @param string $targetStaff 削除対象のスタッフ
  509.      * @param string $deleteTarget 削除対象の時間帯(AM1など)
  510.      * @param array $deleteArea スタッフの対応エリア(削除するエリア)
  511.      */
  512.     //public function deleteStaffFromScheduleArray($reservationDetail, $targetStaffId, $targetStaff, $deleteArea, $deleteDayStart, $deleteDayEnd, $deleteTarget): array
  513.     public function deleteStaffFromScheduleArray($reservationDetailDeleteDay$timePeriodDelete$deleteDay$targetStaffId$targetStaff$deleteArea$deleteTarget$staffFlg): array
  514.     {
  515.         log_debug('deleteStaffFromScheduleArray start');
  516.         switch ($staffFlg) {
  517.             case 'main':
  518.                 log_debug('エコキュートメインスタッフなのでAM1〜PM2まで削除');
  519.                 $deleteTarget $timePeriodDelete['終日'];
  520.                 break;
  521.             case 'sub':
  522.                 log_debug('エコキュートサブスタッフなのでAM1〜AM2まで削除');
  523.                 $deleteTarget $timePeriodDelete['AM'];
  524.                 break;
  525.             default:
  526.                 log_debug('エコキュート以外のスタッフなので通常通り削除');
  527.                 $deleteTarget $timePeriodDelete[$deleteTarget];
  528.         }
  529.         log_debug($targetStaff 'さんの対応エリア' print_r($deleteAreatrue));
  530.         log_debug($targetStaff 'さんの削除スケジュール' .  print_r($deleteTargettrue));
  531.         //dump('削除前のカレンダー(' . $deleteDay . ')' . print_r($reservationDetailDeleteDay, true));
  532.         //スタッフの対応エリアのスケジュール(AM1など)を削除
  533.         foreach ($deleteArea as $targetAreaKey => $targetArea) {
  534.             foreach ($deleteTarget as $deleteTimeKey => $deleteTime) {
  535.                 log_debug($deleteDay ':' $targetArea 'の' $targetStaff 'さんを削除。' $deleteTime 'の' '対象配列キー(スタッフアプリレコードid):' $targetStaffId);
  536.                 if (in_array($targetStaff$reservationDetailDeleteDay[$targetAreaKey][$deleteTime], true)) {
  537.                     unset($reservationDetailDeleteDay[$targetAreaKey][$deleteTime][$targetStaffId]);
  538.                 } else {
  539.                     log_info('削除出来ませんでした。' $targetStaff 'さんが存在しません(予定アプリで別レコードで同じ日付・時間帯を選択した場合も表示される)');
  540.                 }
  541.             }
  542.         }
  543.         log_debug('deleteStaffFromScheduleArray end');
  544.         return $reservationDetailDeleteDay;
  545.     }
  546.     /**
  547.      * 各時間帯のスタッフの空き状況数を計算・表示する
  548.      *
  549.      * @param string $availabilityDisplay 空き状況(カレンダーディスプレイ用)
  550.      * @param int $availabilityCount AMかPMのスタッフ数
  551.      * @param array $schedule1 時間帯のスタッフ数、AM1/PM1用
  552.      * @param array $schedule2 時間帯のスタッフ数、AM2/PM2用
  553.      * @param string $reservable カレンダーディスプレイ用 予約可能
  554.      * @param string $reservationNot カレンダーディスプレイ用 予約不可
  555.      * @param string $few カレンダーディスプレイ用 残りわずか
  556.      */
  557.     public function
  558.     calculateAvailability(&$availabilityDisplay, &$availabilityCount$schedule1$schedule2$reservable$reservationNot$few)
  559.     {
  560.         log_debug('calculateAvailability start');
  561.         //dump($schedule1);
  562.         //dump($schedule2);
  563.         $availabilityCount count($schedule1) + count($schedule2);
  564.         $availabilityCount intval($availabilityCount);
  565.         if ($availabilityCount >= 4) {
  566.             $availabilityDisplay $reservable;
  567.         } elseif ($availabilityCount <= && $availabilityCount >= 1) {
  568.             $availabilityDisplay $few;
  569.         } elseif ($availabilityCount <= 0) {
  570.             $availabilityDisplay $reservationNot;
  571.         }
  572.         log_debug('calculateAvailability end');
  573.     }
  574.     /**
  575.      * ECCUBEカレンダーの営業日の予約不可能日を設定する
  576.      *
  577.      * @param Carbon $few ターゲット日
  578.      * @var array $japanHoliday date型のGoogleカレンダ-の祝日
  579.      * @param array $targetMonthCalendar 対象月のカレンダー
  580.      * @param array $reservationDetail 各日付のAM1~PM2のまでの対応スタッフ詳細。$targetMonthCalendar生成時に使用
  581.      * @param array $subStaffReservationDetail エコキュートの場合、各日付のAM1~PM2のまでのサブスタッフ詳細。$targetMonthCalendar生成時に使用 
  582.      * @var array $areaStaffFromCalender 各エリアの担当スタッフ
  583.      * @var date $targetYmdDateTime カレンダーの対象日
  584.      * @var string $availabilityCount 各時間帯の合計スタッフ数(AMの場合はAM1とAM2の合計)
  585.      * 
  586.      * @var string $searchDay 検索対象の日付。$reservationDetailの検索に使用。
  587.      */
  588.     public function
  589.     setReservationDay($targetMonthCalendar$reservationDetail$subStaffReservationDetailCarbon $targetDate$area$category$reservable$reservationNot$few$subStaffFlg): array
  590.     {
  591.         log_debug('setReservationDay start');
  592.         //dump($reservationDetail);
  593.         //dump($subStaffReservationDetail);
  594.         //初期化 start
  595.         $targetYearString = (string) $targetDate->copy()->year//reservationDetailから予約日を検索するため
  596.         $targetMonthString $targetDate->copy()->format('m'); //reservationDetailから予約日を検索するため
  597.         $availabilityDisplay '';
  598.         $availabilityCount '';
  599.         if ($subStaffFlg) {
  600.             //エコキュートの場合
  601.             for ($i 0$i count($targetMonthCalendar); $i++) {
  602.                 // カレンダー配列の日が空の場合は処理をスキップ
  603.                 if ($targetMonthCalendar[$i]['day'] == '') {
  604.                     continue;
  605.                 }
  606.                 $searchDay $targetYearString '-' $targetMonthString '-' sprintf('%02d'$targetMonthCalendar[$i]['day']);
  607.                 foreach ($reservationDetail[$searchDay] as $areaKey => $areaValue) {
  608.                     //空き状況の計算と表示
  609.                     //AM
  610.                     $schedule2 = []; //予定がある対象日は時間帯関わらず削除されている為、AM2は計算しなくて良い
  611.                     $targetMonthCalendar[$i]['reservation'][$areaKey]['AM1'] = $schedule1 =  $reservationDetail[$searchDay][$areaKey]['AM1'];
  612.                     $this->calculateAvailability($availabilityDisplay$availabilityCount$schedule1$schedule2$reservable$reservationNot$few);
  613.                     $mainStaffCount $availabilityCount;
  614.                     $mainStaffDispllay $availabilityDisplay;
  615.                     if ($mainStaffCount === 1) {
  616.                         //dump('エコキュートのメインスタッフ数が残り1の場合');
  617.                         //残り1名のメインスタッフはサブスタッフにならない為、削除
  618.                         $delete_key array_key_first($reservationDetail[$searchDay][$areaKey]['AM1']);
  619.                         unset($subStaffReservationDetail[$searchDay][$areaKey]['AM1'][$delete_key]);
  620.                         //dump($subStaffReservationDetail[$searchDay][$areaKey]['AM1']);
  621.                     }
  622.                     //サブスタッフ
  623.                     //AM1、AM2のいずれかに予定がある対象日は削除されているため、AM2は計算しなくて良い
  624.                     $targetMonthCalendar[$i]['reservation'][$areaKey]['subStaff']['AM1'] = $schedule1 $subStaffReservationDetail[$searchDay][$areaKey]['AM1'];
  625.                     $this->calculateAvailability($availabilityDisplay$availabilityCount$schedule1$schedule2$reservable$reservationNot$few);
  626.                     //メインスタッフとサブスタッフの人数を比較して、少ない方を表示する.
  627.                     if ($mainStaffCount <= $availabilityCount) {
  628.                         $targetMonthCalendar[$i]['reservation'][$areaKey]['AMCount'] = $mainStaffCount;
  629.                         $targetMonthCalendar[$i]['reservation'][$areaKey]['AM'] = $mainStaffDispllay;
  630.                     } else {
  631.                         $targetMonthCalendar[$i]['reservation'][$areaKey]['AMCount'] = $availabilityCount;
  632.                         $targetMonthCalendar[$i]['reservation'][$areaKey]['AM'] = $availabilityDisplay;
  633.                     }
  634.                     //PM エコキュートの場合は強制的に予約不可にする
  635.                     $targetMonthCalendar[$i]['reservation'][$areaKey]['PM'] = $reservationNot;
  636.                 }
  637.             }
  638.         } else {
  639.             for ($i 0$i count($targetMonthCalendar); $i++) {
  640.                 // カレンダー配列の日が空の場合は処理をスキップ
  641.                 if ($targetMonthCalendar[$i]['day'] == '') {
  642.                     continue;
  643.                 }
  644.                 $searchDay $targetYearString '-' $targetMonthString '-' sprintf('%02d'$targetMonthCalendar[$i]['day']);
  645.                 //dump($reservationDetail[$searchDay]);
  646.                 foreach ($reservationDetail[$searchDay] as $areaKey => $areaValue) {
  647.                     //空き状況の計算と表示
  648.                     //AM
  649.                     $targetMonthCalendar[$i]['reservation'][$areaKey]['AM1'] = $schedule1 =  $reservationDetail[$searchDay][$areaKey]['AM1'];
  650.                     $targetMonthCalendar[$i]['reservation'][$areaKey]['AM2'] = $schedule2 $reservationDetail[$searchDay][$areaKey]['AM2'];
  651.                     $this->calculateAvailability($availabilityDisplay$availabilityCount$schedule1$schedule2$reservable$reservationNot$few);
  652.                     $targetMonthCalendar[$i]['reservation'][$areaKey]['AMCount'] = $availabilityCount;
  653.                     $targetMonthCalendar[$i]['reservation'][$areaKey]['AM'] = $availabilityDisplay;
  654.                     //PM
  655.                     $targetMonthCalendar[$i]['reservation'][$areaKey]['PM1'] = $schedule1 $reservationDetail[$searchDay][$areaKey]['PM1'];
  656.                     $targetMonthCalendar[$i]['reservation'][$areaKey]['PM2'] = $schedule2 $reservationDetail[$searchDay][$areaKey]['PM2'];
  657.                     $this->calculateAvailability($availabilityDisplay$availabilityCount$schedule1$schedule2$reservable$reservationNot$few);
  658.                     $targetMonthCalendar[$i]['reservation'][$areaKey]['PMCount'] = $availabilityCount;
  659.                     $targetMonthCalendar[$i]['reservation'][$areaKey]['PM'] = $availabilityDisplay;
  660.                 }
  661.             }
  662.         }
  663.         log_debug('setReservationDay end');
  664.         return $targetMonthCalendar;
  665.     }
  666.     /**
  667.      * ECCUBEカレンダーの定休日と祝日を予約不可にする
  668.      *
  669.      * @param Carbon $few ターゲット日
  670.      * @param array $japanHoliday date型のGoogleカレンダ-の祝日
  671.      * @param array $targetMonthCalendar 対象月のカレンダー 
  672.      */
  673.     public function
  674.     setReservationHoliDay($targetMonthCalendar$holidayListOfTwoMonths$holidayCarbon $targetDate$area$reservationNot): array
  675.     {
  676.         log_debug('setReservationHoliDay start');
  677.         //ECCUBEカレンダーがDATE型の為、型変換を行う
  678.         $japanHoliday = array();
  679.         foreach ($holiday as $day) {
  680.             $japanHoliday[] = Carbon::parse($day)->toDateTime()->setTime(000);
  681.         }
  682.         for ($i 0$i count($targetMonthCalendar); $i++) {
  683.             // カレンダー配列の日が空の場合は処理をスキップ
  684.             if ($targetMonthCalendar[$i]['day'] == '') {
  685.                 $targetMonthCalendar[$i]['holiday'] = false;
  686.                 $targetMonthCalendar[$i]['today'] = false;
  687.                 continue;
  688.             }
  689.             $targetYmdDateTime = new \DateTime($targetDate->copy()->format('Y-n') . '-' $targetMonthCalendar[$i]['day']); //対象日
  690.             //dump($targetYmdDateTime);
  691.             // カレンダーの日付が定休日リストに存在するかを確認
  692.             $result array_search($targetYmdDateTime$holidayListOfTwoMonths);
  693.             //dump($holidayListOfTwoMonths);
  694.             // 定休日フラグを設定
  695.             if ($result !== false) {
  696.                 $targetMonthCalendar[$i]['holiday'] = true;
  697.                 //定休日は予約不可
  698.                 foreach ($area as $targetarea) {
  699.                     $variable $targetarea['variable'];
  700.                     $targetMonthCalendar[$i]['reservation'][$variable]['AM'] = $reservationNot;
  701.                     $targetMonthCalendar[$i]['reservation'][$variable]['PM'] = $reservationNot;
  702.                 }
  703.             } else {
  704.                 $targetMonthCalendar[$i]['holiday'] = false;
  705.             }
  706.             // 今日フラグを設定
  707.             if ($targetYmdDateTime == new \DateTime($targetDate->copy()->format('Y-n-j'))) {
  708.                 $targetMonthCalendar[$i]['today'] = true;
  709.             } else {
  710.                 $targetMonthCalendar[$i]['today'] = false;
  711.             }
  712.             //対象日が日曜日なら予約不可
  713.             if ($targetMonthCalendar[$i]['dayOfWeek'] === 'Sun') {
  714.                 foreach ($area as $targetarea) {
  715.                     $variable $targetarea['variable'];
  716.                     $targetMonthCalendar[$i]['reservation'][$variable]['AM'] = $reservationNot;
  717.                     $targetMonthCalendar[$i]['reservation'][$variable]['PM'] = $reservationNot;
  718.                 }
  719.             }
  720.             //対象日が祝日なら予約不可
  721.             //dump($targetYmdDateTime);
  722.             //dump($japanHoliday);
  723.             $japanHolidayjudgement array_search($targetYmdDateTime$japanHoliday);
  724.             if ($japanHolidayjudgement !== false) {
  725.                 $targetMonthCalendar[$i]['holiday'] = true;
  726.                 foreach ($area as $targetarea) {
  727.                     $variable $targetarea['variable'];
  728.                     $targetMonthCalendar[$i]['reservation'][$variable]['AM'] = $reservationNot;
  729.                     $targetMonthCalendar[$i]['reservation'][$variable]['PM'] = $reservationNot;
  730.                 }
  731.             }
  732.         }
  733.         log_debug('setReservationHoliDay end');
  734.         //dump($targetMonthCalendar);
  735.         return $targetMonthCalendar;
  736.     }
  737.     /**
  738.      * ECCUBEカレンダーの月初から本日までを予約不可にする
  739.      * 在庫がある商品は本日から予約可能にする。
  740.      * 在庫がない商品は本日から4日までを予約不可にする。
  741.      * 在庫がない場合は在庫の取り寄せ状況を考慮して祝日がある場合は予約不可日数を増やす
  742.      * 在庫あり・なしは$bakusokuflgで判断する。
  743.      * 
  744.      * @param array $targetMonthCalendar 対象月のカレンダー
  745.      * @param string $bakusokuflg 在庫がある場合はtrue、ない場合はfalse 
  746.      * @param Carbon $targetDate 本日の日付と時刻
  747.      * @param string $targetDateCopy 本日。比較対象のため0時にセットする
  748.      * @param string $noReservationDayStart 予約不可日開始日(月初)
  749.      * @param string $noReservationDayEnd 予約不可日終了日、爆速商品で値が変わる為、初期値は前日
  750.      * @param array $japanHoliday 祝日と定休日をまとめた配列。本日〜4日以内に祝日があるか判断する為
  751.      * @param Carbon $amTenFlg 本日のAM10時。現在時刻が10時か比較する為
  752.      * @param string $today 本日。現在時刻が10時か比較する為
  753.      * @param Carbon $noReservationDayEnd 爆速商品以外の場合に不可にする日付の最終日、来月用に仕様する。
  754.      */
  755.     public function
  756.     setNotReservationToday(&$targetMonthCalendar$bakusokuflg$area$holidayListOfTwoMonths$holidayCarbon $targetDate$reservationNot, &$notReservationDay, &$noReservationDayEnd)
  757.     {
  758.         log_debug("setNotReservationToday start");
  759.         //dump($targetDate);
  760.         $targetDateCopy $targetDate->copy()->startOfDay(); //本日。比較対象のため0時にセットする
  761.         $noReservationDayStart $targetDateCopy->copy()->startOfMonth(); //予約不可日開始日(月初)
  762.         $noReservationDayEnd $targetDateCopy->copy()->subDay(); //予約不可日終了日、爆速商品で値が変わる為、初期値は前日
  763.         $today $targetDate->format("Y-m-d"); //本日。比較対象のため0時にセットする
  764.         $amTenFlg $targetDate->copy()->hour(10); // 10時のCarbonオブジェクトを作成
  765.         //dump($noReservationDayStart);
  766.         //dump($noReservationDayEnd);
  767.         //dump($holidayListOfTwoMonths);
  768.         //dump($holiday);
  769.         //ECCUBEカレンダーがDATE型の為、祝日は型変換を行う     
  770.         $japanHoliday = array();
  771.         foreach ($holiday as $day) {
  772.             $japanHoliday[] = Carbon::parse($day)->toDateTime()->setTime(000);
  773.         }
  774.         //dump($japanHoliday);
  775.         //判定の為、定休日と祝日をまとめる
  776.         $japanHoliday array_merge($japanHoliday$holidayListOfTwoMonths);
  777.         //爆速商品(在庫ありかなしか)に合わせて予約不可日を設定する為
  778.         //dump($bakusokuflg);
  779.         if ($bakusokuflg === true) {
  780.             log_info('爆速商品ではない');
  781.             $addDaysCounter 0//予約可能日を当日+平日4日分の日にちにするため
  782.             while ($addDaysCounter 4) {
  783.                 dump($addDaysCounter);
  784.                 $noReservationDayEnd->addDays(1);
  785.                 $dateJudge $noReservationDayEnd->toDate(); //祝日判定用
  786.                 //dump($noReservationDayEnd);
  787.                 //dump($dateJudge);
  788.                 //dump($japanHoliday);
  789.                 //土日の場合
  790.                 if ($noReservationDayEnd->dayOfWeek === Carbon::SATURDAY || $noReservationDayEnd->dayOfWeek === Carbon::SUNDAY) {
  791.                     log_debug($noReservationDayEnd 'は土日なのでカウントしない');
  792.                 }
  793.                 //祝日、定休日の場合、指定日が土日で祝日の場合はカウントしない
  794.                 elseif (array_search($dateJudge$japanHoliday) !== false) {
  795.                     log_debug($noReservationDayEnd 'は祝日,定休日なのでカウントしない');
  796.                 } else {
  797.                     log_debug($noReservationDayEnd 'は平日なのでカウントする');
  798.                     $addDaysCounter++;
  799.                 }
  800.             }
  801.         } elseif ($bakusokuflg === false) {
  802.             log_info('爆速商品');
  803.         } else {
  804.             $errormsg __LINE__ ':爆速商品の判断が出来ません';
  805.             log_error($errormsg);
  806.             $this->sendErrorNotifi->errorNotifi($errormsg);
  807.         }
  808.         $dayEnd $noReservationDayEnd->copy()->addDays()->toDateString();
  809.         //予約不可日を作成する
  810.         $period CarbonPeriod::create($noReservationDayStart$noReservationDayEnd);
  811.         log_notice($noReservationDayStart->toString() . 'から' $noReservationDayEnd->toString() . 'までは予約不可にする');
  812.         foreach ($period as $periodDate) {
  813.             $notReservationDay[] = $periodDate->toDate();
  814.         }
  815.         dump($noReservationDayEnd);
  816.         //カレンダー作成
  817.         for ($i 0$i count($targetMonthCalendar); $i++) {
  818.             // カレンダー配列の日が空の場合は処理をスキップ
  819.             if ($targetMonthCalendar[$i]['day'] == '') {
  820.                 $targetMonthCalendar[$i]['holiday'] = false;
  821.                 $targetMonthCalendar[$i]['today'] = false;
  822.                 continue;
  823.             }
  824.             $targetYmdDateTime = new \DateTime($targetDate->copy()->format('Y-n') . '-' $targetMonthCalendar[$i]['day']); //対象日
  825.             //dump($targetYmdDateTime);
  826.             // カレンダーの日付がお休みか確認
  827.             $result array_search($targetYmdDateTime$notReservationDay);
  828.             //dump($targetMonthCalendar[$i]);
  829.             if ($result !== false) {
  830.                 //予約不可設定
  831.                 foreach ($area as $targetarea) {
  832.                     $variable $targetarea['variable'];
  833.                     $targetMonthCalendar[$i]['reservation'][$variable]['AM'] = $reservationNot;
  834.                     $targetMonthCalendar[$i]['reservation'][$variable]['PM'] = $reservationNot;
  835.                 }
  836.             }
  837.             //現在時刻が10時前後で本日の予約状況を変える為
  838.             $targetYmdDateTime $targetYmdDateTime->format("Y-m-d"); //比較のため
  839.             if ($targetYmdDateTime === $today) {
  840.                 //dump('本日判断');
  841.                 if ($targetDate->lessThan($amTenFlg)) {
  842.                     //dump("AM10時前");
  843.                     foreach ($area as $targetarea) {
  844.                         $variable $targetarea['variable'];
  845.                         $targetMonthCalendar[$i]['reservation'][$variable]['AM'] = $reservationNot;
  846.                     }
  847.                 } else {
  848.                     log_debug("AM10時以降");
  849.                     foreach ($area as $targetarea) {
  850.                         $variable $targetarea['variable'];
  851.                         $targetMonthCalendar[$i]['reservation'][$variable]['AM'] = $reservationNot;
  852.                         $targetMonthCalendar[$i]['reservation'][$variable]['PM'] = $reservationNot;
  853.                     }
  854.                 }
  855.             }
  856.             //4日目以降の処理はしなくて良いため
  857.             if ($targetMonthCalendar[$i]['yearMonthDay'] === $dayEnd) {
  858.                 log_debug("setNotReservationToday end");
  859.                 return;
  860.             }
  861.         }
  862.         log_debug("setNotReservationToday end");
  863.         return;
  864.     }
  865.     /**
  866.      * 来月用ECCUBEカレンダーの月初から予約不可日までを予約不可にする
  867.      * 
  868.      * @param array $targetMonthCalendar 来月のカレンダー
  869.      * @param Carbon $targetDate 本日の日付と時刻
  870.      * @param array $notReservationDay 予約不可日
  871.      */
  872.     public function
  873.     setNotReservationNextCalendar($targetMonthCalendar$areaCarbon $targetDate$reservationNot$notReservationDay$noReservationDayEnd)
  874.     {
  875.         log_debug("setNotReservationNextCalendar start");
  876.         //$targetDate = $targetDate->copy()->addMonth();
  877.         $targetDate $targetDate->copy()->startOfMonth()->addMonth(); //うるう年回避のため、月初にしてから1月加算する
  878.         $lastKey array_key_last($notReservationDay);
  879.         $dayEnd $notReservationDay[$lastKey];
  880.         $noReservationDayStart $targetDate->copy()->startOfMonth(); //予約不可日開始日(月初)
  881.         //予約不可日を作成する
  882.         $period CarbonPeriod::create($noReservationDayStart$noReservationDayEnd);
  883.         log_notice($noReservationDayStart->toString() . 'から' $noReservationDayEnd->toString() . 'までは予約不可にする');
  884.         foreach ($period as $periodDate) {
  885.             $notReservationDay[] = $periodDate->toDate();
  886.         }
  887.         //カレンダー作成
  888.         for ($i 0$i count($targetMonthCalendar); $i++) {
  889.             // カレンダー配列の日が空の場合は処理をスキップ
  890.             if ($targetMonthCalendar[$i]['day'] == '') {
  891.                 $targetMonthCalendar[$i]['holiday'] = false;
  892.                 $targetMonthCalendar[$i]['today'] = false;
  893.                 continue;
  894.             }
  895.             $targetYmdDateTime = new \DateTime($targetDate->copy()->format('Y-n') . '-' $targetMonthCalendar[$i]['day']); //対象日
  896.             // カレンダーの日付がお休みか確認
  897.             dump($targetYmdDateTime);
  898.             $result array_search($targetYmdDateTime$notReservationDay);
  899.             //dump($targetMonthCalendar[$i]);
  900.             if ($result !== false) {
  901.                 //予約不可設定
  902.                 foreach ($area as $targetarea) {
  903.                     $variable $targetarea['variable'];
  904.                     $targetMonthCalendar[$i]['reservation'][$variable]['AM'] = $reservationNot;
  905.                     $targetMonthCalendar[$i]['reservation'][$variable]['PM'] = $reservationNot;
  906.                 }
  907.             }
  908.             //4日目以降の処理はしなくて良いため
  909.             if ($targetMonthCalendar[$i]['yearMonthDay'] === $dayEnd) {
  910.                 log_debug("setNotReservationToday end");
  911.                 return $targetMonthCalendar;
  912.             }
  913.         }
  914.         log_debug("setNotReservationNextCalendar end");
  915.         return $targetMonthCalendar;
  916.     }
  917. }