Swap out button/submit inputs for button elements
[civicrm-core.git] / CRM / Contact / Form / Search.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 * Base Search / View form for *all* listing of multiple
20 * contacts
21 */
3efb5b86 22class CRM_Contact_Form_Search extends CRM_Core_Form_Search {
6a488035 23
d424ffde 24 /**
fe482240 25 * list of valid contexts.
6a488035
TO
26 *
27 * @var array
6a488035 28 */
69078420 29 public static $_validContext = NULL;
6a488035
TO
30
31 /**
fe482240 32 * List of values used when we want to display other objects.
6a488035
TO
33 *
34 * @var array
6a488035 35 */
69078420 36 public static $_modeValues = NULL;
6a488035 37
6a488035 38 /**
fe482240 39 * The contextMenu.
6a488035
TO
40 *
41 * @var array
6a488035
TO
42 */
43 protected $_contextMenu;
44
45 /**
fe482240 46 * The groupId retrieved from the GET vars.
6a488035
TO
47 *
48 * @var int
6a488035
TO
49 */
50 public $_groupID;
51
52 /**
fe482240 53 * The Group ID belonging to Add Member to group ID.
6a488035
TO
54 * retrieved from the GET vars
55 *
56 * @var int
6a488035
TO
57 */
58 protected $_amtgID;
59
60 /**
fe482240 61 * The saved search ID retrieved from the GET vars.
6a488035
TO
62 *
63 * @var int
6a488035
TO
64 */
65 protected $_ssID;
66
6a488035 67 /**
fe482240 68 * The group elements.
6a488035
TO
69 *
70 * @var array
6a488035
TO
71 */
72 public $_group;
73 public $_groupElement;
6a488035
TO
74
75 /**
fe482240 76 * The tag elements.
6a488035
TO
77 *
78 * @var array
6a488035
TO
79 */
80 public $_tag;
81 public $_tagElement;
82
6a488035 83 /**
fe482240 84 * The params used for search.
6a488035
TO
85 *
86 * @var array
6a488035
TO
87 */
88 protected $_params;
89
90 /**
fe482240 91 * The return properties used for search.
6a488035
TO
92 *
93 * @var array
6a488035
TO
94 */
95 protected $_returnProperties;
96
97 /**
fe482240 98 * The sort by character.
6a488035
TO
99 *
100 * @var string
6a488035
TO
101 */
102 protected $_sortByCharacter;
103
104 /**
fe482240 105 * The profile group id used for display.
6a488035 106 *
9f266042 107 * @var int
6a488035
TO
108 */
109 protected $_ufGroupID;
110
c490a46a 111 /**
100fef9d 112 * Csv - common search values
c490a46a
CW
113 *
114 * @var array
c490a46a 115 */
69078420 116 public static $csv = ['contact_type', 'group', 'tag'];
6a488035
TO
117
118 /**
9f266042 119 * How to display the results. Should we display as contributons, members, cases etc.
120 *
121 * @var string
6a488035
TO
122 */
123 protected $_componentMode;
124
125 /**
9f266042 126 * What operator should we use, AND or OR.
127 *
128 * @var string
6a488035
TO
129 */
130 protected $_operator;
131
132 protected $_modeValue;
133
90c13aa2
SL
134 /**
135 * Declare entity reference fields as they will need to be converted to using 'IN'.
136 *
137 * @var array
138 */
be2fb01f 139 protected $entityReferenceFields = ['event_id', 'membership_type_id'];
90c13aa2 140
6a488035 141 /**
fe482240 142 * Name of the selector to use.
69078420 143 * @var string
6a488035 144 */
69078420 145 public static $_selectorName = 'CRM_Contact_Selector';
6a488035
TO
146 protected $_customSearchID = NULL;
147 protected $_customSearchClass = NULL;
148
be2fb01f 149 protected $_openedPanes = [];
d5965a37 150
6e62b28c
TM
151 /**
152 * Explicitly declare the entity api name.
153 */
154 public function getDefaultEntity() {
155 return 'Contact';
156 }
6a488035
TO
157
158 /**
fe482240 159 * Define the set of valid contexts that the search form operates on.
6a488035 160 *
a6c01b45
CW
161 * @return array
162 * the valid context set and the titles
6a488035 163 */
00be9182 164 public static function &validContext() {
6a488035 165 if (!(self::$_validContext)) {
be2fb01f 166 self::$_validContext = [
6a488035
TO
167 'smog' => 'Show members of group',
168 'amtg' => 'Add members to group',
169 'basic' => 'Basic Search',
170 'search' => 'Search',
171 'builder' => 'Search Builder',
172 'advanced' => 'Advanced Search',
173 'custom' => 'Custom Search',
be2fb01f 174 ];
6a488035
TO
175 }
176 return self::$_validContext;
177 }
178
86538308
EM
179 /**
180 * @param $context
181 *
182 * @return bool
183 */
00be9182 184 public static function isSearchContext($context) {
6a488035 185 $searchContext = CRM_Utils_Array::value($context, self::validContext());
dc0aa487 186 return (bool) $searchContext;
6a488035
TO
187 }
188
00be9182 189 public static function setModeValues() {
be2fb01f
CW
190 self::$_modeValues = [
191 CRM_Contact_BAO_Query::MODE_CONTACTS => [
a6c6c64f
JP
192 'selectorName' => self::$_selectorName,
193 'selectorLabel' => ts('Contacts'),
194 'taskFile' => 'CRM/Contact/Form/Search/ResultTasks.tpl',
195 'taskContext' => NULL,
196 'resultFile' => 'CRM/Contact/Form/Selector.tpl',
197 'resultContext' => NULL,
198 'taskClassName' => 'CRM_Contact_Task',
199 'component' => '',
be2fb01f
CW
200 ],
201 CRM_Contact_BAO_Query::MODE_CONTRIBUTE => [
a6c6c64f
JP
202 'selectorName' => 'CRM_Contribute_Selector_Search',
203 'selectorLabel' => ts('Contributions'),
204 'taskFile' => 'CRM/common/searchResultTasks.tpl',
205 'taskContext' => 'Contribution',
206 'resultFile' => 'CRM/Contribute/Form/Selector.tpl',
207 'resultContext' => 'Search',
208 'taskClassName' => 'CRM_Contribute_Task',
209 'component' => 'CiviContribute',
be2fb01f
CW
210 ],
211 CRM_Contact_BAO_Query::MODE_EVENT => [
a6c6c64f
JP
212 'selectorName' => 'CRM_Event_Selector_Search',
213 'selectorLabel' => ts('Event Participants'),
214 'taskFile' => 'CRM/common/searchResultTasks.tpl',
215 'taskContext' => NULL,
216 'resultFile' => 'CRM/Event/Form/Selector.tpl',
217 'resultContext' => 'Search',
218 'taskClassName' => 'CRM_Event_Task',
219 'component' => 'CiviEvent',
be2fb01f
CW
220 ],
221 CRM_Contact_BAO_Query::MODE_ACTIVITY => [
a6c6c64f
JP
222 'selectorName' => 'CRM_Activity_Selector_Search',
223 'selectorLabel' => ts('Activities'),
224 'taskFile' => 'CRM/common/searchResultTasks.tpl',
225 'taskContext' => NULL,
226 'resultFile' => 'CRM/Activity/Form/Selector.tpl',
227 'resultContext' => 'Search',
228 'taskClassName' => 'CRM_Activity_Task',
229 'component' => 'activity',
be2fb01f
CW
230 ],
231 CRM_Contact_BAO_Query::MODE_MEMBER => [
a6c6c64f
JP
232 'selectorName' => 'CRM_Member_Selector_Search',
233 'selectorLabel' => ts('Memberships'),
234 'taskFile' => "CRM/common/searchResultTasks.tpl",
235 'taskContext' => NULL,
236 'resultFile' => 'CRM/Member/Form/Selector.tpl',
237 'resultContext' => 'Search',
238 'taskClassName' => 'CRM_Member_Task',
239 'component' => 'CiviMember',
be2fb01f
CW
240 ],
241 CRM_Contact_BAO_Query::MODE_CASE => [
a6c6c64f
JP
242 'selectorName' => 'CRM_Case_Selector_Search',
243 'selectorLabel' => ts('Cases'),
244 'taskFile' => "CRM/common/searchResultTasks.tpl",
245 'taskContext' => NULL,
246 'resultFile' => 'CRM/Case/Form/Selector.tpl',
247 'resultContext' => 'Search',
248 'taskClassName' => 'CRM_Case_Task',
249 'component' => 'CiviCase',
be2fb01f
CW
250 ],
251 CRM_Contact_BAO_Query::MODE_CONTACTSRELATED => [
a6c6c64f
JP
252 'selectorName' => self::$_selectorName,
253 'selectorLabel' => ts('Related Contacts'),
254 'taskFile' => 'CRM/Contact/Form/Search/ResultTasks.tpl',
255 'taskContext' => NULL,
256 'resultFile' => 'CRM/Contact/Form/Selector.tpl',
257 'resultContext' => NULL,
258 'taskClassName' => 'CRM_Contact_Task',
259 'component' => 'related_contact',
be2fb01f
CW
260 ],
261 CRM_Contact_BAO_Query::MODE_MAILING => [
a6c6c64f
JP
262 'selectorName' => 'CRM_Mailing_Selector_Search',
263 'selectorLabel' => ts('Mailings'),
264 'taskFile' => "CRM/common/searchResultTasks.tpl",
265 'taskContext' => NULL,
266 'resultFile' => 'CRM/Mailing/Form/Selector.tpl',
267 'resultContext' => 'Search',
268 'taskClassName' => 'CRM_Mailing_Task',
269 'component' => 'CiviMail',
be2fb01f
CW
270 ],
271 ];
6a488035
TO
272 }
273
86538308 274 /**
a6c6c64f
JP
275 * Get the metadata for the query mode (this includes task class names)
276 *
86538308
EM
277 * @param int $mode
278 *
a6c6c64f
JP
279 * @return array
280 * @throws \CRM_Core_Exception
86538308 281 */
eda34f9b 282 public static function getModeValue($mode = CRM_Contact_BAO_Query::MODE_CONTACTS) {
a6c6c64f
JP
283 $searchPane = CRM_Utils_Request::retrieve('searchPane', 'String');
284 if (!empty($searchPane)) {
285 $mode = array_search($searchPane, self::getModeToComponentMapping());
286 }
6a488035 287
a6c6c64f 288 self::setModeValues();
48016b17
D
289 // Note $mode might === FALSE because array_search above failed, e.g. for searchPane='location'
290 if (empty(self::$_modeValues[$mode])) {
eda34f9b 291 $mode = CRM_Contact_BAO_Query::MODE_CONTACTS;
6a488035
TO
292 }
293
294 return self::$_modeValues[$mode];
295 }
296
e30af20d 297 /**
298 * Get a mapping of modes to components.
299 *
300 * This will map the integers to the components. Contact has an empty component
301 * an pseudo-components exist for activity & related_contact.
302 *
303 * @return array
304 */
305 public static function getModeToComponentMapping() {
306 $mapping = [];
307 self::setModeValues();
308
309 foreach (self::$_modeValues as $id => $metadata) {
310 $mapping[$id] = $metadata['component'];
311 }
312 return $mapping;
313 }
314
86538308
EM
315 /**
316 * @return array
317 */
00be9182 318 public static function getModeSelect() {
6a488035
TO
319 self::setModeValues();
320
8424571d 321 $enabledComponents = CRM_Core_Component::getEnabledComponents();
be2fb01f 322 $componentModes = [];
6a488035 323 foreach (self::$_modeValues as $id => & $value) {
8424571d
PN
324 if (strpos($value['component'], 'Civi') !== FALSE
325 && !array_key_exists($value['component'], $enabledComponents)
326 ) {
327 continue;
328 }
eda34f9b 329 $componentModes[$id] = $value['selectorLabel'];
6a488035
TO
330 }
331
eda34f9b
MW
332 // unset disabled components
333 if (!array_key_exists('CiviMail', $enabledComponents)) {
334 unset($componentModes[CRM_Contact_BAO_Query::MODE_MAILING]);
335 }
336
337 // unset contributions or participants if user does not have permission on them
6a488035 338 if (!CRM_Core_Permission::access('CiviContribute')) {
eda34f9b 339 unset($componentModes[CRM_Contact_BAO_Query::MODE_CONTRIBUTE]);
6a488035
TO
340 }
341
342 if (!CRM_Core_Permission::access('CiviEvent')) {
eda34f9b
MW
343 unset($componentModes[CRM_Contact_BAO_Query::MODE_EVENT]);
344 }
345
346 if (!CRM_Core_Permission::access('CiviMember')) {
347 unset($componentModes[CRM_Contact_BAO_Query::MODE_MEMBER]);
6a488035
TO
348 }
349
350 if (!CRM_Core_Permission::check('view all activities')) {
eda34f9b 351 unset($componentModes[CRM_Contact_BAO_Query::MODE_ACTIVITY]);
6a488035 352 }
eda34f9b
MW
353
354 return $componentModes;
6a488035
TO
355 }
356
357 /**
32795e82 358 * Builds the list of tasks or actions that a searcher can perform on a result set.
6a488035 359 *
32795e82 360 * @return array
6a488035 361 */
e98a9804 362 public function buildTaskList() {
eda34f9b 363 // amtg = 'Add members to group'
7a3978aa 364 if ($this->_context !== 'amtg') {
eda34f9b
MW
365 $taskParams['deletedContacts'] = FALSE;
366 if ($this->_componentMode == CRM_Contact_BAO_Query::MODE_CONTACTS || $this->_componentMode == CRM_Contact_BAO_Query::MODE_CONTACTSRELATED) {
9c1bc317 367 $taskParams['deletedContacts'] = $this->_formValues['deleted_contacts'] ?? NULL;
7a3978aa 368 }
eda34f9b 369 $className = $this->_modeValue['taskClassName'];
2e1f50d6 370 $taskParams['ssID'] = $this->_ssID ?? NULL;
eda34f9b 371 $this->_taskList += $className::permissionedTaskTitles(CRM_Core_Permission::getPermission(), $taskParams);
32795e82
FG
372 }
373
7a3978aa 374 return $this->_taskList;
32795e82 375 }
6a488035 376
32795e82
FG
377 /**
378 * Build the common elements between the search/advanced form.
6a488035 379 */
00be9182 380 public function buildQuickForm() {
3efb5b86 381 parent::buildQuickForm();
eda34f9b 382
6a488035 383 // some tasks.. what do we want to do with the selected contacts ?
eda34f9b 384 $this->_taskList = $this->buildTaskList();
6a488035
TO
385
386 if (isset($this->_ssID)) {
79d7553f 387 $search_custom_id
388 = CRM_Core_DAO::getFieldValue('CRM_Contact_DAO_SavedSearch', $this->_ssID, 'search_custom_id');
6a488035 389
be2fb01f 390 $savedSearchValues = [
6a488035
TO
391 'id' => $this->_ssID,
392 'name' => CRM_Contact_BAO_SavedSearch::getName($this->_ssID, 'title'),
393 'search_custom_id' => $search_custom_id,
be2fb01f 394 ];
6a488035
TO
395 $this->assign_by_ref('savedSearch', $savedSearchValues);
396 $this->assign('ssID', $this->_ssID);
397 }
398
399 if ($this->_context === 'smog') {
400 // CRM-11788, we might want to do this for all of search where force=1
9c1bc317
CW
401 $formQFKey = $this->_formValues['qfKey'] ?? NULL;
402 $getQFKey = $_GET['qfKey'] ?? NULL;
403 $postQFKey = $_POST['qfKey'] ?? NULL;
6a488035
TO
404 if ($formQFKey && empty($getQFKey) && empty($postQFKey)) {
405 $url = CRM_Utils_System::makeURL('qfKey') . $formQFKey;
406 CRM_Utils_System::redirect($url);
407 }
838d0ff6 408 $permissionForGroup = FALSE;
6a488035
TO
409
410 if (!empty($this->_groupID)) {
6a488035
TO
411 // check if user has permission to edit members of this group
412 $permission = CRM_Contact_BAO_Group::checkPermission($this->_groupID);
413 if ($permission && in_array(CRM_Core_Permission::EDIT, $permission)) {
414 $permissionForGroup = TRUE;
415 }
416
417 // check if _groupID exists, it might not if
418 // we are displaying a hidden group
419 if (!isset($this->_group[$this->_groupID])) {
79d7553f 420 $this->_group[$this->_groupID]
421 = CRM_Core_DAO::getFieldValue('CRM_Contact_DAO_Group', $this->_groupID, 'title');
6a488035
TO
422 }
423
6a488035 424 // set the group title
be2fb01f 425 $groupValues = ['id' => $this->_groupID, 'title' => $this->_group[$this->_groupID]];
6a488035
TO
426 $this->assign_by_ref('group', $groupValues);
427
428 // also set ssID if this is a saved search
429 $ssID = CRM_Core_DAO::getFieldValue('CRM_Contact_DAO_Group', $this->_groupID, 'saved_search_id');
430 $this->assign('ssID', $ssID);
431
432 //get the saved search mapping id
433 if ($ssID) {
434 $this->_ssID = $ssID;
435 $ssMappingId = CRM_Core_DAO::getFieldValue('CRM_Contact_DAO_SavedSearch', $ssID, 'mapping_id');
436 $this->assign('ssMappingID', $ssMappingId);
437 }
438
439 // Set dynamic page title for 'Show Members of Group'
be2fb01f 440 CRM_Utils_System::setTitle(ts('Contacts in Group: %1', [1 => $this->_group[$this->_groupID]]));
6a488035
TO
441 }
442
be2fb01f 443 $group_contact_status = [];
6a488035
TO
444 foreach (CRM_Core_SelectValues::groupContactStatus() as $k => $v) {
445 if (!empty($k)) {
446 $group_contact_status[] = $this->createElement('checkbox', $k, NULL, $v);
447 }
448 }
449 $this->addGroup($group_contact_status,
450 'group_contact_status', ts('Group Status')
451 );
452
e42f6d71 453 $this->assign('permissionEditSmartGroup', CRM_Core_Permission::check('edit groups'));
838d0ff6 454 $this->assign('permissionedForGroup', $permissionForGroup);
6a488035
TO
455 }
456
457 // add the go button for the action form, note it is of type 'next' rather than of type 'submit'
458 if ($this->_context === 'amtg') {
353ffa53
TO
459 // check if _groupID exists, it might not if
460 // we are displaying a hidden group
6a488035
TO
461 if (!isset($this->_group[$this->_amtgID])) {
462 $this->assign('permissionedForGroup', FALSE);
79d7553f 463 $this->_group[$this->_amtgID]
464 = CRM_Core_DAO::getFieldValue('CRM_Contact_DAO_Group', $this->_amtgID, 'title');
6a488035
TO
465 }
466
467 // Set dynamic page title for 'Add Members Group'
be2fb01f 468 CRM_Utils_System::setTitle(ts('Add to Group: %1', [1 => $this->_group[$this->_amtgID]]));
6a488035 469 // also set the group title and freeze the action task with Add Members to Group
be2fb01f 470 $groupValues = ['id' => $this->_amtgID, 'title' => $this->_group[$this->_amtgID]];
6a488035 471 $this->assign_by_ref('group', $groupValues);
cbd83dde 472 $this->add('xbutton', $this->_actionButtonName, ts('Add Contacts to %1', [1 => $this->_group[$this->_amtgID]]),
be2fb01f 473 [
cbd83dde 474 'type' => 'submit',
97e557d7 475 'class' => 'crm-form-submit',
be2fb01f 476 ]
6a488035 477 );
eda34f9b 478 $this->add('hidden', 'task', CRM_Contact_Task::GROUP_ADD);
be2fb01f 479 $selectedRowsRadio = $this->addElement('radio', 'radio_ts', NULL, '', 'ts_sel', ['checked' => 'checked']);
c2d91642 480 $allRowsRadio = $this->addElement('radio', 'radio_ts', NULL, '', 'ts_all');
481 $this->assign('ts_sel_id', $selectedRowsRadio->_attributes['id']);
482 $this->assign('ts_all_id', $allRowsRadio->_attributes['id']);
6a488035 483 }
6a488035 484
be2fb01f 485 $selectedContactIds = [];
9c1bc317 486 $qfKeyParam = $this->_formValues['qfKey'] ?? NULL;
2ebb21f0 487 // We use ajax to handle selections only if the search results component_mode is set to "contacts"
eda34f9b 488 if ($qfKeyParam && ($this->get('component_mode') <= CRM_Contact_BAO_Query::MODE_CONTACTS || $this->get('component_mode') == CRM_Contact_BAO_Query::MODE_CONTACTSRELATED)) {
2ebb21f0 489 $this->addClass('crm-ajax-selection-form');
6a488035 490 $qfKeyParam = "civicrm search {$qfKeyParam}";
b7994703 491 $selectedContactIdsArr = Civi::service('prevnext')->getSelection($qfKeyParam);
6a488035
TO
492 $selectedContactIds = array_keys($selectedContactIdsArr[$qfKeyParam]);
493 }
494
495 $this->assign_by_ref('selectedContactIds', $selectedContactIds);
496
6a488035
TO
497 $rows = $this->get('rows');
498
499 if (is_array($rows)) {
6e8d1c11 500 $this->addRowSelectors($rows);
6a488035
TO
501 }
502
6a488035
TO
503 }
504
505 /**
fe482240 506 * Processing needed for buildForm and later.
86f20cc1 507 *
508 * @throws \CRM_Core_Exception
6a488035 509 */
00be9182 510 public function preProcess() {
6a488035
TO
511 // set the various class variables
512
513 $this->_group = CRM_Core_PseudoConstant::group();
514
6a488035
TO
515 $this->_tag = CRM_Core_BAO_Tag::getTags();
516 $this->_done = FALSE;
517
518 /*
deb50ba3
CW
519 * we allow the controller to set force/reset externally, useful when we are being
520 * driven by the wizard framework
521 */
6a488035 522
8d53ca1d
TO
523 $this->_reset = CRM_Utils_Request::retrieve('reset', 'Boolean');
524
525 $this->_force = CRM_Utils_Request::retrieve('force', 'Boolean');
6a488035
TO
526 $this->_groupID = CRM_Utils_Request::retrieve('gid', 'Positive', $this);
527 $this->_amtgID = CRM_Utils_Request::retrieve('amtgID', 'Positive', $this);
8d53ca1d 528 $this->_ssID = CRM_Utils_Request::retrieve('ssID', 'Positive', $this);
6a488035
TO
529 $this->_sortByCharacter = CRM_Utils_Request::retrieve('sortByCharacter', 'String', $this);
530 $this->_ufGroupID = CRM_Utils_Request::retrieve('id', 'Positive', $this);
eda34f9b
MW
531 $this->_componentMode = CRM_Utils_Request::retrieve('component_mode', 'Positive', $this, FALSE, CRM_Contact_BAO_Query::MODE_CONTACTS, $_REQUEST);
532 $this->_operator = CRM_Utils_Request::retrieve('operator', 'String', $this, FALSE, CRM_Contact_BAO_Query::SEARCH_OPERATOR_AND, 'REQUEST');
6a488035 533
96db3ecb
SL
534 if (!empty($this->_ssID) && !CRM_Core_Permission::check('edit groups')) {
535 CRM_Core_Error::statusBounce(ts('You do not have permission to modify smart groups'));
536 }
537
6a488035
TO
538 /**
539 * set the button names
540 */
6a488035
TO
541 $this->_actionButtonName = $this->getButtonName('next', 'action');
542
6a488035
TO
543 $this->assign('actionButtonName', $this->_actionButtonName);
544
6a488035
TO
545 // if we dont get this from the url, use default if one exsts
546 $config = CRM_Core_Config::singleton();
547 if ($this->_ufGroupID == NULL &&
548 $config->defaultSearchProfileID != NULL
549 ) {
550 $this->_ufGroupID = $config->defaultSearchProfileID;
551 }
552
553 // assign context to drive the template display, make sure context is valid
8d53ca1d 554 $this->_context = CRM_Utils_Request::retrieve('context', 'Alphanumeric', $this, FALSE, 'search');
f3acfdd9 555 if (!array_key_exists($this->_context, self::validContext())) {
6a488035
TO
556 $this->_context = 'search';
557 }
558 $this->set('context', $this->_context);
559 $this->assign('context', $this->_context);
560
561 $this->_modeValue = self::getModeValue($this->_componentMode);
562 $this->assign($this->_modeValue);
563
564 $this->set('selectorName', self::$_selectorName);
565
566 // get user submitted values
567 // get it from controller only if form has been submitted, else preProcess has set this
4eeb9a5b
TO
568 // $this->controller->isModal( ) returns TRUE if page is
569 // valid, i.e all the validations are TRUE
6a488035
TO
570
571 if (!empty($_POST) && !$this->controller->isModal()) {
572 $this->_formValues = $this->controller->exportValues($this->_name);
573
574 $this->normalizeFormValues();
90c13aa2 575 $this->_params = CRM_Contact_BAO_Query::convertFormValues($this->_formValues, 0, FALSE, NULL, $this->entityReferenceFields);
6a488035
TO
576 $this->_returnProperties = &$this->returnProperties();
577
578 // also get the uf group id directly from the post value
579 $this->_ufGroupID = CRM_Utils_Array::value('uf_group_id', $_POST, $this->_ufGroupID);
580 $this->_formValues['uf_group_id'] = $this->_ufGroupID;
581 $this->set('id', $this->_ufGroupID);
582
583 // also get the object mode directly from the post value
584 $this->_componentMode = CRM_Utils_Array::value('component_mode', $_POST, $this->_componentMode);
585
586 // also get the operator from the post value if set
587 $this->_operator = CRM_Utils_Array::value('operator', $_POST, $this->_operator);
588 $this->_formValues['operator'] = $this->_operator;
589 $this->set('operator', $this->_operator);
590 }
591 else {
592 $this->_formValues = $this->get('formValues');
90c13aa2 593 $this->_params = CRM_Contact_BAO_Query::convertFormValues($this->_formValues, 0, FALSE, NULL, $this->entityReferenceFields);
6a488035
TO
594 $this->_returnProperties = &$this->returnProperties();
595 if (!empty($this->_ufGroupID)) {
596 $this->set('id', $this->_ufGroupID);
597 }
598 }
599
600 if (empty($this->_formValues)) {
601 //check if group is a smart group (fix for CRM-1255)
602 if ($this->_groupID) {
603 if ($ssId = CRM_Core_DAO::getFieldValue('CRM_Contact_DAO_Group', $this->_groupID, 'saved_search_id')) {
604 $this->_ssID = $ssId;
605 }
606 }
607
608 // fix for CRM-1907
86f20cc1 609 if (isset($this->_ssID) && $this->_context !== 'smog') {
6a488035
TO
610 // we only retrieve the saved search values if out current values are null
611 $this->_formValues = CRM_Contact_BAO_SavedSearch::getFormValues($this->_ssID);
612
613 //fix for CRM-1505
614 if (CRM_Core_DAO::getFieldValue('CRM_Contact_DAO_SavedSearch', $this->_ssID, 'mapping_id')) {
615 $this->_params = CRM_Contact_BAO_SavedSearch::getSearchParams($this->_ssID);
616 }
617 else {
618 $this->_params = CRM_Contact_BAO_Query::convertFormValues($this->_formValues);
619 }
620 $this->_returnProperties = &$this->returnProperties();
621 }
622 else {
623 if (isset($this->_ufGroupID)) {
624 // also set the uf group id if not already present
625 $this->_formValues['uf_group_id'] = $this->_ufGroupID;
626 }
627 if (isset($this->_componentMode)) {
628 $this->_formValues['component_mode'] = $this->_componentMode;
629 }
630 if (isset($this->_operator)) {
631 $this->_formValues['operator'] = $this->_operator;
632 }
633
634 // FIXME: we should generalise in a way that components could inject url-filters
635 // just like they build their own form elements
be2fb01f 636 foreach ([
69078420
SL
637 'mailing_id',
638 'mailing_delivery_status',
639 'mailing_open_status',
640 'mailing_click_status',
641 'mailing_reply_status',
642 'mailing_optout',
643 'mailing_forward',
644 'mailing_unsubscribe',
645 'mailing_date_low',
646 'mailing_date_high',
0058ef3f
SL
647 'mailing_job_start_date_low',
648 'mailing_job_start_date_high',
649 'mailing_job_start_date_relative',
69078420 650 ] as $mailingFilter) {
6a488035
TO
651 $type = 'String';
652 if ($mailingFilter == 'mailing_id' &&
653 $filterVal = CRM_Utils_Request::retrieve('mailing_id', 'Positive', $this)
654 ) {
be2fb01f 655 $this->_formValues[$mailingFilter] = [$filterVal];
6a488035
TO
656 }
657 elseif ($filterVal = CRM_Utils_Request::retrieve($mailingFilter, $type, $this)) {
658 $this->_formValues[$mailingFilter] = $filterVal;
659 }
660 if ($filterVal) {
661 $this->_openedPanes['Mailings'] = 1;
662 $this->_formValues['hidden_CiviMail'] = 1;
663 }
664 }
665 }
666 }
667 $this->assign('id',
668 CRM_Utils_Array::value('uf_group_id', $this->_formValues)
669 );
eda34f9b 670 $operator = CRM_Utils_Array::value('operator', $this->_formValues, CRM_Contact_BAO_Query::SEARCH_OPERATOR_AND);
6a488035 671 $this->set('queryOperator', $operator);
eda34f9b 672 if ($operator == CRM_Contact_BAO_Query::SEARCH_OPERATOR_OR) {
6a488035
TO
673 $this->assign('operator', ts('OR'));
674 }
675 else {
676 $this->assign('operator', ts('AND'));
677 }
678
679 // show the context menu only when we’re not searching for deleted contacts; CRM-5673
a7488080 680 if (empty($this->_formValues['deleted_contacts'])) {
6a488035 681 $menuItems = CRM_Contact_BAO_Contact::contextMenu();
be2fb01f
CW
682 $primaryActions = CRM_Utils_Array::value('primaryActions', $menuItems, []);
683 $this->_contextMenu = CRM_Utils_Array::value('moreActions', $menuItems, []);
6a488035
TO
684 $this->assign('contextMenu', $primaryActions + $this->_contextMenu);
685 }
686
687 if (!isset($this->_componentMode)) {
688 $this->_componentMode = CRM_Contact_BAO_Query::MODE_CONTACTS;
689 }
6a488035 690 self::$_selectorName = $this->_modeValue['selectorName'];
a6c6c64f 691 self::setModeValues();
6a488035
TO
692
693 $setDynamic = FALSE;
694 if (strpos(self::$_selectorName, 'CRM_Contact_Selector') !== FALSE) {
695 $selector = new self::$_selectorName(
696 $this->_customSearchClass,
697 $this->_formValues,
698 $this->_params,
699 $this->_returnProperties,
700 $this->_action,
ab8a593e 701 FALSE, TRUE,
6a488035
TO
702 $this->_context,
703 $this->_contextMenu
704 );
705 $setDynamic = TRUE;
706 }
707 else {
708 $selector = new self::$_selectorName(
709 $this->_params,
710 $this->_action,
ab8a593e 711 NULL, FALSE, NULL,
6a488035
TO
712 "search", "advanced"
713 );
714 }
715
716 $selector->setKey($this->controller->_key);
717
718 $controller = new CRM_Contact_Selector_Controller($selector,
719 $this->get(CRM_Utils_Pager::PAGE_ID),
720 $this->get(CRM_Utils_Sort::SORT_ID),
721 CRM_Core_Action::VIEW,
722 $this,
723 CRM_Core_Selector_Controller::TRANSFER
724 );
725 $controller->setEmbedded(TRUE);
726 $controller->setDynamicAction($setDynamic);
727
728 if ($this->_force) {
f90f8d32 729 $this->loadMetadata();
6a488035
TO
730 $this->postProcess();
731
732 /*
deb50ba3
CW
733 * Note that we repeat this, since the search creates and stores
734 * values that potentially change the controller behavior. i.e. things
735 * like totalCount etc
736 */
6a488035
TO
737 $controller = new CRM_Contact_Selector_Controller($selector,
738 $this->get(CRM_Utils_Pager::PAGE_ID),
590084ae 739 $this->getSortID(),
6a488035
TO
740 CRM_Core_Action::VIEW, $this, CRM_Core_Selector_Controller::TRANSFER
741 );
742 $controller->setEmbedded(TRUE);
743 $controller->setDynamicAction($setDynamic);
744 }
745
746 $controller->moveFromSessionToTemplate();
747 }
748
6a488035 749 /**
fe482240 750 * Common post processing.
6a488035 751 */
00be9182 752 public function postProcess() {
6a488035
TO
753 /*
754 * sometime we do a postProcess early on, so we dont need to repeat it
755 * this will most likely introduce some more bugs :(
756 */
757
758 if ($this->_done) {
759 return;
760 }
761 $this->_done = TRUE;
762
763 //for prev/next pagination
a3d827a7 764 $crmPID = CRM_Utils_Request::retrieve('crmPID', 'Integer');
6a488035 765
6a488035
TO
766 //get the button name
767 $buttonName = $this->controller->getButtonName();
768
8cc574cf 769 if (isset($this->_ufGroupID) && empty($this->_formValues['uf_group_id'])) {
6a488035
TO
770 $this->_formValues['uf_group_id'] = $this->_ufGroupID;
771 }
772
8cc574cf 773 if (isset($this->_componentMode) && empty($this->_formValues['component_mode'])) {
6a488035
TO
774 $this->_formValues['component_mode'] = $this->_componentMode;
775 }
776
8cc574cf 777 if (isset($this->_operator) && empty($this->_formValues['operator'])) {
6a488035
TO
778 $this->_formValues['operator'] = $this->_operator;
779 }
780
a7488080 781 if (empty($this->_formValues['qfKey'])) {
6a488035
TO
782 $this->_formValues['qfKey'] = $this->controller->_key;
783 }
784
785 if (!CRM_Core_Permission::check('access deleted contacts')) {
786 unset($this->_formValues['deleted_contacts']);
787 }
788
789 $this->set('type', $this->_action);
790 $this->set('formValues', $this->_formValues);
791 $this->set('queryParams', $this->_params);
792 $this->set('returnProperties', $this->_returnProperties);
793
e341bbee 794 if ($buttonName == $this->_actionButtonName) {
6a488035
TO
795 // check actionName and if next, then do not repeat a search, since we are going to the next page
796 // hack, make sure we reset the task values
797 $stateMachine = $this->controller->getStateMachine();
798 $formName = $stateMachine->getTaskFormName();
799 $this->controller->resetPage($formName);
800 return;
801 }
802 else {
804beb78 803 if (array_key_exists($this->getButtonName('refresh'), $_POST) ||
af7e94a6 804 ($this->_force && !$crmPID)
805 ) {
806 //reset the cache table for new search
807 $cacheKey = "civicrm search {$this->controller->_key}";
808 Civi::service('prevnext')->deleteItem(NULL, $cacheKey);
809 }
6a488035
TO
810 $output = CRM_Core_Selector_Controller::SESSION;
811
812 // create the selector, controller and run - store results in session
813 $searchChildGroups = TRUE;
814 if ($this->get('isAdvanced')) {
815 $searchChildGroups = FALSE;
816 }
817
818 $setDynamic = FALSE;
819
820 if (strpos(self::$_selectorName, 'CRM_Contact_Selector') !== FALSE) {
79d7553f 821 $selector = new self::$_selectorName(
6a488035
TO
822 $this->_customSearchClass,
823 $this->_formValues,
824 $this->_params,
825 $this->_returnProperties,
826 $this->_action,
ab8a593e 827 FALSE,
6a488035
TO
828 $searchChildGroups,
829 $this->_context,
830 $this->_contextMenu
831 );
832 $setDynamic = TRUE;
833 }
834 else {
d3e86119 835 $selector = new self::$_selectorName(
6a488035
TO
836 $this->_params,
837 $this->_action,
e60f24eb 838 NULL,
ab8a593e 839 FALSE,
e60f24eb 840 NULL,
6a488035
TO
841 "search",
842 "advanced"
843 );
844 }
845
846 $selector->setKey($this->controller->_key);
847
848 // added the sorting character to the form array
6a488035
TO
849 $config = CRM_Core_Config::singleton();
850 // do this only for contact search
851 if ($setDynamic && $config->includeAlphabeticalPager) {
e166ff79
CW
852 // Don't recompute if we are just paging/sorting
853 if ($this->_reset || (empty($_GET['crmPID']) && empty($_GET['crmSID']) && !$this->_sortByCharacter)) {
6a488035
TO
854 $aToZBar = CRM_Utils_PagerAToZ::getAToZBar($selector, $this->_sortByCharacter);
855 $this->set('AToZBar', $aToZBar);
856 }
857 }
858
6a488035
TO
859 $controller = new CRM_Contact_Selector_Controller($selector,
860 $this->get(CRM_Utils_Pager::PAGE_ID),
590084ae 861 $this->getSortID(),
6a488035
TO
862 CRM_Core_Action::VIEW,
863 $this,
864 $output
865 );
866 $controller->setEmbedded(TRUE);
867 $controller->setDynamicAction($setDynamic);
868 $controller->run();
869 }
870 }
871
86538308 872 /**
e60f24eb 873 * @return NULL
86538308 874 */
00be9182 875 public function &returnProperties() {
6a488035
TO
876 return CRM_Core_DAO::$_nullObject;
877 }
878
879 /**
880 * Return a descriptive name for the page, used in wizard header
881 *
882 * @return string
6a488035 883 */
00be9182 884 public function getTitle() {
6a488035
TO
885 return ts('Search');
886 }
96025800 887
e26bc72e
SL
888 /**
889 * Check Access for a component
890 * @param string $component
891 * @return bool
892 */
893 protected static function checkComponentAccess($component) {
894 $enabledComponents = CRM_Core_Component::getEnabledComponents();
895 if (!array_key_exists($component, $enabledComponents)) {
896 return FALSE;
897 }
898 return CRM_Core_Permission::access($component);
899 }
900
f90f8d32 901 /**
902 * Load metadata for fields on the form.
903 *
904 * @throws \CiviCRM_API3_Exception
905 */
906 protected function loadMetadata() {
f90f8d32 907 // can't by pass acls by passing search criteria in the url.
e26bc72e
SL
908 if (self::checkComponentAccess('CiviContribute')) {
909 $this->addSearchFieldMetadata(['Contribution' => CRM_Contribute_BAO_Query::getSearchFieldMetadata()]);
910 $this->addSearchFieldMetadata(['ContributionRecur' => CRM_Contribute_BAO_ContributionRecur::getContributionRecurSearchFieldMetadata()]);
911 }
50cdce00
SL
912 if (self::checkComponentAccess('CiviPledge')) {
913 $this->addSearchFieldMetadata(['Pledge' => CRM_Pledge_BAO_Query::getSearchFieldMetadata()]);
914 $this->addSearchFieldMetadata(['PledgePayment' => CRM_Pledge_BAO_Query::getPledgePaymentSearchFieldMetadata()]);
915 }
916 if (self::checkComponentAccess('CiviEvent')) {
917 $this->addSearchFieldMetadata(['Participant' => CRM_Event_BAO_Query::getSearchFieldMetadata()]);
918 }
919 if (self::checkComponentAccess('CiviMember')) {
920 $this->addSearchFieldMetadata(['Membership' => CRM_Member_BAO_Query::getSearchFieldMetadata()]);
921 }
922 if (self::checkComponentAccess('CiviGrant')) {
923 $this->addSearchFieldMetadata(['Grant' => CRM_Grant_BAO_Query::getSearchFieldMetadata()]);
924 }
925 if (self::checkComponentAccess('CiviCase')) {
926 $this->addSearchFieldMetadata(['Case' => CRM_Case_BAO_Query::getSearchFieldMetadata()]);
927 }
f90f8d32 928 }
929
6a488035 930}