]> source.dussan.org Git - nextcloud-server.git/commitdiff
Add acceptance tests for unselecting items in dropdown for tags
authorDaniel Calviño Sánchez <danxuliu@gmail.com>
Sat, 8 Jul 2017 13:46:16 +0000 (15:46 +0200)
committerDaniel Calviño Sánchez <danxuliu@gmail.com>
Tue, 11 Jul 2017 11:02:09 +0000 (13:02 +0200)
Signed-off-by: Daniel Calviño Sánchez <danxuliu@gmail.com>
tests/acceptance/config/behat.yml
tests/acceptance/features/app-files.feature
tests/acceptance/features/bootstrap/FilesAppContext.php
tests/acceptance/features/bootstrap/SettingsContext.php [new file with mode: 0644]
tests/acceptance/features/bootstrap/SettingsMenuContext.php

index 865100837612d0edfda39175204d3aaaefa87f20..f9412935e51940d87d1def032d5e52b07ea5c0a1 100644 (file)
@@ -15,6 +15,7 @@ default:
         - FilesSharingAppContext
         - LoginPageContext
         - NotificationContext
+        - SettingsContext
         - SettingsMenuContext
         - UsersSettingsContext
   extensions:
index 8d32508513aaf6e8b795aae9c20366884b01077e..5a0a00955122f7884b889787ffbdfadccf4b4b88 100644 (file)
@@ -68,3 +68,75 @@ Feature: app-files
     And I see that the "Sharing" tab in the details view is eventually loaded
     When I open the input field for tags in the details view
     Then I see that the input field for tags in the details view is shown
+
+  Scenario: create tags using the Administration settings
+    Given I am logged in as the admin
+    And I open the Admin settings
+    And I open the "Workflow" section
+    # The "create" button does nothing before JavaScript was initialized, and
+    # the only way to detect that is waiting for the button to select tags to be
+    # shown.
+    And I see that the button to select tags is shown
+    When I create the tag "tag1" in the settings
+    Then I see that the dropdown for tags in the settings eventually contains the tag "tag1"
+
+  Scenario: add tags using the dropdown in the details view
+    Given I am logged in as the admin
+    And I open the Admin settings
+    And I open the "Workflow" section
+    # The "create" button does nothing before JavaScript was initialized, and
+    # the only way to detect that is waiting for the button to select tags to be
+    # shown.
+    And I see that the button to select tags is shown
+    And I create the tag "tag1" in the settings
+    And I create the tag "tag2" in the settings
+    And I create the tag "tag3" in the settings
+    And I create the tag "tag4" in the settings
+    And I see that the dropdown for tags in the settings eventually contains the tag "tag1"
+    And I see that the dropdown for tags in the settings eventually contains the tag "tag2"
+    And I see that the dropdown for tags in the settings eventually contains the tag "tag3"
+    And I see that the dropdown for tags in the settings eventually contains the tag "tag4"
+    And I log out
+    And I am logged in
+    And I open the details view for "welcome.txt"
+    And I open the input field for tags in the details view
+    # When the input field is opened the dropdown is also opened automatically.
+    When I check the tag "tag2" in the dropdown for tags in the details view
+    And I check the tag "tag4" in the dropdown for tags in the details view
+    Then I see that the tag "tag2" in the dropdown for tags in the details view is checked
+    And I see that the tag "tag4" in the dropdown for tags in the details view is checked
+    And I see that the input field for tags in the details view contains the tag "tag2"
+    And I see that the input field for tags in the details view contains the tag "tag4"
+
+  Scenario: remove tags using the dropdown in the details view
+    Given I am logged in as the admin
+    And I open the Admin settings
+    And I open the "Workflow" section
+    # The "create" button does nothing before JavaScript was initialized, and
+    # the only way to detect that is waiting for the button to select tags to be
+    # shown.
+    And I see that the button to select tags is shown
+    And I create the tag "tag1" in the settings
+    And I create the tag "tag2" in the settings
+    And I create the tag "tag3" in the settings
+    And I create the tag "tag4" in the settings
+    And I see that the dropdown for tags in the settings eventually contains the tag "tag1"
+    And I see that the dropdown for tags in the settings eventually contains the tag "tag2"
+    And I see that the dropdown for tags in the settings eventually contains the tag "tag3"
+    And I see that the dropdown for tags in the settings eventually contains the tag "tag4"
+    And I log out
+    And I am logged in
+    And I open the details view for "welcome.txt"
+    And I open the input field for tags in the details view
+    # When the input field is opened the dropdown is also opened automatically.
+    And I check the tag "tag2" in the dropdown for tags in the details view
+    And I check the tag "tag4" in the dropdown for tags in the details view
+    And I check the tag "tag3" in the dropdown for tags in the details view
+    When I uncheck the tag "tag2" in the dropdown for tags in the details view
+    And I uncheck the tag "tag4" in the dropdown for tags in the details view
+    Then I see that the tag "tag2" in the dropdown for tags in the details view is not checked
+    And I see that the tag "tag4" in the dropdown for tags in the details view is not checked
+    And I see that the tag "tag3" in the dropdown for tags in the details view is checked
+    And I see that the input field for tags in the details view does not contain the tag "tag2"
+    And I see that the input field for tags in the details view does not contain the tag "tag4"
+    And I see that the input field for tags in the details view contains the tag "tag3"
index b9ccb7316ab3163805be9fb6f106a4a4a359fa0e..61f6b115ac6e556520c8e90dd537af990a6530c5 100644 (file)
@@ -104,6 +104,41 @@ class FilesAppContext implements Context, ActorAwareInterface {
                                describedAs("Input field for tags in current section details view in Files app");
        }
 
