basetabslider.lzx
<library>
    <include href="base/baselist.lzx"/>
    <include href="utils/layouts/simplelayout.lzx"/>
    
    <class name="basetabslider" extends="baselist" styleable="false" layout="class: simplelayout" oninit="_setSpacing(spacing)"> 
        
        <attribute name="slideduration" value="300"/>
        
        <attribute name="spacing" value="0" setter="_setSpacing(spacing)"/>
        
        <attribute name="_spacinginitdone" value="false"/>
        
        <attribute name="mintabheight" value="22"/>
        
        <attribute name="opennedtab" value="null"/>
        
        
        <attribute name="inset_top" value="8"/>
        
        
        <attribute name="inset_left" value="8"/>
        
        
        <attribute name="inset_bottom" value="8"/>
        
        
        <attribute name="inset_right" value="8"/>
        
        <attribute name="totalminheight" value="0"/>
        
        <attribute name="availableheight" value="0"/>
        
        <attribute name="hilite" value="false"/>
        
        <attribute name="_defaultview" value="null"/>
        
        <method name="init">
            // all subviews should now be created so call calcheight
            // so the tebelements can reference it
            this.calcAvailableHeight();
            // do inherited and send oninit events
            super.init();
        </method>
        
        <method name="resetHeight">
            this.calcAvailableHeight();
            var opentab = this.getSelection();
            if ( opentab ) {
                if (opentab instanceof Array) {
                    opentab = opentab[0];
                }
                opentab.setAttribute("height",
                    opentab['minheight'] + this.availableheight);
            }
        </method>
        
        <method name="_setSpacing" args="sp">
            // TODO: a max spacing size
            this.spacing = sp;
            if (this._initcomplete) {
                var cv = null;
                if ( this.defaultplacement != null ) {
                    cv = this.searchSubviews( "name", this.defaultplacement );
                } 
                if (cv == null) cv = this;
                cv.layouts[0].setAttribute('spacing', sp);
                if (this._spacinginitdone) {
                    this.resetHeight();
                } else {
                   this._spacinginitdone = true;
               }
            } 
        </method>
        
        <setter name="height" args="h">
            if ( h != height ) {
                super.setAttribute('height', h);
                this.resetHeight();
            }
        </setter>
        
        <method name="select" args="item">
        
            // Override the select method of baselist so a tabelement that is
            // already open does not get a close call
            if ( this._initcomplete ) {
                var sel = this.getSelection();
                if ( sel ) {
                    if (!toggleselected) {
                        if (sel instanceof Array && sel[0] == item) return;
                        if (sel == item) return;
                    }
                }
                this._selector.select( item );
                var v =  this.getValue();
                this.setAttribute('value', v );
                //this is duplicated from baselist, but the superclass method
                //isn't called
                if ( this._hiliteview && this._hiliteview.enabled) {
                    this._hiliteview['setHilite'](false);
                }
                this.setAttribute( 'doesenter', false );
            } else {
                this._initialselection = item;
            }
        
        </method>
        
        <handler name="onaddsubview" method="doAddSubview"/>
        
        <method name="doAddSubview" args="v">
            if ( this.itemclassname == "" ){
                this.__itemclass = v.constructor;
            }
            if ( typeof(v.minheight) == "undefined" ) {
                // no minheight was defined so assign the parent
                // value before calling calcAvailableHeight
                if ( v instanceof lz.basetabelement ) v.setAttribute('minheight', this.mintabheight);
            }
            // v.setAttribute('height', v.minheight);
            this.calcAvailableHeight();
            var opentab = this.getSelection();
            if ( opentab ) {
                if (opentab instanceof Array) {
                    opentab = opentab[0];
                }
                opentab.setAttribute("height",
                    opentab['minheight'] + this.availableheight);
            }
        </method>
        
        <method name="openTab" args="tabelement,withAnimation=false"> 
            var didopen = false;
            var ot = this._selector.getSelection()[0];
            if ( ot != tabelement ){
                this._selector.select(tabelement);
                this.opennedtab = tabelement;
                didopen = true;
            }
            return didopen;
         </method>
        
        <method name="openNext"> 
            var ot = this._selector.getSelection()[0];
            var index = -1;
            for ( var i = 0; i < this._defaultview.subviews.length; i++ ) {
                if ( ot == this._defaultview.subviews[i] ) index = i;
            }
            index += 1;
            if ( index < this._defaultview.subviews.length ) 
                 return openTab(this._defaultview.subviews[index]);
            else
                return false;
         </method>
        
        <method name="addItem" args="txt, value=null"> 
            var v = new lz[this.itemclassname](this,{text:txt, value:value});
            this.onheight.sendEvent(); // force an update
         </method>
        
        <method name="calcAvailableHeight"> 
           var cv = null;
           if ( this.defaultplacement != null ) {
               cv = this.searchSubviews( "name", this.defaultplacement );
           }
           if ( cv == null) cv = this;
           this._defaultview = cv;
           if ( cv['subviews'] ) {
                if ( !cv['layouts'] ) return;
                var tmh = 0;
                for( var m=0; m < cv.subviews.length; m++ ) {
                    var sv = cv.subviews[m]
                    if ( sv.visible && sv instanceof lz.basetabelement ) {
                        tmh += sv['minheight'] + spacing;
                    }
                }
                // don't need spacing on last element
                tmh -= spacing;
                // set the corresponding height of the attributes
                this.setAttribute('availableheight',
                  Math.max(0,cv.height - tmh));
                this.setAttribute('totalminheight',tmh);
            }
         </method>
        <doc>
          <tag name="shortdesc"><text>A non-visual container that animates and coordinates the sliding of basetabelements</text></tag>
          <text>
            <p>A <classname>basetabslider</classname> coordinates the opening and
            closing of its <classname>basetabelement</classname>s contained within
            its scope. <classname>basetabslider</classname> and
            <classname>basetabelement</classname> have no visual aspects. The
            <classname>tabslider</classname> class (which extends
            <classname>basetabslider</classname>) does contain a visual UI and
            shows how <classname>basetabslider</classname> can be extended to
            create your own visual framework for a unique tab slider.</p>
            
            <p>An example of the use of these classes is shown below. In order to
            make the base classes visible, background colors have been assigned to
            them.</p>
            
            <example>
            <canvas height="220">
              <include href="base/basetabslider.lzx"/>
              <basetabslider bgcolor="yellow" width="100" 
                            height="200" mintabheight="40" 
                            spacing="2" slideduration="300">
                <basetabelement clickable="true" bgcolor="red" width="100%"/>
                <basetabelement clickable="true" bgcolor="red" width="100%"/>
                <basetabelement clickable="true" bgcolor="red" width="100%"/>
              </basetabslider>
            </canvas>
            </example>
            
            <p>The <attribute>mintabheight</attribute> specified for a
            <classname>basetabslider</classname> applies to all of its
            <classname>basetabelement</classname>s except those
            <classname>basetabelement</classname>s that define it for themselves,
            as demonstrated in the following example.</p>
            
            <example>
            <canvas height="220">
              <include href="base/basetabslider.lzx"/>
              <basetabslider bgcolor="yellow" width="100" 
                            height="200" mintabheight="40" 
                            spacing="2" slideduration="300">
                <basetabelement clickable="true" bgcolor="red" width="100%"/>
                <basetabelement clickable="true" bgcolor="red" width="100%" minheight="15"/>
                <basetabelement clickable="true" bgcolor="red" width="100%"/>
              </basetabslider>
            </canvas>
            </example>
          </text>
        </doc>
   </class> 
</library>
         Cross References
         Includes
         
         Classes