Last active
May 28, 2021 10:41
-
-
Save tortuetorche/412fbac4f17db5e78e79 to your computer and use it in GitHub Desktop.
Select2 v3.5 helpers for the WebDriver module of Codeception 2.1 (WIP)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?php | |
namespace Helper; | |
// Select2 helpers for the jQuery based replacement for select boxes. | |
// See: http://select2.github.io/select2 | |
// Author: Tortue Torche <[email protected]> | |
// License: MIT | |
// | |
// Installation: | |
// * Put this file in your 'tests/_support/Helper' directory | |
// * Add it in your 'tests/acceptance.suite.yml' file, like this: | |
// class_name: AcceptanceTester | |
// modules: | |
// enabled: | |
// - WebDriver: | |
// url: 'http://localhost:8000' | |
// # ... | |
// - \Helper\Select2 | |
// | |
// * Then run ./vendor/bin/codecept build | |
class Select2 extends \Codeception\Module | |
{ | |
/** | |
* @param $selector | |
* @param $optionText | |
* @param bool $expectedReturn Default to true | |
* | |
* @return string JavaScript | |
*/ | |
protected function _optionIsSelectedForSelect2($selector, $optionText, $expectedReturn = true) | |
{ | |
$returnFlag = $expectedReturn === true ? '' : '!'; | |
return $script = <<<EOT | |
return (function (\$) { | |
var isSelected = false; | |
var values = \$("$selector").select2("data"); | |
values = \$.isArray(values ) ? values : [values]; | |
if (values && values.length > 0) { | |
isSelected = values.some(function (data) { | |
if (data && data.text && data.text === "$optionText") { | |
return data; | |
} | |
}); | |
} | |
return ${returnFlag}isSelected; | |
}(jQuery)); | |
EOT; | |
} | |
/** | |
* Wait until the select2 component is loaded | |
* | |
* @param $selector | |
* @param int $timeout seconds. Default to 5 | |
*/ | |
public function waitForSelect2($selector, $timeout = 5) | |
{ | |
$I = $this->getAcceptanceModule(); | |
$selector = $this->getSelect2Selector($selector); | |
$I->waitForJS('return !!jQuery("'.$selector.'").data("select2");', $timeout); | |
} | |
/** | |
* Checks that the given option is not selected. | |
* | |
* @param $selector | |
* @param $optionText | |
* @param int $timeout seconds. Default to 5 | |
*/ | |
public function dontSeeOptionIsSelectedForSelect2($selector, $optionText, $timeout = 5) | |
{ | |
$I = $this->getAcceptanceModule(); | |
$selector = $this->getSelect2Selector($selector); | |
$this->waitForSelect2($selector, $timeout); | |
$script = $this->_optionIsSelectedForSelect2($selector, $optionText, false); | |
$I->waitForJS($script, $timeout); | |
} | |
/** | |
* Checks that the given option is selected. | |
* | |
* @param $selector | |
* @param $optionText | |
* @param int $timeout seconds. Default to 5 | |
*/ | |
public function seeOptionIsSelectedForSelect2($selector, $optionText, $timeout = 5) | |
{ | |
$I = $this->getAcceptanceModule(); | |
$selector = $this->getSelect2Selector($selector); | |
$this->waitForSelect2($selector, $timeout); | |
$script = $this->_optionIsSelectedForSelect2($selector, $optionText); | |
$I->waitForJS($script, $timeout); | |
} | |
/** | |
* Selects an option in a select2 component. | |
* | |
* $I->selectOptionForSelect2('#my_select2', 'Option value'); | |
* $I->selectOptionForSelect2('#my_select2', ['Option value 1', 'Option value 2']); | |
* $I->selectOptionForSelect2('#my_select2', ['text' => 'Option text']); | |
* $I->selectOptionForSelect2('#my_select2', ['id' => 'Option value', 'text' => 'Option text']); | |
* | |
* @param $selector | |
* @param $option | |
* @param int $timeout seconds. Default to 1 | |
*/ | |
public function selectOptionForSelect2($selector, $option, $timeout = 1) | |
{ | |
$I = $this->getAcceptanceModule(); | |
$selector = $this->getSelect2Selector($selector); | |
$this->waitForSelect2($selector, $timeout); | |
if (is_string($option) || (is_array($option) && array_values($option) === $option)) { | |
// $option is a string or a non associative array | |
$I->executeJS('jQuery("'.$selector.'").select2("val", '.json_encode($option).');', $timeout); | |
} else if(is_array($option)) { | |
$optionId = 'null'; | |
if (isset($option['text']) && empty($option['id'])) { | |
$optionText = $option['text']; | |
$optionId = <<<EOT | |
function() { | |
if (!\$.expr[':'].textEquals) { | |
// Source: http://stackoverflow.com/a/26431267 | |
\$.expr[':'].textEquals = function(el, i, m) { | |
var searchText = m[3]; | |
return $(el).text().trim() === searchText; | |
} | |
} | |
// Find select option by text | |
return \$(\$("$selector").data("select2").select).find("option:textEquals('$optionText'):first").val(); | |
}(); | |
EOT; | |
} | |
$jsonOption = json_encode($option); | |
$script = <<<EOT | |
(function (\$) { | |
var option = $jsonOption; | |
if (!option.id) { | |
option.id = $optionId; | |
} | |
\$("$selector").select2("data", option); | |
\$(\$("$selector").data("select2").select).trigger('change'); | |
}(jQuery)); | |
EOT; | |
$I->executeJS($script, $timeout); | |
} else { | |
$I->fail(); | |
} | |
} | |
/** | |
* Unselect an option in the given select2 component. | |
* | |
* @param $selector | |
* @param $option | |
* @param int $timeout seconds. Default to 1 | |
*/ | |
public function unselectOptionForSelect2($selector, $option = null, $timeout = 1) | |
{ | |
$I = $this->getAcceptanceModule(); | |
$selector = $this->getSelect2Selector($selector); | |
$this->waitForSelect2($selector, $timeout); | |
if ($option && is_string($option)) { | |
$script = <<<EOT | |
(function (\$) { | |
var values = \$("$selector").select2("val"); | |
var index = values.indexOf("$option"); | |
if (index > -1) { | |
values.splice(index, 1); | |
} | |
\$("$selector").select2("val", values); | |
\$(\$("$selector").data("select2").select).trigger('change'); | |
}(jQuery)); | |
EOT; | |
$I->executeJS($script, $timeout); | |
} else { | |
$I->executeJS('jQuery("'.$selector.'").select2("val", "");', $timeout); | |
} | |
} | |
/** | |
* Open the Select2 component | |
* @param string $selector | |
*/ | |
public function openSelect2($selector) | |
{ | |
$I = $this->getAcceptanceModule(); | |
$selector = $this->getSelect2Selector($selector); | |
$this->waitForSelect2($selector); | |
$I->executeJS('jQuery("'.$selector.'").select2("open");'); | |
} | |
/** | |
* Close the Select2 component | |
* @param string $selector | |
*/ | |
public function closeSelect2($selector) | |
{ | |
$I = $this->getAcceptanceModule(); | |
$selector = $this->getSelect2Selector($selector); | |
$this->waitForSelect2($selector); | |
$I->executeJS('jQuery("'.$selector.'").select2("close");'); | |
} | |
protected function getSelect2Selector($selector) | |
{ | |
return preg_replace("/^\#((?!s2id_).+)$/", '#s2id_$1', $selector); | |
} | |
protected function getAcceptanceModule() | |
{ | |
if (! $this->hasModule('WebDriver')) { | |
throw new \Exception("You must enable the WebDriver module", 1); | |
} | |
return $this->getModule('WebDriver'); | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Hi folks,
Many thanks for all your contributions!
I wasn't expected this gist to be so popular!
Here an updated version which works with Codeception 4 (and certainly older version) and Select2 >= 4.0:
https://gist.github.com/tortuetorche/f4754f0867d9d0a272f1a3cea4073295
Have a good day,
Tortue Torche