/*Copyright (c) 2006, Yahoo! Inc. All rights reserved.Code licensed under the BSD License:http://developer.yahoo.com/yui/license.txtversion: 0.11.3*//****************************************************************************//****************************************************************************//****************************************************************************//** * Class providing the customizable functionality of a plug-and-play DHTML * auto complete widget.  Some key features: * <ul> * <li>Navigate with up/down arrow keys and/or mouse to pick a selection</li> * <li>The drop down container can "roll down" or "fly out" via configurable * animation</li> * <li>UI look-and-feel customizable through CSS, including container * attributes, borders, position, fonts, etc</li> * </ul> * * requires YAHOO.util.Dom Dom utility * requires YAHOO.util.Event Event utility * requires YAHOO.widget.DataSource Data source class * see YAHOO.util.Animation Animation utility * see JSON JSON library * * @constructor * @param {element | string} inputEl DOM element reference or string ID of the auto complete input field * @param {element | string} containerEl DOM element reference or string ID of the auto complete &lt;div&gt; *                              container * @param {object} oDataSource Instance of YAHOO.widget.DataSource for query/results * @param {object} oConfigs Optional object literal of config params */YAHOO.widget.AutoComplete = function(inputEl,containerEl,oDataSource,oConfigs) {    if(inputEl && containerEl && oDataSource) {        // Validate data source        if (oDataSource && (oDataSource instanceof YAHOO.widget.DataSource)) {            this.dataSource = oDataSource;        }        else {            return;        }        // Validate input element        if(YAHOO.util.Dom.inDocument(inputEl)) {            if(typeof inputEl == "string") {                    this._sName = "instance" + YAHOO.widget.AutoComplete._nIndex + " " + inputEl;                    this._oTextbox = document.getElementById(inputEl);            }            else {                this._sName = (inputEl.id) ?                    "instance" + YAHOO.widget.AutoComplete._nIndex + " " + inputEl.id:                    "instance" + YAHOO.widget.AutoComplete._nIndex;                this._oTextbox = inputEl;            }        }        else {            return;        }        // Validate container element        if(YAHOO.util.Dom.inDocument(containerEl)) {            if(typeof containerEl == "string") {                    this._oContainer = document.getElementById(containerEl);            }            else {                this._oContainer = containerEl;            }            if(this._oContainer.style.display == "none") {            }        }        else {            return;        }        // Set any config params passed in to override defaults        if (typeof oConfigs == "object") {            for(var sConfig in oConfigs) {                if (sConfig) {                    this[sConfig] = oConfigs[sConfig];                }            }        }        // Initialization sequence        this._initContainer();        this._initProps();        this._initList();        this._initContainerHelpers();        // Set up events        var oSelf = this;        var oTextbox = this._oTextbox;        // Events are actually for the content module within the container        var oContent = this._oContainer._oContent;        // Dom events        YAHOO.util.Event.addListener(oTextbox,"keyup",oSelf._onTextboxKeyUp,oSelf);        YAHOO.util.Event.addListener(oTextbox,"keydown",oSelf._onTextboxKeyDown,oSelf);        YAHOO.util.Event.addListener(oTextbox,"keypress",oSelf._onTextboxKeyPress,oSelf);        YAHOO.util.Event.addListener(oTextbox,"focus",oSelf._onTextboxFocus,oSelf);        YAHOO.util.Event.addListener(oTextbox,"blur",oSelf._onTextboxBlur,oSelf);        YAHOO.util.Event.addListener(oContent,"mouseover",oSelf._onContainerMouseover,oSelf);        YAHOO.util.Event.addListener(oContent,"mouseout",oSelf._onContainerMouseout,oSelf);        YAHOO.util.Event.addListener(oContent,"scroll",oSelf._onContainerScroll,oSelf);        YAHOO.util.Event.addListener(oContent,"resize",oSelf._onContainerResize,oSelf);        if(oTextbox.form) {            YAHOO.util.Event.addListener(oTextbox.form,"submit",oSelf._onFormSubmit,oSelf);        }        // Custom events        this.textboxFocusEvent = new YAHOO.util.CustomEvent("textboxFocus", this);        this.textboxKeyEvent = new YAHOO.util.CustomEvent("textboxKey", this);        this.dataRequestEvent = new YAHOO.util.CustomEvent("dataRequest", this);        this.dataReturnEvent = new YAHOO.util.CustomEvent("dataReturn", this);        this.dataErrorEvent = new YAHOO.util.CustomEvent("dataError", this);        this.containerExpandEvent = new YAHOO.util.CustomEvent("containerExpand", this);        this.typeAheadEvent = new YAHOO.util.CustomEvent("typeAhead", this);        this.itemMouseOverEvent = new YAHOO.util.CustomEvent("itemMouseOver", this);        this.itemMouseOutEvent = new YAHOO.util.CustomEvent("itemMouseOut", this);        this.itemArrowToEvent = new YAHOO.util.CustomEvent("itemArrowTo", this);        this.itemArrowFromEvent = new YAHOO.util.CustomEvent("itemArrowFrom", this);        this.itemSelectEvent = new YAHOO.util.CustomEvent("itemSelect", this);        this.unmatchedItemSelectEvent = new YAHOO.util.CustomEvent("unmatchedItemSelect", this);        this.selectionEnforceEvent = new YAHOO.util.CustomEvent("selectionEnforce", this);        this.containerCollapseEvent = new YAHOO.util.CustomEvent("containerCollapse", this);        this.textboxBlurEvent = new YAHOO.util.CustomEvent("textboxBlur", this);                // Finish up        oTextbox.setAttribute("autocomplete","off");        YAHOO.widget.AutoComplete._nIndex++;    }    // Required arguments were not found    else {    }};/*************************************************************************** * Public member variables ***************************************************************************//** * The data source object that encapsulates the data used for auto completion. * This object should be an inherited object from YAHOO.widget.DataSource. * * @type object */YAHOO.widget.AutoComplete.prototype.dataSource = null;/** * Number of characters that must be entered before querying for results. * Default: 1. * * @type number */YAHOO.widget.AutoComplete.prototype.minQueryLength = 1;/** * Maximum number of results to display in auto complete container. Default: 10. * * @type number */YAHOO.widget.AutoComplete.prototype.maxResultsDisplayed = 10;/** * Number of seconds to delay before submitting a query request.  If a query * request is received before a previous one has completed its delay, the * previous request is cancelled and the new request is set to the delay. * Default: 0.5. * * @type number */YAHOO.widget.AutoComplete.prototype.queryDelay = 0.5;/** * Class name of a highlighted item within the auto complete container. * Default: "yui-ac-highlight". * * @type string */YAHOO.widget.AutoComplete.prototype.highlightClassName = "yui-ac-highlight";/** * Class name of a pre-highlighted item within the auto complete container. * Default: null. * * @type string */YAHOO.widget.AutoComplete.prototype.prehighlightClassName = null;/** * Query delimiter. A single character separator for multiple delimited * selections. Multiple delimiter characteres may be defined as an array of * strings. A null value or empty string indicates that query results cannot * be delimited. This feature is not recommended if you need forceSelection to * be true. Default: null. * * @type string or array */YAHOO.widget.AutoComplete.prototype.delimChar = null;/** * Whether or not the first item in the auto complete container should be * automatically highlighted on expand. Default: true. * * @type boolean */YAHOO.widget.AutoComplete.prototype.autoHighlight = true;/** * Whether or not the auto complete input field should be automatically updated * with the first query result as the user types, auto-selecting the substring * that the user has not typed. Default: false. * * @type boolean */YAHOO.widget.AutoComplete.prototype.typeAhead = false;/** * Whether or not to animate the expansion/collapse of the auto complete * container in the horizontal direction. Default: false. * * @type boolean */YAHOO.widget.AutoComplete.prototype.animHoriz = false;/** * Whether or not to animate the expansion/collapse of the auto complete * container in the vertical direction. Default: true. * * @type boolean */YAHOO.widget.AutoComplete.prototype.animVert = true;/** * Speed of container expand/collapse animation, in seconds. Default: 0.3. * * @type number */YAHOO.widget.AutoComplete.prototype.animSpeed = 0.3;/** * Whether or not to force the user's selection to match one of the query * results. Enabling this feature essentially transforms the auto complete form * input field into a &lt;select&gt; field. This feature is not recommended * with delimiter character(s) defined. Default: false. * * @type boolean */YAHOO.widget.AutoComplete.prototype.forceSelection = false;/** * Whether or not to allow browsers to cache user-typed input in the input * field. Disabling this feature will prevent the widget from setting the * autocomplete="off" on the auto complete input field. When autocomplete="off" * and users click the back button after form submission, user-typed input can * be prefilled by the browser from its cache. This caching of user input may * not be desired for sensitive data, such as credit card numbers, in which * case, implementers should consider setting allowBrowserAutocomplete to false. * Default: true. * * @type boolean */YAHOO.widget.AutoComplete.prototype.allowBrowserAutocomplete = true;/** * Whether or not the auto complete container should always be displayed. * Enabling this feature prevents the toggling of the container to a collapsed * state. Default: false. * * @type boolean */YAHOO.widget.AutoComplete.prototype.alwaysShowContainer = false;/** * Whether or not to use an iFrame to layer over Windows form elements in * IE. Set to true only when the auto complete container will be on top of a * &lt;select&gt; field in IE and thus exposed to the IE z-index bug (i.e., * 5.5 < IE < 7). Default:false. * * @type boolean */YAHOO.widget.AutoComplete.prototype.useIFrame = false;/** * Whether or not the auto complete container should have a shadow. Default:false. * * @type boolean */YAHOO.widget.AutoComplete.prototype.useShadow = false;/*************************************************************************** * Public methods ***************************************************************************/ /** * Public accessor to the unique name of the auto complete instance. * * @return {string} Unique name of the auto complete instance */YAHOO.widget.AutoComplete.prototype.toString = function() {    return "AutoComplete " + this._sName;};/** * Public accessor to the internal array of DOM &lt;li&gt; elements that * display query results within the auto complete container. * * @return {array} Array of &lt;li&gt; elements within the auto complete *                 container */YAHOO.widget.AutoComplete.prototype.getListItems = function() {    return this._aListItems;};/** * Public accessor to the data held in an &lt;li&gt; element of the * auto complete container. * * @return {object or array} Object or array of result data or null */YAHOO.widget.AutoComplete.prototype.getListItemData = function(oListItem) {    if(oListItem._oResultData) {        return oListItem._oResultData;    }    else {        return false;    }};/** * Sets HTML markup for the auto complete container header. This markup will be * inserted within a &lt;div&gt; tag with a class of "ac_hd". * * @param {string} sHeader HTML markup for container header */YAHOO.widget.AutoComplete.prototype.setHeader = function(sHeader) {    if(sHeader) {        if(this._oContainer._oContent._oHeader) {            this._oContainer._oContent._oHeader.innerHTML = sHeader;            this._oContainer._oContent._oHeader.style.display = "block";        }    }    else {        this._oContainer._oContent._oHeader.innerHTML = "";        this._oContainer._oContent._oHeader.style.display = "none";    }};/** * Sets HTML markup for the auto complete container footer. This markup will be * inserted within a &lt;div&gt; tag with a class of "ac_ft". * * @param {string} sFooter HTML markup for container footer */YAHOO.widget.AutoComplete.prototype.setFooter = function(sFooter) {    if(sFooter) {        if(this._oContainer._oContent._oFooter) {            this._oContainer._oContent._oFooter.innerHTML = sFooter;            this._oContainer._oContent._oFooter.style.display = "block";        }    }    else {        this._oContainer._oContent._oFooter.innerHTML = "";        this._oContainer._oContent._oFooter.style.display = "none";    }};/** * Sets HTML markup for the auto complete container body. This markup will be * inserted within a &lt;div&gt; tag with a class of "ac_bd". * * @param {string} sHeader HTML markup for container body */YAHOO.widget.AutoComplete.prototype.setBody = function(sBody) {    if(sBody) {        if(this._oContainer._oContent._oBody) {            this._oContainer._oContent._oBody.innerHTML = sBody;            this._oContainer._oContent._oBody.style.display = "block";            this._oContainer._oContent.style.display = "block";        }    }    else {        this._oContainer._oContent._oBody.innerHTML = "";        this._oContainer._oContent.style.display = "none";    }    this._maxResultsDisplayed = 0;};/** * Overridable method that converts a result item object into HTML markup * for display. Return data values are accessible via the oResultItem object, * and the key return value will always be oResultItem[0]. Markup will be * displayed within &lt;li&gt; element tags in the container. * * @param {object} oResultItem Result item object representing one query result * @param {string} sQuery The current query string * @return {string} HTML markup of formatted result data */YAHOO.widget.AutoComplete.prototype.formatResult = function(oResultItem, sQuery) {    var sResult = oResultItem[0];    if(sResult) {        return sResult;    }    else {        return "";    }};/** * Makes query request to the data source. * * @param {string} sQuery Query string. */YAHOO.widget.AutoComplete.prototype.sendQuery = function(sQuery) {    if(sQuery) {        this._sendQuery(sQuery);    }    else {        return;    }};/*************************************************************************** * Events ***************************************************************************//** * Fired when the auto complete text input box receives focus. Subscribers * receive the following array:<br> *     -  args[0] The auto complete object instance */YAHOO.widget.AutoComplete.prototype.textboxFocusEvent = null;/** * Fired when the auto complete text input box receives key input. Subscribers * receive the following array:<br> *     - args[0] The auto complete object instance *     - args[1] The keycode number */YAHOO.widget.AutoComplete.prototype.textboxKeyEvent = null;/** * Fired when the auto complete instance makes a query to the data source. * Subscribers receive the following array:<br> *     - args[0] The auto complete object instance *     - args[1] The query string */YAHOO.widget.AutoComplete.prototype.dataRequestEvent = null;/** * Fired when the auto complete instance receives query results from the data * source. Subscribers receive the following array:<br> *     - args[0] The auto complete object instance *     - args[1] The query string *     - args[2] Results array */YAHOO.widget.AutoComplete.prototype.dataReturnEvent = null;/** * Fired when the auto complete instance does not receive query results from the * data source due to an error. Subscribers receive the following array:<br> *     - args[0] The auto complete object instance *     - args[1] The query string */YAHOO.widget.AutoComplete.prototype.dataErrorEvent = null;/** * Fired when the auto complete container is expanded. If alwaysShowContainer is * enabled, then containerExpandEvent will be fired when the container is * populated with results. Subscribers receive the following array:<br> *     - args[0] The auto complete object instance */YAHOO.widget.AutoComplete.prototype.containerExpandEvent = null;/** * Fired when the auto complete textbox has been prefilled by the type-ahead * feature. Subscribers receive the following array:<br> *     - args[0] The auto complete object instance *     - args[1] The query string *     - args[2] The prefill string */YAHOO.widget.AutoComplete.prototype.typeAheadEvent = null;/** * Fired when result item has been moused over. Subscribers receive the following * array:<br> *     - args[0] The auto complete object instance *     - args[1] The &lt;li&gt element item moused to */YAHOO.widget.AutoComplete.prototype.itemMouseOverEvent = null;/** * Fired when result item has been moused out. Subscribers receive the * following array:<br> *     - args[0] The auto complete object instance *     - args[1] The &lt;li&gt; element item moused from */YAHOO.widget.AutoComplete.prototype.itemMouseOutEvent = null;/** * Fired when result item has been arrowed to. Subscribers receive the following * array:<br> *     - args[0] The auto complete object instance *     - args[1] The &lt;li&gt; element item arrowed to */YAHOO.widget.AutoComplete.prototype.itemArrowToEvent = null;/** * Fired when result item has been arrowed away from. Subscribers receive the * following array:<br> *     - args[0] The auto complete object instance *     - args[1] The &lt;li&gt; element item arrowed from */YAHOO.widget.AutoComplete.prototype.itemArrowFromEvent = null;/** * Fired when an item is selected via mouse click, ENTER key, or TAB key. * Subscribers receive the following array:<br> *     - args[0] The auto complete object instance *     - args[1] The selected &lt;li&gt; element item *     - args[2] The data returned for the item, either as an object, or mapped from the schema into an array */YAHOO.widget.AutoComplete.prototype.itemSelectEvent = null;/** * Fired when an user selection does not match any of the displayed result items. * Note that this event may not behave as expected when delimiter characters * have been defined. Subscribers receive the following array:<br> *     - args[0] The auto complete object instance *     - args[1] The user selection */YAHOO.widget.AutoComplete.prototype.unmatchedItemSelectEvent = null;/** * Fired if forceSelection is enabled and the user's input has been cleared * because it did not match one of the returned query results. Subscribers * receive the following array:<br> *     - args[0] The auto complete object instance */YAHOO.widget.AutoComplete.prototype.selectionEnforceEvent = null;/** * Fired when the auto complete container is collapsed. If alwaysShowContainer is * enabled, then containerCollapseEvent will be fired when the container is * cleared of results. Subscribers receive the following array:<br> *     - args[0] The auto complete object instance */YAHOO.widget.AutoComplete.prototype.containerCollapseEvent = null;/** * Fired when the auto complete text input box loses focus. Subscribers receive * an array of the following array:<br> *     - args[0] The auto complete object instance */YAHOO.widget.AutoComplete.prototype.textboxBlurEvent = null;/*************************************************************************** * Private member variables ***************************************************************************//** * Internal class variable to index multiple auto complete instances. * * @type number * @private */YAHOO.widget.AutoComplete._nIndex = 0;/** * Name of auto complete instance. * * @type string * @private */YAHOO.widget.AutoComplete.prototype._sName = null;/** * Text input box DOM element. * * @type object * @private */YAHOO.widget.AutoComplete.prototype._oTextbox = null;/** * Whether or not the textbox is currently in focus. If query results come back * but the user has already moved on, do not proceed with auto complete behavior. * * @type boolean * @private */YAHOO.widget.AutoComplete.prototype._bFocused = true;/** * Animation instance for container expand/collapse. * * @type boolean * @private */YAHOO.widget.AutoComplete.prototype._oAnim = null;/** * Container DOM element. * * @type object * @private */YAHOO.widget.AutoComplete.prototype._oContainer = null;/** * Whether or not the auto complete container is currently open. * * @type boolean * @private */YAHOO.widget.AutoComplete.prototype._bContainerOpen = false;/** * Whether or not the mouse is currently over the auto complete * container. This is necessary in order to prevent clicks on container items * from being text input box blur events. * * @type boolean * @private */YAHOO.widget.AutoComplete.prototype._bOverContainer = false;/** * Array of &lt;li&gt; elements references that contain query results within the * auto complete container. * * @type array * @private */YAHOO.widget.AutoComplete.prototype._aListItems = null;/** * Number of &lt;li&gt; elements currently displayed in auto complete container. * * @type number * @private */YAHOO.widget.AutoComplete.prototype._nDisplayedItems = 0;/** * Internal count of &lt;li&gt; elements displayed and hidden in auto complete container. * * @type number * @private */YAHOO.widget.AutoComplete.prototype._maxResultsDisplayed = 0;/** * Current query string * * @type string * @private */YAHOO.widget.AutoComplete.prototype._sCurQuery = null;/** * Past queries this session (for saving delimited queries). * * @type string * @private */YAHOO.widget.AutoComplete.prototype._sSavedQuery = null;/** * Pointer to the currently highlighted &lt;li&gt; element in the container. * * @type object * @private */YAHOO.widget.AutoComplete.prototype._oCurItem = null;/** * Whether or not an item has been selected since the container was populated * with results. Reset to false by _populateList, and set to true when item is * selected. * * @type boolean * @private */YAHOO.widget.AutoComplete.prototype._bItemSelected = false;/** * Key code of the last key pressed in textbox. * * @type number * @private */YAHOO.widget.AutoComplete.prototype._nKeyCode = null;/** * Delay timeout ID. * * @type number * @private */YAHOO.widget.AutoComplete.prototype._nDelayID = -1;/** * Src to iFrame used when useIFrame = true. Supports implementations over SSL * as well. * * @type string * @private */YAHOO.widget.AutoComplete.prototype._iFrameSrc = "javascript:false;";/*************************************************************************** * Private methods ***************************************************************************//** * Updates and validates latest public config properties. * * @private */YAHOO.widget.AutoComplete.prototype._initProps = function() {    // Correct any invalid values    var minQueryLength = this.minQueryLength;    if(isNaN(minQueryLength) || (minQueryLength < 1)) {        minQueryLength = 1;    }    var maxResultsDisplayed = this.maxResultsDisplayed;    if(isNaN(this.maxResultsDisplayed) || (this.maxResultsDisplayed < 1)) {        this.maxResultsDisplayed = 10;    }    var queryDelay = this.queryDelay;    if(isNaN(this.queryDelay) || (this.queryDelay < 0)) {        this.queryDelay = 0.5;    }    var aDelimChar = (this.delimChar) ? this.delimChar : null;    if(aDelimChar) {        if(typeof aDelimChar == "string") {            this.delimChar = [aDelimChar];        }        else if(aDelimChar.constructor != Array) {            this.delimChar = null;        }    }    var animSpeed = this.animSpeed;    if((this.animHoriz || this.animVert) && YAHOO.util.Anim) {        if(isNaN(animSpeed) || (animSpeed < 0)) {            animSpeed = 0.3;        }        if(!this._oAnim ) {            oAnim = new YAHOO.util.Anim(this._oContainer._oContent, {}, this.animSpeed);            this._oAnim = oAnim;        }        else {            this._oAnim.duration = animSpeed;        }    }    if(this.forceSelection && this.delimChar) {    }    if(this.alwaysShowContainer && (this.useShadow || this.useIFrame)) {    }    if(this.alwaysShowContainer) {        this._bContainerOpen = true;    }};/** * Initializes the auto complete container helpers if they are enabled and do * not exist * * @private */YAHOO.widget.AutoComplete.prototype._initContainerHelpers = function() {    if(this.useShadow && !this._oContainer._oShadow) {        var oShadow = document.createElement("div");        oShadow.className = "yui-ac-shadow";        this._oContainer._oShadow = this._oContainer.appendChild(oShadow);    }    if(this.useIFrame && !this._oContainer._oIFrame) {        var oIFrame = document.createElement("iframe");        oIFrame.src = this._iFrameSrc;        oIFrame.frameBorder = 0;        oIFrame.scrolling = "no";        oIFrame.style.position = "absolute";        oIFrame.style.width = "100%";        oIFrame.style.height = "100%";        this._oContainer._oIFrame = this._oContainer.appendChild(oIFrame);    }};/** * Initializes the auto complete container once at object creation * * @private */YAHOO.widget.AutoComplete.prototype._initContainer = function() {    if(!this._oContainer._oContent) {        // The oContent div helps size the iframe and shadow properly        var oContent = document.createElement("div");        oContent.className = "yui-ac-content";        oContent.style.display = "none";        this._oContainer._oContent = this._oContainer.appendChild(oContent);        var oHeader = document.createElement("div");        oHeader.className = "yui-ac-hd";        oHeader.style.display = "none";        this._oContainer._oContent._oHeader = this._oContainer._oContent.appendChild(oHeader);        var oBody = document.createElement("div");        oBody.className = "yui-ac-bd";        this._oContainer._oContent._oBody = this._oContainer._oContent.appendChild(oBody);        var oFooter = document.createElement("div");        oFooter.className = "yui-ac-ft";        oFooter.style.display = "none";        this._oContainer._oContent._oFooter = this._oContainer._oContent.appendChild(oFooter);    }    else {    }};/** * Clears out contents of container body and creates up to * YAHOO.widget.AutoComplete#maxResultsDisplayed &lt;li&gt; elements in an * &lt;ul&gt; element. * * @private */YAHOO.widget.AutoComplete.prototype._initList = function() {    this._aListItems = [];    while(this._oContainer._oContent._oBody.hasChildNodes()) {        var oldListItems = this.getListItems();        if(oldListItems) {            for(var oldi = oldListItems.length-1; oldi >= 0; i--) {                oldListItems[oldi] = null;            }        }        this._oContainer._oContent._oBody.innerHTML = "";    }    var oList = document.createElement("div"); // SO: Changed element from "ul" to "div"        oList = this._oContainer._oContent._oBody.appendChild(oList);    for(var i=0; i<this.maxResultsDisplayed; i++) {        var oItem = document.createElement("span"); // SO: Changed element from "li" to "span"        oItem = oList.appendChild(oItem);        this._aListItems[i] = oItem;        this._initListItem(oItem, i);    }    this._maxResultsDisplayed = this.maxResultsDisplayed;};/** * Initializes each &lt;li&gt; element in the container list. * * @param {object} oItem The &lt;li&gt; DOM element * @param {number} nItemIndex The index of the element * @private */YAHOO.widget.AutoComplete.prototype._initListItem = function(oItem, nItemIndex) {    var oSelf = this;    oItem.style.display = "none";    oItem._nItemIndex = nItemIndex;    oItem.mouseover = oItem.mouseout = oItem.onclick = null;    YAHOO.util.Event.addListener(oItem,"mouseover",oSelf._onItemMouseover,oSelf);    YAHOO.util.Event.addListener(oItem,"mouseout",oSelf._onItemMouseout,oSelf);    YAHOO.util.Event.addListener(oItem,"click",oSelf._onItemMouseclick,oSelf);};/** * Handles &lt;li&gt; element mouseover events in the container. * * @param {event} v The mouseover event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onItemMouseover = function(v,oSelf) {    if(oSelf.prehighlightClassName) {        oSelf._togglePrehighlight(this,"mouseover");    }    else {        oSelf._toggleHighlight(this,"to");    }    oSelf.itemMouseOverEvent.fire(oSelf, this);};/** * Handles &lt;li&gt; element mouseout events in the container. * * @param {event} v The mouseout event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onItemMouseout = function(v,oSelf) {    if(oSelf.prehighlightClassName) {        oSelf._togglePrehighlight(this,"mouseout");    }    else {        oSelf._toggleHighlight(this,"from");    }    oSelf.itemMouseOutEvent.fire(oSelf, this);};/** * Handles &lt;li&gt; element click events in the container. * * @param {event} v The click event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onItemMouseclick = function(v,oSelf) {    // In case item has not been moused over    oSelf._toggleHighlight(this,"to");    oSelf._selectItem(this);};/** * Handles container mouseover events. * * @param {event} v The mouseover event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onContainerMouseover = function(v,oSelf) {    oSelf._bOverContainer = true;};/** * Handles container mouseout events. * * @param {event} v The mouseout event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onContainerMouseout = function(v,oSelf) {    oSelf._bOverContainer = false;    // If container is still active    if(oSelf._oCurItem) {        oSelf._toggleHighlight(oSelf._oCurItem,"to");    }};/** * Handles container scroll events. * * @param {event} v The scroll event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onContainerScroll = function(v,oSelf) {    oSelf._oTextbox.focus();};/** * Handles container resize events. * * @param {event} v The resize event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onContainerResize = function(v,oSelf) {    oSelf._toggleContainerHelpers(oSelf._bContainerOpen);};/** * Handles textbox keydown events of functional keys, mainly for UI behavior. * * @param {event} v The keydown event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onTextboxKeyDown = function(v,oSelf) {    var nKeyCode = v.keyCode;    switch (nKeyCode) {        case 9: // tab            if(oSelf.delimChar && (oSelf._nKeyCode != nKeyCode)) {                if(oSelf._bContainerOpen) {                    YAHOO.util.Event.stopEvent(v);                }            }            // select an item or clear out            if(oSelf._oCurItem) {                oSelf._selectItem(oSelf._oCurItem);            }            else {                oSelf._clearList();            }            break;        case 13: // enter            if(oSelf._nKeyCode != nKeyCode) {                if(oSelf._bContainerOpen) {                    YAHOO.util.Event.stopEvent(v);                }            }            if(oSelf._oCurItem) {                oSelf._selectItem(oSelf._oCurItem);            }            else {                oSelf._clearList();            }            break;        case 27: // esc            oSelf._clearList();            return;        case 39: // right            oSelf._jumpSelection();            break;        case 38: // up            YAHOO.util.Event.stopEvent(v);            oSelf._moveSelection(nKeyCode);            break;        case 40: // down            YAHOO.util.Event.stopEvent(v);            oSelf._moveSelection(nKeyCode);            break;        default:            break;    }};/** * Handles textbox keypress events, for stopEvent in Safari and FF 1.5/Mac * * @param {event} v The keyup event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onTextboxKeyPress = function(v,oSelf) {    var nKeyCode = v.keyCode;    switch (nKeyCode) {    case 9: // tab    case 13: // enter        if((oSelf._nKeyCode != nKeyCode)) {                YAHOO.util.Event.stopEvent(v);        }        break;    case 38: // up    case 40: // down        YAHOO.util.Event.stopEvent(v);        break;    default:        break;    }};/** * Handles textbox keyup events that trigger queries. * * @param {event} v The keyup event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onTextboxKeyUp = function(v,oSelf) {    // Check to see if any of the public properties have been updated    oSelf._initProps();    var nKeyCode = v.keyCode;    oSelf._nKeyCode = nKeyCode;    var sChar = String.fromCharCode(nKeyCode);    var sText = this.value; //string in textbox    // Filter out chars that don't trigger queries    if (oSelf._isIgnoreKey(nKeyCode) || (sText.toLowerCase() == oSelf._sCurQuery)) {        return;    }    else {        oSelf.textboxKeyEvent.fire(oSelf, nKeyCode);    }    // Set timeout on the request    if (oSelf.queryDelay > 0) {        var nDelayID =            setTimeout(function(){oSelf._sendQuery(sText);},(oSelf.queryDelay * 1000));        if (oSelf._nDelayID != -1) {            clearTimeout(oSelf._nDelayID);        }        oSelf._nDelayID = nDelayID;    }    else {        // No delay so send request immediately        oSelf._sendQuery(sText);    }};/** * Whether or not key is functional or should be ignored. Note that the right * arrow key is NOT an ignored key since it triggers queries for certain intl * charsets. * * @param {number} nKeycode Code of key pressed * @return {boolean} Whether or not to be ignore key * @private */YAHOO.widget.AutoComplete.prototype._isIgnoreKey = function(nKeyCode) {    if ((nKeyCode == 9) || (nKeyCode == 13)  || // tab, enter            (nKeyCode == 16) || (nKeyCode == 17) || // shift, ctl            (nKeyCode >= 18 && nKeyCode <= 20) || // alt,pause/break,caps lock            (nKeyCode == 27) || // esc            (nKeyCode >= 33 && nKeyCode <= 35) || // page up,page down,end            (nKeyCode >= 36 && nKeyCode <= 38) || // home,left,up            (nKeyCode == 40) || // down            (nKeyCode >= 44 && nKeyCode <= 45)) { // print screen,insert        return true;    }    return false;};/** * Handles text input box receiving focus. * * @param {event} v The focus event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onTextboxFocus = function (v,oSelf) {    oSelf._oTextbox.setAttribute("autocomplete","off");    oSelf._bFocused = true;    oSelf.textboxFocusEvent.fire(oSelf);};/** * Handles text input box losing focus. * * @param {event} v The focus event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onTextboxBlur = function (v,oSelf) {    // Don't treat as a blur if it was a selection via mouse click    if(!oSelf._bOverContainer || (oSelf._nKeyCode == 9)) {        // Current query needs to be validated        if(!oSelf._bItemSelected) {            if(!oSelf._bContainerOpen || (oSelf._bContainerOpen && !oSelf._textMatchesOption())) {                if(oSelf.forceSelection) {                    oSelf._clearSelection();                }                else {                    oSelf.unmatchedItemSelectEvent.fire(oSelf, oSelf._sCurQuery);                }            }        }        if(oSelf._bContainerOpen) {            oSelf._clearList();        }        oSelf._bFocused = false;        oSelf.textboxBlurEvent.fire(oSelf);    }};/** * Handles form submission event. * * @param {event} v The submit event * @param {object} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._onFormSubmit = function(v,oSelf) {    if(oSelf.allowBrowserAutocomplete) {        oSelf._oTextbox.setAttribute("autocomplete","on");    }    else {        oSelf._oTextbox.setAttribute("autocomplete","off");    }};/** * Makes query request to the data source. * * @param {string} sQuery Query string. * @private */YAHOO.widget.AutoComplete.prototype._sendQuery = function(sQuery) {    // Delimiter has been enabled    var aDelimChar = (this.delimChar) ? this.delimChar : null;    if(aDelimChar) {        // Loop through all possible delimiters and find the latest one        // A " " may be a false positive if they are defined as delimiters AND        // are used to separate delimited queries        var nDelimIndex = -1;        for(var i = aDelimChar.length-1; i >= 0; i--) {            var nNewIndex = sQuery.lastIndexOf(aDelimChar[i]);            if(nNewIndex > nDelimIndex) {                nDelimIndex = nNewIndex;            }        }        // If we think the last delimiter is a space (" "), make sure it is NOT        // a false positive by also checking the char directly before it        if(aDelimChar[i] == " ") {            for (var j = aDelimChar.length-1; j >= 0; j--) {                if(sQuery[nDelimIndex - 1] == aDelimChar[j]) {                    nDelimIndex--;                    break;                }            }        }        // A delimiter has been found so extract the latest query        if (nDelimIndex > -1) {            var nQueryStart = nDelimIndex + 1;            // Trim any white space from the beginning...            while(sQuery.charAt(nQueryStart) == " ") {                nQueryStart += 1;            }            // ...and save the rest of the string for later            this._sSavedQuery = sQuery.substring(0,nQueryStart);            // Here is the query itself            sQuery = sQuery.substr(nQueryStart);        }        else if(sQuery.indexOf(this._sSavedQuery) < 0){            this._sSavedQuery = null;        }    }    // Don't search queries that are too short    if (sQuery.length < this.minQueryLength) {        if (this._nDelayID != -1) {            clearTimeout(this._nDelayID);        }        this._clearList();        return;    }    sQuery = encodeURIComponent(sQuery);    this._nDelayID = -1;    // Reset timeout ID because request has been made    this.dataRequestEvent.fire(this, sQuery);    this.dataSource.getResults(this._populateList, sQuery, this);};/** * Hides all visuals related to the array of &lt;li&gt; elements in the container. * * @private */YAHOO.widget.AutoComplete.prototype._clearList = function() {    this._oContainer._oContent.scrollTop = 0;    var aItems = this._aListItems;    if(aItems && (aItems.length > 0)) {        for(var i = aItems.length-1; i >= 0 ; i--) {            aItems[i].style.display = "none";        }    }    if (this._oCurItem) {        this._toggleHighlight(this._oCurItem,"from");    }    this._oCurItem = null;    this._nDisplayedItems = 0;    this._sCurQuery = null;    this._toggleContainer(false);};/** * Populates the array of &lt;li&gt; elements in the container with query * results. This method is passed to YAHOO.widget.DataSource#getResults as a * callback function so results from the datasource are returned to the * auto complete instance. * * @param {string} sQuery The query string * @param {object} aResults An array of query result objects from the data source * @param {string} oSelf The auto complete instance * @private */YAHOO.widget.AutoComplete.prototype._populateList = function(sQuery, aResults, oSelf) {    if(aResults === null) {        oSelf.dataErrorEvent.fire(oSelf, sQuery);    }    if (!oSelf._bFocused || !aResults) {        return;    }    var isOpera = (navigator.userAgent.toLowerCase().indexOf("opera") != -1);    var contentStyle = oSelf._oContainer._oContent.style;    contentStyle.width = (!isOpera) ? null : "";    contentStyle.height = (!isOpera) ? null : "";    var sCurQuery = decodeURIComponent(sQuery);    oSelf._sCurQuery = sCurQuery;    oSelf._bItemSelected = false;    if(oSelf._maxResultsDisplayed != oSelf.maxResultsDisplayed) {        oSelf._initList();    }    var nItems = Math.min(aResults.length,oSelf.maxResultsDisplayed);    oSelf._nDisplayedItems = nItems;    if (nItems > 0) {        oSelf._initContainerHelpers();        var aItems = oSelf._aListItems;        // Fill items with data        for(var i = nItems-1; i >= 0; i--) {            var oItemi = aItems[i];            var oResultItemi = aResults[i];            oItemi.innerHTML = oSelf.formatResult(oResultItemi, sCurQuery);            oItemi.style.display = "list-item";            oItemi._sResultKey = oResultItemi[0];            oItemi._oResultData = oResultItemi;        }        // Empty out remaining items if any        for(var j = aItems.length-1; j >= nItems ; j--) {            var oItemj = aItems[j];            oItemj.innerHTML = null;            oItemj.style.display = "none";            oItemj._sResultKey = null;            oItemj._oResultData = null;        }        if(oSelf.autoHighlight) {            // Go to the first item            var oFirstItem = aItems[0];            oSelf._toggleHighlight(oFirstItem,"to");            oSelf.itemArrowToEvent.fire(oSelf, oFirstItem);            oSelf._typeAhead(oFirstItem,sQuery);        }        else {            oSelf._oCurItem = null;        }        // Expand the container        oSelf._toggleContainer(true);    }    else {        oSelf._clearList();    }    oSelf.dataReturnEvent.fire(oSelf, sQuery, aResults);};/** * When YAHOO.widget.AutoComplete#bForceSelection is true and the user attempts * leave the text input box without selecting an item from the query results, * the user selection is cleared. * * @private */YAHOO.widget.AutoComplete.prototype._clearSelection = function() {    var sValue = this._oTextbox.value;    var sChar = (this.delimChar) ? this.delimChar[0] : null;    var nIndex = (sChar) ? sValue.lastIndexOf(sChar, sValue.length-2) : -1;    if(nIndex > -1) {        this._oTextbox.value = sValue.substring(0,nIndex);    }    else {         this._oTextbox.value = "";    }    this._sSavedQuery = this._oTextbox.value;    // Fire custom event    this.selectionEnforceEvent.fire(this);};/** * Whether or not user-typed value in the text input box matches any of the * query results. * * @private */YAHOO.widget.AutoComplete.prototype._textMatchesOption = function() {    var foundMatch = false;    for(var i = this._nDisplayedItems-1; i >= 0 ; i--) {        var oItem = this._aListItems[i];        var sMatch = oItem._sResultKey.toLowerCase();        if (sMatch == this._sCurQuery.toLowerCase()) {            foundMatch = true;            break;        }    }    return(foundMatch);};/** * Updates in the text input box with the first query result as the user types, * selecting the substring that the user has not typed. * * @param {object} oItem The &lt;li&gt; element item whose data populates the input field * @param {string} sQuery Query string * @private */YAHOO.widget.AutoComplete.prototype._typeAhead = function(oItem, sQuery) {    // Don't update if turned off    if (!this.typeAhead) {        return;    }    var oTextbox = this._oTextbox;    var sValue = this._oTextbox.value; // any saved queries plus what user has typed    // Don't update with type-ahead if text selection is not supported    if(!oTextbox.setSelectionRange && !oTextbox.createTextRange) {        return;    }    // Select the portion of text that the user has not typed    var nStart = sValue.length;    this._updateValue(oItem);    var nEnd = oTextbox.value.length;    this._selectText(oTextbox,nStart,nEnd);    var sPrefill = oTextbox.value.substr(nStart,nEnd);    this.typeAheadEvent.fire(this,sQuery,sPrefill);};/** * Selects text in a text input box. * * @param {object} oTextbox Text input box element in which to select text * @param {number} nStart Starting index of text string to select * @param {number} nEnd Ending index of text selection * @private */YAHOO.widget.AutoComplete.prototype._selectText = function(oTextbox, nStart, nEnd) {    if (oTextbox.setSelectionRange) { // For Mozilla        oTextbox.setSelectionRange(nStart,nEnd);    }    else if (oTextbox.createTextRange) { // For IE        var oTextRange = oTextbox.createTextRange();        oTextRange.moveStart("character", nStart);        oTextRange.moveEnd("character", nEnd-oTextbox.value.length);        oTextRange.select();    }    else {        oTextbox.select();    }};/** * Syncs auto complete container with its helpers. * * @param {boolean} bShow True if container is expanded, false if collapsed * @private */YAHOO.widget.AutoComplete.prototype._toggleContainerHelpers = function(bShow) {    var bFireEvent = false;    var width = this._oContainer._oContent.offsetWidth + "px";    var height = this._oContainer._oContent.offsetHeight + "px";    if(this.useIFrame && this._oContainer._oIFrame) {        bFireEvent = true;        if(this.alwaysShowContainer || bShow) {            this._oContainer._oIFrame.style.width = width;            this._oContainer._oIFrame.style.height = height;        }        else {            this._oContainer._oIFrame.style.width = 0;            this._oContainer._oIFrame.style.height = 0;        }    }    if(this.useShadow && this._oContainer._oShadow) {        bFireEvent = true;        if(this.alwaysShowContainer || bShow) {            this._oContainer._oShadow.style.width = width;            this._oContainer._oShadow.style.height = height;        }        else {           this._oContainer._oShadow.style.width = 0;            this._oContainer._oShadow.style.height = 0;        }    }};/** * Animates expansion or collapse of the container. * * @param {boolean} bShow True if container should be expanded, false if *                        container should be collapsed * @private */YAHOO.widget.AutoComplete.prototype._toggleContainer = function(bShow) {    // Implementer has container always open so don't mess with it    if(this.alwaysShowContainer) {        // Fire these events to give implementers a hook into the container        // being populated and being emptied        if(bShow) {            this.containerExpandEvent.fire(this);        }        else {            this.containerCollapseEvent.fire(this);        }        this._bContainerOpen = bShow;        return;    }    var oContainer = this._oContainer;    // Container is already closed    if (!bShow && !this._bContainerOpen) {        oContainer._oContent.style.display = "none";        return;    }    // If animation is enabled...    var oAnim = this._oAnim;    if (oAnim && oAnim.getEl() && (this.animHoriz || this.animVert)) {        // If helpers need to be collapsed, do it right away...        // but if helpers need to be expanded, wait until after the container expands        if(!bShow) {            this._toggleContainerHelpers(bShow);        }        if(oAnim.isAnimated()) {            oAnim.stop();        }        // Clone container to grab current size offscreen        var oClone = oContainer._oContent.cloneNode(true);        oContainer.appendChild(oClone);        oClone.style.top = "-9000px";        oClone.style.display = "block";        // Current size of the container is the EXPANDED size        var wExp = oClone.offsetWidth;        var hExp = oClone.offsetHeight;        // Calculate COLLAPSED sizes based on horiz and vert anim        var wColl = (this.animHoriz) ? 0 : wExp;        var hColl = (this.animVert) ? 0 : hExp;        // Set animation sizes        oAnim.attributes = (bShow) ?            {width: { to: wExp }, height: { to: hExp }} :            {width: { to: wColl}, height: { to: hColl }};        // If opening anew, set to a collapsed size...        if(bShow && !this._bContainerOpen) {            oContainer._oContent.style.width = wColl+"px";            oContainer._oContent.style.height = hColl+"px";        }        // Else, set it to its last known size.        else {            oContainer._oContent.style.width = wExp+"px";            oContainer._oContent.style.height = hExp+"px";        }        oContainer.removeChild(oClone);        oClone = null;    	var oSelf = this;    	var onAnimComplete = function() {            // Finish the collapse    		oAnim.onComplete.unsubscribeAll();            if(bShow) {                oSelf.containerExpandEvent.fire(oSelf);            }            else {                oContainer._oContent.style.display = "none";                oSelf.containerCollapseEvent.fire(oSelf);            }            oSelf._toggleContainerHelpers(bShow);     	};        // Display container and animate it        oContainer._oContent.style.display = "block";        oAnim.onComplete.subscribe(onAnimComplete);        oAnim.animate();        this._bContainerOpen = bShow;    }    // Else don't animate, just show or hide    else {        if(bShow) {            oContainer._oContent.style.display = "block";            this.containerExpandEvent.fire(this);        }        else {            oContainer._oContent.style.display = "none";            this.containerCollapseEvent.fire(this);        }        this._toggleContainerHelpers(bShow);        this._bContainerOpen = bShow;   }};/** * Toggles the highlight on or off for an item in the container, and also cleans * up highlighting of any previous item. * * @param {object} oNewItem New The &lt;li&gt; element item to receive highlight *                              behavior * @param {string} sType "mouseover" will toggle highlight on, and "mouseout" *                       will toggle highlight off. * @private */YAHOO.widget.AutoComplete.prototype._toggleHighlight = function(oNewItem, sType) {    var sHighlight = this.highlightClassName;    if(this._oCurItem) {        // Remove highlight from old item        YAHOO.util.Dom.removeClass(this._oCurItem, sHighlight);    }    if((sType == "to") && sHighlight) {        // Apply highlight to new item        YAHOO.util.Dom.addClass(oNewItem, sHighlight);        this._oCurItem = oNewItem;    }};/** * Toggles the pre-highlight on or off for an item in the container. * * @param {object} oNewItem New The &lt;li&gt; element item to receive highlight *                              behavior * @param {string} sType "mouseover" will toggle highlight on, and "mouseout" *                       will toggle highlight off. * @private */YAHOO.widget.AutoComplete.prototype._togglePrehighlight = function(oNewItem, sType) {    if(oNewItem == this._oCurItem) {        return;    }    var sPrehighlight = this.prehighlightClassName;    if((sType == "mouseover") && sPrehighlight) {        // Apply prehighlight to new item        YAHOO.util.Dom.addClass(oNewItem, sPrehighlight);    }    else {        // Remove prehighlight from old item        YAHOO.util.Dom.removeClass(oNewItem, sPrehighlight);    }};/** * Updates the text input box value with selected query result. If a delimiter * has been defined, then the value gets appended with the delimiter. * * @param {object} oItem The &lt;li&gt; element item with which to update the value * @private */YAHOO.widget.AutoComplete.prototype._updateValue = function(oItem) {    var oTextbox = this._oTextbox;    var sDelimChar = (this.delimChar) ? this.delimChar[0] : null;    var sSavedQuery = this._sSavedQuery;    var sResultKey = oItem._sResultKey;    oTextbox.focus();    // First clear text field    oTextbox.value = "";    // Grab data to put into text field    if(sDelimChar) {        if(sSavedQuery) {            oTextbox.value = sSavedQuery;        }        oTextbox.value += sResultKey + sDelimChar;        if(sDelimChar != " ") {            oTextbox.value += " ";        }    }    else { oTextbox.value = sResultKey; }    // scroll to bottom of textarea if necessary    if(oTextbox.type == "textarea") {        oTextbox.scrollTop = oTextbox.scrollHeight;    }    // move cursor to end    var end = oTextbox.value.length;    this._selectText(oTextbox,end,end);    this._oCurItem = oItem;};/** * Selects a result item from the container * * @param {object} oItem The selected &lt;li&gt; element item * @private */YAHOO.widget.AutoComplete.prototype._selectItem = function(oItem) {    this._bItemSelected = true;    this._updateValue(oItem);    this.itemSelectEvent.fire(this, oItem, oItem._oResultData);    this._clearList();};/** * For values updated by type-ahead, the right arrow key jumps to the end * of the textbox, otherwise the container is closed. * * @private */YAHOO.widget.AutoComplete.prototype._jumpSelection = function() {    if(!this.typeAhead) {        return;    }    else {        this._clearList();    }};/** * Triggered by up and down arrow keys, changes the current highlighted * &lt;li&gt; element item. Scrolls container if necessary. * * @param {number} nKeyCode Code of key pressed * @private */YAHOO.widget.AutoComplete.prototype._moveSelection = function(nKeyCode) {    if(this._bContainerOpen) {        // Determine current item's id number        var oCurItem = this._oCurItem;        var nCurItemIndex = -1;        if (oCurItem) {            nCurItemIndex = oCurItem._nItemIndex;        }        var nNewItemIndex = (nKeyCode == 40) ?                (nCurItemIndex + 1) : (nCurItemIndex - 1);        // Out of bounds        if (nNewItemIndex < -2 || nNewItemIndex >= this._nDisplayedItems) {            return;        }        if (oCurItem) {            // Unhighlight current item            this._toggleHighlight(oCurItem, "from");            this.itemArrowFromEvent.fire(this, oCurItem);        }        if (nNewItemIndex == -1) {           // Go back to query (remove type-ahead string)            if(this.delimChar && this._sSavedQuery) {                if (!this._textMatchesOption()) {                    this._oTextbox.value = this._sSavedQuery;                }                else {                    this._oTextbox.value = this._sSavedQuery + this._sCurQuery;                }            }            else {                this._oTextbox.value = this._sCurQuery;            }            this._oCurItem = null;            return;        }        if (nNewItemIndex == -2) {            // Close container            this._clearList();            return;        }        var oNewItem = this._aListItems[nNewItemIndex];        // Scroll the container if necessary        var oContent = this._oContainer._oContent;        var scrollOn = ((YAHOO.util.Dom.getStyle(oContent,"overflow") == "auto") ||            (YAHOO.util.Dom.getStyle(oContent,"overflowY") == "auto"));        if(scrollOn && (nNewItemIndex > -1) &&        (nNewItemIndex < this._nDisplayedItems)) {            // User is keying down            if(nKeyCode == 40) {                // Bottom of selected item is below scroll area...                if((oNewItem.offsetTop+oNewItem.offsetHeight) > (oContent.scrollTop + oContent.offsetHeight)) {                    // Set bottom of scroll area to bottom of selected item                    oContent.scrollTop = (oNewItem.offsetTop+oNewItem.offsetHeight) - oContent.offsetHeight;                }                // Bottom of selected item is above scroll area...                else if((oNewItem.offsetTop+oNewItem.offsetHeight) < oContent.scrollTop) {                    // Set top of selected item to top of scroll area                    oContent.scrollTop = oNewItem.offsetTop;                }            }            // User is keying up            else {                // Top of selected item is above scroll area                if(oNewItem.offsetTop < oContent.scrollTop) {                    // Set top of scroll area to top of selected item                    this._oContainer._oContent.scrollTop = oNewItem.offsetTop;                }                // Top of selected item is below scroll area                else if(oNewItem.offsetTop > (oContent.scrollTop + oContent.offsetHeight)) {                    // Set bottom of selected item to bottom of scroll area                    this._oContainer._oContent.scrollTop = (oNewItem.offsetTop+oNewItem.offsetHeight) - oContent.offsetHeight;                }            }        }        this._toggleHighlight(oNewItem, "to");        this.itemArrowToEvent.fire(this, oNewItem);        if(this.typeAhead) {            this._updateValue(oNewItem);        }    }};/****************************************************************************//****************************************************************************//****************************************************************************//** * Class providing encapsulation of a data source.  *   * @constructor * */YAHOO.widget.DataSource = function() {     /* abstract class */};/*************************************************************************** * Public constants ***************************************************************************//** * Error message for null data responses. * * @type constant * @final */YAHOO.widget.DataSource.prototype.ERROR_DATANULL = "Response data was null";/** * Error message for data responses with parsing errors. * * @type constant * @final */YAHOO.widget.DataSource.prototype.ERROR_DATAPARSE = "Response data could not be parsed";/*************************************************************************** * Public member variables ***************************************************************************//** * Max size of the local cache.  Set to 0 to turn off caching.  Caching is * useful to reduce the number of server connections.  Recommended only for data * sources that return comprehensive results for queries or when stale data is * not an issue. Default: 15. * * @type number */YAHOO.widget.DataSource.prototype.maxCacheEntries = 15;/** * Use this to equate cache matching with the type of matching done by your live * data source. If caching is on and queryMatchContains is true, the cache * returns results that "contain" the query string. By default, * queryMatchContains is set to false, meaning the cache only returns results * that "start with" the query string. Default: false. * * @type boolean */YAHOO.widget.DataSource.prototype.queryMatchContains = false;/** * Data source query subset matching. If caching is on and queryMatchSubset is * true, substrings of queries will return matching cached results. For * instance, if the first query is for "abc" susequent queries that start with * "abc", like "abcd", will be queried against the cache, and not the live data * source. Recommended only for data sources that return comprehensive results * for queries with very few characters. Default: false. * * @type boolean */YAHOO.widget.DataSource.prototype.queryMatchSubset = false;/** * Data source query case-sensitivity matching. If caching is on and * queryMatchCase is true, queries will only return results for case-sensitive * matches. Default: false. * * @type boolean */YAHOO.widget.DataSource.prototype.queryMatchCase = false;/*************************************************************************** * Public methods ***************************************************************************/ /** * Public accessor to the unique name of the data source instance. * * @return {string} Unique name of the data source instance */YAHOO.widget.DataSource.prototype.getName = function() {    return this._sName;}; /** * Public accessor to the unique name of the data source instance. * * @return {string} Unique name of the data source instance */YAHOO.widget.DataSource.prototype.toString = function() {    return "DataSource " + this._sName;};/** * Retrieves query results, first checking the local cache, then making the * query request to the live data source as defined by the function doQuery. * * @param {object} oCallbackFn Callback function defined by oParent object to *                             which to return results  * @param {string} sQuery Query string * @param {object} oParent The object instance that has requested data */YAHOO.widget.DataSource.prototype.getResults = function(oCallbackFn, sQuery, oParent) {        // First look in cache    var aResults = this._doQueryCache(oCallbackFn,sQuery,oParent);        // Not in cache, so get results from server    if(aResults.length === 0) {        this.queryEvent.fire(this, oParent, sQuery);        this.doQuery(oCallbackFn, sQuery, oParent);    }};/** * Abstract method implemented by subclasses to make a query to the live data * source. Must call the callback function with the response returned from the * query. Populates cache (if enabled). * * @param {object} oCallbackFn Callback function implemented by oParent to *                             which to return results  * @param {string} sQuery Query string * @param {object} oParent The object instance that has requested data */YAHOO.widget.DataSource.prototype.doQuery = function(oCallbackFn, sQuery, oParent) {    /* override this */ };/** * Flushes cache. */YAHOO.widget.DataSource.prototype.flushCache = function() {    if(this._aCache) {        this._aCache = [];    }    if(this._aCacheHelper) {        this._aCacheHelper = [];    }    this.cacheFlushEvent.fire(this);};/*************************************************************************** * Events ***************************************************************************//** * Fired when a query is made to the live data source. Subscribers receive the * following array:<br> *     - args[0] The data source instance *     - args[1] The requesting object *     - args[2] The query string */YAHOO.widget.DataSource.prototype.queryEvent = null;/** * Fired when a query is made to the local cache. Subscribers receive the * following array:<br> *     - args[0] The data source instance *     - args[1] The requesting object *     - args[2] The query string */YAHOO.widget.DataSource.prototype.cacheQueryEvent = null;/** * Fired when data is retrieved from the live data source. Subscribers receive * the following array:<br> *     - args[0] The data source instance *     - args[1] The requesting object *     - args[2] The query string *     - args[3] Array of result objects */YAHOO.widget.DataSource.prototype.getResultsEvent = null;    /** * Fired when data is retrieved from the local cache. Subscribers receive the * following array :<br> *     - args[0] The data source instance *     - args[1] The requesting object *     - args[2] The query string *     - args[3] Array of result objects */YAHOO.widget.DataSource.prototype.getCachedResultsEvent = null;/** * Fired when an error is encountered with the live data source. Subscribers * receive the following array:<br> *     - args[0] The data source instance *     - args[1] The requesting object *     - args[2] The query string *     - args[3] Error message string */YAHOO.widget.DataSource.prototype.dataErrorEvent = null;/** * Fired when the local cache is flushed. Subscribers receive the following * array :<br> *     - args[0] The data source instance */YAHOO.widget.DataSource.prototype.cacheFlushEvent = null;/*************************************************************************** * Private member variables ***************************************************************************//** * Internal class variable to index multiple data source instances. * * @type number * @private */YAHOO.widget.DataSource._nIndex = 0;/** * Name of data source instance. * * @type string * @private */YAHOO.widget.DataSource.prototype._sName = null;/** * Local cache of data result objects indexed chronologically. * * @type array * @private */YAHOO.widget.DataSource.prototype._aCache = null;/*************************************************************************** * Private methods ***************************************************************************//** * Initializes data source instance. *   * @private */YAHOO.widget.DataSource.prototype._init = function() {    // Validate and initialize public configs    var maxCacheEntries = this.maxCacheEntries;    if(isNaN(maxCacheEntries) || (maxCacheEntries < 0)) {        maxCacheEntries = 0;    }    // Initialize local cache    if(maxCacheEntries > 0 && !this._aCache) {        this._aCache = [];    }        this._sName = "instance" + YAHOO.widget.DataSource._nIndex;    YAHOO.widget.DataSource._nIndex++;        this.queryEvent = new YAHOO.util.CustomEvent("query", this);    this.cacheQueryEvent = new YAHOO.util.CustomEvent("cacheQuery", this);    this.getResultsEvent = new YAHOO.util.CustomEvent("getResults", this);    this.getCachedResultsEvent = new YAHOO.util.CustomEvent("getCachedResults", this);    this.dataErrorEvent = new YAHOO.util.CustomEvent("dataError", this);    this.cacheFlushEvent = new YAHOO.util.CustomEvent("cacheFlush", this);};/** * Adds a result object to the local cache, evicting the oldest element if the  * cache is full. Newer items will have higher indexes, the oldest item will have * index of 0.  * * @param {object} resultObj  Object literal of data results, including internal *                            properties and an array of result objects * @private */YAHOO.widget.DataSource.prototype._addCacheElem = function(resultObj) {    var aCache = this._aCache;    // Don't add if anything important is missing.    if(!aCache || !resultObj || !resultObj.query || !resultObj.results) {        return;    }        // If the cache is full, make room by removing from index=0    if(aCache.length >= this.maxCacheEntries) {        aCache.shift();    }            // Add to cache, at the end of the array    aCache.push(resultObj);};/** * Queries the local cache for results. If query has been cached, the callback * function is called with the results, and the cached is refreshed so that it * is now the newest element.   * * @param {object} oCallbackFn Callback function defined by oParent object to  *                             which to return results  * @param {string} sQuery Query string * @param {object} oParent The object instance that has requested data * @return {array} aResults Result object from local cache if found, otherwise  *                          null * @private  */YAHOO.widget.DataSource.prototype._doQueryCache = function(oCallbackFn, sQuery, oParent) {    var aResults = [];    var bMatchFound = false;    var aCache = this._aCache;    var nCacheLength = (aCache) ? aCache.length : 0;    var bMatchContains = this.queryMatchContains;        // If cache is enabled...    if((this.maxCacheEntries > 0) && aCache && (nCacheLength > 0)) {        this.cacheQueryEvent.fire(this, oParent, sQuery);        // If case is unimportant, normalize query now instead of in loops        if(!this.queryMatchCase) {            var sOrigQuery = sQuery;            sQuery = sQuery.toLowerCase();        }        // Loop through each cached element's query property...        for(var i = nCacheLength-1; i >= 0; i--) {            var resultObj = aCache[i];            var aAllResultItems = resultObj.results;            // If case is unimportant, normalize match key for comparison            var matchKey = (!this.queryMatchCase) ?                encodeURIComponent(resultObj.query.toLowerCase()):                encodeURIComponent(resultObj.query);                        // If a cached match key exactly matches the query...            if(matchKey == sQuery) {                    // Stash all result objects into aResult[] and stop looping through the cache.                    bMatchFound = true;                    aResults = aAllResultItems;                                        // The matching cache element was not the most recent,                    // so now we need to refresh the cache.                    if(i != nCacheLength-1) {                                                // Remove element from its original location                        aCache.splice(i,1);                        // Add element as newest                        this._addCacheElem(resultObj);                    }                    break;            }            // Else if this query is not an exact match and subset matching is enabled...            else if(this.queryMatchSubset) {                // Loop through substrings of each cached element's query property...                for(var j = sQuery.length-1; j >= 0 ; j--) {                    var subQuery = sQuery.substr(0,j);                                        // If a substring of a cached sQuery exactly matches the query...                    if(matchKey == subQuery) {                                            bMatchFound = true;                                                // Go through each cached result object to match against the query...                        for(var k = aAllResultItems.length-1; k >= 0; k--) {                            var aRecord = aAllResultItems[k];                            var sKeyIndex = (this.queryMatchCase) ?                                encodeURIComponent(aRecord[0]).indexOf(sQuery):                                encodeURIComponent(aRecord[0]).toLowerCase().indexOf(sQuery);                                                        // A STARTSWITH match is when the query is found at the beginning of the key string...                            if((!bMatchContains && (sKeyIndex === 0)) ||                            // A CONTAINS match is when the query is found anywhere within the key string...                            (bMatchContains && (sKeyIndex > -1))) {                                // Stash a match into aResults[].                                aResults.unshift(aRecord);                            }                        }                                                // Add the subset match result set object as the newest element to cache,                        // and stop looping through the cache.                        resultObj = {};                        resultObj.query = sQuery;                        resultObj.results = aResults;                        this._addCacheElem(resultObj);                        break;                    }                }                if(bMatchFound) {                    break;                }            }        }                // If there was a match, send along the results.        if(bMatchFound) {            this.getCachedResultsEvent.fire(this, oParent, sOrigQuery, aResults);            oCallbackFn(sOrigQuery, aResults, oParent);        }    }    return aResults;};/****************************************************************************//****************************************************************************//****************************************************************************//** * Implementation of YAHOO.widget.DataSource using XML HTTP requests that return * query results. * requires YAHOO.util.Connect XMLHTTPRequest library * extends YAHOO.widget.DataSource *   * @constructor * @param {string} sScriptURI Absolute or relative URI to script that returns  *                            query results as JSON, XML, or delimited flat data * @param {array} aSchema Data schema definition of results * @param {object} oConfigs Optional object literal of config params */YAHOO.widget.DS_XHR = function(sScriptURI, aSchema, oConfigs) {    // Set any config params passed in to override defaults    if(typeof oConfigs == "object") {        for(var sConfig in oConfigs) {            this[sConfig] = oConfigs[sConfig];        }    }        // Initialization sequence    if(!aSchema || (aSchema.constructor != Array)) {        return;    }    else {        this.schema = aSchema;    }    this.scriptURI = sScriptURI;    this._init();};YAHOO.widget.DS_XHR.prototype = new YAHOO.widget.DataSource();/*************************************************************************** * Public constants ***************************************************************************//** * JSON data type * * @type constant * @final */YAHOO.widget.DS_XHR.prototype.TYPE_JSON = 0;/** * XML data type * * @type constant * @final */YAHOO.widget.DS_XHR.prototype.TYPE_XML = 1;/** * Flat file data type * * @type constant * @final */YAHOO.widget.DS_XHR.prototype.TYPE_FLAT = 2;/** * Error message for XHR failure. * * @type constant * @final */YAHOO.widget.DS_XHR.prototype.ERROR_DATAXHR = "XHR response failed";/*************************************************************************** * Public member variables ***************************************************************************//** * Number of milliseconds the XHR connection will wait for a server response. A * a value of zero indicates the XHR connection will wait forever. Any value * greater than zero will use the Connection utility's Auto-Abort feature. * Default: 0. * * @type number */YAHOO.widget.DS_XHR.prototype.connTimeout = 0;/** * Absolute or relative URI to script that returns query results. For instance, * queries will be sent to *   <scriptURI>?<scriptQueryParam>=userinput * * @type string */YAHOO.widget.DS_XHR.prototype.scriptURI = null;/** * Query string parameter name sent to scriptURI. For instance, queries will be * sent to *   <scriptURI>?<scriptQueryParam>=userinput * Default: "query". * * @type string */YAHOO.widget.DS_XHR.prototype.scriptQueryParam = "query";/** * String of key/value pairs to append to requests made to scriptURI. Define * this string when you want to send additional query parameters to your script. * When defined, queries will be sent to *   <scriptURI>?<scriptQueryParam>=userinput&<scriptQueryAppend> * Default: "". * * @type string */YAHOO.widget.DS_XHR.prototype.scriptQueryAppend = "";/** * XHR response data type. Other types that may be defined are TYPE_XML and * TYPE_FLAT. Default: TYPE_JSON. * * @type type */YAHOO.widget.DS_XHR.prototype.responseType = YAHOO.widget.DS_XHR.prototype.TYPE_JSON;/** * String after which to strip results. If the results from the XHR are sent * back as HTML, the gzip HTML comment appears at the end of the data and should * be ignored.  Default: "\n&lt;!--" * * @type string */YAHOO.widget.DS_XHR.prototype.responseStripAfter = "\n<!--";/*************************************************************************** * Public methods ***************************************************************************//** * Queries the live data source defined by scriptURI for results. Results are * passed back to a callback function. *   * @param {object} oCallbackFn Callback function defined by oParent object to  *                             which to return results  * @param {string} sQuery Query string * @param {object} oParent The object instance that has requested data */YAHOO.widget.DS_XHR.prototype.doQuery = function(oCallbackFn, sQuery, oParent) {    var isXML = (this.responseType == this.TYPE_XML);    var sUri = this.scriptURI+"?"+this.scriptQueryParam+"="+sQuery;    if(this.scriptQueryAppend.length > 0) {        sUri += "&" + this.scriptQueryAppend;    }    var oResponse = null;        var oSelf = this;    /**     * Sets up ajax request callback     *     * @param {object} oReq          HTTPXMLRequest object     * @private     */    var responseSuccess = function(oResp) {        // Response ID does not match last made request ID.        if(!oSelf._oConn || (oResp.tId != oSelf._oConn.tId)) {            oSelf.dataErrorEvent.fire(oSelf, oParent, sQuery, oSelf.ERROR_DATANULL);            return;        }//DEBUGfor(var foo in oResp) {}        if(!isXML) {            oResp = oResp.responseText;        }        else {             oResp = oResp.responseXML;        }        if(oResp === null) {            oSelf.dataErrorEvent.fire(oSelf, oParent, sQuery, oSelf.ERROR_DATANULL);            return;        }        var aResults = oSelf.parseResponse(sQuery, oResp, oParent);        var resultObj = {};        resultObj.query = decodeURIComponent(sQuery);        resultObj.results = aResults;        if(aResults === null) {            oSelf.dataErrorEvent.fire(oSelf, oParent, sQuery, oSelf.ERROR_DATAPARSE);            return;        }        else {            oSelf.getResultsEvent.fire(oSelf, oParent, sQuery, aResults);            oSelf._addCacheElem(resultObj);            oCallbackFn(sQuery, aResults, oParent);        }    };    var responseFailure = function(oResp) {        oSelf.dataErrorEvent.fire(oSelf, oParent, sQuery, oSelf.ERROR_DATAXHR);        return;    };        var oCallback = {        success:responseSuccess,        failure:responseFailure    };        if(!isNaN(this.connTimeout) && this.connTimeout > 0) {        oCallback.timeout = this.connTimeout;    }        if(this._oConn) {        YAHOO.util.Connect.abort(this._oConn);    }        oSelf._oConn = YAHOO.util.Connect.asyncRequest("GET", sUri, oCallback, null);};/** * Parses raw response data into an array of result objects. The result data key * is always stashed in the [0] element of each result object.  * * @param {string} sQuery Query string * @param {object} oResponse The raw response data to parse * @param {object} oParent The object instance that has requested data * @returns {array} Array of result objects */YAHOO.widget.DS_XHR.prototype.parseResponse = function(sQuery, oResponse, oParent) {    var aSchema = this.schema;    var aResults = [];    var bError = false;    // Strip out comment at the end of results    var nEnd = ((this.responseStripAfter !== "") && (oResponse.indexOf)) ?        oResponse.indexOf(this.responseStripAfter) : -1;    if(nEnd != -1) {        oResponse = oResponse.substring(0,nEnd);    }    switch (this.responseType) {        case this.TYPE_JSON:            var jsonList;            // Divert KHTML clients from JSON lib            if(window.JSON && (navigator.userAgent.toLowerCase().indexOf('khtml')== -1)) {                // Use the JSON utility if available                var jsonObjParsed = JSON.parse(oResponse);                if(!jsonObjParsed) {                    bError = true;                    break;                }                else {                    // eval is necessary here since aSchema[0] is of unknown depth                    jsonList = eval("jsonObjParsed." + aSchema[0]);                }            }            else {                // Parse the JSON response as a string                try {                    // Trim leading spaces                    while (oResponse.substring(0,1) == " ") {                        oResponse = oResponse.substring(1, oResponse.length);                    }                    // Invalid JSON response                    if(oResponse.indexOf("{") < 0) {                        bError = true;                        break;                    }                    // Empty (but not invalid) JSON response                    if(oResponse.indexOf("{}") === 0) {                        break;                    }                    // Turn the string into an object literal...                    // ...eval is necessary here                    var jsonObjRaw = eval("(" + oResponse + ")");                    if(!jsonObjRaw) {                        bError = true;                        break;                    }                    // Grab the object member that contains an array of all reponses...                    // ...eval is necessary here since aSchema[0] is of unknown depth                    jsonList = eval("(jsonObjRaw." + aSchema[0]+")");                }                catch(e) {                    bError = true;                    break;               }            }            if(!jsonList) {                bError = true;                break;            }            if(jsonList.constructor != Array) {                jsonList = [jsonList];            }                        // Loop through the array of all responses...            for(var i = jsonList.length-1; i >= 0 ; i--) {                var aResultItem = [];                var jsonResult = jsonList[i];                // ...and loop through each data field value of each response                for(var j = aSchema.length-1; j >= 1 ; j--) {                    // ...and capture data into an array mapped according to the schema...                    var dataFieldValue = jsonResult[aSchema[j]];                    if(!dataFieldValue) {                        dataFieldValue = "";                    }                    aResultItem.unshift(dataFieldValue);                }                // Capture the array of data field values in an array of results                aResults.unshift(aResultItem);            }            break;        case this.TYPE_XML:            // Get the collection of results            var xmlList = oResponse.getElementsByTagName(aSchema[0]);            if(!xmlList) {                bError = true;                break;            }            // Loop through each result            for(var k = xmlList.length-1; k >= 0 ; k--) {                var result = xmlList.item(k);                var aFieldSet = [];                // Loop through each data field in each result using the schema                for(var m = aSchema.length-1; m >= 1 ; m--) {                    var sValue = null;                    // Values may be held in an attribute...                    var xmlAttr = result.attributes.getNamedItem(aSchema[m]);                    if(xmlAttr) {                        sValue = xmlAttr.value;                    }                    // ...or in a node                    else{                        var xmlNode = result.getElementsByTagName(aSchema[m]);                        if(xmlNode && xmlNode.item(0) && xmlNode.item(0).firstChild) {                            sValue = xmlNode.item(0).firstChild.nodeValue;                        }                        else {                            sValue = "";                        }                    }                    // Capture the schema-mapped data field values into an array                    aFieldSet.unshift(sValue);                }                // Capture each array of values into an array of results                aResults.unshift(aFieldSet);            }            break;        case this.TYPE_FLAT:            if(oResponse.length > 0) {                // Delete the last line delimiter at the end of the data if it exists                var newLength = oResponse.length-aSchema[0].length;                if(oResponse.substr(newLength) == aSchema[0]) {                    oResponse = oResponse.substr(0, newLength);                }                var aRecords = oResponse.split(aSchema[0]);                for(var n = aRecords.length-1; n >= 0; n--) {                    aResults[n] = aRecords[n].split(aSchema[1]);                }            }            break;        default:            break;    }    sQuery = null;    oResponse = null;    oParent = null;    if(bError) {        return null;    }    else {        return aResults;    }};            /*************************************************************************** * Private member variables ***************************************************************************//** * XHR connection object. * * @type object * @private */YAHOO.widget.DS_XHR.prototype._oConn = null;/****************************************************************************//****************************************************************************//****************************************************************************//** * Implementation of YAHOO.widget.DataSource using a native Javascript struct as * its live data source. *   * @constructor * extends YAHOO.widget.DataSource  *   * @param {string} oFunction In-memory Javascript function that returns query *                           results as an array of objects * @param {object} oConfigs Optional object literal of config params */YAHOO.widget.DS_JSFunction = function(oFunction, oConfigs) {    // Set any config params passed in to override defaults    if(typeof oConfigs == "object") {        for(var sConfig in oConfigs) {            this[sConfig] = oConfigs[sConfig];        }    }    // Initialization sequence    if(!oFunction  || (oFunction.constructor != Function)) {        return;    }    else {        this.dataFunction = oFunction;        this._init();    }};YAHOO.widget.DS_JSFunction.prototype = new YAHOO.widget.DataSource();/*************************************************************************** * Public member variables ***************************************************************************//** * In-memory Javascript function that returns query results. * * @type function */YAHOO.widget.DS_JSFunction.prototype.dataFunction = null;/*************************************************************************** * Public methods ***************************************************************************//** * Queries the live data source defined by function for results. Results are * passed back to a callback function. *   * @param {object} oCallbackFn Callback function defined by oParent object to  *                             which to return results  * @param {string} sQuery Query string * @param {object} oParent The object instance that has requested data */YAHOO.widget.DS_JSFunction.prototype.doQuery = function(oCallbackFn, sQuery, oParent) {    var oFunction = this.dataFunction;    var aResults = [];        aResults = oFunction(sQuery);    if(aResults === null) {        this.dataErrorEvent.fire(this, oParent, sQuery, this.ERROR_DATANULL);        return;    }        var resultObj = {};    resultObj.query = decodeURIComponent(sQuery);    resultObj.results = aResults;    this._addCacheElem(resultObj);        this.getResultsEvent.fire(this, oParent, sQuery, aResults);    oCallbackFn(sQuery, aResults, oParent);    return;};/****************************************************************************//****************************************************************************//****************************************************************************//** * Implementation of YAHOO.widget.DataSource using a native Javascript array as * its live data source. * * @constructor * extends YAHOO.widget.DataSource * * @param {string} aData In-memory Javascript array of simple string data * @param {object} oConfigs Optional object literal of config params */YAHOO.widget.DS_JSArray = function(aData, oConfigs) {    // Set any config params passed in to override defaults    if(typeof oConfigs == "object") {        for(var sConfig in oConfigs) {            this[sConfig] = oConfigs[sConfig];        }    }    // Initialization sequence    if(!aData || (aData.constructor != Array)) {        return;    }    else {        this.data = aData;        this._init();    }};YAHOO.widget.DS_JSArray.prototype = new YAHOO.widget.DataSource();/*************************************************************************** * Public member variables ***************************************************************************//** * In-memory Javascript array of strings. * * @type array */YAHOO.widget.DS_JSArray.prototype.data = null;/*************************************************************************** * Public methods ***************************************************************************//** * Queries the live data source defined by data for results. Results are passed * back to a callback function. * * @param {object} oCallbackFn Callback function defined by oParent object to *                             which to return results * @param {string} sQuery Query string * @param {object} oParent The object instance that has requested data */YAHOO.widget.DS_JSArray.prototype.doQuery = function(oCallbackFn, sQuery, oParent) {    var aData = this.data; // the array    var aResults = []; // container for results    var bMatchFound = false;    var bMatchContains = this.queryMatchContains;    if(!this.queryMatchCase) {        sQuery = sQuery.toLowerCase();    }    // Loop through each element of the array...    // which can be a string or an array of strings    for(var i = aData.length-1; i >= 0; i--) {        var aDataset = [];        if(aData[i]) {            if(aData[i].constructor == String) {                aDataset[0] = aData[i];            }            else if(aData[i].constructor == Array) {                aDataset = aData[i];            }        }        if(aDataset[0] && (aDataset[0].constructor == String)) {            var sKeyIndex = (this.queryMatchCase) ?            encodeURIComponent(aDataset[0]).indexOf(sQuery):            encodeURIComponent(aDataset[0]).toLowerCase().indexOf(sQuery);            // A STARTSWITH match is when the query is found at the beginning of the key string...            if((!bMatchContains && (sKeyIndex === 0)) ||            // A CONTAINS match is when the query is found anywhere within the key string...            (bMatchContains && (sKeyIndex > -1))) {                // Stash a match into aResults[].                aResults.unshift(aDataset);            }        }    }    this.getResultsEvent.fire(this, oParent, sQuery, aResults);    oCallbackFn(sQuery, aResults, oParent);};