All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.github.loyada.dollarx.Path.scala Maven / Gradle / Ivy

package com.github.loyada.dollarx

import com.github.loyada.dollarx.util.XpathUtils
import org.openqa.selenium.WebElement
import com.github.loyada.dollarx.ElementPropertiesHelper.{hasHierarchy, transformXpathToCorrectAxis}

object Path {

  val body = new Path(xpath = Some("body"), xpathExplanation = Some("document body"))
  val div = new Path(xpath = Some("div"), xpathExplanation = Some("div"))
  val listItem = new Path(xpath = Some("li"), xpathExplanation = Some("list item"))
  val unorderedList = new Path(xpath = Some("ul"), xpathExplanation = Some("unordered-list"))
  val orderedList = new Path(xpath = Some("ol"), xpathExplanation = Some("ordered-list"))
  val span = new Path(xpath = Some("span"), xpathExplanation = Some("span"))
  val anchor = new Path(xpath = Some("a"), xpathExplanation = Some("anchor"))
  val image = new Path(xpath = Some("img"), xpathExplanation = Some("image"))
  val html = new Path(xpath = Some("html"), xpathExplanation = Some("document"))
  val button = new Path(xpath = Some("button"), xpathExplanation = Some("button"))
  val input = new Path(xpath = Some("input"), xpathExplanation = Some("input"))
  val form = new Path(xpath = Some("form"), xpathExplanation = Some("form"))
  val iframe = new Path(xpath = Some("iframe"), xpathExplanation = Some("iframe"))
  val title = new Path(xpath = Some("title"), xpathExplanation = Some("title"))
  val header1 = new Path(xpath = Some("h1"), xpathExplanation = Some("header1"))
  val header2 = new Path(xpath = Some("h2"), xpathExplanation = Some("header2"))
  val header3 = new Path(xpath = Some("h3"), xpathExplanation = Some("header3"))
  val header4 = new Path(xpath = Some("h4"), xpathExplanation = Some("header4"))
  val header5 = new Path(xpath = Some("h5"), xpathExplanation = Some("header5"))
  val header = header1 or header2 or header3 or header4 or header5
  val element = new Path(xpath = Some("*"), xpathExplanation = Some("any element"))
  val tr = new Path(xpath = Some("tr"), xpathExplanation = Some("table row"))
  val td = new Path(xpath = Some("td"), xpathExplanation = Some("table cell"))
  val th = new Path(xpath = Some("th"), xpathExplanation = Some("table header cell"))
  val table = customElement("table")
  val select = new Path(xpath = Some("select"), xpathExplanation = Some("selection menu"))
  val option = customElement("option")
  val paragraph = new Path(xpath = Some("p"), xpathExplanation = Some("paragraph"))

  def customElement(el: String): Path = new Path(xpath = Some(el), xpathExplanation = Some(el))

  def apply(path: String): Path = {
    new Path(xpath = Some(path))
  }

  implicit def webElementToPath(we: WebElement): Path = {
    new Path(Some(we))
  }

  object first {
    /**
      *
      * @param path
      * @return The first occurrence of path in the document
      */
    def occurrenceOf(path: Path): Path = path(0)
  }

  object last {
    /**
      *
      * @param path
      * @return The last occurrence of path in the document
      */
    def occurrenceOf(path: Path): Path = path(-1)
  }

  case class childNumber(n: Int) {
    /**
      *
      * @param path
      * @return all the elements that are child number n of type path. This is different than path(n), which is global.
      */
    def ofType(path: Path): Path = {
      val newXpath = path.getXPath.get + s"[${n}]"
      val alternateXpath = path.getAlternateXPath.get + s"[${n}]"
      new Path(path.underlyingSource, xpath = Some(newXpath), elementProps = List(),
        xpathExplanation = Some(s"child number $n of type($path)"), alternateXpath = Some(alternateXpath))
    }
  }

}


