3 +--------------------------------------------------------------------+
4 | CiviCRM version 4.7 |
5 +--------------------------------------------------------------------+
6 | Copyright CiviCRM LLC (c) 2004-2016 |
7 +--------------------------------------------------------------------+
8 | This file is a part of CiviCRM. |
10 | CiviCRM is free software; you can copy, modify, and distribute it |
11 | under the terms of the GNU Affero General Public License |
12 | Version 3, 19 November 2007 and the CiviCRM Licensing Exception. |
14 | CiviCRM is distributed in the hope that it will be useful, but |
15 | WITHOUT ANY WARRANTY; without even the implied warranty of |
16 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. |
17 | See the GNU Affero General Public License for more details. |
19 | You should have received a copy of the GNU Affero General Public |
20 | License along with this program; if not, contact CiviCRM LLC |
21 | at info[AT]civicrm[DOT]org. If you have questions about the |
22 | GNU Affero General Public License or the licensing of CiviCRM, |
23 | see the CiviCRM license FAQ at http://civicrm.org/licensing |
24 +--------------------------------------------------------------------+
27 require_once 'CiviTest/CiviSeleniumTestCase.php';
30 * Class WebTest_Contact_SearchBuilderTest
32 class WebTest_Contact_SearchBuilderTest
extends CiviSeleniumTestCase
{
34 protected function setUp() {
38 public function testSearchBuilderOptions() {
39 $this->webtestLogin();
41 $groupName = $this->WebtestAddGroup();
43 // Open the search builder
44 $this->openCiviPage('contact/search/builder', 'reset=1');
46 $this->enterValues(1, 1, 'Contacts', 'Group(s)', NULL, '=', array($groupName));
47 $this->enterValues(1, 2, 'Contacts', 'Country', NULL, '=', array('UNITED STATES'));
48 $this->enterValues(1, 3, 'Individual', 'Gender', NULL, '=', array('Male'));
49 $this->click('_qf_Builder_refresh');
50 $this->waitForPageToLoad();
52 // We should get no results. But check the options are all still set
53 $this->waitForTextPresent('No matches found for:');
54 foreach (array($groupName, 'UNITED STATES', 'Male') as $i => $label) {
55 $this->waitForElementPresent("//span[@id='crm_search_value_1_$i']/select/option[2]");
56 $this->assertSelectedLabel("//span[@id='crm_search_value_1_$i']/select", $label);
60 public function testSearchBuilderRLIKE() {
61 $this->webtestLogin();
64 // We're using Quick Add block on the main page for this.
65 $firstName = substr(sha1(rand()), 0, 7);
66 $this->createDetailContact($firstName);
68 $sortName = "adv$firstName, $firstName";
69 $displayName = "$firstName adv$firstName";
71 $this->_searchBuilder("Postal Code", "100[0-9]", $sortName, "RLIKE");
75 * function to create contact with details (contact details, address, Constituent information ...)
76 * @param null $firstName
78 public function createDetailContact($firstName = NULL) {
81 $firstName = substr(sha1(rand()), 0, 7);
84 // create contact type Individual with subtype
85 // with most of values to required to search
87 $this->openCiviPage('contact/add', array('reset' => 1, 'ct' => 'Individual'), '_qf_Contact_cancel');
89 // --- fill few values in Contact Detail block
90 $this->type("first_name", "$firstName");
91 $this->type("middle_name", "mid$firstName");
92 $this->type("last_name", "adv$firstName");
93 $this->select("contact_sub_type", "label=$Subtype");
94 $this->type("email_1_email", "$firstName@advsearch.co.in");
95 $this->type("phone_1_phone", "123456789");
96 $this->type("external_identifier", "extid$firstName");
98 // --- fill few values in address
99 $this->click("//form[@id='Contact']/div[2]/div[4]/div[1]");
100 $this->waitForElementPresent("address_1_geo_code_2");
101 $this->type("address_1_street_address", "street 1 $firstName");
102 $this->type("address_1_supplemental_address_1", "street supplement 1 $firstName");
103 $this->type("address_1_supplemental_address_2", "street supplement 2 $firstName");
104 $this->type("address_1_city", "city$firstName");
105 $this->type("address_1_postal_code", "100100");
106 $this->select("address_1_country_id", "UNITED STATES");
107 $this->select("address_1_state_province_id", "Alaska");
110 $this->click("_qf_Contact_upload_view");
111 $this->waitForPageToLoad($this->getTimeoutMsec());
112 $this->assertTrue($this->isTextPresent("$firstName adv$firstName"));
115 public function testSearchBuilderContacts() {
116 $this->webtestLogin();
119 $firstName = substr(sha1(rand()), 0, 7);
120 $streetName = "street $firstName";
121 $sortName = "adv$firstName, $firstName";
122 $this->_createContact('Individual', $firstName, "$firstName@advsearch.co.in", $streetName);
123 // search using search builder and advanced search
124 $this->_searchBuilder('Street Address', $streetName, $sortName, '=', '1');
125 $this->_advancedSearch($streetName, $sortName, 'Individual', '1', 'street_address');
128 $orgName = substr(sha1(rand()), 0, 7) . "org";
129 $orgEmail = "ab" . rand() . "@{$orgName}.com";
130 $this->_createContact('Organization', $orgName, $orgEmail, "street $orgName");
131 // search using search builder and advanced search
132 $this->_searchBuilder('Email', $orgEmail, $orgName, '=', '1');
133 $this->_advancedSearch($orgEmail, $orgName, 'Organization', '1', 'email');
136 $householdName = "household" . substr(sha1(rand()), 0, 7);
137 $householdEmail = "h1" . rand() . "@{$householdName}.com";
138 $this->_createContact('Household', $householdName, $householdEmail, "street $householdName");
139 // search using search builder and advanced search
140 $this->_searchBuilder('Email', $householdEmail, $householdName, '=', '1');
141 $this->_advancedSearch($householdEmail, $householdName, 'Household', '1', 'email');
143 $this->openCiviPage("contact/add", "reset=1&ct=Individual");
145 // searching contacts whose email is not set
146 $firstName1 = "00a1" . substr(sha1(rand()), 0, 7);
147 $this->type("first_name", $firstName1);
148 $this->type("last_name", "01adv$firstName1");
150 $this->click("_qf_Contact_upload_view");
151 $this->waitForPageToLoad($this->getTimeoutMsec());
152 $this->openCiviPage("contact/add", "reset=1&ct=Individual");
154 $firstName2 = "00a2" . substr(sha1(rand()), 0, 7);
155 $this->type("first_name", $firstName2);
156 $this->type("last_name", "02adv$firstName2");
158 $this->click("_qf_Contact_upload_view");
159 $this->waitForPageToLoad($this->getTimeoutMsec());
160 $this->openCiviPage("contact/add", "reset=1&ct=Individual");
162 $firstName3 = "00a3" . substr(sha1(rand()), 0, 7);
163 $this->type("first_name", $firstName3);
164 $this->type("last_name", "03adv$firstName3");
166 $this->click("_qf_Contact_upload_view");
167 $this->waitForPageToLoad($this->getTimeoutMsec());
168 $this->_searchBuilder('Email', NULL, NULL, 'IS NULL');
174 foreach ($names as $key => $value) {
175 $this->assertTrue($this->isTextPresent($value));
177 //searching contacts whose phone field is empty
178 $this->_searchBuilder('Phone', NULL, NULL, 'IS EMPTY');
179 foreach ($names as $key => $value) {
180 $this->assertTrue($this->isTextPresent($value));
182 //searching contacts whose phone field is not empty
183 $this->_searchBuilder('Phone', NULL, $firstName, 'IS NOT EMPTY');
184 $this->assertTrue($this->isTextPresent($firstName));
186 $firstName4 = "AB" . substr(sha1(rand()), 0, 7);
187 $postalCode = rand();
188 $this->_createContact('Individual', $firstName4, "$firstName4@advsearch.co.in", NULL, $postalCode);
189 $firstName5 = "CD" . substr(sha1(rand()), 0, 7);
190 $this->_createContact('Individual', $firstName5, "$firstName5@advsearch.co.in", NULL, $postalCode);
191 $firstName6 = "EF" . substr(sha1(rand()), 0, 7);
192 $this->_createContact('Organization', $firstName6, "$firstName6@advsearch.co.in", NULL, $postalCode);
193 $firstName7 = "GH" . substr(sha1(rand()), 0, 7);
194 $this->_createContact('Household', $firstName7, "$firstName7@advsearch.co.in", NULL, $postalCode);
196 // check if the resultset of search builder and advanced search match for the postal code
197 $this->_searchBuilder('Postal Code', $postalCode, NULL, 'LIKE', '4');
198 $this->_advancedSearch($postalCode, NULL, NULL, '4', 'postal_code');
200 $firstName8 = "abcc" . substr(sha1(rand()), 0, 7);
201 $this->_createContact('Individual', $firstName8, "$firstName8@advsearch.co.in", NULL);
202 $this->_searchBuilder('Note(s): Body and Subject', "this is notes by $firstName8 adv$firstName8", $firstName8, 'LIKE');
203 $this->_searchBuilder('Note(s): Subject Only', "this is subject by $firstName8 adv$firstName8", $firstName8, 'LIKE');
204 $this->_searchBuilder('Note(s): Body Only', "this is notes by $firstName8 adv$firstName8", $firstName8, 'LIKE');
205 $this->_advancedSearch("this is notes by $firstName8 adv$firstName8", $firstName8, NULL, NULL, 'note_body', 'notes');
206 $this->_advancedSearch("this is subject by $firstName8 adv$firstName8", $firstName8, NULL, NULL, 'note_subject', 'notes');
207 $this->_advancedSearch("this is notes by $firstName8 adv$firstName8", $firstName8, NULL, NULL, 'note_both', 'notes');
208 $this->_advancedSearch("this is notes by $firstName8 adv$firstName8", $firstName8, NULL, NULL, 'note_both', 'notes');
213 * @param null $fieldValue
218 public function _searchBuilder($field, $fieldValue = NULL, $name = NULL, $op = '=', $count = NULL) {
219 // search builder using contacts(not using contactType)
220 $this->openCiviPage("contact/search/builder", "reset=1");
221 $this->enterValues(1, 1, 'Contacts', $field, NULL, $op, "$fieldValue");
222 $this->click("id=_qf_Builder_refresh");
223 $this->waitForPageToLoad($this->getTimeoutMsec());
224 if (($op == '=' ||
$op == 'LIKE') && $fieldValue) {
225 $this->assertElementContainsText('css=.crm-search-results > table.row-highlight', "$fieldValue");
228 $this->assertElementContainsText('css=.crm-search-results > table.row-highlight', "$name");
231 $this->assertElementContainsText('search-status', "$count Contact");
236 * Enter form values in a Search Builder row.
243 * @param string $value
245 public function enterValues($set, $row, $entity, $field, $loc, $op, $value = '') {
246 if ($set > 1 && $row == 1) {
247 $this->click('addBlock');
250 $this->click("addMore_{$set}");
252 // In the DOM rows are 0 indexed and sets are 1 indexed, so normalizing
255 $this->waitForElementPresent("mapper_{$set}_{$row}_0");
256 $this->select("mapper_{$set}_{$row}_0", "label=$entity");
257 $this->select("mapper_{$set}_{$row}_1", "label=$field");
259 $this->select("mapper_{$set}_{$row}_2", "label=$loc");
261 $this->select("operator_{$set}_{$row}", "value=$op");
262 if (is_array($value)) {
263 $this->waitForElementPresent("css=#crm_search_value_{$set}_{$row} select option + option");
264 foreach ($value as $val) {
270 $select = 'addSelection';
273 $this->$select("css=#crm_search_value_{$set}_{$row} select", "label=$val");
276 elseif ($value && substr($value, 0, 5) == 'date:') {
277 $this->webtestFillDate("value_{$set}_{$row}", trim(substr($value, 5)));
280 $this->type("value_{$set}_{$row}", $value);
285 * @param null $fieldValue
287 * @param null $contactType
291 public function _advancedSearch($fieldValue = NULL, $name = NULL, $contactType = NULL, $count = NULL, $field) {
292 //advanced search by selecting the contactType
293 $this->openCiviPage("contact/search/advanced", "reset=1");
294 if (isset($contactType)) {
295 $this->select("id=contact_type", "value=$contactType");
297 if (substr($field, 0, 5) == 'note_') {
298 $this->click("notes");
299 $this->waitForElementPresent("xpath=//div[@id='notes-search']/table/tbody/tr/td[2]/input[3]");
300 if ($field == 'note_body') {
301 $this->click("CIVICRM_QFID_2_note_option");
303 elseif ($field == 'note_subject') {
304 $this->click("CIVICRM_QFID_3_note_option");
307 $this->click("CIVICRM_QFID_6_note_option");
309 $this->type("note", $fieldValue);
312 $this->click("location");
313 $this->waitForElementPresent("$field");
314 if ($contactType == 'Individual') {
315 $this->type("$field", $fieldValue);
318 $this->type("$field", $fieldValue);
321 $this->click("_qf_Advanced_refresh");
322 $this->waitForPageToLoad($this->getTimeoutMsec());
324 //the search result should be same as the one that we got in search builder
326 $this->assertElementContainsText('Advanced', "$fieldValue");
329 $this->assertElementContainsText('css=.crm-search-results > table.row-highlight', "$name");
332 $this->assertElementContainsText('search-status', "$count Contact");
337 * @param $contactType
338 * @param string $name
340 * @param null $streetName
341 * @param null $postalCode
343 public function _createContact($contactType, $name, $email, $streetName = NULL, $postalCode = NULL) {
344 $this->openCiviPage('contact/add', array('reset' => 1, 'ct' => $contactType), '_qf_Contact_cancel');
346 if ($contactType == 'Individual') {
347 $this->type("first_name", "$name");
348 $this->type("last_name", "adv$name");
349 $name = "$name adv$name";
351 elseif ($contactType == 'Organization') {
352 $this->type("organization_name", $name);
355 $this->type("household_name", $name);
357 $this->click("//form[@id='Contact']/div[2]/div[4]/div[1]");
358 $this->waitForElementPresent("address_1_geo_code_2");
359 $this->type("email_1_email", $email);
360 $this->type("phone_1_phone", "9876543210");
361 $this->type("address_1_street_address", $streetName);
362 $this->select("address_1_country_id", "UNITED STATES");
363 $this->select("address_1_state_province_id", "Alaska");
364 $this->type("address_1_postal_code", $postalCode);
366 $this->click("//form[@id='Contact']/div[2]/div[6]/div[1]");
367 $this->waitForElementPresent("note");
368 $this->type("subject", "this is subject by $name");
369 $this->type("note", "this is notes by $name");
372 $this->click("_qf_Contact_upload_view");
373 $this->waitForPageToLoad($this->getTimeoutMsec());
374 $this->assertTrue($this->isTextPresent("$name has been created."));
378 * Webtest for CRM-12148
380 public function testSearchBuilderfinancialType() {
381 $this->webtestLogin();
383 // add financial type
384 $financialTypeName1 = 'Financial Type' . substr(sha1(rand()), 0, 5);;
385 $financialTypeName2 = 'Financial Type' . substr(sha1(rand()), 0, 5);;
386 $financialType = array(
387 'name' => $financialTypeName1,
388 'is_reserved' => FALSE,
389 'is_deductible' => FALSE,
391 $this->addeditFinancialType($financialType);
392 $financialType['name'] = $financialTypeName2;
393 $this->addeditFinancialType($financialType);
394 //create 6 contribution
395 $this->openCiviPage("contribute/add", "reset=1&action=add&context=standalone", "_qf_Contribution_upload");
396 for ($i = 1; $i <= 6; $i++
) {
398 $financialType = $financialTypeName1;
401 $financialType = $financialTypeName2;
403 // create new contact using dialog
404 $this->createDialogContact();
405 $this->select('financial_type_id', $financialType);
406 $this->type('total_amount', 100 * $i);
407 $this->clickLink('_qf_Contribution_upload_new', '_qf_Contribution_upload_new');
409 $this->openCiviPage("contact/search/builder", "reset=1", "_qf_Builder_refresh");
411 $this->enterValues(1, 1, 'Contribution', 'Financial Type', NULL, '=', array($financialTypeName1));
412 $this->clickLink('_qf_Builder_refresh');
414 $this->assertTrue($this->isTextPresent('3 Contacts'), 'Missing text: ' . '3 Contacts');
416 $this->click("xpath=//div[@class='crm-accordion-header crm-master-accordion-header']");
417 $this->enterValues(1, 1, 'Contribution', 'Financial Type', NULL, '=', array($financialTypeName2));
418 $this->clickLink('_qf_Builder_refresh');
420 $this->assertTrue($this->isTextPresent('3 Contacts'), 'Missing text: ' . '3 Contacts');
422 $this->click("xpath=//div[@class='crm-accordion-header crm-master-accordion-header']");
423 $this->enterValues(1, 1, 'Contribution', 'Financial Type', NULL, 'IN', array(
427 $this->clickLink('_qf_Builder_refresh');
429 $this->assertTrue($this->isTextPresent('6 Contacts'), 'Missing text: ' . '6 Contacts');
433 * Webtest for CRM-12588
435 public function testSearchBuilderMembershipType() {
436 $this->webtestLogin();
438 // create first contact
439 $firstName1 = substr(sha1(rand()), 0, 7);
440 $this->webtestAddContact($firstName1, "Memberson", "Memberson{$firstName1}@memberson.name");
441 $contactName1 = "Memberson, $firstName1";
443 // create Second contact
444 $firstName2 = substr(sha1(rand()), 0, 7);
445 $this->webtestAddContact($firstName2, "Memberson", "Memberson{$firstName2}@memberson.name");
446 $contactName2 = "Memberson, $firstName2";
448 // add membership type
449 $membershipTypes = $this->webtestAddMembershipType();
451 // now add membership
452 $this->openCiviPage("member/add", "reset=1&action=add&context=standalone", "_qf_Membership_upload");
455 $this->webtestFillAutocomplete($firstName1);
457 // fill in Membership Organization
458 $this->waitForElementPresent("membership_type_id[0]");
459 $this->select("membership_type_id[0]", "label={$membershipTypes['member_of_contact']}");
461 // select membership type
462 $this->waitForElementPresent("membership_type_id[1]");
463 $this->select("membership_type_id[1]", "label={$membershipTypes['membership_type']}");
466 $this->type("source", "Membership StandaloneAddTest Webtest");
468 // fill in Start Date
469 $this->webtestFillDate('start_date');
472 $this->clickLink("_qf_Membership_upload");
475 $this->waitForTextPresent("Membership StandaloneAddTest Webtest");
477 // now add membership for second contact
478 $this->openCiviPage("member/add", "reset=1&action=add&context=standalone", "_qf_Membership_upload");
479 $this->webtestFillAutocomplete($firstName2);
480 $this->select("membership_type_id[0]", "label={$membershipTypes['member_of_contact']}");
481 $this->select("membership_type_id[1]", "label={$membershipTypes['membership_type']}");
482 $this->type("source", "Membership StandaloneAddTest Webtest");
483 $this->webtestFillDate('start_date');
485 // fill in Status Override?
486 $this->click("is_override");
487 $this->waitForElementPresent("status_id");
488 $this->select("status_id", "label=Grace");
491 $this->clickLink("_qf_Membership_upload");
492 $this->waitForTextPresent("Membership StandaloneAddTest Webtest");
494 // Open the search builder
495 $this->openCiviPage('contact/search/builder', 'reset=1');
496 $this->enterValues(1, 1, 'Membership', 'Membership Type', NULL, '=', array($membershipTypes['membership_type']));
498 $this->clickLink('_qf_Builder_refresh');
499 $this->waitForAjaxContent();
500 $this->assertElementContainsText("xpath=//div[@id='search-status']/table/tbody/tr[1]/td", "2 Contacts");
502 $this->click("xpath=//div[@class='crm-accordion-header crm-master-accordion-header']");
503 $this->enterValues(1, 2, 'Membership', 'Membership Status', NULL, '=', array('New'));
504 $this->clickLink('_qf_Builder_refresh');
505 $this->waitForAjaxContent();
506 $this->assertElementContainsText("xpath=//div[@id='search-status']/table/tbody/tr[1]/td", "1 Contact");
508 $this->enterValues(1, 2, 'Membership', 'Membership Status', NULL, '=', array('Grace'));
509 $this->clickLink('_qf_Builder_refresh');
510 $this->waitForAjaxContent();
511 $this->assertElementContainsText("xpath=//div[@id='search-status']/table/tbody/tr[1]/td", "1 Contact");
513 $this->click("xpath=//div[@class='crm-accordion-header crm-master-accordion-header']");
514 $this->waitForElementPresent("xpath=//div[@id='map-field']/div[1]/table/tbody/tr[2]/td/a");
515 $this->click("xpath=//div[@id='map-field']/div[1]/table/tbody/tr[2]/td/a");
516 $this->enterValues(1, 2, 'Membership', 'Membership Status', NULL, 'IN', array('New', 'Grace'));
517 $this->clickLink('_qf_Builder_refresh');
518 $this->waitForAjaxContent();
519 $this->assertElementContainsText("xpath=//div[@id='search-status']/table/tbody/tr[1]/td", "2 Contacts");
521 $this->click("xpath=//div[@class='crm-accordion-header crm-master-accordion-header']");
522 $this->waitForElementPresent("xpath=//div[@id='map-field']/div[1]/table/tbody/tr[2]/td/a");
523 $this->click("xpath=//div[@id='map-field']/div[1]/table/tbody/tr[2]/td/a");
524 $this->enterValues(1, 2, 'Membership', 'Membership Status', NULL, 'IN', array('Current', 'Expired'));
525 $this->clickLink('_qf_Builder_refresh');
526 $this->waitForText("xpath=//form[@id='Builder']/div[3]/div/div", "No matches found for");
529 $this->openCiviPage("member/search", "reset=1", "_qf_Search_refresh");
530 $this->select2('membership_type_id', $membershipTypes['membership_type'], TRUE);
531 $this->clickLink('_qf_Search_refresh');
532 $this->assertElementContainsText("xpath=//div[@id='search-status']/table/tbody/tr[1]/td[1]", "2 Result");
534 $this->click("xpath=//div[@class='crm-accordion-header crm-master-accordion-header']");
535 $this->multiselect2("membership_status_id", array("New", "Grace"));
536 $this->clickLink('_qf_Search_refresh');
537 $this->waitForAjaxContent();
538 $this->assertElementContainsText("xpath=//div[@id='search-status']/table/tbody/tr[1]/td", "2 Results");
540 $this->openCiviPage("member/search", "reset=1", "_qf_Search_refresh");
541 $this->waitForAjaxContent();
542 $this->click("xpath=//div[@class='crm-accordion-header crm-master-accordion-header']");
543 $this->select2('membership_type_id', $membershipTypes['membership_type'], TRUE);
544 $this->multiselect2("membership_status_id", array("New"));
545 $this->click('_qf_Search_refresh');
546 $this->waitForAjaxContent();
547 $this->assertElementContainsText("xpath=//div[@id='search-status']/table/tbody/tr[1]/td", "1 Result");