commerce_license-8.x-2.x-dev/src/LicenseAvailabilityCheckerExistingRights.php
src/LicenseAvailabilityCheckerExistingRights.php
<?php namespace Drupal\commerce_license; use Drupal\commerce\Context; use Drupal\commerce\PurchasableEntityInterface; use Drupal\commerce_license\Plugin\Commerce\LicenseType\ExistingRightsFromConfigurationCheckingInterface; use Drupal\commerce_order\AvailabilityCheckerInterface; use Drupal\commerce_order\AvailabilityResult; use Drupal\commerce_order\Entity\OrderItemInterface; use Drupal\Core\Datetime\DateFormatterInterface; use Drupal\Core\Entity\EntityInterface; use Drupal\Core\Entity\EntityTypeManagerInterface; use Drupal\Core\Session\AccountProxyInterface; use Drupal\Core\StringTranslation\StringTranslationTrait; use Drupal\Core\StringTranslation\TranslatableMarkup; /** * Prevents purchase of a license that grants rights the user already has. * * This does not check existing licenses, but checks the granted features * directly. For example, for a role license, this checks whether the user has * the role the license grants, rather than whether they have a license for * that role. * * Using an availability checker rather than an order processor, even though * they currently ultimately do the same thing (as availability checkers are * processed by AvailabilityOrderProcessor, which is itself an order processor), * because eventually availability checkers should deal with hiding the 'add to * cart' form -- see https://www.drupal.org/node/2710107. * * @see \Drupal\commerce_license\LicenseOrderProcessorMultiples */ class LicenseAvailabilityCheckerExistingRights implements AvailabilityCheckerInterface { use StringTranslationTrait; /** * The current active user. * * @var \Drupal\Core\Session\AccountProxyInterface */ protected $currentUser; /** * The entity type manager. * * @var \Drupal\Core\Entity\EntityTypeManagerInterface */ protected $entityTypeManager; /** * The date formatter. * * @var \Drupal\Core\Datetime\DateFormatterInterface */ protected $dateFormatter; /** * Constructs a new LicenseAvailabilityCheckerExistingRights object. * * @param \Drupal\Core\Session\AccountProxyInterface $current_user * The current active user. * @param \Drupal\Core\Entity\EntityTypeManagerInterface $entity_type_manager * The entity type manager. * @param \Drupal\Core\Datetime\DateFormatterInterface $date_formatter * The date formatter. */ public function __construct(AccountProxyInterface $current_user, EntityTypeManagerInterface $entity_type_manager, DateFormatterInterface $date_formatter) { $this->currentUser = $current_user; $this->entityTypeManager = $entity_type_manager; $this->dateFormatter = $date_formatter; } /** * {@inheritdoc} */ public function applies(OrderItemInterface $order_item) { $purchased_entity = $order_item->getPurchasedEntity(); if ($purchased_entity === NULL) { return FALSE; } // This applies only to product variations which have our license trait on // them. Check for the field the trait provides, as checking for the trait // on the bundle is expensive -- see https://www.drupal.org/node/2894805. if (!$purchased_entity->hasField('license_type') || $purchased_entity->get('license_type')->isEmpty()) { return FALSE; } // Don't do an availability check on recurring orders. if ($order_item->getOrder() && $order_item->getOrder()->bundle() === 'recurring') { return FALSE; } // This applies only to license types that implement the interface. $license_type_plugin = $purchased_entity->get('license_type')->first()->getTargetInstance(); return $license_type_plugin instanceof ExistingRightsFromConfigurationCheckingInterface; } /** * {@inheritdoc} */ public function check(OrderItemInterface $order_item, Context $context) { // Hand over to the license type plugin configured in the product variation, // to let it determine whether the user already has what the license would // grant. $customer = $context->getCustomer(); $purchased_entity = $order_item->getPurchasedEntity(); // Load the full user entity for the plugin. $user = $this->entityTypeManager->getStorage('user')->load($customer->id()); if (!$user || !$purchased_entity) { return AvailabilityResult::neutral(); } // Handle license renewal. /** @var \Drupal\commerce_license\Entity\LicenseInterface $existing_license */ $existing_license = $this->entityTypeManager ->getStorage('commerce_license') ->getExistingLicense($purchased_entity, $user->id()); if ($existing_license) { if ($existing_license->canRenew()) { return AvailabilityResult::neutral(); } if (!is_null($existing_license->getRenewalWindowStartTime())) { // Shows a message to indicate window start time, // in case license is renewable, but we're out of its renewable window. $message = $this->getRenewalStartTimeMessage($existing_license->getRenewalWindowStartTime()); return AvailabilityResult::unavailable($message); } $message = $this->t('You have an existing license.'); return AvailabilityResult::unavailable($message); } return $this->checkPurchasable($purchased_entity, $user); } /** * Adds a renewalStartTimeMessage status message to queue. * * @param int|null $renewal_window_start_time * The renewal window start time. * * @return \Drupal\Core\StringTranslation\TranslatableMarkup * The renewal start time message. */ private function getRenewalStartTimeMessage(?int $renewal_window_start_time): TranslatableMarkup { return $this->t('You have an existing license. You will be able to renew your license after @date.', [ '@date' => $this->dateFormatter->format($renewal_window_start_time), ]); } /** * Checks if new license is eligible for purchase. * * Hand over to the license type plugin configured in the product variation, * to let it determine whether the user already has what the license would * grant. Adds a notPurchasableMessage status message to queue. * * @param \Drupal\commerce\PurchasableEntityInterface $entity * The purchased entity. * @param \Drupal\Core\Entity\EntityInterface $user * The user the license would be granted to. * * @return \Drupal\commerce_order\AvailabilityResult * The availability of an order item. * * @throws \Drupal\Core\TypedData\Exception\MissingDataException */ private function checkPurchasable(PurchasableEntityInterface $entity, EntityInterface $user): AvailabilityResult { $license_type_plugin = $entity->get('license_type')->first()->getTargetInstance(); $existing_rights_result = $license_type_plugin->checkUserHasExistingRights($user); if (!$existing_rights_result->hasExistingRights()) { return AvailabilityResult::neutral(); } // Show a message that includes the reason from the rights check. if ($user->id() == $this->currentUser->id()) { $rights_check_message = $existing_rights_result->getOwnerUserMessage(); } else { $rights_check_message = $existing_rights_result->getOtherUserMessage(); } $message = $rights_check_message . ' ' . $this->t('You may not purchase the @product-label product.', [ '@product-label' => $entity->label(), ]); return AvailabilityResult::unavailable($message); } }