+       /**
+        * @return Locator
+        */
+       public static function itemInInputFieldForTagsInCurrentSectionDetailsViewForTag($tag) {
+               return Locator::forThe()->xpath("//span[normalize-space() = '$tag']")->
+                               descendantOf(self::inputFieldForTagsInCurrentSectionDetailsView())->
+                               describedAs("Item in input field for tags in current section details view for tag $tag in Files app");
+       }
+
+       /**
+        * @return Locator
+        */
+       public static function itemInDropdownForTag($tag) {
+               return Locator::forThe()->xpath("//*[contains(concat(' ', normalize-space(@class), ' '), ' select2-result-label ')]//span[normalize-space() = '$tag']/ancestor::li")->
+                               descendantOf(self::select2Dropdown())->
+                               describedAs("Item in dropdown for tag $tag in Files app");
+       }
+
+       /**
+        * @return Locator
+        */
+       public static function checkmarkInItemInDropdownForTag($tag) {
+               return Locator::forThe()->css(".checkmark")->
+                               descendantOf(self::itemInDropdownForTag($tag))->
+                               describedAs("Checkmark in item in dropdown for tag $tag in Files app");
+       }
+
+       /**
+        * @return Locator
+        */
+       private static function select2Dropdown() {
+               return Locator::forThe()->css("#select2-drop")->
+                               describedAs("Select2 dropdown in Files app");
+       }
+
        /**
         * @return Locator
         */
@@ -323,6 +358,24 @@ class FilesAppContext implements Context, ActorAwareInterface {
                $this->actor->find(self::viewFileInFolderMenuItem(), 2)->click();
        }
 
+       /**
+        * @When I check the tag :tag in the dropdown for tags in the details view
+        */
+       public function iCheckTheTagInTheDropdownForTagsInTheDetailsView($tag) {
+               $this->iSeeThatTheTagInTheDropdownForTagsInTheDetailsViewIsNotChecked($tag);
+
+               $this->actor->find(self::itemInDropdownForTag($tag), 10)->click();
+       }
+
+       /**
+        * @When I uncheck the tag :tag in the dropdown for tags in the details view
+        */
+       public function iUncheckTheTagInTheDropdownForTagsInTheDetailsView($tag) {
+               $this->iSeeThatTheTagInTheDropdownForTagsInTheDetailsViewIsChecked($tag);
+
+               $this->actor->find(self::itemInDropdownForTag($tag), 10)->click();
+       }
+
        /**
         * @When I protect the shared link with the password :password
         */
@@ -389,6 +442,46 @@ class FilesAppContext implements Context, ActorAwareInterface {
                                $this->actor->find(self::inputFieldForTagsInCurrentSectionDetailsView(), 10)->isVisible());
        }
 
