OSDN Git Service

Version 0.6.133, fix for closure compiler - ADVANCED_OPTIMIZATIONS
[pettanr/clientJs.git] / 0.6.x / js / 02_dom / 05_XNodeAttr.js
index 5a5f16f..e56b9df 100644 (file)
@@ -47,32 +47,103 @@ X_Node_Attr_HAS_VALUE = {
                SELECT   : true,\r
                BUTTON   : true,\r
                OBJECT   : true,\r
-               PARAM    : true // FlashVars が flash 側から書き換えられるケースがある\r
+               PARAM    : true // FlashVars が flash 側から書き換えられるケースがある??\r
+},\r
+\r
+// <input type=button,hidden,submit,reset,radio,checkbox> の場合、value の値はユーザーで変えることはない\r
+// <input type=text,password,file> はユーザーによって常に変更される HTML5 ではこれにさらにいろいろ加わる\r
+X_Node_Attr_STATIC_VALUE_TYPES = {\r
+       button   : true,\r
+       hidden   : true,\r
+       submit   : true,\r
+       reset    : true,\r
+       radio    : true,\r
+       checkbox : true\r
+},\r
+\r
+// 自由な内容が入るため、参照文字への変換が必要\r
+X_Node_Attr_toChrReferance = {\r
+       value : true,\r
+       title : true,\r
+       alt   : true\r
 },\r
 \r
 X_Node_Attr_renameForTag = {};\r
        // http://nanto.asablo.jp/blog/2005/10/29/123294\r
        // checked -> defaultChecked\r
        // 動的に生成した input 要素を文書ツリーに挿入する前に設定した checked 属性は反映されず、defaultChecked だと反映される\r
