Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

SelectorPath.java 7.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. /*
  2. * Copyright 2000-2013 Vaadin Ltd.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License"); you may not
  5. * use this file except in compliance with the License. You may obtain a copy of
  6. * the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
  12. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
  13. * License for the specific language governing permissions and limitations under
  14. * the License.
  15. */
  16. package com.vaadin.client.debug.internal;
  17. import java.util.HashMap;
  18. import java.util.List;
  19. import java.util.Map;
  20. import com.vaadin.client.ServerConnector;
  21. import com.vaadin.client.componentlocator.ComponentLocator;
  22. import com.vaadin.client.componentlocator.SelectorPredicate;
  23. /**
  24. * A single segment of a selector path pointing to an Element.
  25. * <p>
  26. * This class should be considered internal to the framework and may change at
  27. * any time.
  28. * <p>
  29. *
  30. * @since 7.1.x
  31. */
  32. public class SelectorPath {
  33. private final String path;
  34. private final com.google.gwt.user.client.Element element;
  35. private final ComponentLocator locator;
  36. private static Map<String, Integer> counter = new HashMap<String, Integer>();
  37. private static Map<String, String> legacyNames = new HashMap<String, String>();
  38. static {
  39. legacyNames.put("FilterSelect", "ComboBox");
  40. legacyNames.put("ScrollTable", "Table");
  41. }
  42. protected SelectorPath(ServerConnector c,
  43. com.google.gwt.user.client.Element e) {
  44. element = e;
  45. locator = new ComponentLocator(c.getConnection());
  46. path = locator.getPathForElement(e);
  47. }
  48. public String getPath() {
  49. return path;
  50. }
  51. public com.google.gwt.user.client.Element getElement() {
  52. return element;
  53. }
  54. public ComponentLocator getLocator() {
  55. return locator;
  56. }
  57. /**
  58. * Generate ElementQuery code for Java. Fallback to By.vaadin(path) if
  59. * dealing with LegacyLocator
  60. *
  61. * @return String containing Java code for finding the element described by
  62. * path
  63. */
  64. public String getElementQuery() {
  65. if (path.isEmpty() || locator.isValidForLegacyLocator(path)) {
  66. return getLegacyLocatorQuery();
  67. }
  68. String[] fragments;
  69. String tmpPath = path;
  70. List<SelectorPredicate> postFilters = SelectorPredicate
  71. .extractPostFilterPredicates(path);
  72. if (postFilters.size() > 0) {
  73. tmpPath = tmpPath.substring(1, tmpPath.lastIndexOf(')'));
  74. }
  75. // Generate an ElementQuery
  76. fragments = tmpPath.split("/");
  77. String elementQueryString = "";
  78. int index = 0;
  79. for (SelectorPredicate p : postFilters) {
  80. if (p.getIndex() > 0) {
  81. index = p.getIndex();
  82. }
  83. }
  84. for (int i = 1; i < fragments.length; ++i) {
  85. if (fragments[i].isEmpty()) {
  86. // Recursive searches cause empty fragments
  87. continue;
  88. }
  89. // if i == 1 or previous fragment was empty, search is recursive
  90. boolean recursive = (i > 1 ? fragments[i - 1].isEmpty() : false);
  91. // if elementQueryString is not empty, join the next query with .
  92. String queryFragment = (!elementQueryString.isEmpty() ? "." : "");
  93. // if search is not recursive, add another $ in front of fragment
  94. queryFragment += (!recursive ? "$" : "")
  95. + generateFragment(fragments[i]);
  96. elementQueryString += queryFragment;
  97. }
  98. if (index == 0) {
  99. elementQueryString += ".first()";
  100. } else {
  101. elementQueryString += ".get(" + index + ");";
  102. }
  103. // Return full Java variable assignment and eQuery
  104. return generateJavaVariable(fragments[fragments.length - 1])
  105. + elementQueryString;
  106. }
  107. /**
  108. * Generates a recursive ElementQuery for given path fragment
  109. *
  110. * @return ElementQuery java code as a String
  111. */
  112. private String generateFragment(String fragment) {
  113. // Get Element.class -name
  114. String elementClass = getComponentName(fragment) + "Element.class";
  115. String queryFragment = "$(" + elementClass + ")";
  116. for (SelectorPredicate p : SelectorPredicate
  117. .extractPredicates(fragment)) {
  118. // Add in predicates like .caption and .id
  119. queryFragment += "." + p.getName() + "(\"" + p.getValue() + "\")";
  120. }
  121. return queryFragment;
  122. }
  123. /**
  124. * @since
  125. * @param frags
  126. * @param i
  127. * @return
  128. */
  129. protected String getComponentName(String fragment) {
  130. return fragment.split("\\[")[0];
  131. }
  132. /**
  133. * Generates a legacy locator for SelectorPath.
  134. *
  135. * @return String containing Java code for element search and assignment
  136. */
  137. private String getLegacyLocatorQuery() {
  138. String name;
  139. if (!path.isEmpty()) {
  140. String[] frags = path.split("/");
  141. name = getComponentName(frags[frags.length - 1]).substring(1);
  142. } else {
  143. name = "root";
  144. }
  145. if (legacyNames.containsKey(name)) {
  146. name = legacyNames.get(name);
  147. }
  148. name = getNameWithCount(name);
  149. // Use direct path and elementX naming style.
  150. return "WebElement " + name.substring(0, 1).toLowerCase()
  151. + name.substring(1) + " = getDriver().findElement(By.vaadin(\""
  152. + path + "\"));";
  153. }
  154. /**
  155. * Get variable name with counter for given component name.
  156. *
  157. * @param name
  158. * Component name
  159. * @return name followed by count
  160. */
  161. protected String getNameWithCount(String name) {
  162. if (!counter.containsKey(name)) {
  163. counter.put(name, 0);
  164. }
  165. counter.put(name, counter.get(name) + 1);
  166. name += counter.get(name);
  167. return name;
  168. }
  169. /**
  170. * Generate Java variable assignment from given selector fragment
  171. *
  172. * @param pathFragment
  173. * Selector fragment
  174. * @return piece of java code
  175. */
  176. private String generateJavaVariable(String pathFragment) {
  177. // Get element type and predicates from fragment
  178. List<SelectorPredicate> predicates = SelectorPredicate
  179. .extractPredicates(pathFragment);
  180. String elementType = pathFragment.split("\\[")[0];
  181. String name = getNameFromPredicates(predicates, elementType);
  182. if (name.equals(elementType)) {
  183. name = getNameWithCount(name);
  184. }
  185. // Replace unusable characters
  186. name = name.replaceAll("\\W", "");
  187. // Lowercase the first character of name
  188. return elementType + "Element " + name.substring(0, 1).toLowerCase()
  189. + name.substring(1) + " = ";
  190. }
  191. /**
  192. * Get variable name based on predicates. Fallback to elementType
  193. *
  194. * @param predicates
  195. * Predicates related to element
  196. * @param elementType
  197. * Element type
  198. * @return name for Variable
  199. */
  200. private String getNameFromPredicates(List<SelectorPredicate> predicates,
  201. String elementType) {
  202. String name = elementType;
  203. for (SelectorPredicate p : predicates) {
  204. if ("caption".equals(p.getName())) {
  205. // Caption + elementType is a suitable name
  206. name = p.getValue() + elementType;
  207. } else if ("id".equals(p.getName())) {
  208. // Just id. This is unique, use it.
  209. return p.getValue();
  210. }
  211. }
  212. return name;
  213. }
  214. }