+       /**
+        * @Then I see that the input field for tags in the details view contains the tag :tag
+        */
+       public function iSeeThatTheInputFieldForTagsInTheDetailsViewContainsTheTag($tag) {
+               PHPUnit_Framework_Assert::assertTrue(
+                               $this->actor->find(self::itemInInputFieldForTagsInCurrentSectionDetailsViewForTag($tag), 10)->isVisible());
+       }
+
+       /**
+        * @Then I see that the input field for tags in the details view does not contain the tag :tag
+        */
+       public function iSeeThatTheInputFieldForTagsInTheDetailsViewDoesNotContainTheTag($tag) {
+               $this->iSeeThatTheInputFieldForTagsInTheDetailsViewIsShown();
+
+               try {
+                       PHPUnit_Framework_Assert::assertFalse(
+                                       $this->actor->find(self::itemInInputFieldForTagsInCurrentSectionDetailsViewForTag($tag))->isVisible());
+               } catch (NoSuchElementException $exception) {
+               }
+       }
+
+       /**
+        * @Then I see that the tag :tag in the dropdown for tags in the details view is checked
+        */
+       public function iSeeThatTheTagInTheDropdownForTagsInTheDetailsViewIsChecked($tag) {
+               PHPUnit_Framework_Assert::assertTrue(
+                               $this->actor->find(self::checkmarkInItemInDropdownForTag($tag), 10)->isVisible());
+       }
+
+       /**
+        * @Then I see that the tag :tag in the dropdown for tags in the details view is not checked
+        */
+       public function iSeeThatTheTagInTheDropdownForTagsInTheDetailsViewIsNotChecked($tag) {
+               PHPUnit_Framework_Assert::assertTrue(
+                               $this->actor->find(self::itemInDropdownForTag($tag), 10)->isVisible());
+
+               PHPUnit_Framework_Assert::assertFalse(
+                               $this->actor->find(self::checkmarkInItemInDropdownForTag($tag))->isVisible());
+       }
+
        /**
         * @When I see that the :tabName tab in the details view is eventually loaded
         */
