app/Customize/Controller/ProductController.php line 175

Open in your IDE?
  1. <?php
  2. /*
  3.  * This file is part of EC-CUBE
  4.  *
  5.  * Copyright(c) EC-CUBE CO.,LTD. All Rights Reserved.
  6.  *
  7.  * http://www.ec-cube.co.jp/
  8.  *
  9.  * For the full copyright and license information, please view the LICENSE
  10.  * file that was distributed with this source code.
  11.  */
  12. namespace Customize\Controller;
  13. use Customize\Entity\Floor;
  14. use Customize\Repository\BrandRepository;
  15. use Customize\Repository\MediaCoverageRepository;
  16. use Customize\Service\ArticleService;
  17. use Eccube\Controller\ProductController as ParentController;
  18. use Customize\Repository\ProductRepository;
  19. use Customize\Repository\ProductRelatedRepository;
  20. use Customize\Entity\ProductRelated;
  21. use Customize\Service\FloorDistributionService;
  22. use Eccube\Entity\BaseInfo;
  23. use Eccube\Entity\Master\ProductStatus;
  24. use Eccube\Entity\Product;
  25. use Eccube\Event\EccubeEvents;
  26. use Eccube\Event\EventArgs;
  27. use Eccube\Form\Type\AddCartType;
  28. use Eccube\Form\Type\Master\ProductListMaxType;
  29. use Eccube\Form\Type\Master\ProductListOrderByType;
  30. use Eccube\Form\Type\SearchProductType;
  31. use Eccube\Repository\BaseInfoRepository;
  32. use Customize\Repository\CustomerFavoriteProductRepository;
  33. use Eccube\Repository\Master\ProductListMaxRepository;
  34. use Eccube\Service\CartService;
  35. use Customize\Service\ProductService;
  36. use Eccube\Entity\Master\SaleType;
  37. use Eccube\Entity\ProductClass;
  38. use Eccube\Service\PurchaseFlow\PurchaseContext;
  39. use Eccube\Service\PurchaseFlow\PurchaseFlow;
  40. use Knp\Bundle\PaginatorBundle\Pagination\SlidingPagination;
  41. use Knp\Component\Pager\PaginatorInterface;
  42. use Sensio\Bundle\FrameworkExtraBundle\Configuration\ParamConverter;
  43. use Sensio\Bundle\FrameworkExtraBundle\Configuration\Template;
  44. use Symfony\Component\HttpFoundation\Request;
  45. use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
  46. use Symfony\Component\Routing\Annotation\Route;
  47. use Symfony\Component\Routing\Generator\UrlGeneratorInterface;
  48. use Symfony\Component\Security\Http\Authentication\AuthenticationUtils;
  49. use GuzzleHttp\Client;
  50. use Eccube\Repository\Master\SaleTypeRepository;
  51. class ProductController extends ParentController
  52. {
  53.     /**
  54.      * @var PurchaseFlow
  55.      */
  56.     protected $purchaseFlow;
  57.     /**
  58.      * @var CustomerFavoriteProductRepository
  59.      */
  60.     protected $customerFavoriteProductRepository;
  61.     /**
  62.      * @var CartService
  63.      */
  64.     protected $cartService;
  65.     /**
  66.      * @var ProductService
  67.      */
  68.     protected $productService;
  69.     /**
  70.      * @var ProductRepository
  71.      */
  72.     protected $productRepository;
  73.     /**
  74.      * @var BaseInfo
  75.      */
  76.     protected $BaseInfo;
  77.     /**
  78.      * @var AuthenticationUtils
  79.      */
  80.     protected $helper;
  81.     /**
  82.      * @var ProductListMaxRepository
  83.      */
  84.     protected $productListMaxRepository;
  85.     private $title '';
  86.     private $articleService;
  87.     /** @var mediaCoverageRepository */
  88.     private $mediaCoverageRepository;
  89.     /** @var FloorDistributionService */
  90.     private $floorDistributionService;
  91.     /** @var BrandRepository */
  92.     private $brandRepository;
  93.     /** @var ProductRelatedRepository */
  94.     private $productRelatedRepository;
  95.     /** @var SaleTypeRepository */
  96.     private $saleTypeRepository;
  97.     /**
  98.      * ProductController constructor.
  99.      *
  100.      * @param PurchaseFlow $cartPurchaseFlow
  101.      * @param CustomerFavoriteProductRepository $customerFavoriteProductRepository
  102.      * @param CartService $cartService
  103.      * @param ProductService $productService
  104.      * @param ProductRepository $productRepository
  105.      * @param BaseInfoRepository $baseInfoRepository
  106.      * @param AuthenticationUtils $helper
  107.      * @param ProductListMaxRepository $productListMaxRepository
  108.      * @param ArticleService $articleService
  109.      * @param mediaCoverageRepository $mediaCoverageRepository
  110.      * @param FloorDistributionService $floorDistributionService
  111.      * @param BrandRepository $brandRepository
  112.      * @param ProductRelatedRepository $productRelatedRepository
  113.      * @param SaleTypeRepository $saleTypeRepository
  114.      */
  115.     public function __construct(
  116.         PurchaseFlow $cartPurchaseFlow,
  117.         CustomerFavoriteProductRepository $customerFavoriteProductRepository,
  118.         CartService $cartService,
  119.         ProductService $productService,
  120.         ProductRepository $productRepository,
  121.         BaseInfoRepository $baseInfoRepository,
  122.         AuthenticationUtils $helper,
  123.         ProductListMaxRepository $productListMaxRepository,
  124.         ArticleService $articleService,
  125.         MediaCoverageRepository $mediaCoverageRepository,
  126.         FloorDistributionService $floorDistributionService,
  127.         BrandRepository $brandRepository,
  128.         ProductRelatedRepository $productRelatedRepository,
  129.         SaleTypeRepository $saleTypeRepository
  130.     ) {
  131.         $this->purchaseFlow $cartPurchaseFlow;
  132.         $this->customerFavoriteProductRepository $customerFavoriteProductRepository;
  133.         $this->cartService $cartService;
  134.         $this->productService $productService;
  135.         $this->productRepository $productRepository;
  136.         $this->BaseInfo $baseInfoRepository->get();
  137.         $this->helper $helper;
  138.         $this->productListMaxRepository $productListMaxRepository;
  139.         $this->articleService $articleService;
  140.         $this->mediaCoverageRepository $mediaCoverageRepository;
  141.         $this->floorDistributionService $floorDistributionService;
  142.         $this->brandRepository $brandRepository;
  143.         $this->productRelatedRepository $productRelatedRepository;
  144.         $this->saleTypeRepository $saleTypeRepository;
  145.     }
  146.     /**
  147.      * 商品一覧画面.
  148.      *
  149.      * @Route("/products/list", name="product_list")
  150.      * @Route("/{sale_type}/products/list", name="valuet_product_list")
  151.      * @Route("/{sale_type}/{floor_key}/{brand_identifier}/products/list", name="brand_product_list")
  152.      * @Template("Product/list.twig")
  153.      */
  154.     public function index(Request $requestPaginatorInterface $paginator$sale_type null$floor_key null$brand_identifier null)
  155.     {
  156.         $Brand null;
  157.         if ($floor_key && $brand_identifier) {
  158.             $floor_name FloorDistributionService::floors[$floor_key] ?? null;
  159.             if (!$floor_name) {
  160.                 throw new NotFoundHttpException();
  161.             }
  162.             /** @var \Customize\Entity\Brand $Brand */
  163.             $Brand $this->brandRepository->findOneBy(['identifier' => $brand_identifier]);
  164.             if (!$Brand) {
  165.                 throw new NotFoundHttpException();
  166.             }
  167.             if ($Brand->getFloor()->getName() !== $floor_name) {
  168.                 throw new NotFoundHttpException();
  169.             }
  170.         }
  171.         // Doctrine SQLFilter
  172.         if ($this->BaseInfo->isOptionNostockHidden()) {
  173.             $this->entityManager->getFilters()->enable('option_nostock_hidden');
  174.         }
  175.         // handleRequestは空のqueryの場合は無視するため
  176.         if ($request->getMethod() === 'GET') {
  177.             $request->query->set('pageno'$request->query->get('pageno'''));
  178.         }
  179.         // searchForm
  180.         /* @var $builder \Symfony\Component\Form\FormBuilderInterface */
  181.         $builder $this->formFactory->createNamedBuilder(''SearchProductType::class);
  182.         if ($request->getMethod() === 'GET') {
  183.             $builder->setMethod('GET');
  184.         }
  185.         $event = new EventArgs(
  186.             [
  187.                 'builder' => $builder,
  188.             ],
  189.             $request
  190.         );
  191.         $this->eventDispatcher->dispatch($eventEccubeEvents::FRONT_PRODUCT_INDEX_INITIALIZE);
  192.         /* @var $searchForm \Symfony\Component\Form\FormInterface */
  193.         $searchForm $builder->getForm();
  194.         $searchForm->handleRequest($request);
  195.         // paginator
  196.         $searchData $searchForm->getData();
  197.         if ($Brand) {
  198.             $searchData['brand_id'] = $Brand;
  199.         }
  200.         // 銘店の一品の商品を含むかの判定
  201.         $templatePath $this->floorDistributionService->getFloorTemplateCodeFromPath();
  202.         switch ($templatePath) {
  203.             case Floor::NAME_SPECIAL:
  204.                 $qb $this->productRepository->getQueryBuilderBySearchData($searchData$sale_typeProductClass::SHOW_AT_SPECIAL_ONLY);
  205.                 break;
  206.             case null// 百貨店にいる場合
  207.                 $qb $this->productRepository->getQueryBuilderBySearchData($searchData$sale_typeProductClass::SHOW_AT_BOTH);
  208.                 break;
  209.             default: // 銘店の一品以外の各フロアにいる場合
  210.                 $qb $this->productRepository->getQueryBuilderBySearchData($searchData$sale_typeProductClass::SHOW_AT_FLOOR_ONLY);
  211.                 break;
  212.         }
  213.         $event = new EventArgs(
  214.             [
  215.                 'searchData' => $searchData,
  216.                 'qb' => $qb,
  217.             ],
  218.             $request
  219.         );
  220.         $this->eventDispatcher->dispatch($eventEccubeEvents::FRONT_PRODUCT_INDEX_SEARCH);
  221.         $searchData $event->getArgument('searchData');
  222.         $query $qb->getQuery()
  223.             ->useResultCache(true$this->eccubeConfig['eccube_result_cache_lifetime_short']);
  224.         /** @var SlidingPagination $pagination */
  225.         $pagination $paginator->paginate(
  226.             $query,
  227.             !empty($searchData['pageno']) ? $searchData['pageno'] : 1,
  228.             !empty($searchData['disp_number']) ? $searchData['disp_number']->getId() : $this->productListMaxRepository->findOneBy([],
  229.             ['sort_no' => 'ASC'])->getId(),
  230.             array('wrap-queries' => true)
  231.         );
  232.         // 品切れの商品を後ろに並び替える
  233.         $this->productService->sortProductsByOutOfStock($pagination);
  234.         $ids = [];
  235.         foreach ($pagination as $Product) {
  236.             $ids[] = $Product->getId();
  237.         }
  238.         $ProductsAndClassCategories $this->productRepository->findProductsWithSortedClassCategories($ids'p.id');
  239.         // addCart form
  240.         $forms = [];
  241.         $Toppings = [];
  242.         foreach ($pagination as $Product) {
  243.             /* @var $builder \Symfony\Component\Form\FormBuilderInterface */
  244.             $builder $this->formFactory->createNamedBuilder(
  245.                 '',
  246.                 AddCartType::class,
  247.                 null,
  248.                 [
  249.                     'product' => $ProductsAndClassCategories[$Product->getId()],
  250.                     'allow_extra_fields' => true,
  251.                     'sale_type' => $sale_type,
  252.                 ]
  253.             );
  254.             $addCartForm $builder->getForm();
  255.             $forms[$Product->getId()] = $addCartForm->createView();
  256.             $ToppingGroup $Product->getToppingGroup();
  257.             if ($ToppingGroup) {
  258.                 $ToppingGroupToppings $ToppingGroup->getToppingGroupToppings();
  259.                 foreach ($ToppingGroupToppings as $ToppingGroupTopping) {
  260.                     $Topping $ToppingGroupTopping->getProduct();
  261.                     // 在庫がない、公開ではでない
  262.                     if (!$Topping->getStockFind() || !$Topping->isEnable()) {
  263.                         continue;
  264.                     }
  265.                     $Toppings[$Product->getId()][] = $Topping;
  266.                 }
  267.             }
  268.         }
  269.         // 表示件数
  270.         $builder $this->formFactory->createNamedBuilder(
  271.             'disp_number',
  272.             ProductListMaxType::class,
  273.             null,
  274.             [
  275.                 'required' => false,
  276.                 'allow_extra_fields' => true,
  277.             ]
  278.         );
  279.         if ($request->getMethod() === 'GET') {
  280.             $builder->setMethod('GET');
  281.         }
  282.         $event = new EventArgs(
  283.             [
  284.                 'builder' => $builder,
  285.             ],
  286.             $request
  287.         );
  288.         $this->eventDispatcher->dispatch($eventEccubeEvents::FRONT_PRODUCT_INDEX_DISP);
  289.         $dispNumberForm $builder->getForm();
  290.         $dispNumberForm->handleRequest($request);
  291.         // ソート順
  292.         $builder $this->formFactory->createNamedBuilder(
  293.             'orderby',
  294.             ProductListOrderByType::class,
  295.             null,
  296.             [
  297.                 'required' => false,
  298.                 'allow_extra_fields' => true,
  299.             ]
  300.         );
  301.         if ($request->getMethod() === 'GET') {
  302.             $builder->setMethod('GET');
  303.         }
  304.         $event = new EventArgs(
  305.             [
  306.                 'builder' => $builder,
  307.             ],
  308.             $request
  309.         );
  310.         $this->eventDispatcher->dispatch($eventEccubeEvents::FRONT_PRODUCT_INDEX_ORDER);
  311.         $orderByForm $builder->getForm();
  312.         $orderByForm->handleRequest($request);
  313.         $Category $searchForm->get('category_id')->getData();
  314.         // カテゴリ人気ランキング
  315.         $CategoryRankedProducts $this->getCategoryRankedProducts($Category);
  316.         //Yazawaのカテゴリ記事機能
  317.         $name $this->floorDistributionService->getFloorTemplateCodeFromPath();
  318.         $Articles $Category$this->articleService->getConditionFilteredArticles($Category->getId(), $name): null;
  319.         $Article $Category$this->articleService->getCategoryTopArticle($Category->getId(), $name): null;
  320.         return [
  321.             'subtitle' => $this->getPageTitle($searchData),
  322.             'pagination' => $pagination,
  323.             'search_form' => $searchForm->createView(),
  324.             'disp_number_form' => $dispNumberForm->createView(),
  325.             'order_by_form' => $orderByForm->createView(),
  326.             'forms' => $forms,
  327.             'Category' => $Category,
  328.             'CategoryRankedProducts' => $CategoryRankedProducts,
  329.             'Articles' => $Articles,
  330.             'Article' => $Article,
  331.             'Toppings' => $Toppings,
  332.             'sale_type' => $sale_type,
  333.             'floor_key' => $floor_key,
  334.             'brand_identifier' => $brand_identifier,
  335.             'Brand' => $Brand,
  336.         ];
  337.     }
  338.     /**
  339.      * 商品詳細画面.
  340.      *
  341.      * @Route("/products/detail/{id}", name="product_detail", methods={"GET"}, requirements={"id" = "\d+"})
  342.      * @Route("/{sale_type}/products/detail/{id}", name="valuet_product_detail")
  343.      * @Template("Product/detail.twig")
  344.      * @ParamConverter("Product", options={"repository_method" = "findWithSortedClassCategories"})
  345.      *
  346.      * @param Request $request
  347.      * @param Product $Product
  348.      *
  349.      * @return array
  350.      */
  351.     public function detail(Request $requestProduct $Product$sale_type null)
  352.     {
  353.         // 銘店の一品にリダイレクトするかの判定を実施
  354.         $templatePath $this->floorDistributionService->getFloorTemplateCodeFromPath();
  355.         $ProductClass $Product->getProductClasses()[0];
  356.         switch ($templatePath) {
  357.             case Floor::NAME_SPECIAL:
  358.                 if ($ProductClass->getShowAtSpecial() != ProductClass::SHOW_AT_SPECIAL_ONLY) {
  359.                     return $this->redirectTo($requestarray_flip(FloorDistributionService::floors)[$Product->getShopName()] );
  360.                 }
  361.                 break;
  362.             case null// 百貨店にいる場合
  363.                 if ($ProductClass->getShowAtSpecial() == ProductClass::SHOW_AT_SPECIAL_ONLY) {
  364.                     return $this->redirectTo($requestFloor::NAME_SPECIAL);
  365.                 } else {
  366.                     // デリテイクはリダイレクトしない
  367.                     if (!in_array($sale_type, array("delivery","takeout"))) {
  368.                         return $this->redirectTo($requestarray_flip(FloorDistributionService::floors)[$Product->getShopName()] );
  369.                     }
  370.                 }
  371.                 break;
  372.             default: // 銘店の一品以外の各フロアにいる場合
  373.                 if ($ProductClass->getShowAtSpecial() == ProductClass::SHOW_AT_SPECIAL_ONLY) {
  374.                     return $this->redirectTo($requestFloor::NAME_SPECIAL);
  375.                 }
  376.                 break;
  377.         }
  378.         if (!$this->checkVisibility($Product)) {
  379.             throw new NotFoundHttpException();
  380.         }
  381.         if (!empty($sale_type)) {
  382.             $this->checkProductSaleTypeAndPathCombination($Product$sale_type);
  383.         }
  384.         // トッピング商品は表示不可
  385.         $SaleType $Product->getProductClasses()[0]->getSaleType();
  386.         if ($SaleType->getId() === SaleType::TOPPING) {
  387.             throw new NotFoundHttpException();
  388.         }
  389.         $builder $this->formFactory->createNamedBuilder(
  390.             '',
  391.             AddCartType::class,
  392.             null,
  393.             [
  394.                 'product' => $Product,
  395.                 'id_add_product_id' => false,
  396.                 'sale_type' => $sale_type,
  397.             ]
  398.         );
  399.         $Toppings = [];
  400.         // if ($sale_type !== 'onlineshop') {
  401.             $ToppingGroup $Product->getToppingGroup();
  402.             if ($ToppingGroup) {
  403.                 $ToppingGroupToppings $ToppingGroup->getToppingGroupToppings();
  404.                 foreach ($ToppingGroupToppings as $ToppingGroupTopping) {
  405.                     $Topping $ToppingGroupTopping->getProduct();
  406. // saleTypeがToppingのみ追加
  407.                     if ($Topping->getProductClasses()[0]->getSaleType()->getId() === SaleType::TOPPING) {
  408.                         // 在庫がない、公開ではでない
  409.                         if (!$Topping->getStockFind() || !$Topping->isEnable()) {
  410.                             continue;
  411.                         }
  412.                         $Toppings[$Product->getId()][] = $Topping;
  413.                     }
  414.                 }
  415.             }
  416.         // }
  417.         $event = new EventArgs(
  418.             [
  419.                 'builder' => $builder,
  420.                 'Product' => $Product,
  421.             ],
  422.             $request
  423.         );
  424.         $this->eventDispatcher->dispatch($eventEccubeEvents::FRONT_PRODUCT_DETAIL_INITIALIZE);
  425.         $is_favorite false;
  426.         if ($this->isGranted('ROLE_USER')) {
  427.             $Customer $this->getUser();
  428.             $is_favorite $this->customerFavoriteProductRepository->isFavorite($Customer$Product);
  429.         }
  430.         // カテゴリナビ内に色をつけるためカテゴリの配列を作成
  431.         $ProductCategories $this->productService->productCategories($Product);
  432.         $Brand $Product->getBrand();
  433.         // 関連するメディア掲載履歴を取得
  434.         $MediaCoverages $this->mediaCoverageRepository->getByProductId($Product);
  435.         $ProductCategoriesArray $this->productRepository->getProductCategoriesArray($Product);
  436.         return [
  437.             'title' => $this->title,
  438.             'subtitle' => $Product->getName(),
  439.             'form' => $builder->getForm()->createView(),
  440.             'Product' => $Product,
  441.             'ProductCategories' => $ProductCategories,
  442.             'is_favorite' => $is_favorite,
  443.             'Toppings' => $Toppings,
  444.             'Brand' => $Brand,
  445.             'sale_type' => $sale_type,
  446.             'MediaCoverages' => $MediaCoverages,
  447.             'ProductCategoriesArray' => $ProductCategoriesArray,
  448.         ];
  449.     }
  450.     private function redirectTo(Request $requeststring $floorKey) {
  451.         $params $request->attributes->get('_route_params');
  452.         $params array_merge($params$request->query->all());
  453.         unset($params['sale_type']);
  454.         $url $this->generateUrl('product_detail'$params);
  455.         return $this->redirect('/' $floorKey $url301);
  456.     }
  457.     /**
  458.      * お気に入り追加.
  459.      *
  460.      * @Route("/products/add_favorite/{id}", name="product_add_favorite", requirements={"id" = "\d+"})
  461.      */
  462.     public function addFavorite(Request $requestProduct $Product)
  463.     {
  464.         $this->checkVisibility($Product);
  465.         $event = new EventArgs(
  466.             [
  467.                 'Product' => $Product,
  468.             ],
  469.             $request
  470.         );
  471.         $this->eventDispatcher->dispatch($eventEccubeEvents::FRONT_PRODUCT_FAVORITE_ADD_INITIALIZE);
  472.         $sale_type $request->get('sale_type');
  473.         if ($this->isGranted('ROLE_USER')) {
  474.             $Customer $this->getUser();
  475.             $saleTypes = [
  476.                 'delivery' => SaleType::DELIVERY,
  477.                 'takeout' => SaleType::TAKEOUT,
  478.             ];
  479.             $SaleType null;
  480.             if (array_key_exists($sale_type$saleTypes)) {
  481.                 $SaleType $this->saleTypeRepository->find($saleTypes[$sale_type]);
  482.             } else {
  483.                 foreach ($Product->getProductClasses() as $ProductClass) {
  484.                     if ($ProductClass->getSaleType()) {
  485.                         $SaleType $ProductClass->getSaleType();
  486.                         break;
  487.                     }
  488.                 }
  489.             }
  490.             $this->customerFavoriteProductRepository->addFavoriteWithSaleType($Customer$Product$SaleType);
  491.             $this->session->getFlashBag()->set('product_detail.just_added_favorite'$Product->getId());
  492.             $event = new EventArgs(
  493.                 [
  494.                     'Product' => $Product,
  495.                 ],
  496.                 $request
  497.             );
  498.             $this->eventDispatcher->dispatch($eventEccubeEvents::FRONT_PRODUCT_FAVORITE_ADD_COMPLETE);
  499.             return $this->redirectToRoute('valuet_product_detail', ['id' => $Product->getId(), 'sale_type' => $sale_type]);
  500.         } else {
  501.             // 非会員の場合、ログイン画面を表示
  502.             //  ログイン後の画面遷移先を設定
  503.             $this->setLoginTargetPath($this->generateUrl('product_add_favorite', ['id' => $Product->getId()], UrlGeneratorInterface::ABSOLUTE_URL));
  504.             $this->session->getFlashBag()->set('eccube.add.favorite'true);
  505.             $event = new EventArgs(
  506.                 [
  507.                     'Product' => $Product,
  508.                 ],
  509.                 $request
  510.             );
  511.             $this->eventDispatcher->dispatch($eventEccubeEvents::FRONT_PRODUCT_FAVORITE_ADD_COMPLETE);
  512.             return $this->redirectToRoute('mypage_login');
  513.         }
  514.     }
  515.     /**
  516.      * sale_typeと商品のもつ販売種別が正しいかどうかをチェック
  517.      *
  518.      * @param Product $Product
  519.      * @param string|null $sale_type delivery / takeout / onlineshop
  520.      * @return void
  521.      */
  522.     private function checkProductSaleTypeAndPathCombination(Product $Product$sale_type null)
  523.     {
  524.         /** @var ProductClass $ProductClass */
  525.         foreach ($Product->getProductClasses() as $ProductClass) {
  526.             $SaleType $ProductClass->getSaleType();
  527.             switch ($sale_type) {
  528.                 case 'onlineshop':
  529.                     if (in_array($SaleType->getId(), [SaleType::DELIVERYSaleType::TAKEOUTSaleType::DELIVERY_TAKEOUTSaleType::TOPPING])) {
  530.                         // sale_typeがonlineshopで商品がデリバリー、テイクアウト、デリバリー / テイクアウト、トッピングであればException
  531.                         throw new NotFoundHttpException();
  532.                     }
  533.                     break;
  534.                 case 'delivery':
  535.                     if (!in_array($SaleType->getId(), [SaleType::DELIVERYSaleType::DELIVERY_TAKEOUT])) {
  536.                         // sale_typeがdeliveryで商品がデリバリー、デリバリー / テイクアウトでなければException
  537.                         throw new NotFoundHttpException();
  538.                     }
  539.                     break;
  540.                 case 'takeout':
  541.                     if (!in_array($SaleType->getId(), [SaleType::TAKEOUTSaleType::DELIVERY_TAKEOUT])) {
  542.                         // sale_typeがtakeoutで商品がテイクアウト、デリバリー / テイクアウトでなければException
  543.                         throw new NotFoundHttpException();
  544.                     }
  545.                     break;
  546.             }
  547.         }
  548.     }
  549.     /**
  550.      * 百貨店用商品検索画面.
  551.      *
  552.      * @Route("/search", name="product_search")
  553.      * @Template("Product/search.twig")
  554.      */
  555.     public function search(Request $requestPaginatorInterface $paginator)
  556.     {
  557.         // Doctrine SQLFilter
  558.         if ($this->BaseInfo->isOptionNostockHidden()) {
  559.             $this->entityManager->getFilters()->enable('option_nostock_hidden');
  560.         }
  561.         // searchForm
  562.         /* @var $builder \Symfony\Component\Form\FormBuilderInterface */
  563.         $builder $this->formFactory->createNamedBuilder(''SearchProductType::class);
  564.         if ($request->getMethod() === 'GET') {
  565.             $builder->setMethod('GET');
  566.         }
  567.         /* @var $searchForm \Symfony\Component\Form\FormInterface */
  568.         $searchForm $builder->getForm();
  569.         $searchForm->handleRequest($request);
  570.         // paginator
  571.         $searchData $searchForm->getData();
  572.         $onlineShopPagination $this->searchProductsBySaleType($paginator$searchData'onlineshop');
  573.         $deliveryPagination $this->searchProductsBySaleType($paginator$searchData'delivery');
  574.         $takeoutPagination $this->searchProductsBySaleType($paginator$searchData'takeout');
  575.         // 画面上では参照しないが、Productに紐づくモデルのロードを行うために呼ぶ
  576.         // 他にやり方ありそうだけど思いつかないので暫定
  577.         list($onlineShopForms$onlineShopToppings) = $this->getAddCartFormsAndToppings($onlineShopPagination'onlineshop');
  578.         list($deliveryForms$deliveryToppings) = $this->getAddCartFormsAndToppings($deliveryPagination'delivery');
  579.         list($takeoutForms$takeoutToppings) = $this->getAddCartFormsAndToppings($takeoutPagination'takeout');
  580.         return [
  581.             'subtitle' => $this->getPageTitle($searchData),
  582.             'onlineShopPagination' => $onlineShopPagination,
  583.             'deliveryPagination' => $deliveryPagination,
  584.             'deliveryForms' => $deliveryForms,
  585.             'deliveryToppings' => $deliveryToppings,
  586.             'takeoutPagination' => $takeoutPagination,
  587.             'takeoutForms' => $takeoutForms,
  588.             'takeoutToppings' => $takeoutToppings,
  589.             'search_form' => $searchForm->createView(),
  590.         ];
  591.     }
  592.     public function searchProductsBySaleType($paginator$searchData$sale_type)
  593.     {
  594.         $qb $this->productRepository->getQueryBuilderBySearchData($searchData$sale_typeProductClass::SHOW_AT_BOTH);
  595.         $query $qb->getQuery()
  596.             ->useResultCache(true$this->eccubeConfig['eccube_result_cache_lifetime_short']);
  597.         /** @var SlidingPagination $pagination */
  598.         $pagination $paginator->paginate(
  599.             $query,
  600.             1// 1ページ目固定
  601.             15// 15件固定で取る
  602.             array('wrap-queries' => true)
  603.         );
  604.         // 品切れの商品を後ろに並び替える
  605.         $this->productService->sortProductsByOutOfStock($pagination);
  606.         return $pagination;
  607.     }
  608.     public function getAddCartFormsAndToppings($pagination$sale_type)
  609.     {
  610.         $ids = [];
  611.         foreach ($pagination as $Product) {
  612.             $ids[] = $Product->getId();
  613.         }
  614.         $ProductsAndClassCategories $this->productRepository->findProductsWithSortedClassCategories($ids'p.id');
  615.         // addCart form
  616.         $forms = [];
  617.         $Toppings = [];
  618.         foreach ($pagination as $Product) {
  619.             /* @var $builder \Symfony\Component\Form\FormBuilderInterface */
  620.             $builder $this->formFactory->createNamedBuilder(
  621.                 '',
  622.                 AddCartType::class,
  623.                 null,
  624.                 [
  625.                     'product' => $ProductsAndClassCategories[$Product->getId()],
  626.                     'allow_extra_fields' => true,
  627.                     'sale_type' => $sale_type,
  628.                 ]
  629.             );
  630.             $addCartForm $builder->getForm();
  631.             $forms[$Product->getId()] = $addCartForm->createView();
  632.             $ToppingGroup $Product->getToppingGroup();
  633.             if ($ToppingGroup) {
  634.                 $ToppingGroupToppings $ToppingGroup->getToppingGroupToppings();
  635.                 foreach ($ToppingGroupToppings as $ToppingGroupTopping) {
  636.                     $Topping $ToppingGroupTopping->getProduct();
  637.                     // 在庫がない、公開ではでない
  638.                     if (!$Topping->getStockFind() || !$Topping->isEnable()) {
  639.                         continue;
  640.                     }
  641.                     $Toppings[$Product->getId()][] = $Topping;
  642.                 }
  643.             }
  644.         }
  645.         return [$forms$Toppings];
  646.     }
  647.     public function getCategoryRankedProducts($Category)
  648.     {
  649.         if (empty($Category)) {
  650.             return;
  651.         }
  652.         $rankingProductIds = [
  653.             $Category['category_ranking_first'],
  654.             $Category['category_ranking_second'],
  655.             $Category['category_ranking_third'],
  656.             $Category['category_ranking_fourth']
  657.         ];
  658.         $CategoryRankingProducts = [];
  659.         foreach ($rankingProductIds as $rankingProductId) {
  660.             if (!empty($rankingProductId) && $this->productRepository->find($rankingProductId)) {
  661.                 $CategoryRankingProducts[] = $this->productRepository->findWithSortedClassCategories($rankingProductId'p.id');
  662.             }
  663.         }
  664.         return $CategoryRankingProducts;
  665.     }
  666. }