123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296 |
- define("dojox/layout/RadioGroup", ["dojo/_base/kernel","dojo/_base/declare","dojo/_base/html","dojo/_base/lang","dojo/_base/query",
- "dijit/_Widget","dijit/_Templated","dijit/_Contained","dijit/layout/StackContainer",
- "dojo/fx/easing","dojo/_base/fx","dojo/dom-construct","dojo/dom-class"],function(
- kernel,declare,html,lang,query,Widget,Templated,Contained,StackContainer,easing,baseFx,domConstruct,domClass){
- kernel.experimental("dojox.layout.RadioGroup");
- //
- // dojox.layout.RadioGroup - an experimental (probably poorly named) Layout widget extending StackContainer
- // that accepts ContentPanes as children, and applies aesthetically pleasing responsive transition animations
- // attached to :hover of the Buttons created.
- //
- // FIXME: take the Buttons out of the root template, and allow layoutAlign or similar attrib to use a different
- // template, or build the template dynamically?
- //
- /*=====
- var StackContainer = dijit.layout.StackContainer,
- Templated = dijit._Templated,
- Contained = dijit._Contained,
- Widget = dijit._Widget;
- =====*/
- var RadioGroup = declare("dojox.layout.RadioGroup",[StackContainer,Templated],{
- // summary: A Container that turns its Layout Children into a single Pane and transitions between states
- // onHover of the button
- //
- // duration: Integer
- // used for Fade and Slide RadioGroup's, the duration to run the transition animation. does not affect anything
- // in default RadioGroup
- duration: 750,
- // hasButtons: Boolean
- // toggles internal button making on or off
- hasButtons: false,
- // buttonClass: String
- // The full declared className of the Button widget to use for hasButtons
- buttonClass: "dojox.layout._RadioButton",
-
- // templateString: String
- // the template for our container
- templateString: '<div class="dojoxRadioGroup">'
- +' <div dojoAttachPoint="buttonHolder" style="display:none;">'
- +' <table class="dojoxRadioButtons"><tbody><tr class="dojoxRadioButtonRow" dojoAttachPoint="buttonNode"></tr></tbody></table>'
- +' </div>'
- +' <div class="dojoxRadioView" dojoAttachPoint="containerNode"></div>'
- +'</div>',
- startup: function(){
- // summary: scan the container for children, and make "tab buttons" for them
- this.inherited(arguments);
- this._children = this.getChildren();
- this._buttons = this._children.length;
- this._size = html.coords(this.containerNode);
- if(this.hasButtons){
- html.style(this.buttonHolder, "display", "block");
- }
- },
- _setupChild: function(/* dijit._Widget */child){
- // summary: Creates a hover button for a child node of the RadioGroup
- html.style(child.domNode, "position", "absolute");
- if(this.hasButtons){
-
- var tmp = this.buttonNode.appendChild(domConstruct.create('td'));
- var n = domConstruct.create("div", null, tmp),
- _Button = lang.getObject(this.buttonClass),
- tmpw = new _Button({
- label: child.title,
- page: child
- }, n)
- ;
-
- lang.mixin(child, { _radioButton: tmpw });
- tmpw.startup();
- }
- child.domNode.style.display = "none";
- },
-
- removeChild: function(child){
- if(this.hasButtons && child._radioButton){
- child._radioButton.destroy();
- delete child._radioButton;
- }
- this.inherited(arguments);
- },
-
- // FIXME: shouldn't have to rewriting these, need to take styling out of _showChild and _hideChild
- // and use classes on the domNode in _transition or something similar (in StackContainer)
- _transition: function(/*dijit._Widget*/ newWidget, /*dijit._Widget*/ oldWidget){
- // summary: called when StackContainer receives a selectChild call, used to transition the panes.
- this._showChild(newWidget);
- if(oldWidget){
- this._hideChild(oldWidget);
- }
- // Size the new widget, in case this is the first time it's being shown,
- // or I have been resized since the last time it was shown.
- // page must be visible for resizing to work
- if(this.doLayout && newWidget.resize){
- newWidget.resize(this._containerContentBox || this._contentBox);
- }
- },
- _showChild: function(/*dijit._Widget*/ page){
- // summary: show the selected child widget
- var children = this.getChildren();
- page.isFirstChild = (page == children[0]);
- page.isLastChild = (page == children[children.length-1]);
- page.selected = true;
- page.domNode.style.display="";
- if(page._onShow){
- page._onShow(); // trigger load in ContentPane
- }else if(page.onShow){
- page.onShow();
- }
- },
- _hideChild: function(/*dijit._Widget*/ page){
- // summary: hide the specified child widget
- page.selected = false;
- page.domNode.style.display="none";
- if(page.onHide){
- page.onHide();
- }
- }
- });
- declare("dojox.layout.RadioGroupFade", RadioGroup, {
- // summary: An extension on a stock RadioGroup, that fades the panes.
- _hideChild: function(page){
- // summary: hide the specified child widget
- baseFx.fadeOut({
- node:page.domNode,
- duration:this.duration,
- onEnd: lang.hitch(this,"inherited", arguments, arguments)
- }).play();
- },
- _showChild: function(page){
- // summary: show the specified child widget
- this.inherited(arguments);
- html.style(page.domNode, "opacity", 0);
- baseFx.fadeIn({
- node:page.domNode,
- duration:this.duration
- }).play();
- }
- });
- declare("dojox.layout.RadioGroupSlide", RadioGroup, {
- // summary: A Sliding Radio Group
- // description:
- // An extension on a stock RadioGroup widget, sliding the pane
- // into view from being hidden. The entry direction is randomized
- // on each view
- //
- // easing: Function
- // A hook to override the default easing of the pane slides.
- easing: "dojo.fx.easing.backOut",
- // zTop: Integer
- // A z-index to apply to the incoming pane
- zTop: 99,
-
- constructor: function(){
- if(lang.isString(this.easing)){
- this.easing = lang.getObject(this.easing);
- }
- },
-
- _positionChild: function(page){
- // summary: set the child out of view immediately after being hidden
- // FIXME: is there a real "size" floating around always?
- if(!this._size){ return; }
-
- // there should be a contest: obfuscate this function as best you can.
- var rA = true, rB = true;
- switch(page.slideFrom){
- case "bottom" : rB = !rB; break;
- case "right" : rA = !rA; rB = !rB; break;
- case "top" : break;
- case "left" : rA = !rA; break;
- default:
- rA = Math.round(Math.random());
- rB = Math.round(Math.random());
- break;
- }
- var prop = rA ? "top" : "left",
- val = (rB ? "-" : "") + (this._size[rA ? "h" : "w" ] + 20) + "px";
-
- html.style(page.domNode, prop, val);
- },
- _showChild: function(page){
- // summary: Slide in the selected child widget
-
- var children = this.getChildren();
- page.isFirstChild = (page == children[0]);
- page.isLastChild = (page == children[children.length-1]);
- page.selected = true;
- html.style(page.domNode,{
- zIndex: this.zTop, display:""
- })
- if(this._anim && this._anim.status()=="playing"){
- this._anim.gotoPercent(100,true);
- }
-
- this._anim = baseFx.animateProperty({
- node:page.domNode,
- properties: {
- left: 0,
- top: 0
- },
- duration: this.duration,
- easing: this.easing,
- onEnd: lang.hitch(page, function(){
- if(this.onShow){ this.onShow(); }
- if(this._onShow){ this._onShow(); }
- }),
- beforeBegin: lang.hitch(this, "_positionChild", page)
- });
- this._anim.play();
- },
- _hideChild: function(page){
- // summary: reset the position of the hidden pane out of sight
- page.selected = false;
- page.domNode.style.zIndex = this.zTop - 1;
- if(page.onHide){
- page.onHide();
- }
- }
-
- });
- declare("dojox.layout._RadioButton",[Widget,Templated,Contained],{
- // summary: The Buttons for a RadioGroup
- //
- // description: A private widget used to manipulate the StackContainer (RadioGroup*). Don't create directly.
- //
-
- // label: String
- // the Text Label of the button
- label: "",
- // domNode to tell parent to select
- page: null,
- templateString: '<div dojoAttachPoint="focusNode" class="dojoxRadioButton"><span dojoAttachPoint="titleNode" class="dojoxRadioButtonLabel">${label}</span></div>',
-
- startup: function(){
- // summary: start listening to mouseOver
- this.connect(this.domNode, "onmouseenter", "_onMouse");
- },
-
- _onMouse: function(/* Event */e){
- // summary: set the selected child on hover, and set our hover state class
- this.getParent().selectChild(this.page);
- this._clearSelected();
- domClass.add(this.domNode,"dojoxRadioButtonSelected");
- },
- _clearSelected: function(){
- // summary: remove hover state class from sibling Buttons. This is easier (and more reliable)
- // than setting up an additional connection to onMouseOut
-
- // FIXME: this relies on the template being [div][span]node[/span][/div]
- query(".dojoxRadioButtonSelected", this.domNode.parentNode.parentNode)
- .removeClass("dojoxRadioButtonSelected")
- ;
- }
-
- });
- lang.extend(Widget,{
- // slideFrom: String
- // A parameter needed by RadioGroupSlide only. An optional paramter to force
- // the ContentPane to slide in from a set direction. Defaults
- // to "random", or specify one of "top", "left", "right", "bottom"
- // to slideFrom top, left, right, or bottom.
- slideFrom: "random"
- })
- });
|