2019-07-17 20:08:50 +00:00
|
|
|
/*
|
2019-07-17 20:16:19 +00:00
|
|
|
Copyright (c) 2009, Yahoo! Inc. All rights reserved.
|
2019-07-17 20:08:50 +00:00
|
|
|
Code licensed under the BSD License:
|
|
|
|
http://developer.yahoo.net/yui/license.txt
|
2019-07-17 20:16:19 +00:00
|
|
|
version: 2.7.0
|
2019-07-17 20:08:50 +00:00
|
|
|
*/
|
|
|
|
(function() {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The tabview module provides a widget for managing content bound to tabs.
|
|
|
|
* @module tabview
|
|
|
|
* @requires yahoo, dom, event, element
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
var Y = YAHOO.util,
|
|
|
|
Dom = Y.Dom,
|
|
|
|
Event = Y.Event,
|
|
|
|
document = window.document,
|
2019-07-17 20:08:50 +00:00
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
// STRING CONSTANTS
|
|
|
|
ACTIVE = 'active',
|
|
|
|
ACTIVE_INDEX = 'activeIndex',
|
|
|
|
ACTIVE_TAB = 'activeTab',
|
|
|
|
CONTENT_EL = 'contentEl',
|
|
|
|
ELEMENT = 'element',
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A widget to control tabbed views.
|
|
|
|
* @namespace YAHOO.widget
|
|
|
|
* @class TabView
|
|
|
|
* @extends YAHOO.util.Element
|
|
|
|
* @constructor
|
|
|
|
* @param {HTMLElement | String | Object} el(optional) The html
|
|
|
|
* element that represents the TabView, or the attribute object to use.
|
|
|
|
* An element will be created if none provided.
|
|
|
|
* @param {Object} attr (optional) A key map of the tabView's
|
|
|
|
* initial attributes. Ignored if first arg is attributes object.
|
|
|
|
*/
|
2019-07-17 20:16:19 +00:00
|
|
|
TabView = function(el, attr) {
|
2019-07-17 20:08:50 +00:00
|
|
|
attr = attr || {};
|
|
|
|
if (arguments.length == 1 && !YAHOO.lang.isString(el) && !el.nodeName) {
|
|
|
|
attr = el; // treat first arg as attr object
|
|
|
|
el = attr.element || null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!el && !attr.element) { // create if we dont have one
|
2019-07-17 20:16:19 +00:00
|
|
|
el = this._createTabViewElement(attr);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
2019-07-17 20:16:19 +00:00
|
|
|
TabView.superclass.constructor.call(this, el, attr);
|
2019-07-17 20:08:50 +00:00
|
|
|
};
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
YAHOO.extend(TabView, Y.Element, {
|
2019-07-17 20:08:50 +00:00
|
|
|
/**
|
|
|
|
* The className to add when building from scratch.
|
|
|
|
* @property CLASSNAME
|
|
|
|
* @default "navset"
|
|
|
|
*/
|
|
|
|
CLASSNAME: 'yui-navset',
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The className of the HTMLElement containing the TabView's tab elements
|
|
|
|
* to look for when building from existing markup, or to add when building
|
|
|
|
* from scratch.
|
|
|
|
* All childNodes of the tab container are treated as Tabs when building
|
|
|
|
* from existing markup.
|
|
|
|
* @property TAB_PARENT_CLASSNAME
|
|
|
|
* @default "nav"
|
|
|
|
*/
|
|
|
|
TAB_PARENT_CLASSNAME: 'yui-nav',
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The className of the HTMLElement containing the TabView's label elements
|
|
|
|
* to look for when building from existing markup, or to add when building
|
|
|
|
* from scratch.
|
|
|
|
* All childNodes of the content container are treated as content elements when
|
|
|
|
* building from existing markup.
|
|
|
|
* @property CONTENT_PARENT_CLASSNAME
|
|
|
|
* @default "nav-content"
|
|
|
|
*/
|
|
|
|
CONTENT_PARENT_CLASSNAME: 'yui-content',
|
|
|
|
|
|
|
|
_tabParent: null,
|
|
|
|
_contentParent: null,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds a Tab to the TabView instance.
|
|
|
|
* If no index is specified, the tab is added to the end of the tab list.
|
|
|
|
* @method addTab
|
|
|
|
* @param {YAHOO.widget.Tab} tab A Tab instance to add.
|
|
|
|
* @param {Integer} index The position to add the tab.
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
addTab: function(tab, index) {
|
2019-07-17 20:16:19 +00:00
|
|
|
var tabs = this.get('tabs'),
|
|
|
|
before = this.getTab(index),
|
|
|
|
tabParent = this._tabParent,
|
|
|
|
contentParent = this._contentParent,
|
|
|
|
tabElement = tab.get(ELEMENT),
|
|
|
|
contentEl = tab.get(CONTENT_EL);
|
|
|
|
|
2019-07-17 20:08:50 +00:00
|
|
|
if (!tabs) { // not ready yet
|
|
|
|
this._queue[this._queue.length] = ['addTab', arguments];
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
index = (index === undefined) ? tabs.length : index;
|
|
|
|
|
|
|
|
if ( before ) {
|
|
|
|
tabParent.insertBefore(tabElement, before.get(ELEMENT));
|
|
|
|
} else {
|
|
|
|
tabParent.appendChild(tabElement);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( contentEl && !Dom.isAncestor(contentParent, contentEl) ) {
|
|
|
|
contentParent.appendChild(contentEl);
|
|
|
|
}
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
if ( !tab.get(ACTIVE) ) {
|
2019-07-17 20:08:50 +00:00
|
|
|
tab.set('contentVisible', false, true); /* hide if not active */
|
|
|
|
} else {
|
2019-07-17 20:16:19 +00:00
|
|
|
this.set(ACTIVE_TAB, tab, true);
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
this._initTabEvents(tab);
|
|
|
|
tabs.splice(index, 0, tab);
|
|
|
|
},
|
2019-07-17 20:08:50 +00:00
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
_initTabEvents: function(tab) {
|
|
|
|
tab.addListener( tab.get('activationEvent'), tab._onActivate, this, tab);
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
tab.addListener('activationEventChange', function(e) {
|
|
|
|
if (e.prevValue != e.newValue) {
|
2019-07-17 20:16:19 +00:00
|
|
|
tab.removeListener(e.prevValue, tab._onActivate);
|
|
|
|
tab.addListener(e.newValue, tab._onActivate, this, tab);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Routes childNode events.
|
|
|
|
* @method DOMEventHandler
|
|
|
|
* @param {event} e The Dom event that is being handled.
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
DOMEventHandler: function(e) {
|
2019-07-17 20:16:19 +00:00
|
|
|
var target = Event.getTarget(e),
|
|
|
|
tabParent = this._tabParent,
|
|
|
|
tabs = this.get('tabs'),
|
|
|
|
tab,
|
|
|
|
tabEl,
|
|
|
|
contentEl;
|
|
|
|
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
if (Dom.isAncestor(tabParent, target) ) {
|
|
|
|
for (var i = 0, len = tabs.length; i < len; i++) {
|
|
|
|
tabEl = tabs[i].get(ELEMENT);
|
2019-07-17 20:16:19 +00:00
|
|
|
contentEl = tabs[i].get(CONTENT_EL);
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
if ( target == tabEl || Dom.isAncestor(tabEl, target) ) {
|
|
|
|
tab = tabs[i];
|
|
|
|
break; // note break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tab) {
|
|
|
|
tab.fireEvent(e.type, e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the Tab instance at the specified index.
|
|
|
|
* @method getTab
|
|
|
|
* @param {Integer} index The position of the Tab.
|
|
|
|
* @return YAHOO.widget.Tab
|
|
|
|
*/
|
|
|
|
getTab: function(index) {
|
|
|
|
return this.get('tabs')[index];
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the index of given tab.
|
|
|
|
* @method getTabIndex
|
|
|
|
* @param {YAHOO.widget.Tab} tab The tab whose index will be returned.
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
getTabIndex: function(tab) {
|
2019-07-17 20:16:19 +00:00
|
|
|
var index = null,
|
|
|
|
tabs = this.get('tabs');
|
2019-07-17 20:08:50 +00:00
|
|
|
for (var i = 0, len = tabs.length; i < len; ++i) {
|
|
|
|
if (tab == tabs[i]) {
|
|
|
|
index = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return index;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes the specified Tab from the TabView.
|
|
|
|
* @method removeTab
|
|
|
|
* @param {YAHOO.widget.Tab} item The Tab instance to be removed.
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
removeTab: function(tab) {
|
2019-07-17 20:16:19 +00:00
|
|
|
var tabCount = this.get('tabs').length,
|
|
|
|
index = this.getTabIndex(tab);
|
|
|
|
|
|
|
|
if ( tab === this.get(ACTIVE_TAB) ) {
|
|
|
|
if (tabCount > 1) { // select another tab
|
|
|
|
if (index + 1 === tabCount) { // if last, activate previous
|
|
|
|
this.set(ACTIVE_INDEX, index - 1);
|
|
|
|
} else { // activate next tab
|
|
|
|
this.set(ACTIVE_INDEX, index + 1);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
2019-07-17 20:16:19 +00:00
|
|
|
} else { // no more tabs
|
|
|
|
this.set(ACTIVE_TAB, null);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this._tabParent.removeChild( tab.get(ELEMENT) );
|
2019-07-17 20:16:19 +00:00
|
|
|
this._contentParent.removeChild( tab.get(CONTENT_EL) );
|
2019-07-17 20:08:50 +00:00
|
|
|
this._configs.tabs.value.splice(index, 1);
|
2019-07-17 20:16:19 +00:00
|
|
|
|
|
|
|
tab.fireEvent('remove', { type: 'remove', tabview: this });
|
2019-07-17 20:08:50 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Provides a readable name for the TabView instance.
|
|
|
|
* @method toString
|
|
|
|
* @return String
|
|
|
|
*/
|
|
|
|
toString: function() {
|
|
|
|
var name = this.get('id') || this.get('tagName');
|
|
|
|
return "TabView " + name;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The transiton to use when switching between tabs.
|
|
|
|
* @method contentTransition
|
|
|
|
*/
|
|
|
|
contentTransition: function(newTab, oldTab) {
|
2019-07-17 20:16:19 +00:00
|
|
|
if (newTab) {
|
|
|
|
newTab.set('contentVisible', true);
|
|
|
|
}
|
|
|
|
if (oldTab) {
|
|
|
|
oldTab.set('contentVisible', false);
|
|
|
|
}
|
2019-07-17 20:08:50 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* setAttributeConfigs TabView specific properties.
|
|
|
|
* @method initAttributes
|
|
|
|
* @param {Object} attr Hash of initial attributes
|
|
|
|
*/
|
|
|
|
initAttributes: function(attr) {
|
|
|
|
TabView.superclass.initAttributes.call(this, attr);
|
|
|
|
|
|
|
|
if (!attr.orientation) {
|
|
|
|
attr.orientation = 'top';
|
|
|
|
}
|
|
|
|
|
|
|
|
var el = this.get(ELEMENT);
|
|
|
|
|
|
|
|
if (!Dom.hasClass(el, this.CLASSNAME)) {
|
|
|
|
Dom.addClass(el, this.CLASSNAME);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The Tabs belonging to the TabView instance.
|
|
|
|
* @attribute tabs
|
|
|
|
* @type Array
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig('tabs', {
|
|
|
|
value: [],
|
|
|
|
readOnly: true
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The container of the tabView's label elements.
|
|
|
|
* @property _tabParent
|
|
|
|
* @private
|
|
|
|
* @type HTMLElement
|
|
|
|
*/
|
|
|
|
this._tabParent =
|
|
|
|
this.getElementsByClassName(this.TAB_PARENT_CLASSNAME,
|
2019-07-17 20:16:19 +00:00
|
|
|
'ul' )[0] || this._createTabParent();
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The container of the tabView's content elements.
|
|
|
|
* @property _contentParent
|
|
|
|
* @type HTMLElement
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._contentParent =
|
|
|
|
this.getElementsByClassName(this.CONTENT_PARENT_CLASSNAME,
|
2019-07-17 20:16:19 +00:00
|
|
|
'div')[0] || this._createContentParent();
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* How the Tabs should be oriented relative to the TabView.
|
|
|
|
* @attribute orientation
|
|
|
|
* @type String
|
|
|
|
* @default "top"
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig('orientation', {
|
|
|
|
value: attr.orientation,
|
|
|
|
method: function(value) {
|
|
|
|
var current = this.get('orientation');
|
|
|
|
this.addClass('yui-navset-' + value);
|
|
|
|
|
|
|
|
if (current != value) {
|
|
|
|
this.removeClass('yui-navset-' + current);
|
|
|
|
}
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
if (value === 'bottom') {
|
2019-07-17 20:08:50 +00:00
|
|
|
this.appendChild(this._tabParent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The index of the tab currently active.
|
|
|
|
* @attribute activeIndex
|
|
|
|
* @type Int
|
|
|
|
*/
|
2019-07-17 20:16:19 +00:00
|
|
|
this.setAttributeConfig(ACTIVE_INDEX, {
|
2019-07-17 20:08:50 +00:00
|
|
|
value: attr.activeIndex,
|
|
|
|
method: function(value) {
|
|
|
|
},
|
|
|
|
validator: function(value) {
|
2019-07-17 20:16:19 +00:00
|
|
|
var ret = true;
|
|
|
|
if (value && this.getTab(value).get('disabled')) { // cannot activate if disabled
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
return ret;
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The tab currently active.
|
|
|
|
* @attribute activeTab
|
|
|
|
* @type YAHOO.widget.Tab
|
|
|
|
*/
|
2019-07-17 20:16:19 +00:00
|
|
|
this.setAttributeConfig(ACTIVE_TAB, {
|
2019-07-17 20:08:50 +00:00
|
|
|
value: attr.activeTab,
|
|
|
|
method: function(tab) {
|
2019-07-17 20:16:19 +00:00
|
|
|
var activeTab = this.get(ACTIVE_TAB);
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
if (tab) {
|
2019-07-17 20:16:19 +00:00
|
|
|
tab.set(ACTIVE, true);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
if (activeTab && activeTab !== tab) {
|
|
|
|
activeTab.set(ACTIVE, false);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
if (activeTab && tab !== activeTab) { // no transition if only 1
|
2019-07-17 20:08:50 +00:00
|
|
|
this.contentTransition(tab, activeTab);
|
|
|
|
} else if (tab) {
|
|
|
|
tab.set('contentVisible', true);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
validator: function(value) {
|
2019-07-17 20:16:19 +00:00
|
|
|
var ret = true;
|
|
|
|
if (value && value.get('disabled')) { // cannot activate if disabled
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
return ret;
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
this.on('activeTabChange', this._onActiveTabChange);
|
|
|
|
this.on('activeIndexChange', this._onActiveIndexChange);
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
YAHOO.log('attributes initialized', 'info', 'TabView');
|
|
|
|
if ( this._tabParent ) {
|
2019-07-17 20:16:19 +00:00
|
|
|
this._initTabs();
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Due to delegation we add all DOM_EVENTS to the TabView container
|
|
|
|
// but IE will leak when unsupported events are added, so remove these
|
|
|
|
this.DOM_EVENTS.submit = false;
|
|
|
|
this.DOM_EVENTS.focus = false;
|
|
|
|
this.DOM_EVENTS.blur = false;
|
|
|
|
|
|
|
|
for (var type in this.DOM_EVENTS) {
|
|
|
|
if ( YAHOO.lang.hasOwnProperty(this.DOM_EVENTS, type) ) {
|
|
|
|
this.addListener.call(this, type, this.DOMEventHandler);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
/**
|
|
|
|
* Removes selected state from the given tab if it is the activeTab
|
|
|
|
* @method deselectTab
|
|
|
|
* @param {Int} index The tab index to deselect
|
|
|
|
*/
|
|
|
|
deselectTab: function(index) {
|
|
|
|
if (this.getTab(index) === this.get('activeTab')) {
|
|
|
|
this.set('activeTab', null);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Makes the tab at the given index the active tab
|
|
|
|
* @method selectTab
|
|
|
|
* @param {Int} index The tab index to be made active
|
|
|
|
*/
|
|
|
|
selectTab: function(index) {
|
|
|
|
this.set('activeTab', this.getTab(index));
|
|
|
|
},
|
|
|
|
|
|
|
|
_onActiveTabChange: function(e) {
|
|
|
|
var activeIndex = this.get(ACTIVE_INDEX),
|
2019-07-17 20:08:50 +00:00
|
|
|
newIndex = this.getTabIndex(e.newValue);
|
|
|
|
|
|
|
|
if (activeIndex !== newIndex) {
|
2019-07-17 20:16:19 +00:00
|
|
|
if (!(this.set(ACTIVE_INDEX, newIndex)) ) { // NOTE: setting
|
2019-07-17 20:08:50 +00:00
|
|
|
// revert if activeIndex update fails (cancelled via beforeChange)
|
2019-07-17 20:16:19 +00:00
|
|
|
this.set(ACTIVE_TAB, e.prevValue);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
_onActiveIndexChange: function(e) {
|
2019-07-17 20:08:50 +00:00
|
|
|
// no set if called from ActiveTabChange event
|
2019-07-17 20:16:19 +00:00
|
|
|
if (e.newValue !== this.getTabIndex(this.get(ACTIVE_TAB))) {
|
|
|
|
if (!(this.set(ACTIVE_TAB, this.getTab(e.newValue))) ) { // NOTE: setting
|
2019-07-17 20:08:50 +00:00
|
|
|
// revert if activeTab update fails (cancelled via beforeChange)
|
2019-07-17 20:16:19 +00:00
|
|
|
this.set(ACTIVE_INDEX, e.prevValue);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
}
|
2019-07-17 20:16:19 +00:00
|
|
|
},
|
2019-07-17 20:08:50 +00:00
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
/**
|
|
|
|
* Creates Tab instances from a collection of HTMLElements.
|
|
|
|
* @method _initTabs
|
|
|
|
* @private
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
_initTabs: function() {
|
|
|
|
var tabs = Dom.getChildren(this._tabParent),
|
|
|
|
contentElements = Dom.getChildren(this._contentParent),
|
|
|
|
activeIndex = this.get(ACTIVE_INDEX),
|
|
|
|
tab,
|
|
|
|
attr,
|
|
|
|
active;
|
|
|
|
|
|
|
|
for (var i = 0, len = tabs.length; i < len; ++i) {
|
|
|
|
attr = {};
|
|
|
|
|
|
|
|
if (contentElements[i]) {
|
|
|
|
attr.contentEl = contentElements[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
tab = new YAHOO.widget.Tab(tabs[i], attr);
|
|
|
|
this.addTab(tab);
|
|
|
|
|
|
|
|
if (tab.hasClass(tab.ACTIVE_CLASSNAME) ) {
|
|
|
|
active = tab;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (activeIndex) {
|
|
|
|
this.set(ACTIVE_TAB, this.getTab(activeIndex));
|
|
|
|
} else {
|
|
|
|
this._configs.activeTab.value = active; // dont invoke method
|
|
|
|
this._configs.activeIndex.value = this.getTabIndex(active);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
2019-07-17 20:16:19 +00:00
|
|
|
},
|
2019-07-17 20:08:50 +00:00
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
_createTabViewElement: function(attr) {
|
|
|
|
var el = document.createElement('div');
|
|
|
|
|
|
|
|
if ( this.CLASSNAME ) {
|
|
|
|
el.className = this.CLASSNAME;
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
2019-07-17 20:16:19 +00:00
|
|
|
|
|
|
|
YAHOO.log('TabView Dom created', 'info', 'TabView');
|
|
|
|
return el;
|
|
|
|
},
|
2019-07-17 20:08:50 +00:00
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
_createTabParent: function(attr) {
|
|
|
|
var el = document.createElement('ul');
|
2019-07-17 20:08:50 +00:00
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
if ( this.TAB_PARENT_CLASSNAME ) {
|
|
|
|
el.className = this.TAB_PARENT_CLASSNAME;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.get(ELEMENT).appendChild(el);
|
|
|
|
|
|
|
|
return el;
|
|
|
|
},
|
2019-07-17 20:08:50 +00:00
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
_createContentParent: function(attr) {
|
|
|
|
var el = document.createElement('div');
|
2019-07-17 20:08:50 +00:00
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
if ( this.CONTENT_PARENT_CLASSNAME ) {
|
|
|
|
el.className = this.CONTENT_PARENT_CLASSNAME;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.get(ELEMENT).appendChild(el);
|
|
|
|
|
|
|
|
return el;
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
2019-07-17 20:16:19 +00:00
|
|
|
});
|
|
|
|
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
YAHOO.widget.TabView = TabView;
|
|
|
|
})();
|
|
|
|
|
|
|
|
(function() {
|
2019-07-17 20:16:19 +00:00
|
|
|
var Y = YAHOO.util,
|
|
|
|
Dom = Y.Dom,
|
|
|
|
Lang = YAHOO.lang,
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
// STRING CONSTANTS
|
2019-07-17 20:16:19 +00:00
|
|
|
ACTIVE_TAB = 'activeTab',
|
|
|
|
LABEL = 'label',
|
2019-07-17 20:08:50 +00:00
|
|
|
LABEL_EL = 'labelEl',
|
|
|
|
CONTENT = 'content',
|
2019-07-17 20:16:19 +00:00
|
|
|
CONTENT_EL = 'contentEl',
|
2019-07-17 20:08:50 +00:00
|
|
|
ELEMENT = 'element',
|
|
|
|
CACHE_DATA = 'cacheData',
|
|
|
|
DATA_SRC = 'dataSrc',
|
|
|
|
DATA_LOADED = 'dataLoaded',
|
|
|
|
DATA_TIMEOUT = 'dataTimeout',
|
|
|
|
LOAD_METHOD = 'loadMethod',
|
|
|
|
POST_DATA = 'postData',
|
2019-07-17 20:16:19 +00:00
|
|
|
DISABLED = 'disabled',
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A representation of a Tab's label and content.
|
|
|
|
* @namespace YAHOO.widget
|
|
|
|
* @class Tab
|
|
|
|
* @extends YAHOO.util.Element
|
|
|
|
* @constructor
|
|
|
|
* @param element {HTMLElement | String} (optional) The html element that
|
|
|
|
* represents the TabView. An element will be created if none provided.
|
|
|
|
* @param {Object} properties A key map of initial properties
|
|
|
|
*/
|
2019-07-17 20:16:19 +00:00
|
|
|
Tab = function(el, attr) {
|
2019-07-17 20:08:50 +00:00
|
|
|
attr = attr || {};
|
|
|
|
if (arguments.length == 1 && !Lang.isString(el) && !el.nodeName) {
|
|
|
|
attr = el;
|
|
|
|
el = attr.element;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!el && !attr.element) {
|
2019-07-17 20:16:19 +00:00
|
|
|
el = this._createTabElement(attr);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
this.loadHandler = {
|
|
|
|
success: function(o) {
|
|
|
|
this.set(CONTENT, o.responseText);
|
|
|
|
},
|
|
|
|
failure: function(o) {
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
Tab.superclass.constructor.call(this, el, attr);
|
|
|
|
|
|
|
|
this.DOM_EVENTS = {}; // delegating to tabView
|
|
|
|
};
|
|
|
|
|
|
|
|
YAHOO.extend(Tab, YAHOO.util.Element, {
|
|
|
|
/**
|
|
|
|
* The default tag name for a Tab's inner element.
|
|
|
|
* @property LABEL_INNER_TAGNAME
|
|
|
|
* @type String
|
|
|
|
* @default "em"
|
|
|
|
*/
|
|
|
|
LABEL_TAGNAME: 'em',
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The class name applied to active tabs.
|
|
|
|
* @property ACTIVE_CLASSNAME
|
|
|
|
* @type String
|
|
|
|
* @default "selected"
|
|
|
|
*/
|
|
|
|
ACTIVE_CLASSNAME: 'selected',
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The class name applied to active tabs.
|
|
|
|
* @property ACTIVE_CLASSNAME
|
|
|
|
* @type String
|
|
|
|
* @default "selected"
|
|
|
|
*/
|
|
|
|
HIDDEN_CLASSNAME: 'yui-hidden',
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The title applied to active tabs.
|
|
|
|
* @property ACTIVE_TITLE
|
|
|
|
* @type String
|
|
|
|
* @default "active"
|
|
|
|
*/
|
|
|
|
ACTIVE_TITLE: 'active',
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The class name applied to disabled tabs.
|
|
|
|
* @property DISABLED_CLASSNAME
|
|
|
|
* @type String
|
|
|
|
* @default "disabled"
|
|
|
|
*/
|
|
|
|
DISABLED_CLASSNAME: DISABLED,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The class name applied to dynamic tabs while loading.
|
|
|
|
* @property LOADING_CLASSNAME
|
|
|
|
* @type String
|
|
|
|
* @default "disabled"
|
|
|
|
*/
|
|
|
|
LOADING_CLASSNAME: 'loading',
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Provides a reference to the connection request object when data is
|
|
|
|
* loaded dynamically.
|
|
|
|
* @property dataConnection
|
|
|
|
* @type Object
|
|
|
|
*/
|
|
|
|
dataConnection: null,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Object containing success and failure callbacks for loading data.
|
|
|
|
* @property loadHandler
|
|
|
|
* @type object
|
|
|
|
*/
|
|
|
|
loadHandler: null,
|
|
|
|
|
|
|
|
_loading: false,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Provides a readable name for the tab.
|
|
|
|
* @method toString
|
|
|
|
* @return String
|
|
|
|
*/
|
|
|
|
toString: function() {
|
2019-07-17 20:16:19 +00:00
|
|
|
var el = this.get(ELEMENT),
|
|
|
|
id = el.id || el.tagName;
|
2019-07-17 20:08:50 +00:00
|
|
|
return "Tab " + id;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* setAttributeConfigs TabView specific properties.
|
|
|
|
* @method initAttributes
|
|
|
|
* @param {Object} attr Hash of initial attributes
|
|
|
|
*/
|
|
|
|
initAttributes: function(attr) {
|
|
|
|
attr = attr || {};
|
|
|
|
Tab.superclass.initAttributes.call(this, attr);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The event that triggers the tab's activation.
|
|
|
|
* @attribute activationEvent
|
|
|
|
* @type String
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig('activationEvent', {
|
|
|
|
value: attr.activationEvent || 'click'
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The element that contains the tab's label.
|
|
|
|
* @attribute labelEl
|
|
|
|
* @type HTMLElement
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig(LABEL_EL, {
|
2019-07-17 20:16:19 +00:00
|
|
|
value: attr[LABEL_EL] || this._getLabelEl(),
|
2019-07-17 20:08:50 +00:00
|
|
|
method: function(value) {
|
2019-07-17 20:16:19 +00:00
|
|
|
value = Dom.get(value);
|
2019-07-17 20:08:50 +00:00
|
|
|
var current = this.get(LABEL_EL);
|
|
|
|
|
|
|
|
if (current) {
|
|
|
|
if (current == value) {
|
|
|
|
return false; // already set
|
|
|
|
}
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
current.parentNode.replaceChild(value, current);
|
|
|
|
this.set(LABEL, value.innerHTML);
|
|
|
|
}
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The tab's label text (or innerHTML).
|
|
|
|
* @attribute label
|
|
|
|
* @type String
|
|
|
|
*/
|
2019-07-17 20:16:19 +00:00
|
|
|
this.setAttributeConfig(LABEL, {
|
|
|
|
value: attr.label || this._getLabel(),
|
2019-07-17 20:08:50 +00:00
|
|
|
method: function(value) {
|
|
|
|
var labelEl = this.get(LABEL_EL);
|
|
|
|
if (!labelEl) { // create if needed
|
2019-07-17 20:16:19 +00:00
|
|
|
this.set(LABEL_EL, this._createLabelEl());
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
labelEl.innerHTML = value;
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The HTMLElement that contains the tab's content.
|
|
|
|
* @attribute contentEl
|
|
|
|
* @type HTMLElement
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig(CONTENT_EL, {
|
2019-07-17 20:16:19 +00:00
|
|
|
value: attr[CONTENT_EL] || document.createElement('div'),
|
2019-07-17 20:08:50 +00:00
|
|
|
method: function(value) {
|
2019-07-17 20:16:19 +00:00
|
|
|
value = Dom.get(value);
|
2019-07-17 20:08:50 +00:00
|
|
|
var current = this.get(CONTENT_EL);
|
|
|
|
|
|
|
|
if (current) {
|
2019-07-17 20:16:19 +00:00
|
|
|
if (current === value) {
|
2019-07-17 20:08:50 +00:00
|
|
|
return false; // already set
|
|
|
|
}
|
2019-07-17 20:16:19 +00:00
|
|
|
if (!this.get('selected')) {
|
|
|
|
Dom.addClass(value, 'yui-hidden');
|
|
|
|
}
|
|
|
|
current.parentNode.replaceChild(value, current);
|
|
|
|
this.set(CONTENT, value.innerHTML);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The tab's content.
|
|
|
|
* @attribute content
|
|
|
|
* @type String
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig(CONTENT, {
|
2019-07-17 20:16:19 +00:00
|
|
|
value: attr[CONTENT],
|
2019-07-17 20:08:50 +00:00
|
|
|
method: function(value) {
|
|
|
|
this.get(CONTENT_EL).innerHTML = value;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The tab's data source, used for loading content dynamically.
|
|
|
|
* @attribute dataSrc
|
|
|
|
* @type String
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig(DATA_SRC, {
|
|
|
|
value: attr.dataSrc
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not content should be reloaded for every view.
|
|
|
|
* @attribute cacheData
|
|
|
|
* @type Boolean
|
|
|
|
* @default false
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig(CACHE_DATA, {
|
|
|
|
value: attr.cacheData || false,
|
|
|
|
validator: Lang.isBoolean
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The method to use for the data request.
|
|
|
|
* @attribute loadMethod
|
|
|
|
* @type String
|
|
|
|
* @default "GET"
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig(LOAD_METHOD, {
|
|
|
|
value: attr.loadMethod || 'GET',
|
|
|
|
validator: Lang.isString
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not any data has been loaded from the server.
|
|
|
|
* @attribute dataLoaded
|
|
|
|
* @type Boolean
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig(DATA_LOADED, {
|
|
|
|
value: false,
|
|
|
|
validator: Lang.isBoolean,
|
|
|
|
writeOnce: true
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Number if milliseconds before aborting and calling failure handler.
|
|
|
|
* @attribute dataTimeout
|
|
|
|
* @type Number
|
|
|
|
* @default null
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig(DATA_TIMEOUT, {
|
|
|
|
value: attr.dataTimeout || null,
|
|
|
|
validator: Lang.isNumber
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Arguments to pass when POST method is used
|
|
|
|
* @attribute postData
|
|
|
|
* @default null
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig(POST_DATA, {
|
|
|
|
value: attr.postData || null
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not the tab is currently active.
|
|
|
|
* If a dataSrc is set for the tab, the content will be loaded from
|
|
|
|
* the given source.
|
|
|
|
* @attribute active
|
|
|
|
* @type Boolean
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig('active', {
|
|
|
|
value: attr.active || this.hasClass(this.ACTIVE_CLASSNAME),
|
|
|
|
method: function(value) {
|
|
|
|
if (value === true) {
|
|
|
|
this.addClass(this.ACTIVE_CLASSNAME);
|
|
|
|
this.set('title', this.ACTIVE_TITLE);
|
|
|
|
} else {
|
|
|
|
this.removeClass(this.ACTIVE_CLASSNAME);
|
|
|
|
this.set('title', '');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
validator: function(value) {
|
|
|
|
return Lang.isBoolean(value) && !this.get(DISABLED) ;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not the tab is disabled.
|
|
|
|
* @attribute disabled
|
|
|
|
* @type Boolean
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig(DISABLED, {
|
|
|
|
value: attr.disabled || this.hasClass(this.DISABLED_CLASSNAME),
|
|
|
|
method: function(value) {
|
|
|
|
if (value === true) {
|
|
|
|
Dom.addClass(this.get(ELEMENT), this.DISABLED_CLASSNAME);
|
|
|
|
} else {
|
|
|
|
Dom.removeClass(this.get(ELEMENT), this.DISABLED_CLASSNAME);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
validator: Lang.isBoolean
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The href of the tab's anchor element.
|
|
|
|
* @attribute href
|
|
|
|
* @type String
|
|
|
|
* @default '#'
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig('href', {
|
|
|
|
value: attr.href ||
|
|
|
|
this.getElementsByTagName('a')[0].getAttribute('href', 2) || '#',
|
|
|
|
method: function(value) {
|
|
|
|
this.getElementsByTagName('a')[0].href = value;
|
|
|
|
},
|
|
|
|
validator: Lang.isString
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The Whether or not the tab's content is visible.
|
|
|
|
* @attribute contentVisible
|
|
|
|
* @type Boolean
|
|
|
|
* @default false
|
|
|
|
*/
|
|
|
|
this.setAttributeConfig('contentVisible', {
|
|
|
|
value: attr.contentVisible,
|
|
|
|
method: function(value) {
|
|
|
|
if (value) {
|
|
|
|
Dom.removeClass(this.get(CONTENT_EL), this.HIDDEN_CLASSNAME);
|
|
|
|
|
|
|
|
if ( this.get(DATA_SRC) ) {
|
|
|
|
// load dynamic content unless already loading or loaded and caching
|
|
|
|
if ( !this._loading && !(this.get(DATA_LOADED) && this.get(CACHE_DATA)) ) {
|
|
|
|
this._dataConnect();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Dom.addClass(this.get(CONTENT_EL), this.HIDDEN_CLASSNAME);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
validator: Lang.isBoolean
|
|
|
|
});
|
|
|
|
YAHOO.log('attributes initialized', 'info', 'Tab');
|
|
|
|
},
|
|
|
|
|
|
|
|
_dataConnect: function() {
|
2019-07-17 20:16:19 +00:00
|
|
|
if (!Y.Connect) {
|
2019-07-17 20:08:50 +00:00
|
|
|
YAHOO.log('YAHOO.util.Connect dependency not met',
|
|
|
|
'error', 'Tab');
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Dom.addClass(this.get(CONTENT_EL).parentNode, this.LOADING_CLASSNAME);
|
|
|
|
this._loading = true;
|
2019-07-17 20:16:19 +00:00
|
|
|
this.dataConnection = Y.Connect.asyncRequest(
|
2019-07-17 20:08:50 +00:00
|
|
|
this.get(LOAD_METHOD),
|
|
|
|
this.get(DATA_SRC),
|
|
|
|
{
|
|
|
|
success: function(o) {
|
|
|
|
YAHOO.log('content loaded successfully', 'info', 'Tab');
|
|
|
|
this.loadHandler.success.call(this, o);
|
|
|
|
this.set(DATA_LOADED, true);
|
|
|
|
this.dataConnection = null;
|
|
|
|
Dom.removeClass(this.get(CONTENT_EL).parentNode,
|
|
|
|
this.LOADING_CLASSNAME);
|
|
|
|
this._loading = false;
|
|
|
|
},
|
|
|
|
failure: function(o) {
|
|
|
|
YAHOO.log('loading failed: ' + o.statusText, 'error', 'Tab');
|
|
|
|
this.loadHandler.failure.call(this, o);
|
|
|
|
this.dataConnection = null;
|
|
|
|
Dom.removeClass(this.get(CONTENT_EL).parentNode,
|
|
|
|
this.LOADING_CLASSNAME);
|
|
|
|
this._loading = false;
|
|
|
|
},
|
|
|
|
scope: this,
|
|
|
|
timeout: this.get(DATA_TIMEOUT)
|
|
|
|
},
|
|
|
|
|
|
|
|
this.get(POST_DATA)
|
|
|
|
);
|
2019-07-17 20:16:19 +00:00
|
|
|
},
|
|
|
|
_createTabElement: function(attr) {
|
|
|
|
var el = document.createElement('li'),
|
|
|
|
a = document.createElement('a'),
|
|
|
|
label = attr.label || null,
|
|
|
|
labelEl = attr.labelEl || null;
|
|
|
|
|
|
|
|
a.href = attr.href || '#'; // TODO: Use Dom.setAttribute?
|
|
|
|
el.appendChild(a);
|
|
|
|
|
|
|
|
if (labelEl) { // user supplied labelEl
|
|
|
|
if (!label) { // user supplied label
|
|
|
|
label = this._getLabel();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
labelEl = this._createLabelEl();
|
|
|
|
}
|
|
|
|
|
|
|
|
a.appendChild(labelEl);
|
|
|
|
|
|
|
|
YAHOO.log('creating Tab Dom', 'info', 'Tab');
|
|
|
|
return el;
|
|
|
|
},
|
|
|
|
|
|
|
|
_getLabelEl: function() {
|
|
|
|
return this.getElementsByTagName(this.LABEL_TAGNAME)[0];
|
|
|
|
},
|
|
|
|
|
|
|
|
_createLabelEl: function() {
|
|
|
|
var el = document.createElement(this.LABEL_TAGNAME);
|
|
|
|
return el;
|
|
|
|
},
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
_getLabel: function() {
|
|
|
|
var el = this.get(LABEL_EL);
|
|
|
|
|
|
|
|
if (!el) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
return el.innerHTML;
|
|
|
|
},
|
|
|
|
|
|
|
|
_onActivate: function(e, tabview) {
|
|
|
|
var tab = this,
|
|
|
|
silent = false;
|
|
|
|
|
|
|
|
|
|
|
|
Y.Event.preventDefault(e);
|
|
|
|
if (tab === tabview.get(ACTIVE_TAB)) {
|
|
|
|
silent = true; // dont fire activeTabChange if already active
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
2019-07-17 20:16:19 +00:00
|
|
|
tabview.set(ACTIVE_TAB, tab, silent);
|
2019-07-17 20:08:50 +00:00
|
|
|
}
|
2019-07-17 20:16:19 +00:00
|
|
|
});
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
/**
|
|
|
|
* Fires when a tab is removed from the tabview
|
|
|
|
* @event remove
|
|
|
|
* @type CustomEvent
|
|
|
|
* @param {Event} An event object with fields for "type" ("remove")
|
|
|
|
* and "tabview" (the tabview instance it was removed from)
|
|
|
|
*/
|
2019-07-17 20:08:50 +00:00
|
|
|
|
|
|
|
YAHOO.widget.Tab = Tab;
|
|
|
|
})();
|
|
|
|
|
2019-07-17 20:16:19 +00:00
|
|
|
YAHOO.register("tabview", YAHOO.widget.TabView, {version: "2.7.0", build: "1799"});
|