diff --git a/tests/acceptance/features/bootstrap/SettingsContext.php b/tests/acceptance/features/bootstrap/SettingsContext.php
new file mode 100644 (file)
index 0000000..edbb6a9
--- /dev/null
@@ -0,0 +1,152 @@
+<?php
+
+/**
+ *
+ * @copyright Copyright (c) 2017, Daniel Calviño Sánchez (danxuliu@gmail.com)
+ *
+ * @license GNU AGPL version 3 or any later version
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Affero General Public License as
+ * published by the Free Software Foundation, either version 3 of the
+ * License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Affero General Public License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ *
+ */
+
+use Behat\Behat\Context\Context;
+
+class SettingsContext implements Context, ActorAwareInterface {
+
+       use ActorAware;
+
+       /**
+        * @return Locator
+        */
+       public static function systemTagsSelectTagButton() {
+               return Locator::forThe()->id("s2id_systemtag")->
+                               describedAs("Select tag button in system tags section in Administration Settings");
+       }
+
+       /**
+        * @return Locator
+        */
+       public static function systemTagsItemInDropdownForTag($tag) {
+               return Locator::forThe()->xpath("//*[contains(concat(' ', normalize-space(@class), ' '), ' select2-result-label ')]//span[normalize-space() = '$tag']/ancestor::li")->
+                               descendantOf(self::select2Dropdown())->
+                               describedAs("Item in dropdown for tag $tag in system tags section in Administration Settings");
+       }
+
+       /**
+        * @return Locator
+        */
+       private static function select2Dropdown() {
+               return Locator::forThe()->css("#select2-drop")->
+                               describedAs("Select2 dropdown in Settings");
+       }
+
+       /**
+        * @return Locator
+        */
+       private static function select2DropdownMask() {
+               return Locator::forThe()->css("#select2-drop-mask")->
+                               describedAs("Select2 dropdown mask in Settings");
+       }
+
+       /**
+        * @return Locator
+        */
+       public static function systemTagsTagNameInput() {
+               return Locator::forThe()->id("systemtag_name")->
+                               describedAs("Tag name input in system tags section in Administration Settings");
+       }
+
+       /**
+        * @return Locator
+        */
+       public static function systemTagsCreateOrUpdateButton() {
+               return Locator::forThe()->id("systemtag_submit")->
+                               describedAs("Create/Update button in system tags section in Administration Settings");
+       }
+
+       /**
+        * @return Locator
+        */
+       public static function systemTagsResetButton() {
+               return Locator::forThe()->id("systemtag_reset")->
+                               describedAs("Reset button in system tags section in Administration Settings");
+       }
+
+       /**
+        * @When I create the tag :tag in the settings
+        */
+       public function iCreateTheTagInTheSettings($tag) {
+               $this->actor->find(self::systemTagsResetButton(), 10)->click();
+               $this->actor->find(self::systemTagsTagNameInput())->setValue($tag);
+               $this->actor->find(self::systemTagsCreateOrUpdateButton())->click();
+       }
+
+       /**
+        * @Then I see that the button to select tags is shown
+        */
+       public function iSeeThatTheButtonToSelectTagsIsShown() {
+               PHPUnit_Framework_Assert::assertTrue($this->actor->find(self::systemTagsSelectTagButton(), 10)->isVisible());
+       }
+
+       /**
+        * @Then I see that the dropdown for tags in the settings eventually contains the tag :tag
+        */
+       public function iSeeThatTheDropdownForTagsInTheSettingsEventuallyContainsTheTag($tag) {
+               // When the dropdown is opened it is not automatically updated if new
+               // tags are added to the server, and when a tag is created, no explicit
+               // feedback is provided to the user about the completion of that
+               // operation (that is, when the tag is added to the server). Therefore,
+               // to verify that creating a tag does in fact add it to the server it is
+               // necessary to repeatedly open the dropdown until the tag is shown in
+               // the dropdown (or the limit of tries is reached).
+
+               PHPUnit_Framework_Assert::assertTrue($this->actor->find(self::systemTagsSelectTagButton(), 10)->isVisible());
+
+               $actor = $this->actor;
+
+               $tagFoundInDropdownCallback = function() use($actor, $tag) {
+                       // Open the dropdown to look for the tag.
+                       $actor->find(self::systemTagsSelectTagButton())->click();
+
+                       // When the dropdown is opened it is initially empty, and its
+                       // contents are updated once received from the server. Therefore, a
+                       // timeout must be used when looking for the tags.
+                       try {
+                               $tagFound = $this->actor->find(self::systemTagsItemInDropdownForTag($tag), 10)->isVisible();
+                       } catch (NoSuchElementException $exception) {
+                               $tagFound = false;
+                       }
+
+                       // Close again the dropdown after looking for the tag. When a
+                       // dropdown is opened Select2 creates a special element that masks
+                       // every other element but the dropdown to get all mouse clicks;
+                       // this is used by Select2 to close the dropdown when the user
+                       // clicks outside it.
+                       $actor->find(self::select2DropdownMask())->click();
+
+                       return $tagFound;
+               };
+
+               $numberOfTries = 5;
+               for ($i = 0; $i < $numberOfTries; $i++) {
+                       if ($tagFoundInDropdownCallback()) {
+                               return;
+                       }
+               }
+
+               PHPUnit_Framework_Assert::fail("The dropdown in system tags section in Administration Settings does not contain the tag $tag after $numberOfTries tries");
+       }
+
+}
index 1ff5d94e98f3f59651bb7ff49b220c7de90a048c..cc0fc81cf783c6f43746bc67650928235c9ea31c 100644 (file)
@@ -43,6 +43,13 @@ class SettingsMenuContext implements Context, ActorAwareInterface {
                                describedAs("Settings menu");
        }
 
+       /**
+        * @return Locator
+        */
+       public static function adminMenuItem() {
+               return self::menuItemFor("Admin");
+       }
+
        /**
         * @return Locator
         */
@@ -73,6 +80,15 @@ class SettingsMenuContext implements Context, ActorAwareInterface {
                $this->actor->find(self::settingsMenuButton(), 10)->click();
        }
 
+       /**
+        * @When I open the Admin settings
+        */
+       public function iOpenTheAdminSettings() {
+               $this->iOpenTheSettingsMenu();
+
+               $this->actor->find(self::adminMenuItem(), 2)->click();
+       }
+
        /**
         * @When I open the User settings
         */