contacts_events-8.x-1.x-dev/modules/teams/src/Entity/TeamApplication.php
modules/teams/src/Entity/TeamApplication.php
<?php
namespace Drupal\contacts_events_teams\Entity;
use Drupal\contacts_events\Entity\Ticket;
use Drupal\contacts_events\Entity\TicketInterface;
use Drupal\Core\Entity\EntityStorageInterface;
use Drupal\Core\Field\BaseFieldDefinition;
use Drupal\Core\Entity\ContentEntityBase;
use Drupal\Core\Entity\EntityChangedTrait;
use Drupal\Core\Entity\EntityTypeInterface;
use Drupal\Core\Field\FieldStorageDefinitionInterface;
use Drupal\Core\Session\AccountInterface;
use Drupal\field\Entity\FieldConfig;
use Drupal\user\EntityOwnerInterface;
use Drupal\user\UserInterface;
/**
* Defines the Team application entity.
*
* @ingroup contacts_events_teams
*
* @ContentEntityType(
* id = "c_events_team_app",
* label = @Translation("Team application"),
* label_plural = @Translation("Team applications"),
* label_collection = @Translation("Team applications"),
* bundle_label = @Translation("Team application form"),
* handlers = {
* "view_builder" = "Drupal\Core\Entity\EntityViewBuilder",
* "list_builder" = "Drupal\contacts_events_teams\TeamApplicationListBuilder",
* "views_data" = "Drupal\contacts_events_teams\Entity\TeamApplicationViewsData",
* "form" = {
* "default" = "Drupal\contacts_events_teams\Form\TeamApplicationForm",
* "edit" = "Drupal\contacts_events_teams\Form\TeamApplicationForm",
* "delete" = "Drupal\contacts_events_teams\Form\TeamApplicationDeleteForm",
* },
* "access" = "Drupal\contacts_events_teams\TeamApplicationAccessControlHandler",
* "route_provider" = {
* "html" = "Drupal\contacts_events_teams\TeamApplicationHtmlRouteProvider",
* },
* },
* base_table = "c_events_team_app",
* admin_permission = "administer team application entities",
* entity_keys = {
* "id" = "id",
* "bundle" = "type",
* "uuid" = "uuid",
* "uid" = "user_id",
* },
* links = {
* "collection" = "/event/{contacts_event}/teams/applications",
* "canonical" = "/event/{contacts_event}/teams/{c_events_team}/applications/{c_events_team_app}",
* "edit-form" = "/event/{contacts_event}/teams/{c_events_team}/applications/{c_events_team_app}/edit"
* },
* bundle_entity_type = "c_events_team_app_type",
* field_ui_base_route = "entity.c_events_team_app_type.edit_form"
* )
*/
class TeamApplication extends ContentEntityBase implements EntityOwnerInterface {
use EntityChangedTrait;
/**
* Creates a new team application, for the specified team and ticket.
*
* @param \Drupal\contacts_events\Entity\TicketInterface $ticket
* The ticket that will be associated with the team application.
* @param \Drupal\contacts_events_teams\Entity\TeamInterface $team
* The team that the applicant is applying for.
* @param bool $auto_accept
* Whether the application should be automatically accepted.
* Defaults to FALSE.
*
* @return \Drupal\contacts_events_teams\Entity\TeamApplication
* The new application.
*/
public static function createForTicket(TicketInterface $ticket, TeamInterface $team, $auto_accept = FALSE) {
$application = TeamApplication::create([
'type' => $team->get('form')->target_id,
'user_id' => $ticket->getTicketHolderId(),
'team' => $team->id(),
'ticket' => $ticket->id(),
'event' => $ticket->getEvent()->id(),
'state' => $auto_accept ? 'approved' : 'draft',
]);
return $application;
}
/**
* Changes the team that an application is for.
*
* If the new team has the same application form as the old team,
* this application will be updated with the new team, saved and returned.
*
* If the new team has a different application form, a new application will
* be created, field values copied across (where they can), saved and
* returned. This application will be archived and saved.
*
* @param \Drupal\contacts_events_teams\Entity\TeamApplication $existing_application
* The application for which the team to be changed.
* @param \Drupal\contacts_events_teams\Entity\Team $team
* The team to change to.
* @param bool $auto_accept
* Whether the new application form should be automatically accepted
* (TRUE) or needs to be filled in by the applicant (FALSE).
* Defaults to FALSE.
*
* @return \Drupal\contacts_events_teams\Entity\TeamApplication
* Either this application with the team updated, or a new team application.
*/
public static function changeTeam(TeamApplication $existing_application, Team $team, $auto_accept = FALSE) {
$application_status = $auto_accept ? 'approved' : 'draft';
// If the two teams use the same application form, just change this
// application's team and reset its status to either draft/approved.
if ($existing_application->bundle() == $team->get('form')->target_id) {
$existing_application->set('team', $team->id());
$existing_application->set('state', $application_status);
$existing_application->save();
return $existing_application;
}
else {
// If the two teams *don't* use the same application form, create
// a new application.
$new_application = $existing_application->cloneWithDifferentTeam($team);
$new_application->set('state', $application_status);
$new_application->save();
// Archive old form.
$existing_application->set('state', 'archived');
// Flag that we're in the middle of a transfer process.
// The TeamTicketStateSubscriber will normally reset a ticket back to
// pending when an application is archived. Setting a surrogate property
// on the application will prevent this
// (see TeamTicketStateSubscriber::updateTicketState)
$existing_application->is_transferring = TRUE;
$existing_application->save();
return $new_application;
}
}
/**
* {@inheritdoc}
*/
public static function preCreate(EntityStorageInterface $storage_controller, array &$values) {
parent::preCreate($storage_controller, $values);
$values += [
'user_id' => \Drupal::currentUser()->id(),
'state' => 'draft',
];
}
/**
* {@inheritdoc}
*/
public function getCreatedTime() {
return $this->get('created')->value;
}
/**
* {@inheritdoc}
*/
public function setCreatedTime($timestamp) {
$this->set('created', $timestamp);
return $this;
}
/**
* {@inheritdoc}
*/
public function getOwner() {
return $this->get('user_id')->entity;
}
/**
* {@inheritdoc}
*/
public function getOwnerId() {
return $this->get('user_id')->target_id;
}
/**
* {@inheritdoc}
*/
public function setOwnerId($uid) {
$this->set('user_id', $uid);
return $this;
}
/**
* {@inheritdoc}
*/
public function setOwner(UserInterface $account) {
$this->set('user_id', $account->id());
return $this;
}
/**
* Whether the team application has been submitted.
*
* @return bool
* Whether the team application has been submitted.
*/
public function isSubmitted() : bool {
// If it's not draft, it's been submitted.
return $this->state->value && $this->state->value != 'draft';
}
/**
* Gets the team for this application.
*
* @return \Drupal\contacts_events_teams\Entity\Team|null
* The team.
*/
public function getTeam() : ?Team {
return $this->get('team')->entity;
}
/**
* The ticket associated with this team application.
*
* @return \Drupal\contacts_events\Entity\Ticket|null
* The ticket associated with this team application.
*/
public function getTicket() : ?Ticket {
return $this->get('ticket')->entity;
}
/**
* {@inheritdoc}
*/
protected function urlRouteParameters($rel) {
$parameters = parent::urlRouteParameters($rel);
$parameters['c_events_team'] = $this->get('team')->target_id;
$parameters['contacts_event'] = $this->get('event')->target_id;
return $parameters;
}
/**
* {@inheritdoc}
*/
public static function baseFieldDefinitions(EntityTypeInterface $entity_type) {
$fields = parent::baseFieldDefinitions($entity_type);
$fields['state'] = BaseFieldDefinition::create('state')
->setLabel(t('Application Status'))
->setDescription(t('The team application state.'))
->setRequired(TRUE)
->setSetting('max_length', 255)
->setSetting('workflow', 'contacts_events_teams_application_process')
->setDisplayOptions('view', [
'label' => 'hidden',
'type' => 'state_transition_form',
'weight' => 10,
])
->setDisplayConfigurable('form', TRUE)
->setDisplayConfigurable('view', TRUE);
$fields['team'] = BaseFieldDefinition::create('entity_reference')
->setLabel(t('Team'))
->setSetting('target_type', 'c_events_team')
->setSetting('handler', 'default')
->setDisplayOptions('view', [
'label' => 'inline',
'type' => 'entity_reference_label',
'weight' => 0,
])
->setDisplayConfigurable('form', TRUE)
->setDisplayConfigurable('view', TRUE);
$fields['ticket'] = BaseFieldDefinition::create('entity_reference')
->setLabel(t('Ticket'))
->setDescription(t('Ticket'))
->setSetting('target_type', 'contacts_ticket')
->setSetting('handler', 'default')
->setDisplayConfigurable('form', TRUE)
->setDisplayConfigurable('view', TRUE);
$fields['event'] = BaseFieldDefinition::create('entity_reference')
->setLabel(t('Event'))
->setDescription(t('The Event'))
->setSetting('target_type', 'contacts_event')
->setSetting('handler', 'default')
->setDisplayOptions('view', [
'label' => 'inline',
'type' => 'entity_reference_label',
'weight' => 0,
])
->setDisplayConfigurable('form', TRUE)
->setDisplayConfigurable('view', TRUE);
$fields['user_id'] = BaseFieldDefinition::create('entity_reference')
->setLabel(t('User'))
->setDescription(t('The user ID of author of the Team application entity.'))
->setSetting('target_type', 'user')
->setSetting('handler', 'default')
->setDisplayOptions('view', [
'label' => 'inline',
'type' => 'author',
'weight' => 0,
])
->setDisplayConfigurable('form', FALSE)
->setDisplayConfigurable('view', TRUE);
$fields['created'] = BaseFieldDefinition::create('created')
->setLabel(t('Created'))
->setDescription(t('The time that the entity was created.'));
$fields['changed'] = BaseFieldDefinition::create('changed')
->setLabel(t('Changed'))
->setDescription(t('The time that the entity was last edited.'));
$fields['status_log'] = BaseFieldDefinition::create('status_log')
->setLabel(t('Status History'))
->setCardinality(FieldStorageDefinitionInterface::CARDINALITY_UNLIMITED)
->setDescription(t('A log of when the status was changed and by whom.'))
->setSettings([
'max_length' => 50,
'source_field' => 'state',
])
->setDisplayOptions('view', [
'weight' => 20,
'type' => 'status_log_list',
])
->setDisplayConfigurable('view', FALSE)
->setDisplayConfigurable('form', FALSE);
return $fields;
}
/**
* Retreives a collection of transition IDs that the user can apply manually.
*
* @param \Drupal\Core\Session\AccountInterface $account
* The account.
*
* @return array
* Array of transition IDs that the user can invoke.
*/
public function getManuallyAllowedTransitions(AccountInterface $account) {
/** @var \Drupal\state_machine\Plugin\Field\FieldType\StateItem $state */
$state = $this->state->first();
$transitions = array_keys($state->getTransitions());
$staff_only_transitions = [
'submit',
'accept',
'approve',
'reject',
'reject_notify',
'archive',
'reset_to_not_submitted',
];
if ($account->hasPermission('manage all contacts events team applications')) {
return array_intersect($transitions, $staff_only_transitions);
}
$team_leader_transitions = [
'accept',
'reject',
];
if ($this->getTeam() && $this->getTeam()->isTeamLeader($account->id())) {
return array_intersect($transitions, $team_leader_transitions);
}
return [];
}
/**
* Creates a new application based on an existing one, for a different team.
*
* Sets up the new application for the same ticketholder as the other
* application, but for a different team.
*
* Copies application questions and references (defined as config fields)
* across where both applications have the same fields (they may not, if the
* team has different application forms that have different questions
* defined).
*
* @param \Drupal\contacts_events_teams\Entity\Team $team
* The team that the applicant is applying for.
*
* @return \Drupal\contacts_events_teams\Entity\TeamApplication
* The new team application.
*/
protected function cloneWithDifferentTeam(Team $team) {
$ticket = $this->getTicket();
$application = TeamApplication::createForTicket($ticket, $team);
// Find all config fields on the original team application. If the new
// application has a field with the same name, copy the value across.
// Only want to do this for config fields (ie the fields that have
// the questions defined).
foreach ($this->getFields(FALSE) as $field) {
if ($field->getFieldDefinition() instanceof FieldConfig && $application->hasField($field->getName())) {
$application->set($field->getName(), $field->getValue());
}
}
return $application;
}
}
