_contactId = $this->get('contactId'); $this->_contactType = CRM_Core_DAO::getFieldValue('CRM_Contact_DAO_Contact', $this->_contactId, 'contact_type'); $this->_relationshipId = $this->get('id'); $this->_rtype = CRM_Utils_Request::retrieve('rtype', 'String', $this); $this->_rtypeId = CRM_Utils_Request::retrieve('relTypeId', 'String', $this); $this->_display_name_a = CRM_Core_DAO::getFieldValue('CRM_Contact_DAO_Contact', $this->_contactId, 'display_name'); $this->assign('display_name_a', $this->_display_name_a); //get the relationship values. $this->_values = array(); if ($this->_relationshipId) { $params = array('id' => $this->_relationshipId); CRM_Core_DAO::commonRetrieve('CRM_Contact_DAO_Relationship', $params, $this->_values); } // Check for permissions if (in_array($this->_action, array(CRM_Core_Action::ADD, CRM_Core_Action::UPDATE, CRM_Core_Action::DELETE))) { if (!CRM_Contact_BAO_Contact_Permission::allow($this->_contactId, CRM_Core_Permission::EDIT) && !CRM_Contact_BAO_Contact_Permission::allow($this->_values['contact_id_b'], CRM_Core_Permission::EDIT)) { CRM_Core_Error::statusBounce(ts('You do not have the necessary permission to edit this contact.')); } } // Set page title based on action switch ($this->_action) { case CRM_Core_Action::VIEW: CRM_Utils_System::setTitle(ts('View Relationship for %1', array(1 => $this->_display_name_a))); break; case CRM_Core_Action::ADD: CRM_Utils_System::setTitle(ts('Add Relationship for %1', array(1 => $this->_display_name_a))); break; case CRM_Core_Action::UPDATE: CRM_Utils_System::setTitle(ts('Edit Relationship for %1', array(1 => $this->_display_name_a))); break; case CRM_Core_Action::DELETE: CRM_Utils_System::setTitle(ts('Delete Relationship for %1', array(1 => $this->_display_name_a))); break; } $this->_caseId = CRM_Utils_Request::retrieve('caseID', 'Integer', $this); if (!$this->_rtypeId) { $params = CRM_Utils_Request::exportValues(); if (isset($params['relationship_type_id'])) { $this->_rtypeId = $params['relationship_type_id']; } elseif (!empty($this->_values)) { $this->_rtypeId = $this->_values['relationship_type_id'] . '_' . $this->_rtype; } } //get the relationship type id $this->_relationshipTypeId = str_replace(array('_a_b', '_b_a'), array('', ''), $this->_rtypeId); //get the relationship type if (!$this->_rtype) { $this->_rtype = str_replace($this->_relationshipTypeId . '_', '', $this->_rtypeId); } //need to assign custom data type and subtype to the template - FIXME: explain why $this->assign('customDataType', 'Relationship'); $this->assign('customDataSubType', $this->_relationshipTypeId); $this->assign('entityID', $this->_relationshipId); //use name as it remain constant, CRM-3336 $this->_allRelationshipNames = CRM_Core_PseudoConstant::relationshipType('name'); // Current employer? if ($this->_action & CRM_Core_Action::UPDATE) { if ($this->_allRelationshipNames[$this->_relationshipTypeId]["name_a_b"] == 'Employee of') { $this->_isCurrentEmployer = $this->_values['contact_id_b'] == CRM_Core_DAO::getFieldValue('CRM_Contact_DAO_Contact', $this->_values['contact_id_a'], 'employer_id'); } } // when custom data is included in this page if (!empty($_POST['hidden_custom'])) { CRM_Custom_Form_CustomData::preProcess($this, NULL, $this->_relationshipTypeId, 1, 'Relationship', $this->_relationshipId); CRM_Custom_Form_CustomData::buildQuickForm($this); CRM_Custom_Form_CustomData::setDefaultValues($this); } } /** * Set default values for the form. */ public function setDefaultValues() { $defaults = array(); if ($this->_action & CRM_Core_Action::UPDATE) { if (!empty($this->_values)) { $defaults['relationship_type_id'] = $this->_rtypeId; if (!empty($this->_values['start_date'])) { list($defaults['start_date']) = CRM_Utils_Date::setDateDefaults($this->_values['start_date']); } if (!empty($this->_values['end_date'])) { list($defaults['end_date']) = CRM_Utils_Date::setDateDefaults($this->_values['end_date']); } $defaults['description'] = CRM_Utils_Array::value('description', $this->_values); $defaults['is_active'] = CRM_Utils_Array::value('is_active', $this->_values); // The javascript on the form will swap these fields if it is a b_a relationship, so we compensate here $defaults['is_permission_a_b'] = CRM_Utils_Array::value('is_permission_' . $this->_rtype, $this->_values); $defaults['is_permission_b_a'] = CRM_Utils_Array::value('is_permission_' . strrev($this->_rtype), $this->_values); $defaults['is_current_employer'] = $this->_isCurrentEmployer; // Load info about the related contact $contact = new CRM_Contact_DAO_Contact(); if ($this->_rtype == 'a_b' && $this->_values['contact_id_a'] == $this->_contactId) { $contact->id = $this->_values['contact_id_b']; } else { $contact->id = $this->_values['contact_id_a']; } if ($contact->find(TRUE)) { $defaults['related_contact_id'] = $contact->id; $this->_display_name_b = $contact->display_name; $this->assign('display_name_b', $this->_display_name_b); } $noteParams = array( 'entity_id' => $this->_relationshipId, 'entity_table' => 'civicrm_relationship', 'limit' => 1, 'version' => 3, ); $note = civicrm_api('Note', 'getsingle', $noteParams); $defaults['note'] = CRM_Utils_Array::value('note', $note); } } else { $defaults['is_active'] = $defaults['is_current_employer'] = 1; $defaults['relationship_type_id'] = $this->_rtypeId; } $this->_enabled = $defaults['is_active']; return $defaults; } /** * Add the rules for form. */ public function addRules() { if (!($this->_action & CRM_Core_Action::DELETE)) { $this->addFormRule(array('CRM_Contact_Form_Relationship', 'dateRule')); } } /** * Build the form object. */ public function buildQuickForm() { if ($this->_action & CRM_Core_Action::DELETE) { $this->addButtons(array( array( 'type' => 'next', 'name' => ts('Delete'), 'isDefault' => TRUE, ), array( 'type' => 'cancel', 'name' => ts('Cancel'), ), ) ); return; } // Select list $relationshipList = CRM_Contact_BAO_Relationship::getContactRelationshipType($this->_contactId, $this->_rtype, $this->_relationshipId); // Metadata needed on clientside $this->assign('relationshipData', self::getRelationshipTypeMetadata($relationshipList)); foreach ($this->_allRelationshipNames as $id => $vals) { if ($vals['name_a_b'] === 'Employee of') { $this->assign('employmentRelationship', $id); break; } } $this->addField('relationship_type_id', array('options' => array('' => ts('- select -')) + $relationshipList, 'class' => 'huge', 'placeholder' => '- select -'), TRUE); $label = $this->_action & CRM_Core_Action::ADD ? ts('Contact(s)') : ts('Contact'); $contactField = $this->addField('related_contact_id', array('label' => $label, 'name' => 'contact_id_b', 'multiple' => TRUE, 'create' => TRUE), TRUE); // This field cannot be updated if ($this->_action & CRM_Core_Action::UPDATE) { $contactField->freeze(); } $this->add('advcheckbox', 'is_current_employer', $this->_contactType == 'Organization' ? ts('Current Employee') : ts('Current Employer')); $this->addField('start_date', array('label' => ts('Start Date'), 'formatType' => 'searchDate')); $this->addField('end_date', array('label' => ts('End Date'), 'formatType' => 'searchDate')); $this->addField('is_active', array('label' => ts('Enabled?'), 'type' => 'advcheckbox')); $this->addField('is_permission_a_b'); $this->addField('is_permission_b_a'); $this->addField('description', array('label' => ts('Description'))); CRM_Contact_Form_Edit_Notes::buildQuickForm($this); if ($this->_action & CRM_Core_Action::VIEW) { $this->addButtons(array( array( 'type' => 'cancel', 'name' => ts('Done'), ), )); } else { // make this form an upload since we don't know if the custom data injected dynamically is of type file etc. $this->addButtons(array( array( 'type' => 'upload', 'name' => ts('Save Relationship'), 'isDefault' => TRUE, ), array( 'type' => 'cancel', 'name' => ts('Cancel'), ), )); } } /** * This function is called when the form is submitted. */ public function postProcess() { // Store the submitted values in an array. $params = $this->controller->exportValues($this->_name); // CRM-14612 - Don't use adv-checkbox as it interferes with the form js $params['is_permission_a_b'] = CRM_Utils_Array::value('is_permission_a_b', $params, 0); $params['is_permission_b_a'] = CRM_Utils_Array::value('is_permission_b_a', $params, 0); // action is taken depending upon the mode if ($this->_action & CRM_Core_Action::DELETE) { CRM_Contact_BAO_Relationship::del($this->_relationshipId); // CRM-15881 UPDATES // Since the line above nullifies the organization_name and employer_id fiels in the contact record, we need to reload all blocks to reflect this chage on the user interface. $this->ajaxResponse['reloadBlocks'] = array('#crm-contactinfo-content'); return; } $relationshipTypeParts = explode('_', $params['relationship_type_id']); $params['relationship_type_id'] = $relationshipTypeParts[0]; if (!$this->_rtype) { // Do we need to wrap this in an if - when is rtype used & is relationship_type_id always set then? $this->_rtype = $params['relationship_type_id']; } $params['contact_id_' . $relationshipTypeParts[1]] = $this->_contactId; // Update mode (always single) if ($this->_action & CRM_Core_Action::UPDATE) { $update = TRUE; $params['id'] = $this->_relationshipId; $ids['relationship'] = $this->_relationshipId; $relation = CRM_Contact_BAO_Relationship::getRelationshipByID($this->_relationshipId); if ($relation->contact_id_a == $this->_contactId) { // I couldn't replicate this path in testing. See below. $params['contact_id_a'] = $this->_contactId; $params['contact_id_b'] = array($params['related_contact_id']); $outcome = CRM_Contact_BAO_Relationship::createMultiple($params, $relationshipTypeParts[1]); $relationshipIds = $outcome['relationship_ids']; } else { // The only reason we have changed this to use the api & not the above is that this was broken. // Recommend extracting all of update into a function that uses the api // and ensuring api / bao take care of 'other stuff' in this form // the contact_id_a & b can't be changed on this form so don't really need setting. $params['contact_id_b'] = $this->_contactId; $params['contact_id_a'] = $params['related_contact_id']; $result = civicrm_api3('relationship', 'create', $params); $relationshipIds = array($result['id']); } $ids['contactTarget'] = ($relation->contact_id_a == $this->_contactId) ? $relation->contact_id_b : $relation->contact_id_a; // @todo this belongs in the BAO. if ($this->_isCurrentEmployer) { // if relationship type changes, relationship is disabled, or "current employer" is unchecked, // clear the current employer. CRM-3235. $relChanged = $params['relationship_type_id'] != $this->_values['relationship_type_id']; if (!$params['is_active'] || !$params['is_current_employer'] || $relChanged) { // CRM-15881 UPDATES // If not is_active then is_current_employer needs to be set false as well! Logically a contact cannot be a current employee of a disabled employer relationship. // If this is not done, then the below process will go ahead and disable the organization_name and employer_id fields in the contact record (which is what is wanted) but then further down will be re-enabled becuase is_current_employer is not false, therefore undoing what was done correctly. if (!$params['is_active']) { $params['is_current_employer'] = FALSE; } CRM_Contact_BAO_Contact_Utils::clearCurrentEmployer($this->_values['contact_id_a']); // Refresh contact summary if in ajax mode $this->ajaxResponse['reloadBlocks'] = array('#crm-contactinfo-content'); } } if (empty($outcome['saved']) && !empty($update)) { $outcome['saved'] = $update; } $this->setMessage($outcome); } // Create mode (could be 1 or more relationships) else { $params['contact_id_' . $relationshipTypeParts[2]] = explode(',', $params['related_contact_id']); $outcome = CRM_Contact_BAO_Relationship::createMultiple($params, $relationshipTypeParts[1]); $relationshipIds = $outcome['relationship_ids']; if (empty($outcome['saved']) && !empty($update)) { $outcome['saved'] = $update; } $this->setMessage($outcome); } // if this is called from case view, //create an activity for case role removal.CRM-4480 // @todo this belongs in the BAO. if ($this->_caseId) { CRM_Case_BAO_Case::createCaseRoleActivity($this->_caseId, $relationshipIds, $params['contact_check'], $this->_contactId); } // Save notes // @todo this belongs in the BAO. if ($this->_action & CRM_Core_Action::UPDATE || $params['note']) { foreach ($relationshipIds as $id) { $noteParams = array( 'entity_id' => $id, 'entity_table' => 'civicrm_relationship', ); $existing = civicrm_api3('note', 'get', $noteParams); if (!empty($existing['id'])) { $noteParams['id'] = $existing['id']; } $noteParams['note'] = $params['note']; $noteParams['contact_id'] = $this->_contactId; if (!empty($existing['id']) || $params['note']) { $action = $params['note'] ? 'create' : 'delete'; civicrm_api3('note', $action, $noteParams); } } } $params['relationship_ids'] = $relationshipIds; // Refresh contact tabs which might have been affected $this->ajaxResponse['updateTabs'] = array( '#tab_member' => CRM_Contact_BAO_Contact::getCountComponent('membership', $this->_contactId), '#tab_contribute' => CRM_Contact_BAO_Contact::getCountComponent('contribution', $this->_contactId), ); // Set current employee/employer relationship, CRM-3532 if ($params['is_current_employer'] && $this->_allRelationshipNames[$params['relationship_type_id']]["name_a_b"] == 'Employee of') { $employerParams = array(); foreach ($relationshipIds as $id) { // Fixme this is dumb why do we have to look this up again? $rel = CRM_Contact_BAO_Relationship::getRelationshipByID($id); $employerParams[$rel->contact_id_a] = $rel->contact_id_b; } // @todo this belongs in the BAO. CRM_Contact_BAO_Contact_Utils::setCurrentEmployer($employerParams); // Refresh contact summary if in ajax mode $this->ajaxResponse['reloadBlocks'] = array('#crm-contactinfo-content'); } } /** * Date validation. * * @param array $params * (reference ) an assoc array of name/value pairs. * * @return bool|array * mixed true or array of errors */ public static function dateRule($params) { $errors = array(); // check start and end date if (!empty($params['start_date']) && !empty($params['end_date'])) { $start_date = CRM_Utils_Date::format(CRM_Utils_Array::value('start_date', $params)); $end_date = CRM_Utils_Date::format(CRM_Utils_Array::value('end_date', $params)); if ($start_date && $end_date && (int ) $end_date < (int ) $start_date) { $errors['end_date'] = ts('The relationship end date cannot be prior to the start date.'); } } return empty($errors) ? TRUE : $errors; } /** * Set Status message to reflect outcome of the update action. * * @param array $outcome * Outcome of save action - including * - 'valid' : Number of valid relationships attempted. * - 'invalid' : Number of invalid relationships attempted. * - 'duplicate' : Number of duplicate relationships attempted. * - 'saved' : boolean of whether save was successful */ protected function setMessage($outcome) { if (!empty($outcome['valid']) && empty($outcome['saved'])) { CRM_Core_Session::setStatus(ts('Relationship created.', array( 'count' => $outcome['valid'], 'plural' => '%count relationships created.', )), ts('Saved'), 'success'); } if (!empty($outcome['invalid'])) { CRM_Core_Session::setStatus(ts('%count relationship record was not created due to an invalid contact type.', array( 'count' => $outcome['invalid'], 'plural' => '%count relationship records were not created due to invalid contact types.', )), ts('%count invalid relationship record', array( 'count' => $outcome['invalid'], 'plural' => '%count invalid relationship records', ))); } if (!empty($outcome['duplicate'])) { CRM_Core_Session::setStatus(ts('One relationship was not created because it already exists.', array( 'count' => $outcome['duplicate'], 'plural' => '%count relationships were not created because they already exist.', )), ts('%count duplicate relationship', array( 'count' => $outcome['duplicate'], 'plural' => '%count duplicate relationships', ))); } if (!empty($outcome['saved'])) { CRM_Core_Session::setStatus(ts('Relationship record has been updated.'), ts('Saved'), 'success'); } } /** * @param $relationshipList * @return array */ public static function getRelationshipTypeMetadata($relationshipList) { $contactTypes = CRM_Contact_BAO_ContactType::contactTypeInfo(TRUE); $allRelationshipNames = CRM_Core_PseudoConstant::relationshipType('name'); $jsData = array(); // Get just what we need to keep the dom small $whatWeWant = array_flip(array( 'contact_type_a', 'contact_type_b', 'contact_sub_type_a', 'contact_sub_type_b', )); foreach ($allRelationshipNames as $id => $vals) { if (isset($relationshipList["{$id}_a_b"]) || isset($relationshipList["{$id}_b_a"])) { $jsData[$id] = array_filter(array_intersect_key($allRelationshipNames[$id], $whatWeWant)); // Add user-friendly placeholder foreach (array('a', 'b') as $x) { $type = !empty($jsData[$id]["contact_sub_type_$x"]) ? $jsData[$id]["contact_sub_type_$x"] : CRM_Utils_Array::value("contact_type_$x", $jsData[$id]); $jsData[$id]["placeholder_$x"] = $type ? ts('- select %1 -', array(strtolower($contactTypes[$type]['label']))) : ts('- select contact -'); } } } return $jsData; } }