-       // 先頭にスペース\r
-function X_Node_Attr_objToAttrText( obj ){\r
-               var noValue = X_Node_Attr_noValue,\r
-                       attrs = [ '' ], n = 0, p, v;\r
-               if( !obj ) return ''; // Opera7\r
-               for( p in obj ){\r
-                       v = obj[ p ];\r
-                       if( p === 'value' ){\r
-                               v = v.split( '"' ).join( '&quot;' ).split( '>' ).join( '&gt;' ).split( '<' ).join( '&lt;' );\r
+       // ロードイベントを拾うために、要素生成時にネットワーク関連の属性を設定しない。\r
+       //  -> src (img, iframe, ), link の href, <param name="movie" src=>\r
+       // \r
+function X_Node_Attr_objToAttrText( that, skipNetworkForElmCreation ){\r
+       var obj     = that._attrs,\r
+               noValue = X_Node_Attr_noValue,\r
+               attrs   = [ '' ], // 先頭にスペース\r
+               plain   = X_EMPTY_OBJECT,\r
+               n = 0, k, check;\r
+\r
+       if( skipNetworkForElmCreation ){\r
+               delete that._newAttrs;\r
+               // このあとで _newAttr にネットワーク系の属性を控える, attrText には加えない\r
+       } else {\r
+               that._flags &= ~X_Node_State.OLD_ATTRTEXT;\r
+               // 完全な attrText\r
+       };\r
+       \r
+       if( !obj ){ // Opera7\r
+               delete that._attrText;\r
+               return '';\r
+       };\r
+       \r
+       for( k in obj ){\r
+               if( plain[ k ] ) continue;\r
+               \r
+               if( skipNetworkForElmCreation ){\r
+                       check = false;\r
+                       switch( that._tag + k ){\r
+                               case 'PARAMvalue' :\r
+                                       check = obj[ 'name' ] !== 'movie';\r
+                               case 'INPUTsrc'  :\r
+                                       check = check || ( obj[ 'type' ] !== 'image' );\r
+                               case 'LINKhref' :\r
+                                       check = check || ( obj[ 'rel' ] !== 'stylesheet' );\r
+                                       \r
+                                       if( !check ) break;\r
+\r
+                               case 'IMGsrc'     :\r
+                               case 'IFRAMEsrc'  :\r
+                               case 'FRAMEsrc'   :\r
+                               case 'SCRIPTsrc'  :\r
+                               case 'EMBEDsrc'   :\r
+                               case 'OBJECTdata' :\r
+                               case 'BGSOUNDsrc' :\r
+                               case 'APPLETcode' :\r
+                               //case 'AUDIOsrc' :\r
+                               //case 'VIDEOsrc' :\r
+                                       if( !that._newAttrs ) that._newAttrs = {};\r
+                                       that._newAttrs[ k ] = obj[ k ];\r
+                                       continue;\r
                        };\r
-                       attrs[ ++n ] = noValue[ p ] ? p : [ p, '="', v, '"' ].join( '' );\r
                };\r
-               return 0 < n ? attrs.join( ' ' ) : '';\r
+\r
+               attrs[ ++n ] = noValue[ k ] ? k : [\r
+                       k, '="',\r
+                       X_Node_Attr_toChrReferance[ k ] ? X_String_toChrReferance( obj[ k ] ) : obj[ k ],\r
+                       '"' ].join( '' );\r
+       };\r
+       \r
+       if( 0 < n ){\r
+               return that._attrText = attrs.join( ' ' );\r
+       };\r
+       delete that._attrText;\r
+       return '';\r
 };\r
 \r
 (function( renameForDOM, renameForTag ){\r
-       var name, i;\r
-       for( name in renameForDOM ){\r
-               renameForTag[ renameForDOM[ name ] ] = name;\r
+       var k, i;\r
+       for( k in renameForDOM ){\r
+               //if( X_EMPTY_OBJECT[ k ] ) continue;\r
+               renameForTag[ renameForDOM[ k ] ] = k;\r
        };\r
 })( X_Node_Attr_renameForDOM, X_Node_Attr_renameForTag );\r
 \r
@@ -80,37 +151,37 @@ function X_Node_Attr_objToAttrText( obj ){
 \r
 /* --------------------------------------\r
  *  attribute\r
- *  X_Node_Attr_toIndex に定義されている 属性の場合\r
  * \r
  * http://nanto.asablo.jp/blog/2005/10/29/123294\r
  * className, onclick等 はここで設定しない\r
  * \r
  */\r
 Node.prototype.attr = function( nameOrObj /* v */ ){\r
-       var attrs = this._attrs, newAttrs, f, p, elm, v;\r
+       var attrs = this._attrs, newAttrs, f, k, elm, v;\r
        \r
-       if( this._xnodeType !== 1 ) return this;\r
+       if( !this._tag ) return this;\r
        \r
-       if( nameOrObj && X.Type.isObject( nameOrObj ) ){\r
+       if( nameOrObj && X_Type_isObject( nameOrObj ) ){\r
                attrs || ( attrs = this._attrs = {} );\r
                newAttrs = this._newAttrs || ( this._newAttrs = {} );\r
                \r
-               for( p in nameOrObj ){\r
-                       if( X_Node_Attr_setAttr( this, attrs, newAttrs, p, nameOrObj[ p ] ) === true ) f = true;\r
+               for( k in nameOrObj ){\r
+                       //if( X_EMPTY_OBJECT[ k ] ) continue;\r
+                       if( X_Node_Attr_setAttr( this, attrs, newAttrs, k, nameOrObj[ k ] ) === true ) f = true;\r
                };\r
                if( f ){\r
-                       this._attrText = false;\r
-                       this._dirty |= X_Node_Dirty.ATTR;\r
-                       this._root && X_Node_reserveUpdate();                   \r
+                       delete this._attrText;\r
+                       this._flags |= X_Node_State.DIRTY_ATTR | X_Node_State.OLD_ATTRTEXT;\r
+                       this._flags & X_Node_State.IN_TREE && X_Node_reserveUpdate();\r
                };\r
                return this;\r
        } else\r
        if( 1 < arguments.length ){\r
                // setter\r
                if( X_Node_Attr_setAttr( this, attrs || ( this._attrs = {} ), this._newAttrs || ( this._newAttrs = {} ), nameOrObj, arguments[ 1 ] ) === true ){\r
-                       this._attrText = false;\r
-                       this._dirty |= X_Node_Dirty.ATTR;\r
-                       this._root && X_Node_reserveUpdate();\r
+                       delete this._attrText;\r
+                       this._flags |= X_Node_State.DIRTY_ATTR | X_Node_State.OLD_ATTRTEXT;\r
+                       this._flags & X_Node_State.IN_TREE && X_Node_reserveUpdate();\r
                };\r
                return this;\r
        } else\r
@@ -128,16 +199,23 @@ Node.prototype.attr = function( nameOrObj /* v */ ){
                        case 'style' :\r
                        case 'cssText' :\r
                                return this.cssText();\r
-                       case 'text' :\r
-                               return this.text();\r
-                       case 'html' :\r
-                               return this.html();\r
+\r
+                       case 'src' : // src は遷移して変化する, name も?\r
+                               if( this._tag !== 'IFRAME' ) break;\r
+                               if( this._newAttrs && X_Object_inObject( nameOrObj, this._newAttrs ) ) return this._newAttrs[ nameOrObj ];\r
+                               if( elm = X_UA_DOM.IE4 ? this._rawObject || X_Node__ie4getRawNode( this ) : this._rawObject ){\r
+                                       if( !attrs ) attrs = this._attrs = {};\r
+                                       return attrs[ nameOrObj ] = elm[ nameOrObj ]; // getAttribute( nameOrObj )?\r
+                               };\r
+                               break;\r
+                               \r
                        case 'selected' :\r
                                // kquery.js : safariのバグ対策\r
                                // if ($.browser.safari && key === "selected" && tmp) tmp.selectedIndex;\r
                                // 親ノードの selectedIndex の getter を呼んでおくと値が正しくなる、ということ?( by itozyun )\r
-                               if( X_UA.WebKit ) this._rawObject.parentNode.selectedIndex;\r
+                               if( X_UA[ 'WebKit' ] ) this._rawObject.parentNode && this._rawObject.parentNode.selectedIndex;\r
                        case 'value' :\r
+                               if( this._tag === 'INPUT' && X_Node_Attr_STATIC_VALUE_TYPES[ attrs[ 'type' ] ] ) break;\r
                        case 'checked' :\r
                        case 'disabled' :                       \r
                        case 'selectedIndex' :\r
@@ -162,10 +240,11 @@ function X_Node_Attr_setAttr( that, attrs, newAttrs, name, v ){
                        return;\r
                case 'id' :\r
                        v = ( v !== 'ie4uid' + that._uid ) ? v : undefined;\r
+                       // TODO unique の check\r
                        if( v !== that._id ){\r
                                that._id = v;\r
-                               that._dirty |= X_Node_Dirty.ID;\r
-                               that._root && X_Node_reserveUpdate();\r
+                               that._flags |= X_Node_State.DIRTY_ID;\r
+                               that._flags & X_Node_State.IN_TREE && X_Node_reserveUpdate();\r
                        };\r
                        return; \r
                case 'class' :\r
@@ -190,7 +269,7 @@ function X_Node_Attr_setAttr( that, attrs, newAttrs, name, v ){
        \r
        if( v == null ){\r
                newAttrs[ name ] = undefined;\r
-               if( attrs.hasOwnProperty( name ) ) delete attrs[ name ];\r
+               if( X_Object_inObject( name, attrs ) ) delete attrs[ name ];\r
        } else {\r
                newAttrs[ name ] = attrs[ name ] = v;\r
        };\r