Merge pull request #18467 from mattwire/membershiptestfix
[civicrm-core.git] / CRM / Activity / Page / AJAX.php
CommitLineData
6a488035
TO
1<?php
2/*
3 +--------------------------------------------------------------------+
bc77d7c0 4 | Copyright CiviCRM LLC. All rights reserved. |
6a488035 5 | |
bc77d7c0
TO
6 | This work is published under the GNU AGPLv3 license with some |
7 | permitted exceptions and without any warranty. For full license |
8 | and copyright information, see https://civicrm.org/licensing |
6a488035 9 +--------------------------------------------------------------------+
d25dd0ee 10 */
6a488035
TO
11
12/**
13 *
14 * @package CRM
ca5cec67 15 * @copyright CiviCRM LLC https://civicrm.org/licensing
6a488035
TO
16 */
17
18/**
19 * This class contains all the function that are called using AJAX (jQuery)
20 */
21class CRM_Activity_Page_AJAX {
62d3ee27 22
00be9182 23 public static function getCaseActivity() {
5d817a13
MM
24 // Should those params be passed through the validateParams method?
25 $caseID = CRM_Utils_Type::validate($_GET['caseID'], 'Integer');
26 $contactID = CRM_Utils_Type::validate($_GET['cid'], 'Integer');
27 $userID = CRM_Utils_Type::validate($_GET['userID'], 'Integer');
28 $context = CRM_Utils_Type::validate(CRM_Utils_Array::value('context', $_GET), 'String');
6a488035 29
be2fb01f 30 $optionalParameters = [
9c3f979f
MM
31 'source_contact_id' => 'Integer',
32 'status_id' => 'Integer',
33 'activity_deleted' => 'Boolean',
34 'activity_type_id' => 'Integer',
ab9eca19
CW
35 // "Date" validation fails because it expects only numbers with no hyphens
36 'activity_date_low' => 'Alphanumeric',
37 'activity_date_high' => 'Alphanumeric',
be2fb01f 38 ];
6a488035 39
9c3f979f 40 $params = CRM_Core_Page_AJAX::defaultSortAndPagerParams();
be2fb01f 41 $params += CRM_Core_Page_AJAX::validateParams([], $optionalParameters);
6a488035
TO
42
43 // get the activities related to given case
44 $activities = CRM_Case_BAO_Case::getCaseActivity($caseID, $params, $contactID, $context, $userID);
45
ad280fb6 46 CRM_Utils_JSON::output($activities);
6a488035
TO
47 }
48
00be9182 49 public static function getCaseGlobalRelationships() {
9c3f979f 50 $params = CRM_Core_Page_AJAX::defaultSortAndPagerParams();
1d85d241 51
6a488035 52 // get the activities related to given case
be2fb01f 53 $globalGroupInfo = [];
1d85d241 54
6a488035 55 // get the total row count
9c3f979f 56 CRM_Case_BAO_Case::getGlobalContacts($globalGroupInfo, NULL, FALSE, TRUE, NULL, NULL);
6a488035 57 // limit the rows
1d154485 58 $relGlobal = CRM_Case_BAO_Case::getGlobalContacts($globalGroupInfo, $params['sortBy'] ?? NULL, $showLinks = TRUE, FALSE, $params['offset'], $params['rp']);
1d85d241 59
be2fb01f 60 $relationships = [];
ad280fb6
JL
61 // after sort we can update username fields to be a url
62 foreach ($relGlobal as $key => $value) {
be2fb01f 63 $relationship = [];
ad280fb6
JL
64 $relationship['sort_name'] = $value['sort_name'];
65 $relationship['phone'] = $value['phone'];
66 $relationship['email'] = $value['email'];
1d85d241 67
ad280fb6
JL
68 array_push($relationships, $relationship);
69 }
70
be2fb01f 71 $globalRelationshipsDT = [];
ad280fb6 72 $globalRelationshipsDT['data'] = $relationships;
9c3f979f
MM
73 $globalRelationshipsDT['recordsTotal'] = count($relationships);
74 $globalRelationshipsDT['recordsFiltered'] = count($relationships);
aab589e2 75
ad280fb6 76 CRM_Utils_JSON::output($globalRelationshipsDT);
1d85d241
DL
77 }
78
00be9182 79 public static function getCaseClientRelationships() {
353ffa53 80 $caseID = CRM_Utils_Type::escape($_GET['caseID'], 'Integer');
6a488035 81 $contactID = CRM_Utils_Type::escape($_GET['cid'], 'Integer');
1d85d241 82
9c3f979f 83 $params = CRM_Core_Page_AJAX::defaultSortAndPagerParams();
6a488035
TO
84
85 // Retrieve ALL client relationships
86 $relClient = CRM_Contact_BAO_Relationship::getRelationship($contactID,
87 CRM_Contact_BAO_Relationship::CURRENT,
88 0, 0, 0, NULL, NULL, FALSE
89 );
1d85d241 90
6a488035 91 $caseRelationships = CRM_Case_BAO_Case::getCaseRoles($contactID, $caseID);
1d85d241 92
6a488035
TO
93 // Now build 'Other Relationships' array by removing relationships that are already listed under Case Roles
94 // so they don't show up twice.
be2fb01f 95 $clientRelationships = [];
6a488035
TO
96 foreach ($relClient as $r) {
97 if (!array_key_exists($r['id'], $caseRelationships)) {
98 $clientRelationships[] = $r;
99 }
100 }
1d85d241 101
6a488035 102 // sort clientRelationships array using jquery call params
1d154485
JP
103 $sortArray = [];
104 if (!empty($params['_raw_values']['sort'])) {
105 foreach ($clientRelationships as $key => $row) {
106 $sortArray[$key] = $row[$params['_raw_values']['sort'][0]];
107 }
108 $sort_type = "SORT_" . strtoupper($params['_raw_values']['order'][0]);
109 array_multisort($sortArray, constant($sort_type), $clientRelationships);
6a488035 110 }
1d85d241 111
be2fb01f 112 $relationships = [];
6a488035 113 // after sort we can update username fields to be a url
22e263ad 114 foreach ($clientRelationships as $key => $value) {
be2fb01f 115 $relationship = [];
ad280fb6
JL
116 $relationship['relation'] = $value['relation'];
117 $relationship['name'] = '<a href=' . CRM_Utils_System::url('civicrm/contact/view',
353ffa53 118 'action=view&reset=1&cid=' . $clientRelationships[$key]['cid']) . '>' . $clientRelationships[$key]['name'] . '</a>';
ad280fb6
JL
119 $relationship['phone'] = $value['phone'];
120 $relationship['email'] = $value['email'];
121
122 array_push($relationships, $relationship);
6a488035 123 }
1d85d241 124
be2fb01f 125 $clientRelationshipsDT = [];
ad280fb6 126 $clientRelationshipsDT['data'] = $relationships;
9c3f979f
MM
127 $clientRelationshipsDT['recordsTotal'] = count($relationships);
128 $clientRelationshipsDT['recordsFiltered'] = count($relationships);
aab589e2 129
ad280fb6 130 CRM_Utils_JSON::output($clientRelationshipsDT);
1d85d241
DL
131 }
132
00be9182 133 public static function getCaseRoles() {
353ffa53 134 $caseID = CRM_Utils_Type::escape($_GET['caseID'], 'Integer');
6a488035 135 $contactID = CRM_Utils_Type::escape($_GET['cid'], 'Integer');
1d85d241 136
9c3f979f 137 $params = CRM_Core_Page_AJAX::defaultSortAndPagerParams();
6a488035
TO
138
139 $caseRelationships = CRM_Case_BAO_Case::getCaseRoles($contactID, $caseID);
140 $caseTypeName = CRM_Case_BAO_Case::getCaseType($caseID, 'name');
141 $xmlProcessor = new CRM_Case_XMLProcessor_Process();
353ffa53 142 $caseRoles = $xmlProcessor->get($caseTypeName, 'CaseRoles');
1d85d241 143
6a488035 144 $hasAccessToAllCases = CRM_Core_Permission::check('access all cases and activities');
1d85d241 145
6a488035 146 $managerRoleId = $xmlProcessor->getCaseManagerRoleId($caseTypeName);
ad280fb6 147
6a488035 148 foreach ($caseRelationships as $key => $value) {
3b1c37fe 149 // This role has been filled
41cf58d3 150 unset($caseRoles[$value['relation_type'] . '_' . $value['relationship_direction']]);
b44e3f84 151 // mark original case relationships record to use on setting edit links below
6a488035
TO
152 $caseRelationships[$key]['source'] = 'caseRel';
153 }
1d85d241 154
6a488035
TO
155 $caseRoles['client'] = CRM_Case_BAO_Case::getContactNames($caseID);
156
157 // move/transform caseRoles array data to caseRelationships
158 // for sorting and display
5b6db2c7 159 // CRM-14466 added cid to the non-client array to avoid php notice
22e263ad 160 foreach ($caseRoles as $id => $value) {
6a488035 161 if ($id != "client") {
be2fb01f 162 $rel = [];
6a488035
TO
163 $rel['relation'] = $value;
164 $rel['relation_type'] = $id;
165 $rel['name'] = '(not assigned)';
166 $rel['phone'] = '';
167 $rel['email'] = '';
168 $rel['source'] = 'caseRoles';
169 $caseRelationships[] = $rel;
0db6c3e1
TO
170 }
171 else {
22e263ad 172 foreach ($value as $clientRole) {
be2fb01f 173 $relClient = [];
5eb4e891 174 $relClient['relation'] = ts('Client');
6a488035
TO
175 $relClient['name'] = $clientRole['sort_name'];
176 $relClient['phone'] = $clientRole['phone'];
177 $relClient['email'] = $clientRole['email'];
178 $relClient['cid'] = $clientRole['contact_id'];
179 $relClient['source'] = 'contact';
180 $caseRelationships[] = $relClient;
181 }
182 }
183 }
184
185 // sort clientRelationships array using jquery call params
1d154485
JP
186 $sortArray = [];
187 if (!empty($params['_raw_values']['sort'])) {
188 foreach ($caseRelationships as $key => $row) {
189 $sortArray[$key] = $row[$params['_raw_values']['sort'][0]];
190 }
191 $sort_type = "SORT_" . strtoupper($params['_raw_values']['order'][0]);
192 array_multisort($sortArray, constant($sort_type), $caseRelationships);
6a488035 193 }
6a488035 194
be2fb01f 195 $relationships = [];
5b6db2c7 196
6a488035 197 // set user name, email and edit columns links
ad280fb6 198 foreach ($caseRelationships as $key => &$row) {
3b1c37fe
CW
199 $typeLabel = $row['relation'];
200 // Add "<br />(Case Manager)" to label
41cf58d3 201 if (!empty($row['relation_type']) && !empty($row['relationship_direction']) && $row['relation_type'] . '_' . $row['relationship_direction'] == $managerRoleId) {
3b1c37fe
CW
202 $row['relation'] .= '<br />' . '(' . ts('Case Manager') . ')';
203 }
6a488035 204 // view user links
3662628a 205 if (!empty($row['cid'])) {
86bfa4f6 206 $row['name'] = '<a class="view-contact" title="' . ts('View Contact') . '" href=' . CRM_Utils_System::url('civicrm/contact/view',
353ffa53 207 'action=view&reset=1&cid=' . $row['cid']) . '>' . $row['name'] . '</a>';
5b6db2c7 208 }
6a488035 209 // email column links/icon
c91df8b4 210 if ($row['email']) {
13a3d214 211 $row['email'] = '<a class="crm-hover-button crm-popup" href="' . CRM_Utils_System::url('civicrm/activity/email/add', 'reset=1&action=add&atype=3&cid=' . $row['cid']) . '&caseid=' . $caseID . '" title="' . ts('Send an Email') . '"><i class="crm-i fa-envelope" aria-hidden="true"></i></a>';
6a488035
TO
212 }
213 // edit links
3662628a 214 $row['actions'] = '';
6a488035 215 if ($hasAccessToAllCases) {
e3756c36
CW
216 $contactType = empty($row['relation_type']) ? '' : (string) CRM_Core_DAO::getFieldValue('CRM_Contact_DAO_RelationshipType', $row['relation_type'], 'contact_type_b');
217 $contactType = $contactType == 'Contact' ? '' : $contactType;
22e263ad 218 switch ($row['source']) {
32864ccf 219 case 'caseRel':
be2fb01f 220 $row['actions'] = '<a href="#editCaseRoleDialog" title="' . ts('Reassign %1', [1 => $typeLabel]) . '" class="crm-hover-button case-miniform" data-contact_type="' . $contactType . '" data-rel_type="' . $row['relation_type'] . '_' . $row['relationship_direction'] . '" data-cid="' . $row['cid'] . '" data-rel_id="' . $row['rel_id'] . '"data-key="' . CRM_Core_Key::get('civicrm/ajax/relation') . '">' .
13a3d214 221 '<i class="crm-i fa-pencil" aria-hidden="true"></i>' .
353ffa53 222 '</a>' .
be2fb01f 223 '<a href="#deleteCaseRoleDialog" title="' . ts('Remove %1', [1 => $typeLabel]) . '" class="crm-hover-button case-miniform" data-contact_type="' . $contactType . '" data-rel_type="' . $row['relation_type'] . '_' . $row['relationship_direction'] . '" data-cid="' . $row['cid'] . '" data-key="' . CRM_Core_Key::get('civicrm/ajax/delcaserole') . '">' .
92fcb95f 224 '<span class="icon delete-icon"></span>' .
353ffa53 225 '</a>';
32864ccf 226 break;
1d85d241 227
32864ccf 228 case 'caseRoles':
be2fb01f 229 $row['actions'] = '<a href="#editCaseRoleDialog" title="' . ts('Assign %1', [1 => $typeLabel]) . '" class="crm-hover-button case-miniform" data-contact_type="' . $contactType . '" data-rel_type="' . $row['relation_type'] . '_a_b" data-key="' . CRM_Core_Key::get('civicrm/ajax/relation') . '">' .
13a3d214 230 '<i class="crm-i fa-pencil" aria-hidden="true"></i>' .
353ffa53 231 '</a>';
32864ccf 232 break;
6a488035 233 }
6a488035 234 }
ad280fb6
JL
235 unset($row['cid']);
236 unset($row['relation_type']);
237 unset($row['rel_id']);
238 unset($row['client_id']);
239 unset($row['source']);
240 array_push($relationships, $row);
6a488035 241 }
ad280fb6
JL
242 $params['total'] = count($relationships);
243
be2fb01f 244 $caseRelationshipsDT = [];
ad280fb6
JL
245 $caseRelationshipsDT['data'] = $relationships;
246 $caseRelationshipsDT['recordsTotal'] = $params['total'];
247 $caseRelationshipsDT['recordsFiltered'] = $params['total'];
248
249 CRM_Utils_JSON::output($caseRelationshipsDT);
ffd93213 250
1d85d241
DL
251 }
252
00be9182 253 public static function convertToCaseActivity() {
be2fb01f
CW
254 $params = ['caseID', 'activityID', 'contactID', 'newSubject', 'targetContactIds', 'mode'];
255 $vals = [];
6a488035 256 foreach ($params as $param) {
9c1bc317 257 $vals[$param] = $_POST[$param] ?? NULL;
6a488035
TO
258 }
259
ecdef330 260 CRM_Utils_JSON::output(self::_convertToCaseActivity($vals));
6a488035
TO
261 }
262
ffd93213 263 /**
c490a46a 264 * @param array $params
ffd93213
EM
265 *
266 * @return array
267 */
00be9182 268 public static function _convertToCaseActivity($params) {
6a488035 269 if (!$params['activityID'] || !$params['caseID']) {
be2fb01f 270 return (['error_msg' => 'required params missing.']);
6a488035
TO
271 }
272
273 $otherActivity = new CRM_Activity_DAO_Activity();
274 $otherActivity->id = $params['activityID'];
275 if (!$otherActivity->find(TRUE)) {
be2fb01f 276 return (['error_msg' => 'activity record is missing.']);
6a488035
TO
277 }
278 $actDateTime = CRM_Utils_Date::isoToMysql($otherActivity->activity_date_time);
279
7808aae6 280 // Create new activity record.
6a488035 281 $mainActivity = new CRM_Activity_DAO_Activity();
be2fb01f 282 $mainActVals = [];
6a488035
TO
283 CRM_Core_DAO::storeValues($otherActivity, $mainActVals);
284
7808aae6 285 // Get new activity subject.
6a488035
TO
286 if (!empty($params['newSubject'])) {
287 $mainActVals['subject'] = $params['newSubject'];
288 }
289
290 $mainActivity->copyValues($mainActVals);
291 $mainActivity->id = NULL;
292 $mainActivity->activity_date_time = $actDateTime;
7808aae6 293 // Make sure this is current revision.
6a488035 294 $mainActivity->is_current_revision = TRUE;
9a30d940
MWMC
295 $mainActivity->original_id = $otherActivity->id;
296 $otherActivity->is_current_revision = FALSE;
6a488035
TO
297
298 $mainActivity->save();
299 $mainActivityId = $mainActivity->id;
300 CRM_Activity_BAO_Activity::logActivityAction($mainActivity);
6a488035 301
7808aae6
SB
302 // Mark previous activity as deleted. If it was a non-case activity
303 // then just change the subject.
be2fb01f 304 if (in_array($params['mode'], [
353ffa53 305 'move',
79d7553f 306 'file',
be2fb01f 307 ])) {
6a488035
TO
308 $caseActivity = new CRM_Case_DAO_CaseActivity();
309 $caseActivity->case_id = $params['caseID'];
310 $caseActivity->activity_id = $otherActivity->id;
311 if ($params['mode'] == 'move' || $caseActivity->find(TRUE)) {
312 $otherActivity->is_deleted = 1;
313 }
314 else {
be2fb01f 315 $otherActivity->subject = ts('(Filed on case %1)', [
c5c263ca 316 1 => $params['caseID'],
be2fb01f 317 ]) . ' ' . $otherActivity->subject;
6a488035 318 }
6a488035
TO
319 $otherActivity->save();
320
6a488035 321 }
6a488035 322
be2fb01f 323 $targetContacts = [];
6a488035
TO
324 if (!empty($params['targetContactIds'])) {
325 $targetContacts = array_unique(explode(',', $params['targetContactIds']));
326 }
f813f78e 327
44f817d4 328 $activityContacts = CRM_Activity_BAO_ActivityContact::buildOptions('record_type_id', 'validate');
a24b3694 329 $sourceID = CRM_Utils_Array::key('Activity Source', $activityContacts);
330 $assigneeID = CRM_Utils_Array::key('Activity Assignees', $activityContacts);
331 $targetID = CRM_Utils_Array::key('Activity Targets', $activityContacts);
332
7b1ec1c6 333 $sourceContactID = CRM_Activity_BAO_Activity::getSourceContactID($params['activityID']);
be2fb01f 334 $src_params = [
7b1ec1c6 335 'activity_id' => $mainActivityId,
336 'contact_id' => $sourceContactID,
21dfd5f5 337 'record_type_id' => $sourceID,
be2fb01f 338 ];
7b1ec1c6 339 CRM_Activity_BAO_ActivityContact::create($src_params);
340
6a488035 341 foreach ($targetContacts as $key => $value) {
be2fb01f 342 $targ_params = [
6a488035 343 'activity_id' => $mainActivityId,
1d85d241 344 'contact_id' => $value,
21dfd5f5 345 'record_type_id' => $targetID,
be2fb01f 346 ];
1d85d241 347 CRM_Activity_BAO_ActivityContact::create($targ_params);
6a488035
TO
348 }
349
c012436d
BS
350 //CRM-21114 retrieve assignee contacts from original case; allow overriding from params
351 $assigneeContacts = CRM_Activity_BAO_ActivityContact::retrieveContactIdsByActivityId($params['activityID'], $assigneeID);
6a488035
TO
352 if (!empty($params['assigneeContactIds'])) {
353 $assigneeContacts = array_unique(explode(',', $params['assigneeContactIds']));
354 }
5402bfb8 355 foreach ($assigneeContacts as $value) {
be2fb01f 356 $assigneeParams = [
6a488035 357 'activity_id' => $mainActivityId,
1d85d241 358 'contact_id' => $value,
21dfd5f5 359 'record_type_id' => $assigneeID,
be2fb01f 360 ];
1d85d241 361 CRM_Activity_BAO_ActivityContact::create($assigneeParams);
6a488035
TO
362 }
363
7808aae6 364 // Attach newly created activity to case.
6a488035
TO
365 $caseActivity = new CRM_Case_DAO_CaseActivity();
366 $caseActivity->case_id = $params['caseID'];
367 $caseActivity->activity_id = $mainActivityId;
368 $caseActivity->save();
369 $error_msg = $caseActivity->_lastError;
6a488035
TO
370
371 $params['mainActivityId'] = $mainActivityId;
372 CRM_Activity_BAO_Activity::copyExtendedActivityData($params);
388995aa 373 CRM_Utils_Hook::post('create', 'CaseActivity', $caseActivity->id, $caseActivity);
6a488035 374
be2fb01f 375 return (['error_msg' => $error_msg, 'newId' => $mainActivity->id]);
6a488035
TO
376 }
377
f2ac86d1 378 /**
379 * Get activities for the contact.
380 *
a413c3db 381 * @throws \CRM_Core_Exception
f2ac86d1 382 */
00be9182 383 public static function getContactActivity() {
be2fb01f 384 $requiredParameters = [
9c3f979f 385 'cid' => 'Integer',
be2fb01f 386 ];
6a488035 387
be2fb01f 388 $optionalParameters = [
9c3f979f 389 'context' => 'String',
23289ddd 390 'activity_type_id' => 'Integer',
391 'activity_type_exclude_id' => 'Integer',
49d4d222 392 'activity_status_id' => 'String',
6e793248 393 'activity_date_time_relative' => 'String',
394 'activity_date_time_low' => 'String',
395 'activity_date_time_high' => 'String',
be2fb01f 396 ];
6a488035 397
9c3f979f 398 $params = CRM_Core_Page_AJAX::defaultSortAndPagerParams();
5d817a13 399 $params += CRM_Core_Page_AJAX::validateParams($requiredParameters, $optionalParameters);
6a488035 400
e8691b2c
MM
401 // To be consistent, the cid parameter should be renamed to contact_id in
402 // the template file, see templates/CRM/Activity/Selector/Selector.tpl
403 $params['contact_id'] = $params['cid'];
404 unset($params['cid']);
405
6a488035
TO
406 // get the contact activities
407 $activities = CRM_Activity_BAO_Activity::getContactActivitySelector($params);
408
c7d77593 409 foreach ($activities['data'] as $key => $value) {
7808aae6 410 // Check if recurring activity.
b53cbfbc 411 if (!empty($value['is_recurring_activity'])) {
04374d9d 412 $repeat = $value['is_recurring_activity'];
be2fb01f 413 $activities['data'][$key]['activity_type'] .= '<br/><span class="bold">' . ts('Repeating (%1 of %2)', [1 => $repeat[0], 2 => $repeat[1]]) . '</span>';
97c7504f 414 }
415 }
416
6a488035 417 // store the activity filter preference CRM-11761
a6d192c8 418 if (Civi::settings()->get('preserve_activity_tab_filter') && ($userID = CRM_Core_Session::getLoggedInContactID())) {
419 unset($optionalParameters['context']);
420 foreach ($optionalParameters as $searchField => $dataType) {
c3a9ccbb 421 $formSearchField = $searchField;
a413c3db 422 if ($searchField === 'activity_type_id') {
c3a9ccbb
JP
423 $formSearchField = 'activity_type_filter_id';
424 }
a413c3db 425 elseif ($searchField === 'activity_type_exclude_id') {
c3a9ccbb
JP
426 $formSearchField = 'activity_type_exclude_filter_id';
427 }
a6d192c8 428 if (!empty($params[$searchField])) {
c1d3e301 429 $activityFilter[$formSearchField] = $params[$searchField];
be2fb01f 430 if (in_array($searchField, ['activity_date_time_low', 'activity_date_time_high'])) {
e04e6d91 431 $activityFilter['activity_date_time_relative'] = 0;
a6d192c8 432 }
a413c3db 433 elseif ($searchField === 'activity_status_id') {
a6d192c8 434 $activityFilter['status_id'] = explode(',', $activityFilter[$searchField]);
435 }
436 }
a6d192c8 437 }
f3548d18 438
c3c679b0 439 Civi::contactSettings()->set('activity_tab_filter', $activityFilter);
6a488035 440 }
1d85d241 441
7d12de7f 442 CRM_Utils_JSON::output($activities);
6a488035 443 }
96025800 444
6a488035 445}