blob: 503644e94a26889b1b602f0acb52516c3c096eec [file] [log] [blame]
/*
Copyright (c) 2004-2006, The Dojo Foundation
All Rights Reserved.
Licensed under the Academic Free License version 2.1 or above OR the
modified BSD license. For more information on Dojo licensing, see:
http://dojotoolkit.org/community/licensing.shtml
*/
dojo.provide("dojo.widget.TabContainer");
dojo.require("dojo.lang.func");
dojo.require("dojo.widget.*");
dojo.require("dojo.widget.PageContainer");
dojo.require("dojo.event.*");
dojo.require("dojo.html.selection");
dojo.require("dojo.widget.html.layout");
dojo.widget.defineWidget("dojo.widget.TabContainer", dojo.widget.PageContainer, {
// summary
// A TabContainer is a container that has multiple panes, but shows only
// one pane at a time. There are a set of tabs corresponding to each pane,
// where each tab has the title (aka label) of the pane, and optionally a close button.
//
// Publishes topics <widgetId>-addChild, <widgetId>-removeChild, and <widgetId>-selectChild
// (where <widgetId> is the id of the TabContainer itself.
// labelPosition: String
// Defines where tab labels go relative to tab content.
// "top", "bottom", "left-h", "right-h"
labelPosition: "top",
// closeButton: String
// If closebutton=="tab", then every tab gets a close button.
// DEPRECATED: Should just say closable=true on each
// pane you want to be closable.
closeButton: "none",
templateString: null, // override setting in PageContainer
templatePath: dojo.uri.dojoUri("src/widget/templates/TabContainer.html"),
templateCssPath: dojo.uri.dojoUri("src/widget/templates/TabContainer.css"),
// selectedTab: String
// initially selected tab (widgetId)
// DEPRECATED: use selectedChild instead.
selectedTab: "",
postMixInProperties: function() {
if(this.selectedTab){
dojo.deprecated("selectedTab deprecated, use selectedChild instead, will be removed in", "0.5");
this.selectedChild=this.selectedTab;
}
if(this.closeButton!="none"){
dojo.deprecated("closeButton deprecated, use closable='true' on each child instead, will be removed in", "0.5");
}
dojo.widget.TabContainer.superclass.postMixInProperties.apply(this, arguments);
},
fillInTemplate: function() {
// create the tab list that will have a tab (a.k.a. tab button) for each tab panel
this.tablist = dojo.widget.createWidget("TabController",
{
id: this.widgetId + "_tablist",
labelPosition: this.labelPosition,
doLayout: this.doLayout,
containerId: this.widgetId
}, this.tablistNode);
dojo.widget.TabContainer.superclass.fillInTemplate.apply(this, arguments);
},
postCreate: function(args, frag) {
dojo.widget.TabContainer.superclass.postCreate.apply(this, arguments);
// size the container pane to take up the space not used by the tabs themselves
this.onResized();
},
_setupChild: function(tab){
if(this.closeButton=="tab" || this.closeButton=="pane"){
// TODO: remove in 0.5
tab.closable=true;
}
dojo.html.addClass(tab.domNode, "dojoTabPane");
dojo.widget.TabContainer.superclass._setupChild.apply(this, arguments);
},
onResized: function(){
// Summary: Configure the content pane to take up all the space except for where the tabs are
if(!this.doLayout){ return; }
// position the labels and the container node
var labelAlign=this.labelPosition.replace(/-h/,"");
var children = [
{domNode: this.tablist.domNode, layoutAlign: labelAlign},
{domNode: this.containerNode, layoutAlign: "client"}
];
dojo.widget.html.layout(this.domNode, children);
if(this.selectedChildWidget){
var containerSize = dojo.html.getContentBox(this.containerNode);
this.selectedChildWidget.resizeTo(containerSize.width, containerSize.height);
}
},
selectTab: function(tab, callingWidget){
dojo.deprecated("use selectChild() rather than selectTab(), selectTab() will be removed in", "0.5");
this.selectChild(tab, callingWidget);
},
onKey: function(e){
// summary
// Keystroke handling for keystrokes on the tab panel itself (that were bubbled up to me)
// Ctrl-up: focus is returned from the pane to the tab button
// Alt-del: close tab
if(e.keyCode == e.KEY_UP_ARROW && e.ctrlKey){
// set focus to current tab
var button = this.correspondingTabButton || this.selectedTabWidget.tabButton;
button.focus();
dojo.event.browser.stopEvent(e);
}else if(e.keyCode == e.KEY_DELETE && e.altKey){
if (this.selectedChildWidget.closable){
this.closeChild(this.selectedChildWidget);
dojo.event.browser.stopEvent(e);
}
}
},
destroy: function(){
this.tablist.destroy();
dojo.widget.TabContainer.superclass.destroy.apply(this, arguments);
}
});
dojo.widget.defineWidget(
"dojo.widget.TabController",
dojo.widget.PageController,
{
// summary
// Set of tabs (the things with labels and a close button, that you click to show a tab panel).
// Lets the user select the currently shown pane in a TabContainer or PageContainer.
// TabController also monitors the TabContainer, and whenever a pane is
// added or deleted updates itself accordingly.
templateString: "<div wairole='tablist' dojoAttachEvent='onKey'></div>",
// labelPosition: String
// Defines where tab labels go relative to tab content.
// "top", "bottom", "left-h", "right-h"
labelPosition: "top",
doLayout: true,
// class: String
// Class name to apply to the top dom node
"class": "",
// buttonWidget: String
// the name of the tab widget to create to correspond to each page
buttonWidget: "TabButton",
postMixInProperties: function() {
if(!this["class"]){
this["class"] = "dojoTabLabels-" + this.labelPosition + (this.doLayout ? "" : " dojoTabNoLayout");
}
dojo.widget.TabController.superclass.postMixInProperties.apply(this, arguments);
}
}
);
dojo.widget.defineWidget("dojo.widget.TabButton", dojo.widget.PageButton,
{
// summary
// A tab (the thing you click to select a pane).
// Contains the title (aka label) of the pane, and optionally a close-button to destroy the pane.
// This is an internal widget and should not be instantiated directly.
templateString: "<div class='dojoTab' dojoAttachEvent='onClick'>"
+"<div dojoAttachPoint='innerDiv'>"
+"<span dojoAttachPoint='titleNode' tabIndex='-1' waiRole='tab'>${this.label}</span>"
+"<span dojoAttachPoint='closeButtonNode' class='close closeImage' style='${this.closeButtonStyle}'"
+" dojoAttachEvent='onMouseOver:onCloseButtonMouseOver; onMouseOut:onCloseButtonMouseOut; onClick:onCloseButtonClick'></span>"
+"</div>"
+"</div>",
postMixInProperties: function(){
this.closeButtonStyle = this.closeButton ? "" : "display: none";
dojo.widget.TabButton.superclass.postMixInProperties.apply(this, arguments);
},
fillInTemplate: function(){
dojo.html.disableSelection(this.titleNode);
dojo.widget.TabButton.superclass.fillInTemplate.apply(this, arguments);
},
onCloseButtonClick: function(/*Event*/ evt){
// since the close button is located inside the select button, make sure that the select
// button doesn't inadvertently get an onClick event
evt.stopPropagation();
dojo.widget.TabButton.superclass.onCloseButtonClick.apply(this, arguments);
}
});
dojo.widget.defineWidget(
"dojo.widget.a11y.TabButton",
dojo.widget.TabButton,
{
// summary
// Tab for display in high-contrast mode (where background images don't show up).
// This is an internal widget and shouldn't be instantiated directly.
imgPath: dojo.uri.dojoUri("src/widget/templates/images/tab_close.gif"),
templateString: "<div class='dojoTab' dojoAttachEvent='onClick;onKey'>"
+"<div dojoAttachPoint='innerDiv'>"
+"<span dojoAttachPoint='titleNode' tabIndex='-1' waiRole='tab'>${this.label}</span>"
+"<img class='close' src='${this.imgPath}' alt='[x]' style='${this.closeButtonStyle}'"
+" dojoAttachEvent='onClick:onCloseButtonClick'>"
+"</div>"
+"</div>"
}
);