123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592 |
- <?php
- /**
- * Base include file for SimpleTest
- * @package SimpleTest
- * @subpackage WebTester
- * @version $Id$
- */
- /**#@+
- * include other SimpleTest class files
- */
- require_once(dirname(__FILE__) . '/page.php');
- require_once(dirname(__FILE__) . '/user_agent.php');
- /**#@-*/
- /**
- * A composite page. Wraps a frameset page and
- * adds subframes. The original page will be
- * mostly ignored. Implements the SimplePage
- * interface so as to be interchangeable.
- * @package SimpleTest
- * @subpackage WebTester
- */
- class SimpleFrameset {
- private $frameset;
- private $frames;
- private $focus;
- private $names;
- /**
- * Stashes the frameset page. Will make use of the
- * browser to fetch the sub frames recursively.
- * @param SimplePage $page Frameset page.
- */
- function __construct($page) {
- $this->frameset = $page;
- $this->frames = array();
- $this->focus = false;
- $this->names = array();
- }
- /**
- * Adds a parsed page to the frameset.
- * @param SimplePage $page Frame page.
- * @param string $name Name of frame in frameset.
- * @access public
- */
- function addFrame($page, $name = false) {
- $this->frames[] = $page;
- if ($name) {
- $this->names[$name] = count($this->frames) - 1;
- }
- }
- /**
- * Replaces existing frame with another. If the
- * frame is nested, then the call is passed down
- * one level.
- * @param array $path Path of frame in frameset.
- * @param SimplePage $page Frame source.
- * @access public
- */
- function setFrame($path, $page) {
- $name = array_shift($path);
- if (isset($this->names[$name])) {
- $index = $this->names[$name];
- } else {
- $index = $name - 1;
- }
- if (count($path) == 0) {
- $this->frames[$index] = &$page;
- return;
- }
- $this->frames[$index]->setFrame($path, $page);
- }
- /**
- * Accessor for current frame focus. Will be
- * false if no frame has focus. Will have the nested
- * frame focus if any.
- * @return array Labels or indexes of nested frames.
- * @access public
- */
- function getFrameFocus() {
- if ($this->focus === false) {
- return array();
- }
- return array_merge(
- array($this->getPublicNameFromIndex($this->focus)),
- $this->frames[$this->focus]->getFrameFocus());
- }
- /**
- * Turns an internal array index into the frames list
- * into a public name, or if none, then a one offset
- * index.
- * @param integer $subject Internal index.
- * @return integer/string Public name.
- * @access private
- */
- protected function getPublicNameFromIndex($subject) {
- foreach ($this->names as $name => $index) {
- if ($subject == $index) {
- return $name;
- }
- }
- return $subject + 1;
- }
- /**
- * Sets the focus by index. The integer index starts from 1.
- * If already focused and the target frame also has frames,
- * then the nested frame will be focused.
- * @param integer $choice Chosen frame.
- * @return boolean True if frame exists.
- * @access public
- */
- function setFrameFocusByIndex($choice) {
- if (is_integer($this->focus)) {
- if ($this->frames[$this->focus]->hasFrames()) {
- return $this->frames[$this->focus]->setFrameFocusByIndex($choice);
- }
- }
- if (($choice < 1) || ($choice > count($this->frames))) {
- return false;
- }
- $this->focus = $choice - 1;
- return true;
- }
- /**
- * Sets the focus by name. If already focused and the
- * target frame also has frames, then the nested frame
- * will be focused.
- * @param string $name Chosen frame.
- * @return boolean True if frame exists.
- * @access public
- */
- function setFrameFocus($name) {
- if (is_integer($this->focus)) {
- if ($this->frames[$this->focus]->hasFrames()) {
- return $this->frames[$this->focus]->setFrameFocus($name);
- }
- }
- if (in_array($name, array_keys($this->names))) {
- $this->focus = $this->names[$name];
- return true;
- }
- return false;
- }
- /**
- * Clears the frame focus.
- * @access public
- */
- function clearFrameFocus() {
- $this->focus = false;
- $this->clearNestedFramesFocus();
- }
- /**
- * Clears the frame focus for any nested frames.
- * @access private
- */
- protected function clearNestedFramesFocus() {
- for ($i = 0; $i < count($this->frames); $i++) {
- $this->frames[$i]->clearFrameFocus();
- }
- }
- /**
- * Test for the presence of a frameset.
- * @return boolean Always true.
- * @access public
- */
- function hasFrames() {
- return true;
- }
- /**
- * Accessor for frames information.
- * @return array/string Recursive hash of frame URL strings.
- * The key is either a numerical
- * index or the name attribute.
- * @access public
- */
- function getFrames() {
- $report = array();
- for ($i = 0; $i < count($this->frames); $i++) {
- $report[$this->getPublicNameFromIndex($i)] =
- $this->frames[$i]->getFrames();
- }
- return $report;
- }
- /**
- * Accessor for raw text of either all the pages or
- * the frame in focus.
- * @return string Raw unparsed content.
- * @access public
- */
- function getRaw() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getRaw();
- }
- $raw = '';
- for ($i = 0; $i < count($this->frames); $i++) {
- $raw .= $this->frames[$i]->getRaw();
- }
- return $raw;
- }
- /**
- * Accessor for plain text of either all the pages or
- * the frame in focus.
- * @return string Plain text content.
- * @access public
- */
- function getText() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getText();
- }
- $raw = '';
- for ($i = 0; $i < count($this->frames); $i++) {
- $raw .= ' ' . $this->frames[$i]->getText();
- }
- return trim($raw);
- }
- /**
- * Accessor for last error.
- * @return string Error from last response.
- * @access public
- */
- function getTransportError() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getTransportError();
- }
- return $this->frameset->getTransportError();
- }
- /**
- * Request method used to fetch this frame.
- * @return string GET, POST or HEAD.
- * @access public
- */
- function getMethod() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getMethod();
- }
- return $this->frameset->getMethod();
- }
- /**
- * Original resource name.
- * @return SimpleUrl Current url.
- * @access public
- */
- function getUrl() {
- if (is_integer($this->focus)) {
- $url = $this->frames[$this->focus]->getUrl();
- $url->setTarget($this->getPublicNameFromIndex($this->focus));
- } else {
- $url = $this->frameset->getUrl();
- }
- return $url;
- }
- /**
- * Page base URL.
- * @return SimpleUrl Current url.
- * @access public
- */
- function getBaseUrl() {
- if (is_integer($this->focus)) {
- $url = $this->frames[$this->focus]->getBaseUrl();
- } else {
- $url = $this->frameset->getBaseUrl();
- }
- return $url;
- }
- /**
- * Expands expandomatic URLs into fully qualified
- * URLs for the frameset page.
- * @param SimpleUrl $url Relative URL.
- * @return SimpleUrl Absolute URL.
- * @access public
- */
- function expandUrl($url) {
- return $this->frameset->expandUrl($url);
- }
- /**
- * Original request data.
- * @return mixed Sent content.
- * @access public
- */
- function getRequestData() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getRequestData();
- }
- return $this->frameset->getRequestData();
- }
- /**
- * Accessor for current MIME type.
- * @return string MIME type as string; e.g. 'text/html'
- * @access public
- */
- function getMimeType() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getMimeType();
- }
- return $this->frameset->getMimeType();
- }
- /**
- * Accessor for last response code.
- * @return integer Last HTTP response code received.
- * @access public
- */
- function getResponseCode() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getResponseCode();
- }
- return $this->frameset->getResponseCode();
- }
- /**
- * Accessor for last Authentication type. Only valid
- * straight after a challenge (401).
- * @return string Description of challenge type.
- * @access public
- */
- function getAuthentication() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getAuthentication();
- }
- return $this->frameset->getAuthentication();
- }
- /**
- * Accessor for last Authentication realm. Only valid
- * straight after a challenge (401).
- * @return string Name of security realm.
- * @access public
- */
- function getRealm() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getRealm();
- }
- return $this->frameset->getRealm();
- }
- /**
- * Accessor for outgoing header information.
- * @return string Header block.
- * @access public
- */
- function getRequest() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getRequest();
- }
- return $this->frameset->getRequest();
- }
- /**
- * Accessor for raw header information.
- * @return string Header block.
- * @access public
- */
- function getHeaders() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getHeaders();
- }
- return $this->frameset->getHeaders();
- }
- /**
- * Accessor for parsed title.
- * @return string Title or false if no title is present.
- * @access public
- */
- function getTitle() {
- return $this->frameset->getTitle();
- }
- /**
- * Accessor for a list of all fixed links.
- * @return array List of urls as strings.
- * @access public
- */
- function getUrls() {
- if (is_integer($this->focus)) {
- return $this->frames[$this->focus]->getUrls();
- }
- $urls = array();
- foreach ($this->frames as $frame) {
- $urls = array_merge($urls, $frame->getUrls());
- }
- return array_values(array_unique($urls));
- }
- /**
- * Accessor for URLs by the link label. Label will match
- * regardess of whitespace issues and case.
- * @param string $label Text of link.
- * @return array List of links with that label.
- * @access public
- */
- function getUrlsByLabel($label) {
- if (is_integer($this->focus)) {
- return $this->tagUrlsWithFrame(
- $this->frames[$this->focus]->getUrlsByLabel($label),
- $this->focus);
- }
- $urls = array();
- foreach ($this->frames as $index => $frame) {
- $urls = array_merge(
- $urls,
- $this->tagUrlsWithFrame(
- $frame->getUrlsByLabel($label),
- $index));
- }
- return $urls;
- }
- /**
- * Accessor for a URL by the id attribute. If in a frameset
- * then the first link found with that ID attribute is
- * returned only. Focus on a frame if you want one from
- * a specific part of the frameset.
- * @param string $id Id attribute of link.
- * @return string URL with that id.
- * @access public
- */
- function getUrlById($id) {
- foreach ($this->frames as $index => $frame) {
- if ($url = $frame->getUrlById($id)) {
- if (! $url->gettarget()) {
- $url->setTarget($this->getPublicNameFromIndex($index));
- }
- return $url;
- }
- }
- return false;
- }
- /**
- * Attaches the intended frame index to a list of URLs.
- * @param array $urls List of SimpleUrls.
- * @param string $frame Name of frame or index.
- * @return array List of tagged URLs.
- * @access private
- */
- protected function tagUrlsWithFrame($urls, $frame) {
- $tagged = array();
- foreach ($urls as $url) {
- if (! $url->getTarget()) {
- $url->setTarget($this->getPublicNameFromIndex($frame));
- }
- $tagged[] = $url;
- }
- return $tagged;
- }
- /**
- * Finds a held form by button label. Will only
- * search correctly built forms.
- * @param SimpleSelector $selector Button finder.
- * @return SimpleForm Form object containing
- * the button.
- * @access public
- */
- function getFormBySubmit($selector) {
- return $this->findForm('getFormBySubmit', $selector);
- }
- /**
- * Finds a held form by image using a selector.
- * Will only search correctly built forms. The first
- * form found either within the focused frame, or
- * across frames, will be the one returned.
- * @param SimpleSelector $selector Image finder.
- * @return SimpleForm Form object containing
- * the image.
- * @access public
- */
- function getFormByImage($selector) {
- return $this->findForm('getFormByImage', $selector);
- }
- /**
- * Finds a held form by the form ID. A way of
- * identifying a specific form when we have control
- * of the HTML code. The first form found
- * either within the focused frame, or across frames,
- * will be the one returned.
- * @param string $id Form label.
- * @return SimpleForm Form object containing the matching ID.
- * @access public
- */
- function getFormById($id) {
- return $this->findForm('getFormById', $id);
- }
- /**
- * General form finder. Will search all the frames or
- * just the one in focus.
- * @param string $method Method to use to find in a page.
- * @param string $attribute Label, name or ID.
- * @return SimpleForm Form object containing the matching ID.
- * @access private
- */
- protected function findForm($method, $attribute) {
- if (is_integer($this->focus)) {
- return $this->findFormInFrame(
- $this->frames[$this->focus],
- $this->focus,
- $method,
- $attribute);
- }
- for ($i = 0; $i < count($this->frames); $i++) {
- $form = $this->findFormInFrame(
- $this->frames[$i],
- $i,
- $method,
- $attribute);
- if ($form) {
- return $form;
- }
- }
- $null = null;
- return $null;
- }
- /**
- * Finds a form in a page using a form finding method. Will
- * also tag the form with the frame name it belongs in.
- * @param SimplePage $page Page content of frame.
- * @param integer $index Internal frame representation.
- * @param string $method Method to use to find in a page.
- * @param string $attribute Label, name or ID.
- * @return SimpleForm Form object containing the matching ID.
- * @access private
- */
- protected function findFormInFrame($page, $index, $method, $attribute) {
- $form = $this->frames[$index]->$method($attribute);
- if (isset($form)) {
- $form->setDefaultTarget($this->getPublicNameFromIndex($index));
- }
- return $form;
- }
- /**
- * Sets a field on each form in which the field is
- * available.
- * @param SimpleSelector $selector Field finder.
- * @param string $value Value to set field to.
- * @return boolean True if value is valid.
- * @access public
- */
- function setField($selector, $value) {
- if (is_integer($this->focus)) {
- $this->frames[$this->focus]->setField($selector, $value);
- } else {
- for ($i = 0; $i < count($this->frames); $i++) {
- $this->frames[$i]->setField($selector, $value);
- }
- }
- }
- /**
- * Accessor for a form element value within a page.
- * @param SimpleSelector $selector Field finder.
- * @return string/boolean A string if the field is
- * present, false if unchecked
- * and null if missing.
- * @access public
- */
- function getField($selector) {
- for ($i = 0; $i < count($this->frames); $i++) {
- $value = $this->frames[$i]->getField($selector);
- if (isset($value)) {
- return $value;
- }
- }
- return null;
- }
- }
- ?>
|