X-Git-Url: http://git.sourceforge.jp/view?a=blobdiff_plain;f=0.6.x%2Fjs%2F01_core%2F02_XUA.js;h=426b486147c93e7d02c99ce8f51446adc056c76d;hb=a3d03e96ad8c0392ef683eb6c64421e094b96958;hp=1167846de09a7025814e813fcc023753ae5449e5;hpb=4533342ee7fb302488e76a207a129bb18866c00f;p=pettanr%2FclientJs.git diff --git a/0.6.x/js/01_core/02_XUA.js b/0.6.x/js/01_core/02_XUA.js index 1167846..426b486 100644 --- a/0.6.x/js/01_core/02_XUA.js +++ b/0.6.x/js/01_core/02_XUA.js @@ -2,12 +2,21 @@ // ------------------------------------------------------------------------- // // ------------ local variables -------------------------------------------- // // ------------------------------------------------------------------------- // -var X_UA = (function( n, undefined ){ - var acme = {}, - dua = n.userAgent, - dav = n.appVersion, + +/** + * UserAgent に関する定数を保持する。 + * @namespace X.UA + * @alias X.UA + * @type {object} + */ +var X_UA = X[ 'UA' ] = {}, + X_UA_classNameForHTML = 'js-enabled '; + +(function(){ + var dua = navigator.userAgent, + dav = navigator.appVersion, tv = parseFloat(dav), - sys = n.platform, + sys = navigator.platform, tridentToVer, i, j, v; console.log( ' userAgent : ' + dua ); @@ -23,267 +32,642 @@ var X_UA = (function( n, undefined ){ v = dua.split( 'OS ' )[ 1 ].split( '_' ); i = window.devicePixelRatio === 1; - - acme.iOSMajor = parseFloat( v[ 0 ] ) || 0; - acme.iOSMinor = parseFloat( v[ 1 ] ) || 0; - acme.iOSPatch = parseFloat( v[ 2 ] ) || 0; - - acme.iOS = acme.iOSMajor + acme.iOSMinor / 10; + /** + * @alias X.UA.iOSMajor + * @type {number} + */ + X_UA[ 'iOSMajor' ] = parseFloat( v[ 0 ] ) || 0; + /** + * @alias X.UA.iOSMinor + * @type {number} + */ + X_UA[ 'iOSMinor' ] = parseFloat( v[ 1 ] ) || 0; + /** + * @alias X.UA.iOSPatch + * @type {number} + */ + X_UA[ 'iOSPatch' ] = parseFloat( v[ 2 ] ) || 0; + /** + * @alias X.UA.iOS + * @type {number} + */ + X_UA[ 'iOS' ] = X_UA[ 'iOSMajor' ] + X_UA[ 'iOSMinor' ] / 10; if( screen.width === screen.height * 1.5 || screen.width * 1.5 === screen.height ){ v = true; // 4:3 model }; if( sys === 'iPhone' ){ - acme.iPhone = true; - if( v ) acme.iPhone_4s = true; // 4s以下 - if( v && i ) acme.iPhone_3GS = true; // 3GS以下 - alert( 'iPhone ' + ( acme.iPhone_3GS ? '3GS以下' : acme.iPhone_4s ? '4s以下' : '5以上' ) ); + /** + * @alias X.UA.iPhone + * @type {boolean} + */ + X_UA[ 'iPhone' ] = true; + if( v ){ + /** + * iPhone4s以下 + * @alias X.UA.iPhone_4s + * @type {boolean} + */ + X_UA[ 'iPhone_4s' ] = true; + + if( i ){ + /** + * iPhone3GS以下 + * @alias X.UA.iPhone_3GS + * @type {boolean} + */ + X_UA[ 'iPhone_3GS' ] = true; + }; + }; + + //alert( 'iPhone ' + ( X_UA[ 'iPhone_3GS' ] ? '3GS以下' : X_UA[ 'iPhone_4s' ] ? '4s以下' : '5以上' ) ); }; - if( sys === 'iPad' ){ - acme.iPad = true; - if( i ) acme.iPod_2Mini1 = true; + if( sys === 'iPad' ){ + /** + * @alias X.UA.iPad + * @type {boolean} + */ + X_UA[ 'iPad' ] = true; + if( i ){ + /** + * iPad2以下または初代iPad mini 以下 + * @alias X.UA.iPad_2Mini1 + * @type {boolean} + */ + X_UA[ 'iPad_2Mini1' ] = true; + }; }; - if( sys === 'iPod' ){ - acme.iPod = true; - if( v ) acme.iPod_4 = true; - if( v && i ) acme.iPod_3 = true; - alert( 'iPod touch ' + ( acme.iPod_3 ? '3以下' : acme.iPod_4 ? '4以下' : '5以上' ) ); + if( sys === 'iPod' ){ + /** + * @alias X.UA.iPod + * @type {boolean} + */ + X_UA[ 'iPod' ] = true; + + if( v ){ + /** + * iPod4以下 + * @alias X.UA.iPod_4 + * @type {boolean} + */ + X_UA[ 'iPod_4' ] = true; + + if( i ){ + /** + * iPod3以下 + * @alias X.UA.iPod_3 + * @type {boolean} + */ + X_UA[ 'iPod_3' ] = true; + }; + }; + //alert( 'iPod touch ' + ( X_UA[ 'iPod_3' ] ? '3以下' : X_UA[ 'iPod_4' ] ? '4以下' : '5以上' ) ); }; - console.log( '>> iOS : ' + acme.iOS ); + console.log( '>> iOS : ' + X_UA[ 'iOS' ] ); } else if( dua.indexOf( 'hp-tablet' ) !== -1 || dua.indexOf( 'webOS' ) !== -1 ){ - // http://user-agent-string.info/list-of-ua/os-detail?os=webOS - acme.webOS = true; // webOS + /** + * http://user-agent-string.info/list-of-ua/os-detail?os=webOS + * @alias X.UA.webOS + * @type {boolean} + */ + X_UA[ 'webOS' ] = true; // webOS } else if( sys.indexOf( 'Win' ) + 1 ){ - console.log( 'Win' ); - acme.Windows = true; - if( sys === 'Win16' ) acme.Win16 = true; - if( sys === 'Win32' ) acme.Win32 = true; - if( sys === 'Win64' ) acme.Win64 = true; - if( sys === 'WinCE' ) acme.WinCE = true; + + switch( sys ){ + case 'WinCE' : + /** + * @alias X.UA.WinCE + * @type {boolean} + */ + X_UA[ sys ] = true; + break; + case 'Win16' : + case 'Win32' : + case 'Win64' : + /** + * @alias X.UA.Win16 + * @alias X.UA.Win32 + * @alias X.UA.Win64 + * @type {boolean} + */ + X_UA[ sys ] = true; + + if( v = dav.split( 'Windows NT 10' )[ 1 ] ){ + switch( v.substr( 0, 2 ) ){ + case '.0' : v = 10; break; + default : v = '?'; + }; + } else + if( v = dav.split( 'Windows NT ' )[ 1 ] ){ + switch( v.substr( 0, 3 ) ){ + case '6.3' : v = 8.1; break; + case '6.2' : v = 8; break; + case '6.1' : v = 7; break; + case '6.0' : v = 'Vista'; break; + case '5.2' : v = '2003|XP64'; break; + case '5.1' : v = v.indexOf( '5.1; SV1' ) ? 'XP' : 'XPSP2'; break; + case '5.0' : v = v.indexOf( '5.01' ) ? 2000 : '2kSP1'; break; + case '4.0' : v = 'NT'; break; + default : v = '?'; + }; + } else + if( v = dav.split( 'Windows ' )[ 1 ] ){ + switch( v.substr( 0, 2 ) ){ + case '98' : v = v.indexOf( '98; Win 9x 4.90' ) ? '98|98SE' : 'ME'; break; + case '95' : v = 95; break; + case '3.' : v = parseFloat( v ); break; + default : v = '?'; + }; + } else { + v = '?'; + }; + + /** + * 10, 8.1, 8, 7, Vista, 2003|XP64, XPSP2, XP, 2kSP1, 2000, ME, 98|98SE, 95, ? + * @alias X.UA.Windows + * @type {number|string} + */ + X_UA[ 'Windows' ] = v; + break; + }; + + // winRT } else if( sys.indexOf( 'Mac' ) + 1 ){ console.log( 'Mac' ); - acme.Mac = true; - if( sys === 'MacPPC' || sys === 'MacPowerPC' ) acme.MacPPC = true; - if( sys === 'Mac68K' ) acme.Mac68K = true; - if( sys === 'MacIntel' ) acme.MacIntel = true; + /** + * @alias X.UA.Mac + * @type {boolean} + */ + X_UA[ 'Mac' ] = true; + switch( sys ){ + case 'MacPowerPC' : + /** + * @alias X.UA.MacPPC + * @type {boolean} + */ + X_UA[ 'MacPPC' ] = true; + break; + case 'MacPPC' : + case 'Mac68K' : + case 'MacIntel' : + /** + * @alias X.UA.MacPPC + * @alias X.UA.Mac68K + * @alias X.UA.MacIntel + * @type {boolean} + */ + X_UA[ sys ] = true; + }; } else - if( sys.indexOf( 'Linux' ) + 1 ){ + if( ( sys.indexOf( 'Linux' ) + 1 ) || ( sys.indexOf( 'Android' ) + 1 ) ){ console.log( 'Linux' ); - acme.Linux = true; + /** + * @alias X.UA.Linux + * @type {boolean} + */ + X_UA[ 'Linux' ] = true; - if( ( i = dua.indexOf( 'Android' ) ) !== -1 ){ - acme.Android = parseFloat( dua.substr( i + 8 ) ) || 0; - console.log( '>> Android : ' + acme.Android ); + if( dua.indexOf( 'Android ' ) !== -1 ){ + v = dua.split( 'Android ' )[ 1 ].split( '.' ); + /** + * @alias X.UA.AndroidMajor + * @type {number} + */ + X_UA[ 'AndroidMajor' ] = parseFloat( v[ 0 ] ) || 0; + /** + * @alias X.UA.AndroidMinor + * @type {number} + */ + X_UA[ 'AndroidMinor' ] = parseFloat( v[ 1 ] ) || 0; + /** + * @alias X.UA.AndroidPatch + * @type {number} + */ + X_UA[ 'AndroidPatch' ] = parseFloat( v[ 2 ] ) || 0; + /** + * Firefox で Version が取れない! + * http://bizmakoto.jp/bizid/articles/1207/31/news004.html + * Chrome Android 4.0以上 Google + * Dolphin Browser HD Android 2.0.1以上 Mobotap + * Firefox Android 2.2以上 Mozilla + * Opera Mobile Android 1.6以上 Opera Software ASA + * Sleipnir Mobile Android 2.1以上 Fenrir + * @alias X.UA.Android + * @type {number} + */ + X_UA[ 'Android' ] = X_UA[ 'AndroidMajor' ] + X_UA[ 'AndroidMinor' ] / 10; + console.log( '>> Android : ' + X_UA[ 'Android' ] ); }; }; - -/* - * http://bizmakoto.jp/bizid/articles/1207/31/news004.html -Chrome Android 4.0以上 Google -Dolphin Browser HD Android 2.0.1以上 Mobotap -Firefox Android 2.2以上 Mozilla -Opera Mobile Android 1.6以上 Opera Software ASA -Sleipnir Mobile Android 2.1以上 Fenrir - */ if( window.opera ){ i = dua.indexOf( 'Opera' ); // Opera/ j = dua.indexOf( 'Version/' ); - acme.Opera = Math.max( + /** + * @alias X.UA.Opera + * @type {number} + */ + X_UA[ 'Opera' ] = v = Math.max( i !== -1 ? parseFloat( dua.substr( i + 6 ) ) : 0, j !== -1 ? parseFloat( dua.substr( j + 8 ) ) : 0, tv ); - // closure compiler で minify するとOpera7で動かない - // --compilation_level WHITESPACE_ONLY --formatting pretty_print <- 動く - acme.Opera7 = acme.Opera < 8; - acme.Opera78 = acme.Opera < 9; - acme.OperaMini = 0 < dua.indexOf('Opera Mini'); - acme.OperaMobile = 0 < dua.indexOf('Opera Mobi'); - acme.OperaTablet = 0 < dua.indexOf('Opera Tablet'); - acme.Wii = dua.indexOf( 'Nintendo Wii' ) !== -1; - acme.NDS = dua.indexOf( 'Nitro' ) !== -1; - console.log( '>> Opera : ' + acme.Opera ); - return acme; + /** + * memo:closure compiler で minify するとOpera7で動かない + * --compilation_level WHITESPACE_ONLY --formatting pretty_print <- 動く + * @alias X.UA.Opera7 + * @type {boolean} + */ + X_UA[ 'Opera7' ] = v < 8; + /** + * @alias X.UA.Opera78 + * @type {boolean} + */ + X_UA[ 'Opera78' ] = v < 9; + /** + * @alias X.UA.OperaMini + * @type {boolean} + */ + X_UA[ 'OperaMini' ] = 0 < dua.indexOf('Opera Mini'); + /** + * @alias X.UA.OperaMobile + * @type {boolean} + */ + X_UA[ 'OperaMobile' ] = 0 < dua.indexOf('Opera Mobi'); + /** + * @alias X.UA.OperaTablet + * @type {boolean} + */ + X_UA[ 'OperaTablet' ] = 0 < dua.indexOf('Opera Tablet'); + /** + * @alias X.UA.Wii + * @type {boolean} + */ + X_UA[ 'Wii' ] = dua.indexOf( 'Nintendo Wii' ) !== -1; + /** + * @alias X.UA.NDS + * @type {boolean} + */ + X_UA[ 'NDS' ] = dua.indexOf( 'Nitro' ) !== -1; + console.log( '>> Opera : ' + v ); + return; }; // Mozilla/5.0 (Windows NT 6.3; Trident/7.0; rv:11.0) like Gecko if( document.all || dav.indexOf( 'Trident/' ) !== -1 ){ - acme.ActiveX = !!window[ 'ActiveXObject' ]; - acme._IE = parseFloat(dua.split('MSIE ')[1]) || parseFloat(dua.split('rv:')[1]) || parseFloat(dav.split('MSIE ')[1]) || 0; - // IE11 の互換モードの dav にも Trident/7.0 が書かれているため互換モードか?判定ができる - // 互換モードでは Silverlight でエラーが出る? - acme.Trident = parseFloat(dav.split('Trident/')[1]) || 0; - acme.IE = document.documentMode || acme._IE; - tridentToVer = acme.Trident ? ( acme.Trident + 4 | 0 ) : acme._IE; - acme.IECompat = /* acme.IE !== acme._IE || */ tridentToVer !== acme._IE && tridentToVer; - acme.IE4 = acme.IE && acme.IE < 5; - acme.IE5678 = 5 <= acme.IE && acme.IE < 9; - acme.IE5 = 5 <= acme.IE && acme.IE < 5.5; - acme.IE55 = 5.5 <= acme.IE && acme.IE < 6; - acme.IE5x = acme.IE5 || acme.IE55; - acme.IE6 = 6 <= acme.IE && acme.IE < 7; - acme.IE7 = 7 <= acme.IE && acme.IE < 8; - acme.IE8 = 8 <= acme.IE && acme.IE < 9; - acme.IE9 = 9 <= acme.IE && acme.IE < 10; - acme.MacIE = dua.indexOf( 'Mac_PowerPC' ) !== -1 || dua.indexOf( 'Mac_PPC' ) !== -1 || dua.indexOf( 'Mac_68K' ) !== -1; - acme.IEMobile = dua.toLowerCase().indexOf( 'iemobile' ) !== -1 || acme.WinCE; - acme.WinPhone = dua.toLowerCase().indexOf( 'windows phone' ) !== -1; - console.log( '>> IE : ' + acme.IE + ' ActiveX : ' + acme.ActiveX ); + /** + * @alias X.UA.ActiveX + * @type {boolean} + */ + X_UA[ 'ActiveX' ] = !!window[ 'ActiveXObject' ]; + /** + * documentモードを考慮しないIEのバージョン + * @alias X.UA._IE + * @type {number} + */ + X_UA[ '_IE' ] = parseFloat(dua.split('MSIE ')[1]) || parseFloat(dua.split('rv:')[1]) || parseFloat(dav.split('MSIE ')[1]) || 0; + /** + * IE11 の互換モードの navigator.appVersion にも Trident/7.0 が書かれているため互換モードか?判定ができる + * 互換モードでは Silverlight でエラーが出る? + * @alias X.UA.Trident + * @type {number} + */ + X_UA[ 'Trident' ] = parseFloat(dav.split('Trident/')[1]) || 0; + + tridentToVer = X_UA[ 'Trident' ] ? ( X_UA[ 'Trident' ] + 4 | 0 ) : X_UA[ '_IE' ]; + /** + * IE10 以上の互換モードを使用している場合、そのバージョン + * @alias X.UA.IEHost + * @type {number} + */ + if( tridentToVer !== X_UA[ '_IE' ] ) X_UA[ 'IEHost' ] = tridentToVer; + /** + * documentモードを考慮したIEのバージョン + * @alias X.UA.IE + * @type {number} + */ + X_UA[ 'IE' ] = v = document.documentMode || tridentToVer; + /** + * @alias X.UA.IE4 + * @type {boolean} + */ + if( v && v < 4.5 ) X_UA[ 'IE4' ] = true; + /** + * @alias X.UA.IE45 + * @type {boolean} + */ + if( v && 4.5 <= v && v < 5 ) X_UA[ 'IE45' ] = true; + /** + * @alias X.UA.IE4x + * @type {boolean} + */ + if( X_UA[ 'IE4' ] || X_UA[ 'IE45' ] ) X_UA[ 'IE4x' ] = true; + /** + * @alias X.UA.IE5 + * @type {boolean} + */ + if( 5 <= v && v < 5.5 ) X_UA[ 'IE5' ] = true; + /** + * @alias X.UA.IE55 + * @type {boolean} + */ + if( 5.5 <= v && v < 6 ) X_UA[ 'IE55' ] = true; + /** + * @alias X.UA.IE5x + * @type {boolean} + */ + if( X_UA[ 'IE5' ] || X_UA[ 'IE55' ] ) X_UA[ 'IE5x' ] = true; + /** + * @alias X.UA.IE6 + * @type {boolean} + */ + if( 6 <= v && v < 7 ) X_UA[ 'IE6' ] = true; + /** + * @alias X.UA.IE7 + * @type {boolean} + */ + if( 7 <= v && v < 8 ) X_UA[ 'IE7' ] = true; + /** + * @alias X.UA.IE8 + * @type {boolean} + */ + if( 8 <= v && v < 9 ) X_UA[ 'IE8' ] = true; + /** + * @alias X.UA.IE9 + * @type {boolean} + */ + if( 9 <= v && v < 10 ) X_UA[ 'IE9' ] = true; + /** + * @alias X.UA.MacIE + * @type {boolean} + */ + if( X_UA[ 'Mac' ] ) X_UA[ 'MacIE' ] = true; + /** + * @alias X.UA.IEMobile + * @type {boolean} + */ + if( dua.toLowerCase().indexOf( 'iemobile' ) !== -1 || X_UA[ 'WinCE' ] ) X_UA[ 'IEMobile' ] = true; + /** + * @alias X.UA.WinPhone + * @type {boolean} + */ + if( dua.toLowerCase().indexOf( 'windows phone' ) !== -1 || 0 < dav.indexOf( 'ZuneWP' ) ) X_UA[ 'WinPhone' ] = true; // ZuneWP は IEM のデスクトップモードで登場する + + console.log( '>> IE : ' + v + ' ActiveX : ' + X_UA[ 'ActiveX' ] + ' IEHost : ' + X_UA[ 'IEHost' ] ); // TODO XBox360, XBox1, Modern or Desktop, Standalone - return acme; + return; }; - // http://qa.support.sony.jp/solution/S0812181056444/common/nfb34_dom_200jp/dom_dom0_JP.html + // if( ( i = dua.indexOf( 'NetFront\/' ) !== -1 ) ){ - acme.NetFront = parseFloat( dua.substr( i + 9 ) ) || 0; - console.log( '>> NetFront : ' + acme.NetFront ); - return acme; + /** + * http://qa.support.sony.jp/solution/S0812181056444/common/nfb34_dom_200jp/dom_dom0_JP.html + * @alias X.UA.NetFront + * @type {number} + */ + X_UA[ 'NetFront' ] = parseFloat( dua.substr( i + 9 ) ) || 0.1; + console.log( '>> NetFront : ' + X_UA[ 'NetFront' ] ); + return; }; - if( acme.Linux && tv === 2 && dua.indexOf( 'Sony\/COM2\/' ) !== -1 ){ - acme.NetFront = 3.4; - console.log( '>> NetFront : ' + acme.NetFront ); - return acme; + if( X_UA[ 'Linux' ] && tv === 2 && dua.indexOf( 'Sony\/COM2\/' ) !== -1 ){ + X_UA[ 'NetFront' ] = 3.4; + console.log( '>> NetFront : ' + X_UA[ 'NetFront' ] ); + return; }; - // http://www.useragentstring.com/pages/Playstation%203/ - // Mozilla/5.0 (PLAYSTATION 3; 3.55) - // Mozilla/4.0 (PS3 (PlayStation 3); 1.00) if( ( i = dua.toUpperCase().indexOf( 'PLAYSTATION 3' ) !== -1 ) ){ - acme.PS3 = parseFloat( dua.substr( i + 15 ) ) || 0; - console.log( '>> PS3 : ' + acme.PS3 ); - return acme; + /** + * PlayStation 3 システムバージョン 4.10 未満の SONY 独自ブラウザ + * http://www.useragentstring.com/pages/Playstation%203/ + * Mozilla/5.0 (PLAYSTATION 3; 3.55) + * Mozilla/4.0 (PS3 (PlayStation 3); 1.00) + * https://github.com/Famous/famous/blob/1a02c8084587d80519ea4bd3b55649ab32ee2e65/examples/assets/lib/require.js + * PS3 ブラウザのロードイベントについて + * @alias X.UA.PS3 + * @type {number} + */ + X_UA[ 'PS3' ] = parseFloat( dua.substr( i + 15 ) ) || 0.1; + console.log( '>> PS3 : ' + X_UA[ 'PS3' ] ); + return; }; - // http://www.useragentstring.com/pages/iCab/ - // iCab/3.0.2 (Macintosh; U; PPC Mac OS X) - // Mozilla/5.0 (Macintosh; U; PPC Mac OS; en) iCab 3 if( ( i = dua.indexOf( 'iCab' ) !== -1 ) ){ - acme.iCab = parseFloat( dua.substr( i + 5 ) ) || 0; - console.log( '>> iCab : ' + acme.iCab ); - return acme; + /** + * http://www.useragentstring.com/pages/iCab/ + * iCab/3.0.2 (Macintosh; U; PPC Mac OS X) + * Mozilla/5.0 (Macintosh; U; PPC Mac OS; en) iCab 3 + * @alias X.UA.iCab + * @type {number} + */ + X_UA[ 'iCab' ] = parseFloat( dua.substr( i + 5 ) ) || 0.1; + console.log( '>> iCab : ' + X_UA[ 'iCab' ] ); + return; }; if( 0 < dua.indexOf( 'Gecko\/' ) && ( i = dua.indexOf( 'rv:' ) ) ){ v = dua.substr( i + 3 ).split( '.' ); - acme.Gecko = parseFloat( v[ 0 ] ) || 0 + + /** + * メジャーバージョン + マイナーバージョン + * @alias X.UA.Gecko + * @type {number} + */ + X_UA[ 'Gecko' ] = parseFloat( v[ 0 ] ) || 0 + ( parseFloat( v[ 1 ] ) || 0 ) / 10 + ( parseFloat( v[ 2 ] ) || 0 ) / 100; - - acme.GeckoMajor = parseFloat( v[ 0 ] ) || 0; - acme.GeckoMinor = parseFloat( v[ 1 ] ) || 0; - acme.GeckoPatch = parseFloat( v[ 2 ] ) || 0; + /** + * @alias X.UA.GeckoMajor + * @type {number} + */ + X_UA[ 'GeckoMajor' ] = parseFloat( v[ 0 ] ) || 0; + /** + * @alias X.UA.GeckoMinor + * @type {number} + */ + X_UA[ 'GeckoMinor' ] = parseFloat( v[ 1 ] ) || 0; + /** + * @alias X.UA.GeckoPatch + * @type {number} + */ + X_UA[ 'GeckoPatch' ] = parseFloat( v[ 2 ] ) || 0; //Fennec - // Mozilla/5.0 (Android; Linux armv7l; rv:9.0) Gecko/20111216 Firefox/9.0 Fennec/9.0 if( ( i = dua.indexOf( 'Fennec/' ) ) !== -1 ){ - acme.Fennec = parseFloat( dua.substr( i + 7 ) ); - console.log( '>> Fennec : ' + acme.Fennec + ', Gecko : ' + acme.Gecko ); - return acme; + /** + * Mozilla/5.0 (Android; Linux armv7l; rv:9.0) Gecko/20111216 Firefox/9.0 Fennec/9.0 + * @alias X.UA.Fennec + * @type {number} + */ + X_UA[ 'Fennec' ] = parseFloat( dua.substr( i + 7 ) ); + console.log( '>> Fennec : ' + X_UA[ 'Fennec' ] + ', Gecko : ' + X_UA[ 'Gecko' ] ); + return; }; //Firefox //Netscape //Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:0.9.4.1) Gecko/20020508 Netscape6/6.2.3 if( ( i = dua.indexOf( 'Netscape6/' ) ) !== -1 ){ - acme.NN = parseFloat( dua.substr( i + 10 ) ) || 6; - acme.NN6 = true; - console.log( '>> NN : ' + acme.NN + ', Gecko : ' + acme.Gecko ); - return acme; + /** + * @alias X.UA.NN + * @type {number} + */ + X_UA[ 'NN' ] = parseFloat( dua.substr( i + 10 ) ) || 6; + /** + * @alias X.UA.NN6 + * @type {boolean} + */ + X_UA[ 'NN6' ] = true; + console.log( '>> NN : ' + X_UA[ 'NN' ] + ', Gecko : ' + X_UA[ 'Gecko' ] ); + return; } else //Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.7.2) Gecko/20040804 Netscape/7.2 (ax) //Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.7.5) Gecko/20070321 Netscape/8.1.3 if( ( i = dua.indexOf( 'Netscape/' ) ) !== -1 ){ - acme.NN = parseFloat( dua.substr( i + 9 ) ) || 7; - console.log( '>> NN : ' + acme.NN + ', Gecko : ' + acme.Gecko ); - return acme; + X_UA[ 'NN' ] = parseFloat( dua.substr( i + 9 ) ) || 7; + console.log( '>> NN : ' + X_UA[ 'NN' ] + ', Gecko : ' + X_UA[ 'Gecko' ] ); + return; } else //Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.8.1.12) Gecko/20080219 Firefox/2.0.0.12 Navigator/9.0.0.6 if( ( i = dua.indexOf( 'Navigator/' ) ) !== -1 ){ - acme.NN = parseFloat( dua.substr( i + 10 ) ) || 9; - console.log( '>> NN : ' + acme.NN + ', Gecko : ' + acme.Gecko ); - return acme; + X_UA[ 'NN' ] = parseFloat( dua.substr( i + 10 ) ) || 9; + console.log( '>> NN : ' + X_UA[ 'NN' ] + ', Gecko : ' + X_UA[ 'Gecko' ] ); + return; }; - console.log( '>> Gecko : ' + acme.Gecko ); + console.log( '>> Gecko : ' + X_UA[ 'Gecko' ] ); }; + if( ( dua.indexOf( 'Linux; U; Android ' ) !== -1 || dua.indexOf( 'Linux; Android ' ) !== -1 ) && + ( dua.indexOf( 'Chrome\/' ) === -1 || dua.indexOf( 'Version\/' ) !== -1 ) ){ // Chrome/ を含まない または Version/ を含む + + /* if( window.chrome ){ // Android3.1 のAOSPブラウザで .chrome がいた、、、 + //X_UA[ 'Blink' ] = X_UA[ 'ChromeWK' ] = tv; + } else */ + if( dua.indexOf( 'Version\/' ) === -1 && ( v = parseFloat( dua.split( 'Chrome\/' )[ 1 ] ) ) ){ + /** + * Android 標準ブラウザ Chrome Webkit ラップブラウザ + * @alias X.UA.ChromeWK + * @type {number} + */ + X_UA[ 'ChromeWK' ] = v; + } else + // http://uupaa.hatenablog.com/entry/2014/04/15/163346 + // Chrome WebView は Android 4.4 の時点では WebGL や WebAudio など一部の機能が利用できません(can i use)。 + // また UserAgent が書き換え可能なため、旧来のAOSPブラウザの UserAgent を偽装した形で配布されているケースがあります。 + // http://caniuse.com/#compare=chrome+40,android+4.2-4.3,android+4.4,android+4.4.3-4.4.4,and_chr+45 + // CustomElement の有無で判定 + if( document[ 'registerElement' ] ){ + // UA が偽装された ChromeWK + X_UA[ 'ChromeWK' ] = tv; + alert( 'UA が偽装された Chrome Webkit' ); + } else { + /** + * Android 標準ブラウザ AOSP + * @alias X.UA.AOSP + * @type {number} + */ + X_UA[ 'AOSP' ] = X_UA[ 'Android' ]; + }; + + i = parseFloat( dua.split( 'WebKit\/' )[ 1 ] ); + /** + * @alias X.UA.AndroidWebkit + * @type {number} + */ + X_UA[ 'AndroidWebkit' ] = i; + //alert( 'AudioSprite調査:Android標準ブラウザ Webkit Version ' + i ); + + /* + * http://www.flexfirm.jp/blog/article/402 + * Sブラウザ + * SC-04E、SC-01F、SC-02F、 SC-04F、SCL22、SCL23など + */ + } else + // TODO Blink if( window.chrome ){ - acme.Blink = tv; - console.log( '>>Blink : ' + acme.Blink ); + /** + * @alias X.UA.Blink + * @type {number} + */ + X_UA[ 'Blink' ] = parseFloat( dua.split( 'Chrome/' )[ 1 ] ); + + if( v = parseFloat( dua.split( 'OPR/' )[ 1 ] ) ){ + /** + * @alias X.UA.BlinkOpera + * @type {number} + */ + X_UA[ 'BlinkOpera' ] = v; + }; + console.log( '>>Blink : ' + X_UA[ 'Blink' ] ); } else if( dav.indexOf( 'Konqueror' ) !== -1 ){ - acme.Khtml = tv; - console.log( '>>Khtml : ' + acme.Khtml ); + /** + * @alias X.UA.Khtml + * @type {number} + */ + X_UA[ 'Khtml' ] = tv; + console.log( '>>Khtml : ' + X_UA[ 'Khtml' ] ); } else - if( ( i = dua.indexOf( 'Android ' ) ) !== -1 ){ - acme.AndroidBrowser = i = parseFloat( dua.substr( i + 8 ) ) || 0; - acme.AndroidBrowser1 = 1 <= i && i < 2; - acme.AndroidBrowser2 = 2 <= i && i < 3; - acme.AndroidBrowser3 = 3 <= i && i < 4; - acme.AndroidBrowser4 = 4 <= i && i < 5; - acme.AndroidBrowser5 = 5 <= i && i < 6; - console.log( '>> AndroidBrowser : ' + acme.Android ); - } else + if( i = parseFloat(dua.split('WebKit\/')[1]) ){ - acme.WebKit = i; + /** + * @alias X.UA.WebKit + * @type {number} + */ + X_UA[ 'WebKit' ] = i; - acme.Chrome = parseFloat(dua.split('Chrome\/')[1]) || undefined; + if( v = parseFloat(dua.split('Chrome\/')[1]) ){ + /** + * @alias X.UA.Chrome + * @type {number} + */ + X_UA[ 'Chrome' ] = v; + }; // TODO webkit Opera - console.log( '>>Webkit : ' + acme.WebKit ); + console.log( '>>Webkit : ' + X_UA[ 'WebKit' ] ); - if( i && !acme.Chrome && dua.indexOf( 'Safari' ) !== -1 ){ + if( i && !X_UA[ 'Chrome' ] && dua.indexOf( 'Safari' ) !== -1 ){ if( dav.indexOf( 'Version/' ) !== -1 ){ - acme.Safari = parseFloat( dav.split('Version/')[1] ); + /** + * @alias X.UA.Safari + * @type {number} + */ + X_UA[ 'Safari' ] = parseFloat( dav.split('Version/')[1] ); } else { - if( i < 73 ){ - acme.Safari = 0.8; - } else - if( i < 85 ){ - acme.Safari = 0.9; - } else - if( i < 100 ){ - acme.Safari = 1; - } else - if( i < 125 ){ - acme.Safari = 1.1; - } else - if( i < 312 ){ - acme.Safari = 1.2; - } else - if( i < 412 ){ - acme.Safari = 1.3; - } else - if( i <= 419.3 ){ - acme.Safari = 2; - } else - if( i <= 525.13 ){ - acme.Safari = 3; - } else - if( i <= 525.25 ){ - acme.Safari = 3.1; - } else if( i <= 528.16 ){ - acme.Safari = 3.2; + X_UA[ 'Safari' ] = i < 73 ? 0.8 : + i < 85 ? 0.9 : + i < 100 ? 1 : + i < 125 ? 1.1 : + i < 312 ? 1.2 : + i < 412 ? 1.3 : + i <= 419.3 ? 2 : + i <= 525.13 ? 3 : + i <= 525.25 ? 3.1 : 3.2; }; }; }; - console.log( '>> Webkit : ' + acme.WebKit + ' Safari : ' + acme.Safari ); + console.log( '>> Webkit : ' + X_UA[ 'WebKit' ] + ' Safari : ' + X_UA[ 'Safari' ] ); }; - //http://www.useragentstring.com/pages/Iris/ - if( dua.toLowerCase().indexOf( 'iris' ) !== -1 ) acme.Iris = true; + if( dua.toLowerCase().indexOf( 'iris' ) !== -1 ){ + /** + * http://www.useragentstring.com/pages/Iris/ + * @alias X.UA.Iris + * @type {boolean} + */ + X_UA[ 'Iris' ] = true; + }; if( // Kobo Mozilla/5.0 (Linux; U; Android 2.0; en-us;) AppleWebKit/533.1 (KHTML, like Gecko) Verson/4.0 Mobile Safari/533.1 (Kobo Touch) dua.indexOf( 'Kobo' ) !== -1 || @@ -291,25 +675,68 @@ Sleipnir Mobile Android 2.1以上 Fenrir dua.indexOf( 'Kindle' ) !== -1 || // Sony Reader Mozilla/5.0 (Linux; U; ja-jp; EBRD1101; EXT) AppleWebKit/533.1 (KHTML, like Gecko) Version/4.0 Mobile Safari/533.1 dua.indexOf( 'EBRD' ) !== -1 - ) acme.EInk = true; - - return acme; -})( navigator ), + ){ + /** + * Kobo, Kindle, Sony Reader + * @alias X.UA.EInk + * @type {boolean} + */ + X_UA[ 'EInk' ] = true; + }; +})(); + +(function(){ + var k, v; + if( X_UA[ 'IE45' ] || X_UA[ 'IE4' ] ){ + if( X_UA[ 'Mac' ] ){ + X_UA_classNameForHTML = 'Mac'; + } else + if( X_UA[ 'WinCE' ] ){ + // TODO CE3 の ie4 と WM の ie4 の分岐 + X_UA_classNameForHTML = 'WinCE'; + } else + if( X_UA[ 'Win' ] ){ + X_UA_classNameForHTML = 'Win'; + } else { + X_UA_classNameForHTML = 'Other'; + }; + + X_UA_classNameForHTML += 'IE4'; + + if( X_UA[ 'IE45' ] ){ + X_UA_classNameForHTML += '5'; + }; + + if( X_UA[ 'ActiveX' ] ){ + X_UA_classNameForHTML += 'ActiveX'; + }; + + } else { + for( k in X_UA ){ + v = X_UA[ k ]; + if( v ){ + X_UA_classNameForHTML += k + ' '; + if( v !== true ){ + X_UA_classNameForHTML += k + v + ' '; + }; + }; + }; + }; +})(); -X_UA_DOM = {}, -X_UA_EVENT = {}, -X_UA_HID = {}; -//X_UA.IECompat && alert( X_UA.IE + ' ' + X_UA._IE + ' Tri:' + X_UA.Trident ); +var X_UA_DOM = {}, + X_UA_EVENT = {}, + X_UA_HID = {}; /* * http://d.hatena.ne.jp/t-uchima/20051003/p1 * MacIEにはattachEventが一応あるけどwindow.attachEventとdocument.attachEventしかなく他の要素にはattachEventはない。 */ -if( X_UA.IE4 && X_UA.IE < 5 ){ // ie4 & iemobi4 & macie4.x +if( X_UA[ 'IE4' ] && X_UA[ 'IE' ] < 5 ){ // ie4 & iemobi4 & macie4.x X_UA_DOM.IE4 = true; X_UA_EVENT.IE4 = true; } else -if( X_UA.MacIE ){ +if( X_UA[ 'MacIE' ] ){ X_UA_DOM.W3C = true; X_UA_EVENT.IE = true; } else @@ -323,22 +750,17 @@ if( document.getElementById ){ } else { X_UA_EVENT.DOM0 = true; }; -} else -if( document.layers ){ - -} else { - }; if( navigator.msPointerEnabled || navigator.pointerEnabled ) X_UA_HID.POINTER = true; if( !X_UA_HID.POINTER && window.ontouchstart !== undefined ) X_UA_HID.TOUCH = true; -//alert(X_UA.Safari + ' ' + X_UA.WebKit + '\n\n' + navigator.userAgent + '\n\n' + navigator.appVersion + '\n\n' + navigator.platform ); +//alert(X_UA[ 'Safari' ] + ' ' + X_UA[ 'WebKit' ] + '\n\n' + navigator.userAgent + '\n\n' + navigator.appVersion + '\n\n' + navigator.platform ); // Safari 3.1 未満は開発コンソールがない! // http://shimax.cocolog-nifty.com/search/2006/09/safarijavascrip_c54d.html /* -if( X_UA.Safari && X_UA.WebKit < 525.13 ){ +if( X_UA[ 'Safari' ] && X_UA[ 'WebKit' ] < 525.13 ){ window.onerror = function( x, y, z ){ var n = String.fromCharCode( 10 ); alert('window.onerrorによるエラーの捕捉:' + n + x + n + y + 'の' + z + '行目付近です。'); @@ -346,21 +768,32 @@ if( X_UA.Safari && X_UA.WebKit < 525.13 ){ }; };*/ -// ------------------------------------------------------------------------- // -// --- interface ----------------------------------------------------------- // -// ------------------------------------------------------------------------- // -X.UA = X_UA; - - +// TODO 構文のサポート instanceof, in, try catch -if( X_UA.IE < 7 ){ // error @ NN7.2 - // bonus: hotfix for IE6 SP1 (bug KB823727) - // multipleIEs IE6 standalone 版では不可, IE5.5 は可,,, - X_UA.IE4 || X_UA.MacIE ? +if( X_UA[ 'IE' ] < 7 ){ // error @ NN7.2 + X_UA[ 'IE4' ] || X_UA[ 'MacIE' ] ? document.execCommand && document.execCommand( 'BackgroundImageCache', false, true ) : (function(){ - X_UA.IE_EXECOM_ERROR = eval( 'var a=1;try{document.execCommand&&document.execCommand("BackgroundImageCache",!1,!0)}catch(e){a=0}!a' ); + /** + * ie7 以下で実行する document.execCommand( 'BackgroundImageCache', false, true ) の失敗。 + * bonus: hotfix for IE6 SP1 (bug KB823727) + * multipleIEs IE6 standalone 版では不可, IE5.5 は可,,, + * @alias X.UA.ieExeComError */ + X_UA[ 'ieExeComError' ] = eval( 'var a=1;try{document.execCommand&&document.execCommand("BackgroundImageCache",!1,!0)}catch(e){a=0}!a' ); })(); - X_UA.IE_EXECOM_ERROR && alert( 'document.execCommand error!' ); + //X_UA[ 'ieExeComError' ] && alert( 'document.execCommand error!' ); }; + +/* + * HTML5 に対応しない IE8 以下でも の下に
を作ることができる + * その際に
の直前に改行文字が出現するが childNodes は長さ 1 で
だけの模様、、、 +X_UA_ATagWrapDiv = (function( e, h ){ + e = document.createElement( 'div' ); + e.innerHTML = h = '
'; + console.log( e.innerHTML.length + '\n' + e.firstChild.tagName ); + return e.childNodes.length === 1; +})(); + +console.log( 'HTML5? ' + X_UA_ATagWrapDiv ); */ +