2010-06-01 13:45:04 -07:00
|
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is MozMill Test code.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is Mozilla Foundation.
|
|
|
|
* Portions created by the Initial Developer are Copyright (C) 2009
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
* Henrik Skupin <hskupin@mozilla.com>
|
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
|
|
* the provisions above, a recipient may use your version of this file under
|
|
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
|
|
*
|
|
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @fileoverview
|
|
|
|
* The ToolbarAPI adds support for accessing and interacting with toolbar elements
|
|
|
|
*
|
|
|
|
* @version 1.0.0
|
|
|
|
*/
|
|
|
|
|
2010-09-17 17:18:06 -07:00
|
|
|
const MODULE_NAME = 'ToolbarAPI';
|
2010-06-01 13:45:04 -07:00
|
|
|
|
2010-09-17 17:18:06 -07:00
|
|
|
const RELATIVE_ROOT = '.';
|
|
|
|
const MODULE_REQUIRES = ['UtilsAPI'];
|
2010-06-01 13:45:04 -07:00
|
|
|
|
2010-09-17 17:18:06 -07:00
|
|
|
const TIMEOUT = 5000;
|
|
|
|
|
|
|
|
const AUTOCOMPLETE_POPUP = '/id("main-window")/id("mainPopupSet")/id("PopupAutoCompleteRichResult")';
|
|
|
|
const URLBAR_CONTAINER = '/id("main-window")/id("tab-view-deck")/{"flex":"1"}' +
|
|
|
|
'/id("navigator-toolbox")/id("nav-bar")/id("urlbar-container")';
|
|
|
|
const URLBAR_INPUTBOX = URLBAR_CONTAINER + '/id("urlbar")/anon({"class":"autocomplete-textbox-container"})' +
|
|
|
|
'/anon({"anonid":"textbox-input-box"})';
|
|
|
|
const CONTEXT_MENU = URLBAR_INPUTBOX + '/anon({"anonid":"input-box-contextmenu"})';
|
2010-06-01 13:45:04 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*
|
|
|
|
* @param {MozmillController} controller
|
|
|
|
* MozMillController of the window to operate on
|
|
|
|
*/
|
|
|
|
function autoCompleteResults(controller)
|
|
|
|
{
|
|
|
|
this._controller = controller;
|
|
|
|
this.popup = this.getElement({type: "popup"});
|
|
|
|
this._results = this.getElement({type: "results"});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* AutoComplete Result class
|
|
|
|
*/
|
|
|
|
autoCompleteResults.prototype = {
|
|
|
|
/**
|
|
|
|
* Returns all autocomplete results
|
|
|
|
*
|
|
|
|
* @returns Autocomplete results
|
|
|
|
* @type {Array of ElemBase}
|
|
|
|
*/
|
|
|
|
get allResults() {
|
|
|
|
var results = [];
|
|
|
|
for (ii = 0; ii < this.length; ii++) {
|
|
|
|
results.push(this.getResult(ii));
|
|
|
|
}
|
|
|
|
return results;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the controller of the current window
|
|
|
|
*
|
|
|
|
* @returns Mozmill Controller
|
|
|
|
* @type MozMillController
|
|
|
|
*/
|
|
|
|
get controller() {
|
|
|
|
return this._controller;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if the autocomplete popup is open
|
|
|
|
*
|
|
|
|
* @returns True if the panel is open
|
|
|
|
* @type {boolean}
|
|
|
|
*/
|
|
|
|
get isOpened() {
|
|
|
|
return (this.popup.getNode().state == 'open');
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the amount of autocomplete entries
|
|
|
|
*
|
|
|
|
* @returns Number of all entries
|
|
|
|
* @type {number}
|
|
|
|
*/
|
|
|
|
get length() {
|
|
|
|
return this._results.getNode().itemCount;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the currently selected index
|
|
|
|
*
|
|
|
|
* @returns Selected index
|
|
|
|
* @type {number}
|
|
|
|
*/
|
|
|
|
get selectedIndex() {
|
|
|
|
return this._results.getNode().selectedIndex;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the visible autocomplete results
|
|
|
|
*
|
|
|
|
* @returns Results
|
|
|
|
* @type {Array of ElemBase}
|
|
|
|
*/
|
|
|
|
get visibleResults() {
|
|
|
|
var results = [];
|
|
|
|
for (ii = 0; ii < this.length; ii++) {
|
|
|
|
var result = this.getResult(ii);
|
|
|
|
if (!result.getNode().hasAttribute("collapsed"))
|
|
|
|
results.push(result);
|
|
|
|
}
|
|
|
|
return results;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the underlined text of all results from the text or URL
|
|
|
|
*
|
|
|
|
* @param {ElemBase} result
|
|
|
|
* Autocomplete result which has to be checked
|
|
|
|
* @param {string} type
|
|
|
|
* Type of element to check (text or url)
|
|
|
|
*
|
|
|
|
* @returns An array of substrings which are underlined
|
|
|
|
* @type {Array of string}
|
|
|
|
*/
|
|
|
|
getUnderlinedText : function autoCompleteResults_getUnderlinedText(result, type) {
|
|
|
|
this._controller.assertJS("subject.resultNode != null",
|
|
|
|
{resultNode: result.getNode()});
|
|
|
|
|
|
|
|
// Get the description element of the given title or url
|
|
|
|
var description = null;
|
|
|
|
switch (type) {
|
|
|
|
case "title":
|
|
|
|
description = result.getNode().boxObject.firstChild.childNodes[1].childNodes[0];
|
|
|
|
break;
|
|
|
|
case "url":
|
2010-09-17 17:18:06 -07:00
|
|
|
description = result.getNode().boxObject.lastChild.childNodes[2].childNodes[0];
|
2010-06-01 13:45:04 -07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Error(arguments.callee.name + ": Type unknown - " + type);
|
|
|
|
}
|
|
|
|
|
|
|
|
let values = [ ];
|
|
|
|
for each (node in description.childNodes) {
|
|
|
|
if (node.nodeName == 'span') {
|
|
|
|
// Only add underlined text to the results
|
|
|
|
values.push(node.innerHTML);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return values;
|
|
|
|
},
|
|
|
|
|
2010-09-17 17:18:06 -07:00
|
|
|
/**
|
|
|
|
* Gets all the needed external DTD urls as an array
|
|
|
|
*
|
|
|
|
* @returns Array of external DTD urls
|
|
|
|
* @type [string]
|
|
|
|
*/
|
|
|
|
getDtds : function autoCompleteResults_getDtds() {
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
|
2010-06-01 13:45:04 -07:00
|
|
|
/**
|
|
|
|
* Retrieve an UI element based on the given spec
|
|
|
|
*
|
|
|
|
* @param {object} spec
|
|
|
|
* Information of the UI element which should be retrieved
|
|
|
|
* type: General type information
|
|
|
|
* subtype: Specific element or property
|
|
|
|
* value: Value of the element or property
|
|
|
|
* @returns Element which has been created
|
|
|
|
* @type {ElemBase}
|
|
|
|
*/
|
|
|
|
getElement : function autoCompleteResults_getElement(spec) {
|
|
|
|
var elem = null;
|
|
|
|
|
|
|
|
switch (spec.type) {
|
|
|
|
/**
|
|
|
|
* subtype: subtype to match
|
|
|
|
* value: value to match
|
|
|
|
*/
|
|
|
|
case "popup":
|
2010-09-17 17:18:06 -07:00
|
|
|
elem = new elementslib.Lookup(this._controller.window.document, AUTOCOMPLETE_POPUP);
|
2010-06-01 13:45:04 -07:00
|
|
|
break;
|
|
|
|
case "results":
|
|
|
|
elem = new elementslib.Lookup(this._controller.window.document,
|
2010-09-17 17:18:06 -07:00
|
|
|
AUTOCOMPLETE_POPUP + '/anon({"anonid":"richlistbox"})');
|
2010-06-01 13:45:04 -07:00
|
|
|
break;
|
|
|
|
case "result":
|
|
|
|
elem = new elementslib.Elem(this._results.getNode().getItemAtIndex(spec.value));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Error(arguments.callee.name + ": Unknown element type - " + spec.type);
|
|
|
|
}
|
|
|
|
|
|
|
|
return elem;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the autocomplete result element of the given index
|
|
|
|
*
|
|
|
|
* @param {number} index
|
|
|
|
* Index of the result to return
|
|
|
|
* @returns Autocomplete result element
|
|
|
|
* @type {ElemBase}
|
|
|
|
*/
|
|
|
|
getResult : function autoCompleteResults_getResult(index) {
|
|
|
|
return this.getElement({type: "result", value: index});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*
|
|
|
|
* @param {MozmillController} controller
|
|
|
|
* MozMillController of the window to operate on
|
|
|
|
*/
|
|
|
|
function locationBar(controller)
|
|
|
|
{
|
|
|
|
this._controller = controller;
|
|
|
|
this._autoCompleteResults = new autoCompleteResults(controller);
|
2010-09-17 17:18:06 -07:00
|
|
|
this._utilsApi = collector.getModule('UtilsAPI');
|
2010-06-01 13:45:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Location Bar class
|
|
|
|
*/
|
|
|
|
locationBar.prototype = {
|
|
|
|
/**
|
|
|
|
* Returns the autocomplete object
|
|
|
|
*
|
|
|
|
* @returns Autocomplete object
|
|
|
|
* @type {object}
|
|
|
|
*/
|
|
|
|
get autoCompleteResults() {
|
|
|
|
return this._autoCompleteResults;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the controller of the current window
|
|
|
|
*
|
|
|
|
* @returns Mozmill controller
|
|
|
|
* @type {MozMillController}
|
|
|
|
*/
|
|
|
|
get controller() {
|
|
|
|
return this._controller;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the urlbar element
|
|
|
|
*
|
|
|
|
* @returns URL bar
|
|
|
|
* @type {ElemBase}
|
|
|
|
*/
|
|
|
|
get urlbar() {
|
|
|
|
return this.getElement({type: "urlbar"});
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the currently shown URL
|
|
|
|
*
|
|
|
|
* @returns Text inside the location bar
|
|
|
|
* @type {string}
|
|
|
|
*/
|
|
|
|
get value() {
|
|
|
|
return this.urlbar.value;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clear the location bar
|
|
|
|
*/
|
|
|
|
clear : function locationBar_clear() {
|
|
|
|
this.focus({type: "shortcut"});
|
|
|
|
this._controller.keypress(this.urlbar, "VK_DELETE", {});
|
|
|
|
this._controller.waitForEval("subject.value == ''",
|
2010-09-17 17:18:06 -07:00
|
|
|
TIMEOUT, 100, this.urlbar.getNode());
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Close the context menu of the urlbar input field
|
|
|
|
*/
|
|
|
|
closeContextMenu : function locationBar_closeContextMenu() {
|
|
|
|
var menu = this.getElement({type: "contextMenu"});
|
|
|
|
this._controller.keypress(menu, "VK_ESCAPE", {});
|
2010-06-01 13:45:04 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if the location bar contains the given text
|
|
|
|
*
|
|
|
|
* @param {string} text
|
|
|
|
* Text which should be checked against
|
|
|
|
*/
|
|
|
|
contains : function locationBar_contains(text) {
|
|
|
|
return this.urlbar.getNode().value.indexOf(text) != -1;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Focus the location bar
|
|
|
|
*
|
|
|
|
* @param {object} event
|
|
|
|
* Focus the location bar with the given event (click or shortcut)
|
|
|
|
*/
|
|
|
|
focus : function locationBar_focus(event) {
|
|
|
|
switch (event.type) {
|
|
|
|
case "click":
|
|
|
|
this._controller.click(this.urlbar);
|
|
|
|
break;
|
|
|
|
case "shortcut":
|
2010-09-17 17:18:06 -07:00
|
|
|
var cmdKey = this._utilsApi.getEntity(this.getDtds(), "openCmd.commandkey");
|
|
|
|
this._controller.keypress(null, cmdKey, {accelKey: true});
|
2010-06-01 13:45:04 -07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Error(arguments.callee.name + ": Unkown event type - " + event.type);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait until the location bar has been focused
|
|
|
|
this._controller.waitForEval("subject.getAttribute('focused') == 'true'",
|
2010-09-17 17:18:06 -07:00
|
|
|
TIMEOUT, 100, this.urlbar.getNode());
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets all the needed external DTD urls as an array
|
|
|
|
*
|
|
|
|
* @returns Array of external DTD urls
|
|
|
|
* @type [string]
|
|
|
|
*/
|
|
|
|
getDtds : function locationBar_getDtds() {
|
|
|
|
var dtds = ["chrome://branding/locale/brand.dtd",
|
|
|
|
"chrome://browser/locale/browser.dtd"];
|
|
|
|
return dtds;
|
2010-06-01 13:45:04 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve an UI element based on the given spec
|
|
|
|
*
|
|
|
|
* @param {object} spec
|
|
|
|
* Information of the UI element which should be retrieved
|
|
|
|
* type: General type information
|
|
|
|
* subtype: Specific element or property
|
|
|
|
* value: Value of the element or property
|
|
|
|
* @returns Element which has been created
|
|
|
|
* @type ElemBase
|
|
|
|
*/
|
|
|
|
getElement : function locationBar_getElement(spec) {
|
|
|
|
var elem = null;
|
|
|
|
|
|
|
|
switch(spec.type) {
|
|
|
|
/**
|
|
|
|
* subtype: subtype to match
|
|
|
|
* value: value to match
|
|
|
|
*/
|
2010-09-17 17:18:06 -07:00
|
|
|
case "contextMenu":
|
|
|
|
elem = new elementslib.Lookup(this._controller.window.document, CONTEXT_MENU);
|
|
|
|
break;
|
|
|
|
case "contextMenu_entry":
|
|
|
|
elem = new elementslib.Lookup(this._controller.window.document, CONTEXT_MENU +
|
|
|
|
'/{"cmd":"cmd_' + spec.subtype + '"}');
|
|
|
|
break;
|
2010-06-01 13:45:04 -07:00
|
|
|
case "favicon":
|
|
|
|
elem = new elementslib.ID(this._controller.window.document, "page-proxy-favicon");
|
|
|
|
break;
|
|
|
|
case "feedButton":
|
|
|
|
elem = new elementslib.ID(this._controller.window.document, "feed-button");
|
|
|
|
break;
|
|
|
|
case "goButton":
|
2010-09-17 17:18:06 -07:00
|
|
|
elem = new elementslib.ID(this._controller.window.document, "urlbar-go-button");
|
2010-06-01 13:45:04 -07:00
|
|
|
break;
|
|
|
|
case "historyDropMarker":
|
|
|
|
elem = new elementslib.Lookup(this._controller.window.document,
|
2010-09-17 17:18:06 -07:00
|
|
|
URLBAR_CONTAINER + '/id("urlbar")/anon({"anonid":"historydropmarker"})');
|
2010-06-01 13:45:04 -07:00
|
|
|
break;
|
|
|
|
case "identityBox":
|
|
|
|
elem = new elementslib.ID(this._controller.window.document, "identity-box");
|
|
|
|
break;
|
|
|
|
case "starButton":
|
|
|
|
elem = new elementslib.ID(this._controller.window.document, "star-button");
|
|
|
|
break;
|
|
|
|
case "urlbar":
|
|
|
|
elem = new elementslib.ID(this._controller.window.document, "urlbar");
|
|
|
|
break;
|
2010-09-17 17:18:06 -07:00
|
|
|
case "urlbar_input":
|
|
|
|
elem = new elementslib.Lookup(this._controller.window.document, URLBAR_INPUTBOX +
|
|
|
|
'/anon({"anonid":"input"})');
|
|
|
|
break;
|
2010-06-01 13:45:04 -07:00
|
|
|
default:
|
|
|
|
throw new Error(arguments.callee.name + ": Unknown element type - " + spec.type);
|
|
|
|
}
|
|
|
|
|
|
|
|
return elem;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Load the given URL
|
|
|
|
*
|
|
|
|
* @param {string} url
|
|
|
|
* URL of web page to load
|
|
|
|
*/
|
|
|
|
loadURL : function locationBar_loadURL(url) {
|
|
|
|
this.focus({type: "shortcut"});
|
|
|
|
this.type(url);
|
|
|
|
this._controller.keypress(this.urlbar, "VK_RETURN", {});
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Toggles between the open and closed state of the auto-complete popup
|
|
|
|
*/
|
|
|
|
toggleAutocompletePopup : function locationBar_toggleAutocompletePopup() {
|
|
|
|
var dropdown = this.getElement({type: "historyDropMarker"});
|
|
|
|
var stateOpen = this.autoCompleteResults.isOpened;
|
|
|
|
|
|
|
|
this._controller.click(dropdown);
|
|
|
|
this._controller.waitForEval("subject.isOpened == " + stateOpen,
|
2010-09-17 17:18:06 -07:00
|
|
|
TIMEOUT, 100, this.autoCompleteResults);
|
2010-06-01 13:45:04 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Type the given text into the location bar
|
|
|
|
*
|
|
|
|
* @param {string} text
|
|
|
|
* Text to enter into the location bar
|
|
|
|
*/
|
|
|
|
type : function locationBar_type(text) {
|
|
|
|
this._controller.type(this.urlbar, text);
|
|
|
|
this.contains(text);
|
|
|
|
}
|
|
|
|
}
|