class Path(val underlyingSource: Option[WebElement] = None, val xpath: Option[String] = None, val insideXpath: Option[String] = None,
           val elementProps: List[ElementProperty] = Nil, val xpathExplanation: Option[String] = None, val describedBy: Option[String] = None,
           val alternateXpath: Option[String] = None) {


  def getXPath: Option[String] = {
    if (xpath.isEmpty && elementProps.isEmpty && insideXpath.isEmpty) {
      None
    } else {
      val processedXpath = (if (insideXpath.isDefined) (insideXpath.get + "//") else "") + xpath.getOrElse("*")
      val props = elementProps.map(e => s"[${e.toXpath}]").mkString("")
      Some(processedXpath + props)
    }
  }

  def getAlternateXPath: Option[String] = {
    if (xpath.isEmpty && elementProps.isEmpty && insideXpath.isEmpty) {
      None
    } else {
      val props = elementProps.map(e => s"[${e.toXpath}]").mkString("")
      Some(alternateXpath.getOrElse(xpath.getOrElse("*")) + props)
    }
  }

  private def getXPathWithoutInsideClause: Option[String] = {
    if (xpath.isEmpty && elementProps.isEmpty) {
      None
    }
    else {
      val props: String = elementProps.map(e => s"[${e.toXpath}]").mkString("")
      Some(xpath.getOrElse("*") + props)
    }
  }

  def getUnderlyingSource(): Option[WebElement] = underlyingSource

  def getElementProperties = elementProps

  /**
    *
    * @param n
    * @return always a single element, since this is simply nth element of type path in the document.
    */
  def apply(n: Int) = {
    val prefix = if (n == 0) "the first occurrence of " else {
      if (n == -1) "the last occurence of " else s"occurrence number ${n + 1} of "
    }
    val pathString = this.toString()
    val wrapped = if (pathString.contains(" ")) s"($pathString)" else pathString
    val index: String = if (n== -1) "last()" else s"${n+1}"
    new Path(underlyingSource, Some(s"(//${getXPath.get})[$index]"), xpathExplanation = Some(prefix + wrapped),
      alternateXpath = Some(s"(//${getAlternateXPath.get})[$index]"))
  }

  def that(props: ElementProperty*): Path = {
    if (describedBy.isDefined) {
      new Path(underlyingSource, xpath = getXPathWithoutInsideClause, elementProps = List(props: _*), xpathExplanation = describedBy, insideXpath = insideXpath,
        alternateXpath = alternateXpath)
    } else {
      new Path(underlyingSource, xpath = xpath, elementProps = elementProps ++ props, xpathExplanation = xpathExplanation, insideXpath = insideXpath, alternateXpath = alternateXpath)
    }
  }

  def and(props: ElementProperty*): Path = that(props: _*)

  def unary_!(): Path = new Path(underlyingSource,
    Some(XpathUtils.DoesNotExistInEntirePage(getXPath.getOrElse(""))),
    xpathExplanation = Some(s"anything except (${toString()})"),
    alternateXpath = Some(XpathUtils.DoesNotExistInEntirePage(getAlternateXPath.getOrElse(""))) )

  def or(path: Path) = {
    verifyRelationBetweenElements(path)
    new Path(underlyingSource, Some(s"*[(self::${transformXpathToCorrectAxis(this).get}) | (self::${transformXpathToCorrectAxis(path).get})]"),
      alternateXpath =  Some(s"*[(self::${getAlternateXPath.get}) | (self::${path.getAlternateXPath.get})]"),
      xpathExplanation = Some(s"${wrapIfNeeded(this)} or ${wrapIfNeeded(path)}"))
  }


  def withClass(cssClass: String): Path = {
    createNewWithAdditionalProperty(ElementProperties.hasClass(cssClass))
  }

  def withClasses(cssClasses: String*): Path = {
    createNewWithAdditionalProperty(ElementProperties.hasClasses(cssClasses: _*))
  }

  def withTextContaining(txt: String): Path = {
    createNewWithAdditionalProperty(ElementProperties.hasTextContaining(txt))
  }

  private def createNewWithAdditionalProperty(prop: ElementProperty) = {
    if (describedBy.isEmpty) {
      new Path(underlyingSource, xpath, elementProps = elementProps :+ prop, insideXpath = insideXpath,
        xpathExplanation = xpathExplanation, alternateXpath = alternateXpath)
    } else {
      new Path(underlyingSource, getXPath, insideXpath = insideXpath, elementProps = List(prop), xpathExplanation = describedBy, describedBy = describedBy,
        alternateXpath = alternateXpath)
    }
  }

  def withText(txt: String): Path = {
    createNewWithAdditionalProperty(ElementProperties.hasText(txt))
  }

  def withId(id: String): Path = {
    createNewWithAdditionalProperty(ElementProperties.hasId(id))
  }

  def inside(path: Path): Path = {
    verifyRelationBetweenElements(path)
    val newXPath = getXPathWithoutInsideClause.getOrElse("")
    val (correctedXpathForIndex, correctInsidePath) = if (newXPath startsWith ("(")) {
      ((newXPath + s"[${ElementProperties.is.inside(path).toXpath}]"), None)
    } else (newXPath, Some(path.getXPath.get + (if (insideXpath.isDefined) ("//" + insideXpath.get) else "")))
    new Path(path.getUnderlyingSource(),
      xpath = Some(correctedXpathForIndex),
      insideXpath = correctInsidePath,
      alternateXpath = this.that(ElementProperties.is.descendantOf(path)).getAlternateXPath,
      xpathExplanation = Some(toString + s", inside ${wrapIfNeeded(path)}"))

  }

  def insideTopLevel: Path = {
    if (getXPath.isEmpty) throw new IllegalArgumentException("must have a non-empty xpath")
    new Path(
      getUnderlyingSource(),
      xpath = Some(XpathUtils.insideTopLevel(getXPath.get)),
      describedBy = Some(toString()),
      alternateXpath = Some(XpathUtils.insideTopLevel(getAlternateXPath.get)))
  }

  private def wrapIfNeeded(path: Path): String = {
    if ((path.toString.trim.contains(" "))) "(" + path + ")" else path.toString
  }

  def childOf(path: Path) = {
    verifyRelationBetweenElements(path)
    createWithSimpleRelation(path, "child")
  }

  def parentOf(path: Path) = {
    verifyRelationBetweenElements(path)
    createWithSimpleRelation(path, "parent")
  }

  def containing(path: Path) = ancestorOf(path)

  def containedIn(path: Path) = descendantOf(path)

  def ancestorOf(path: Path) = {
    verifyRelationBetweenElements(path)
    createWithSimpleRelation(path, "ancestor")
  }

  def descendantOf(path: Path) = {
    verifyRelationBetweenElements(path)
    createWithSimpleRelation(path, "descendant")
  }

  def afterSibling(path: Path) = {
    verifyRelationBetweenElements(path)
    createWithHumanReadableRelation(path, "following-sibling", "after the sibling")
  }

  def after(path: Path) = {
    verifyRelationBetweenElements(path)
    createWithHumanReadableRelation(path, "following", "after")
  }

  def beforeSibling(path: Path) = {
    verifyRelationBetweenElements(path)
    createWithHumanReadableRelation(path, "preceding-sibling", "before the sibling")
  }

  def before(path: Path): Path = {
    verifyRelationBetweenElements(path)
    createWithHumanReadableRelation(path, "preceding", "before")
  }

  private def verifyRelationBetweenElements(path: Path) {
    if (path.getUnderlyingSource().isDefined || !getXPath.isDefined || !path.getXPath.isDefined) throw new IllegalArgumentException()
  }

  def describedBy(txt: String) = {
    val descriptionAsOption = Some(txt)
    new Path(underlyingSource, xpath = xpath, elementProps = elementProps, insideXpath = insideXpath,
      xpathExplanation = xpathExplanation,
      alternateXpath = alternateXpath,
      describedBy = descriptionAsOption)
  }

  def toXpath() = "xpath: " + getXPath.getOrElse("")

  private def createWithSimpleRelation(path: Path, relation: String): Path = {
    verifyRelationBetweenElements(path)
    val myXpath: String = getXPath.get
    val isInside: Boolean = insideXpath.isDefined
    val processedXpath: String = if (isInside) s"*[ancestor::${insideXpath.get} and self::${xpath.getOrElse("*")}]" else myXpath
    val newAlternateXpath = getAlternateXPath.get + s"[${ElementPropertiesHelper.oppositeRelation(relation)}::${path.getAlternateXPath.get}]"
    val useAlternateXpath = hasHierarchy(processedXpath)
    val newXpath = if (useAlternateXpath) newAlternateXpath else path.getXPath.get + "/" + relation + "::" + processedXpath
    new Path(underlyingSource = underlyingSource,
      xpath = Some(newXpath),
      alternateXpath = Some(newAlternateXpath),
      xpathExplanation = Some(toString + ", " +  relation + " of " + path.toString))
  }

  private def createWithHumanReadableRelation(path: Path, xpathRelation: String, humanReadableRelation: String): Path = {
    verifyRelationBetweenElements(path)
    val myXpath: String = getXPath.get
    val isInside: Boolean = insideXpath.isDefined
    val processedXpath: String = if (isInside) s"${getXPathWithoutInsideClause.get}[ancestor::${insideXpath.get}]" else myXpath
    val newAlternateXpath = getAlternateXPath.get + s"[${ElementPropertiesHelper.oppositeRelation(xpathRelation)}::${path.getAlternateXPath.get}]"
    val useAlternateXpath = hasHierarchy(processedXpath)
    val newXpath = if (useAlternateXpath) newAlternateXpath else (path.getXPath.get + "/" + xpathRelation + "::" + processedXpath)
    new Path(underlyingSource = underlyingSource,
      xpath = Some(newXpath),
      alternateXpath = Some(newAlternateXpath),
      xpathExplanation = Some(toString + ", " + humanReadableRelation + " " + wrapIfNeeded(path)))
  }

  override def toString() = {
    def getXpathExplanationForToString: Option[String] = {
      if (xpath.isDefined) {
        if (xpathExplanation.isDefined) xpathExplanation else Some("xpath: \"" + xpath.get + "\"")
      }
      else None
    }

    def getPropertiesToStringForLength1: Option[String] = {
      val firstProp = elementProps.head.toString
      val thatMaybe: String = if ((firstProp.startsWith("has ") || firstProp.startsWith("is ") || firstProp.startsWith("not "))) "that " else ""
      Some(thatMaybe + elementProps.head)
    }

    def getPropertiesToStringForLengthLargerThan2: Option[String] = {
      val propsAsList: String = elementProps.map(e => e.toString).mkString(", ")
      if (xpathExplanation.isDefined && xpathExplanation.get.contains("with properties") || elementProps.size == 1) {
        Some("and " + propsAsList)
      }
      else {
        Some("that [" + propsAsList + "]")
      }
    }


    if (describedBy.isDefined && describedBy != xpathExplanation) {
      describedBy.get
    } else {
      val underlyingOption = if (underlyingSource.isDefined) Some(s"under reference element ${underlyingSource.get}") else None
      val xpathOption = getXpathExplanationForToString

      val propsOption: Option[String] = {
        if (elementProps.size == 1 && (!xpathOption.getOrElse("").contains(", ") || xpathOption == describedBy)) {
          getPropertiesToStringForLength1
        } else if (elementProps.size == 2 && !xpathOption.getOrElse("").contains(" ")) {
          Some(s"that ${elementProps.head}, and ${elementProps.last}")
        } else if (elementProps.size > 1 || xpathOption.getOrElse("").contains(" ") && elementProps.nonEmpty) {
          getPropertiesToStringForLengthLargerThan2
        } else None
      }

      if (xpathExplanation.isDefined && underlyingOption.isEmpty && propsOption.isEmpty) {
        xpathExplanation.get
      } else {
        List(underlyingOption, xpathOption, propsOption).flatten.mkString(", ")
      }
    }
  }


}







© 2015 - 2025 Weber Informatics LLC | Privacy Policy