1 //{+xhr"XHR,XDR,MSXMLによる通信"(XMLHTTPRequest, XDomainRequest, ActiveX-MSXML を使った通信)[+net]
\r
3 // https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest
\r
4 // https://web.archive.org/web/20071101021832/http://web.paulownia.jp/script/ajax/xmlhttp4.html
\r
5 // https://web.archive.org/web/20091029170015/http://wiki.paulownia.jp/ajax/xmlhttprequest
\r
7 * http://ponpon-village.net/ajax/xmlhttp.htm
\r
8 * IE のバージョンによっては、ActiveXObject("Msxml2.XMLHTTP.5.0") , ActiveXObject("Msxml2.XMLHTTP.4.0") ,
\r
9 ActiveXObject("Msxml2.XMLHTTP.3.0") , ActiveXObject("Msxml2.XMLHTTP") なども使用出来る。
\r
11 http://vird2002.s8.xrea.com/javascript/XMLHttpRequest.html
\r
13 new ActiveXObject( 'Msxml2.XMLHTTP.3.0' ); // バージョン3.0 広範に利用されているので、今後も bugfix を行う
\r
14 new ActiveXObject( 'Msxml2.XMLHTTP.6.0' ); // バージョン6.0 は最新版なので bugfix を続ける
\r
16 // --- 使うべきではないオブジェクト
\r
17 new ActiveXObject( 'Microsoft.XMLHTTP' ); // Microsoft接頭辞は古いので指定すべきではない
\r
18 new ActiveXObject( 'Msxml.XMLHTTP' ); // Msxml2接頭辞を指定すべき
\r
19 new ActiveXObject( 'Msxml2.XMLHTTP' ); // バージョンを省略すると 3.0 として扱われるので、バージョンは明記すべき
\r
20 new ActiveXObject( 'Msxml2.XMLHTTP.4.0' ); // バージョン4.0 は bugfix が行われないので、3.0 か 6.0 を指定すべき
\r
21 new ActiveXObject( 'Msxml2.XMLHTTP.5.0' ); // バージョン5.0 は bugfix が行われないので、3.0 か 6.0 を指定すべき
\r
23 [IE][Javascript][Json] IE+Jsonではまった
\r
24 http://d.hatena.ne.jp/khiker/20081026/javascript_json
\r
25 > AddCharset utf-8 json
\r
26 > AddType text/javascript json
\r
28 JavaScriptでJSONをeval
\r
29 http://d.hatena.ne.jp/sshi/20060904/p1
\r
31 itozyun 2014-10-30 20:55:41
\r
32 basic 認証のかかったhtml を表示して、そのjsが xhr をすると Android1.6 では 401 error が返る。Android 2.3 では解決している。
\r
33 Android1.6- の XHR で 401 エラーが返った場合は、iframe に xml を表示させてその内容を取ればサーバ側の対応無しでいけるかも?
\r
34 Android2 にも xdomain な GET が一回しかできない問題 gears 使えない?
\r
36 IE9 で 画像バイナリの取得 VBA をかましている
\r
37 http://web.archive.org/web/20130808105151/http://gurimmer.lolipop.jp/daihakken/2012/05/22/javascriptajaxxmlhttprequest%E3%82%92%E4%BD%BF%E3%81%A3%E3%81%9Fajax%E3%81%AE%E3%82%A8%E3%83%B3%E3%82%B3%E3%83%BC%E3%83%89
\r
38 http://d.hatena.ne.jp/maachang/20130221/1361427565
\r
40 http://web.archive.org/web/20130531162446/http://gurimmer.lolipop.jp/daihakken/2012/06/25/ajaxjavascript%E3%83%8D%E3%82%A4%E3%83%86%E3%82%A3%E3%83%96xmlhttp%E3%82%B5%E3%83%9D%E3%83%BC%E3%83%88%E3%81%A8%E3%81%AF/
\r
42 IE8 以下で xhr の失敗率が高い問題
\r
43 http://tkengo-totoro.blogspot.jp/2011/11/iexmlhttprequest.html
\r
44 TODO クライアント側にもリトライ機構を入れてみる
\r
47 TODO xml の取得には XMLDOM を使用する
\r
48 var activex = JKL.ParseXML.HTTP.ACTIVEX_XMLHTTP; // IXMLHttpRequest
\r
49 if ( this.method == "GET" && ! this.textmode ) {
\r
50 // use IXMLDOMElement to accept any mime types
\r
51 // because overrideMimeType() is not available on IE6
\r
52 activex = JKL.ParseXML.HTTP.ACTIVEX_XMLDOM; // IXMLDOMElement
\r
54 // debug.print( "new ActiveXObject( '"+activex+"' )" );
\r
55 this.req = new ActiveXObject( activex );
\r
58 var // Opera7.6+, Safari1.2+, khtml3.?+, Gecko0.9.7+
\r
59 // ie9- ではローカルリソースには MSXML を使う
\r
60 X_XHR_createW3C = window[ 'XMLHttpRequest' ] && function(){ return X_XHR_w3c || ( X_XHR_w3c = new XMLHttpRequest() ); },
\r
61 X_XHR_w3c = X_XHR_createW3C && X_XHR_createW3C(),
\r
62 X_XHR_cors = X_XHR_w3c && X_XHR_w3c.withCredentials !== undefined,
\r
63 X_XHR_progress = X_XHR_w3c && X_XHR_w3c.onprogress !== undefined,
\r
64 X_XHR_upload = X_XHR_w3c && !!X_XHR_w3c.upload,
\r
66 X_XHR_createXDR = window[ 'XDomainRequest' ] && function(){ return X_XHR_xdr || ( X_XHR_xdr = new XDomainRequest() ); },
\r
67 X_XHR_xdr = X_XHR_createXDR && X_XHR_createXDR(),
\r
70 X_XHR_msXMLName = '',
\r
73 // ie11の互換モード(7,8)の msxml はいまいち動かない
\r
74 X_XHR_createMSXML = X_UA[ 'ActiveX' ] && ( X_UA[ 'IE5x' ] || X_UA[ 'IE6' ] || X_URL_IS_LOCAL ) &&
\r
75 function(){ return X_Script_createActiveXObjectSafty( X_XHR_msXMLName ); },
\r
79 X_XHR_neverReuse = X_UA[ 'IE' ] < 9, // ie7,8 の xhr はリユース不可。msxml はリユース可能。
\r
81 X_XHR_TYPE_FLASH = 8,
\r
82 X_XHR_TYPE_GADGET = 16;
\r
84 if( X_XHR_createMSXML ){
\r
88 n = [ m + ".6.0", m + ".3.0", m + ".5.0", m + ".4.0", m, "Microsoft" + x ],
\r
89 v = [ 6, 3, 5, 4, 2, 1 ],
\r
93 a = X_Script_createActiveXObjectSafty( n[ ++i ] );
\r
95 X_XHR_msXMLVer = v[ i ];
\r
96 X_XHR_msXMLName = n[ i ];
\r
101 X_XHR_createMSXML = null;
\r
107 'W3C' : X_XHR_createW3C ? 1 : 0,
\r
108 'MSXML' : X_XHR_createMSXML ? 2 : 0,
\r
109 'XDR' : X_XHR_createXDR ? 4 : 0,
\r
112 * http://hakuhin.jp/as/import.html
\r
113 * ファイルの読み込みについて(4 or 5 or 6+)
\r
114 * http://hakuhin.jp/as/javascript.html
\r
115 * Flash から JavaScript にアクセスする(3+)
\r
117 'FLASH' : 4 <= X_Plugin_FLASH_VERSION ? 8 : 0,
\r
119 'GADGET' : 5.5 <= X_UA[ 'IE' ] || !X_UA[ 'IE' ] ? 16 : 0,
\r
122 * https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest
\r
123 * Progress Events Chrome7, firefox3.5, ie10, opera12, Safari?, Chrome for Android 0.16
\r
125 'PROGRESS' : X_XHR_progress,
\r
127 'UPLOAD_PROGRESS' : X_XHR_upload,
\r
129 // or gadget proxy or flash
\r
130 'CORS' : X_XHR_xdr || X_XHR_cors,
\r
132 'BINARY' : X_Script_VBS_ENABLED
\r
135 if( X_XHR_msXMLVer ) X[ 'XHR' ][ 'MSXML_VERSION' ] = X_XHR_msXMLVer;
\r
137 if( X_XHR_w3c || X_XHR_msXML ){
\r
139 X_TEMP.X_XHR_init = function(){
\r
140 X_XHR = X_Class_override( X_EventDispatcher(), X_TEMP.X_XHR_params, true );
\r
142 delete X_TEMP.X_XHR_init;
\r
143 delete X_TEMP.X_XHR_params;
\r
148 X_TEMP.X_XHR_params = {
\r
150 '_rawType' : X_EventDispatcher_EVENT_TARGET_XHR,
\r
163 load : function( obj ){
\r
164 var raw = X_XHR[ '_rawObject' ],
\r
165 method = obj[ 'method' ],
\r
166 url = obj[ 'url' ],
\r
167 async = obj[ 'async' ] !== false,
\r
168 username = obj[ 'username' ],
\r
169 password = obj[ 'password' ],
\r
170 headers = obj[ 'headers' ] || {},
\r
171 postdata = obj[ 'postdata' ] || '',
\r
172 timeout = obj[ 'timeout' ] || 20000,
\r
173 noCache = obj[ 'cache' ] !== true,
\r
174 dataType = X_XHR._dataType = obj[ 'dataType' ],
\r
175 xDomain = !X_URL_isSameDomain( url ),
\r
176 isFile = X_URL_isLocal( url ),
\r
180 if( !raw || xDomain !== X_XHR._isXDR || ( X_XHR_createMSXML && isFile !== X_XHR._isMsXML ) ){
\r
181 raw && X_XHR[ 'unlisten' ]( [ 'load', 'readystatechange', 'progress', 'error', 'timeout' ] );
\r
183 X_XHR[ '_rawObject' ] = raw = xDomain ?
\r
185 X_XHR_createW3C() :
\r
189 ( X_XHR_createMSXML ?
\r
190 ( X_XHR_msXML = X_XHR_msXML || X_XHR_createMSXML() ):
\r
194 X_XHR_createW3C() :
\r
195 ( X_XHR_msXML = X_XHR_msXML || X_XHR_createMSXML() );
\r
197 // raw === XDR これは error になるのでフラグに控える
\r
198 X_XHR._isXDR = X_XHR_createXDR && xDomain;
\r
199 X_XHR._isMsXML = !X_XHR_createW3C || ( isFile && X_XHR_createMSXML );
\r
202 raw.open( method, url, async, username, password );
\r
204 if( raw.responseType !== undefined ){
\r
205 switch( dataType ){
\r
209 raw.responseType = 'text';
\r
211 case 'json' : // firefox9- は moz-json
\r
212 raw.responseType = X_UA[ 'Gecko' ] < 10 ? 'moz-json' : X_UA[ 'Gecko' ] ? dataType : ''; // Iron 37 でエラー
\r
219 raw.responseType = 'document';
\r
222 case 'arraybuffer' :
\r
223 // jpeg,jpg,png,gif,mp3,ogg...
\r
224 raw.responseType = dataType;
\r
229 // http://www.quirksmode.org/blog/archives/2005/09/xmlhttp_notes_r_1.html
\r
230 if( !X_XHR._isMsXML && raw.overrideMimeType ){
\r
231 type = X_URL_getEXT( url ) || dataType;
\r
239 tmp = 'application/json';
\r
244 type = type || 'webm';
\r
246 type = type || 'ogg';
\r
250 tmp = 'audio/' + type;
\r
254 tmp = 'audio/x-' + type;
\r
262 tmp = 'text/plain; charset=x-user-defined';
\r
266 if( tmp = obj[ 'mimeType' ] || tmp ) raw.overrideMimeType( tmp );
\r
269 if( !X_XHR._isXDR && ( X_XHR._isMsXML ? 3 <= X_XHR_msXMLVer : raw.setRequestHeader ) ){ // msxml は setRequestHeader getter がいけない
\r
273 headers[ 'Pragma' ] = 'no-cache';
\r
274 headers[ 'Cache-Control' ] = 'no-cache';
\r
275 headers[ 'If-Modified-Since' ] = 'Thu, 01 Jun 1970 00:00:00 GMT';
\r
278 // http://nakigao.sitemix.jp/blog/?p=2040
\r
279 // json 取得時に SafariでHTTP/412のエラー。但し相手が audio の場合この指定があるとロードに失敗する。 iOS8.2, iOS7.1 では遭遇せず
\r
280 if( dataType === 'json' ){
\r
281 headers[ 'If-Modified-Since' ] = ( new Date ).toUTCString();
\r
284 // http://boscono.hatenablog.com/entry/2013/12/23/152851
\r
285 if ( !xDomain && !headers[ 'X-Requested-With' ] ) {
\r
286 headers[ 'X-Requested-With' ] = 'XMLHttpRequest';
\r
289 if( method === 'POST' && !headers[ 'Content-Type' ] ){
\r
290 headers[ 'Content-Type' ] = 'application/x-www-form-urlencoded';
\r
294 for( p in headers ){
\r
295 //if( X_EMPTY_OBJECT[ p ] ) continue;
\r
296 //console.log( headers[ p ] );
\r
297 headers[ p ] !== undefined && raw.setRequestHeader( p, headers[ p ] + '' ); // Opera8.01+, MSXML3+
\r
301 if( !X_XHR._isMsXML && raw.timeout !== undefined ){
\r
302 raw.timeout = timeout;
\r
304 X_XHR._timerID = X_Timer_once( timeout, X_XHR.onTimeout );
\r
307 // send 前にフラグを立てる,回線が早いと raw.send() 内で onload -> _busy = false ののち、 _busy = true するため。
\r
308 X_XHR._busy = true;
\r
310 raw.send( X_Type_isString( postdata ) ? postdata : X_String_serialize( postdata ) );
\r
312 if( !async || raw.readyState === 4 ){
\r
313 X_Timer_once( 32, X_XHR, [ { type : 'readystatechange' } ] );
\r
316 if( X_XHR._isMsXML ){
\r
317 raw[ 'onreadystatechange' ] = X_XHR.handleEvent;
\r
319 if( X_XHR_progress || X_XHR._isXDR ){
\r
320 X_XHR[ 'listen' ]( [ 'load', 'progress', 'error', 'timeout' ] ); //, 'abort'
\r
322 if( X_UA[ 'IE8' ] ){
\r
323 X_XHR[ 'listen' ]( [ 'readystatechange', 'error', 'timeout' ] );
\r
325 if( X_UA[ 'IE7' ] ){
\r
326 X_XHR[ 'listen' ]( [ 'readystatechange', 'error' ] );
\r
328 X_XHR[ 'listen' ]( [ 'load', 'readystatechange', 'error', 'timeout' ] ); //, 'abort'
\r
331 if( X_XHR_upload ){
\r
332 raw.upload.addEventListener( 'progress', X_XHR.onUploadProgress );
\r
337 cancel : function(){
\r
338 /* X_XHR[ '_rawObject' ].abort && */ X_XHR[ '_rawObject' ].abort();
\r
339 X_XHR._canceled = true;
\r
342 reset : function(){
\r
344 X_XHR._method = X_XHR._dataType = '';
\r
345 X_XHR._canceled = X_XHR._busy = X_XHR._error = false;
\r
346 X_XHR._timerID && X_Timer_remove( X_XHR._timerID );
\r
347 X_XHR._percent = X_XHR._timerID = 0;
\r
349 // XMLHttpRequest の使い方
\r
350 // http://webos-goodies.jp/archives/50548720.html
\r
351 // XMLHttpRequest オブジェクトを再利用する際も、 abort メソッドを呼び出す必要があるようです。
\r
352 /* X_XHR[ '_rawObject' ].abort && */ X_XHR[ '_rawObject' ].abort();
\r
354 // XMLHttpRequest で順番にリソースを取得する
\r
355 // http://note.chiebukuro.yahoo.co.jp/detail/n16248
\r
356 // Opera 10.10 と Safari 4.1 はエラーが起きた XHR を再利用できないので毎回作る
\r
359 // domes.lingua.heliohost.org/dom-intro/load-save2.html
\r
360 // 規定上は open() を呼び出すと XMLHttpRequest オブジェクトが未送信状態に戻りますが、
\r
361 // Opera 10.10、Safari 4.1 では、同一オリジン制限に違反した XMLHttpRequest オブジェクトは再度 open() しても未送信状態に戻りません。
\r
363 // Timeout した Gecko の xhr.response に触るとエラー??
\r
365 if( X_XHR._error || ( X_XHR_neverReuse && !X_XHR._isMsXML ) ){
\r
367 if( X_XHR_upload ){
\r
368 X_XHR_w3c.upload.removeEventListener( 'progress', X_XHR.onUploadProgress );
\r
371 // ie7 は xhr object を再利用できない。但し send のあとに alert を挟むと動いた、、、
\r
372 // ie7モード(IE11) では再利用可能、、、
\r
374 X_EventDispatcher_toggleAllEvents( X_XHR, false );
\r
375 X_XHR[ '_rawObject' ] = null;
\r
377 if( X_XHR._isXDR ){
\r
379 delete X_XHR._isXDR;
\r
384 X_XHR[ 'unlisten' ]( [ 'load', 'readystatechange', 'progress', 'error', 'timeout' ] );
\r
388 handleEvent : function( e ){
\r
389 var raw = X_XHR[ '_rawObject' ],
\r
390 live = !X_XHR._canceled,
\r
391 headers, status, text, data;
\r
393 switch( e && e.type || 'readystatechange' ){
\r
395 * http://memopad.bitter.jp/w3c/ajax/ajax_xmlhttprequest_onreadystatechange.html
\r
396 readyState XMLHttpRequest のステータスを保持する。0 から 4 までに変化する:
\r
401 4: リクエストは終了してレスポンスの準備が完了
\r
403 404: Page not found
\r
405 If-Modified-Sinceヘッダを利用してWebページのキャッシュを行うXMLHttpRequestラッパー
\r
406 http://www.semblog.org/msano/archives/000407.html
\r
408 case 'readystatechange' :
\r
409 //if( !X.XHR.PROGRESS ){
\r
410 switch( raw.readyState ){
\r
414 case 2 : // 0% ajaxstart
\r
415 live && X_XHR[ 'asyncDispatch' ]( { type : X_EVENT_PROGRESS, 'percent' : 0 } );
\r
418 live && X_XHR[ 'asyncDispatch' ]( { type : X_EVENT_PROGRESS, 'percent' : X_XHR._percent < 99.9 ? 99.9 : ( X_XHR._percent + 100 ) / 2 } );
\r
422 if( X_XHR._percent === 100 ) return; // Opera8 readystatechange が2重に発生
\r
433 if( !X_XHR._busy ) return;
\r
435 X_XHR._percent = 100;
\r
436 X_XHR._busy = false;
\r
437 status = raw.status;
\r
439 // TODO GET_FULL_HEADERS
\r
440 // https://msdn.microsoft.com/en-us/library/ms766595%28v=vs.85%29.aspx
\r
441 // Implemented in: MSXML 3.0 and MSXML 6.0
\r
442 if( X_XHR._isXDR ){
\r
443 headers = { 'Content-Type' : raw.contentType };
\r
445 if( ( X_XHR._isMsXML ? 3 <= X_XHR_msXMLVer : raw.setRequestHeader ) && ( headers = raw.getAllResponseHeaders() ) ){
\r
446 headers = X_XHR_parseResponseHeaders( headers );
\r
449 // https://code.google.com/p/fakeworker-js/source/browse/src/javascript/fakeworker.js
\r
451 ( !status && location.protocol === 'file:' ) ||
\r
452 // IE 6.0 でローカルファイルにアクセスした
\r
453 ( status < 100 && ( status = 200 ) ) ||
\r
454 ( 200 <= status && status < 400 ) ||
\r
455 //status === 304 ||
\r
456 ( status === 1223 && ( status = 204 ) ) ||
\r
457 ( X_UA[ 'Webkit' ] && status === undefined ) // safari: /webkit/.test(userAgent)
\r
460 * opera8, safari2, khtml3 で utf8 日本語文字列の文字化け
\r
462 * http://www.kawa.net/works/js/jkl/parsexml.html
\r
464 text = raw[ 'responseText' ];
\r
465 // Safari and Konqueror cannot understand the encoding of text files.
\r
466 if( text && ( X_UA[ 'Webkit' ] < 420 || X_UA[ 'KHTML' ] < 4 ) ){
\r
467 text = escape( text );
\r
468 if ( !text.match( '%u' ) && esc.match( '%' ) ){
\r
469 text = decodeURIComponent( text );
\r
474 // parse json, html, xml, text, script, css
\r
475 switch( X_XHR._dataType ){
\r
478 data = raw[ 'responseText' ];
\r
482 data = raw[ 'response' ] || raw[ 'responseText' ];
\r
483 // eval() を使っているけど JSON の無いブラウザは XDomain な XHR はできないのでよしとする。
\r
484 // XDomain な XHR の際は Flash 等で代替し、その中に Json parser も組み込む。
\r
485 // http://d.hatena.ne.jp/sshi/20060904/p1
\r
486 if( !X_Type_isObject( data ) ) data = X_JSON_parseTrustableString( data );
\r
492 // svg, vml, xaml, xul, mxml ??
\r
493 data = raw[ 'responseXML' ] || raw[ 'response' ] || raw[ 'responseText' ]; // とりあえず
\r
496 case 'arraybuffer' :
\r
497 // TODO resoponceBody if( X_UA[ 'IE' ] < 10 )
\r
498 // http://d.hatena.ne.jp/maachang/20130221/1361427565
\r
499 data = raw[ 'response' ] || raw[ 'responseText' ]; // とりあえず
\r
502 X_XHR[ 'asyncDispatch' ]( 32, { type : X_EVENT_SUCCESS, status : status || 200, response : data, 'headers' : headers || null } );
\r
504 X_XHR[ 'asyncDispatch' ]( 32, { type : X_EVENT_ERROR, status : status || 400, 'headers' : headers || null } );
\r
509 if( e.lengthComputable ){
\r
510 X_XHR._percent = e.loaded / e.total * 100;
\r
511 live && X_XHR._percent < 100 && X_XHR[ 'asyncDispatch' ]( { type : X_EVENT_PROGRESS, 'percent' : X_XHR._percent } );
\r
516 //console.dir( e );
\r
517 X_XHR._busy = false;
\r
518 X_XHR._error = X_UA[ 'Opera' ] || X_UA[ 'Webkit' ] ;
\r
519 live && X_XHR[ 'asyncDispatch' ]( 32, { type : X_EVENT_ERROR, status : raw.status } );
\r
522 case 'timeout' : // Gecko 12.0 https://developer.mozilla.org/ja/docs/XMLHttpRequest/Synchronous_and_Asynchronous_Requests
\r
523 X_XHR._busy = false;
\r
524 X_XHR._error = !!X_UA[ 'Gecko' ];
\r
525 X_XHR[ 'asyncDispatch' ]( { type : X_EVENT_ERROR, 'timeout' : true, status : 408 } );
\r
531 * http://www.kawa.net/works/js/jkl/parsexml.html
\r
533 // ================================================================
\r
534 // method: documentElement()
\r
535 // return: XML DOM in response body
\r
537 JKL.ParseXML.HTTP.prototype.documentElement = function() {
\r
538 // debug.print( "documentElement: "+this.req );
\r
539 if ( ! this.req ) return;
\r
540 if ( this.req.responseXML ) {
\r
541 return this.req.responseXML.documentElement; // XMLHTTPRequest
\r
543 return this.req.documentElement; // IXMLDOMDocument
\r
548 onTimeout : function(){
\r
549 var raw = X_XHR[ '_rawObject' ],
\r
550 live = !X_XHR._canceled || !X_XHR._busy;
\r
552 if( live || raw.readyState < 3 ){
\r
553 X_XHR._busy = false;
\r
554 live && X_XHR[ 'asyncDispatch' ]( { type : X_EVENT_ERROR, 'timeout' : true, status : 408 } );
\r
556 X_XHR._timerID = 0;
\r
559 onUploadProgress : X_XHR_upload && function( e ){
\r
560 !X_XHR._canceled &&
\r
561 X_XHR[ 'asyncDispatch' ]( {
\r
562 type : X_EVENT_PROGRESS,
\r
563 'percent' : X_XHR._percent,
\r
564 'uploadPercent' : e.loaded / e.total * 100
\r
568 // 同期リクエストでなければならない場合, unload, beforeunload時
\r
571 * https://gist.github.com/mmazer/5404301
\r
573 * XmlHttpRequest's getAllResponseHeaders() method returns a string of response
\r
574 * headers according to the format described here:
\r
575 * http://www.w3.org/TR/XMLHttpRequest/#the-getallresponseheaders-method
\r
576 * This method parses that string into a user-friendly key/value pair object.
\r
578 * http://hakuhin.jp/js/xmlhttprequest.html#XHR_GET_ALL_RESPONSE_HEADERS
\r
579 * 複数の情報が存在する場合、改行で区切られています。
\r
582 function X_XHR_parseResponseHeaders( headerStr ){
\r
583 var headers = {}, headerPairs, i = 0, l, headerPair, index, key, val;
\r
585 if( !headerStr ) return headers;
\r
587 headerPairs = headerStr.split( '\u000d\u000a' );
\r
588 for( l = headerPairs.length; i < l ; ++i ){
\r
589 headerPair = headerPairs[i];
\r
590 index = headerPair.indexOf( '\u003a\u0020' );
\r
592 key = headerPair.substring( 0, index );
\r
593 val = headerPair.substring( index + 2 );
\r
594 headers[ key ] = val.split( '\r\n' ).join( '\n' ).split( '\n' );
\r