1 // TODO onlineevent offlineevent, netspeed
\r
2 // local への通信に対しては、netspeed を更新しない
\r
4 // http://bugs.jquery.com/ticket/2709
\r
5 // <head><base> がある場合、<script>の追加に失敗する
\r
10 * <p>state() メソッドだけを持つ。通信用のプロパティは X.Pair によって隠蔽されています。
\r
12 * <p>kill() で通信待ち中はキャンセル&破棄、通信中の場合は通信の中断&破棄を行う。
\r
15 * <dt>X.Event.PROGRESS<dd>通信進行状況
\r
16 * <dt>X.Event.SUCCESS<dd>通信成功
\r
17 * <dt>X.Event.ERROR<dd>通信エラー タイムアウトの場合、e.timeout == true で分かる。
\r
18 * <dt>X.Event.CANCELED<dd>通信のユーザー、プログラムによるキャンセル。SUCCESS, ERROR, COMPLETE 後に kill()してもCANCELEDは呼ばれません。
\r
19 * <dt>X.Event.COMPLETE<dd>通信完了。SUCCESS, ERROR, CANCELED 後に発生。
\r
21 * <p>X.Net インスタンスは COMPLETE 後に自動で破棄される。
\r
25 * <dt>type<dd>'xhr', 'jsonp', 'image', 'img'
\r
26 * <dt>xhr<dd>URL { url : 'hoge', type : 'xhr' } の省略形
\r
27 * <dt>jsonp<dd>URL { url : 'hoge', type : 'jsonp' } の省略形
\r
28 * <dt>image, img<dd>URL { url : 'hoge', type : 'image' } の省略形
\r
30 * <h4>XHR 用プロパティ</h4>
\r
32 * <dt>method<dd>'GET', 'POST' 未指定かつ postdata を設定している場合、'POST' になる。
\r
33 * <dt>params<dd>url パラメータを object で渡すことが出来る。
\r
34 * <dt>postdata<dd>string, object の場合は X.String.serialize される。
\r
35 * <dt>async<dd>boolean
\r
36 * <dt>username<dd>BASIC 認証
\r
37 * <dt>password<dd>BASIC 認証
\r
38 * <dt>headers<dd>object xhr.setRequestHeader する値
\r
39 * <dt>timeout<dd>タイムアウト ms
\r
40 * <dt>cache<dd>headers[ 'Pragma' ] = 'no-cache' 等を設定するか?
\r
41 * <dt>dataType<dd>'text', 'json', 'xml', 'blob', 'arraybuffer' 等。xhr.responseType に指定する値
\r
42 * <dt>mimeType<dd>'text/xml', 'audio/mpeg' 等。xhr.overrideMimeType する値
\r
43 * <dt>auth<dd>X.OAuth2 インスタンス(OAuth2 サービスの定義)
\r
44 * <dt>getFullHeaders<dd>getAllResponseHeaders() をパースしたハッシュを返す。値は配列になっている。XDR は Content-Type しか取得でいない。
\r
45 * <dt>canUse<dd>未実装。gadget proxy, YQL, <del>YPipes</del> 等のマッシュアップの許可。現在は test : 'gadget' としている
\r
48 * <h4>JSONP 用プロパティ</h4>
\r
50 * <dt>params<dd>url パラメータを object で渡すことが出来る。
\r
51 * <dt>callbackName<dd>callback(json) コールバック名が固定されている際に指定。または &callback=hoge 以外の名前でコールバックを指定する場合に params と callbackName に書いておく。url パラメータに callback が無く、callbackName もない場合、フレームワーク内で自動で設定される
\r
52 * <dt>charset<dd>ページと異なるjsonpを読み込む場合に指定 'EUC-JP', 'Shift-JIS' 等 script タグの charset に入る。https://code.google.com/p/ajaxzip3/issues/detail?id=5
\r
53 * <dt>useFireWall<dd>異なるドメインに jsonp を読み込んだ後、xdomain iframe 通信を使ってデータを受け取る。不正なコードの実行を防ぐことが出来る、未実装
\r
56 * <h4>Form 用プロパティ</h4>
\r
58 * <dt>params<dd>url パラメータを object で渡すことが出来る。
\r
59 * <dt>method<dd>'GET' or 'POST'
\r
60 * <dt>target<dd>'_self', '_parent', '_top' の場合、ページから離脱する。target を指定せず同一ドメインの場合 response に body.innerHTML が返る。
\r
64 * @class 各種ネットワーク機能をラップしインターフェイスを共通化する。
\r
66 * @extends {EventDispatcher}
\r
67 * @example // XHR - GET
\r
68 * var net = X.Net( { xhr : urlString } )
\r
69 * .listen( X.Event.PROGRESS )
\r
70 * .listenOnce( [ X.Event.SUCCESS, X.Event.ERROR, X.Event.TIMEOUT, X.Event.CANCELED ] );
\r
73 * var net = X.Net( urlString );
\r
76 * var net = X.Net( { xhr : urlString, postdata : myData } );
\r
79 * var net = X.Net( { jsonp : urlString, params : params, callbackName : callbackName, charset : charset, useFireWall : false } );
\r
82 * var net = X.Net( { form : urlString, method : 'POST', target : '_self', params : {} } );
\r
84 * // Image preload & getSize
\r
85 * var net = X.Net( { image : src, sizeDetection : true } );
\r
87 * // load <script>, <link>
\r
89 X[ 'Net' ] = X_EventDispatcher[ 'inherits' ](
\r
94 'Constructor' : function( urlOrObject, opt_options ){
\r
95 var opt, url, type, auth;
\r
97 if( X_Type_isObject( opt = urlOrObject ) ){
\r
99 if( X_Type_isString( url = opt[ 'xhr' ] ) ){
\r
100 type = X_NET_TYPE_XHR;
\r
104 if( X_Type_isString( url = opt[ 'jsonp' ] ) ){
\r
105 type = X_NET_TYPE_JSONP;
\r
109 if( X_Type_isString( url = opt[ 'img' ] || opt[ 'image' ] ) ){
\r
110 type = X_NET_TYPE_IMAGE;
\r
114 if( X_Type_isString( url = opt[ 'form' ] ) ){
\r
115 type = X_NET_TYPE_FORM;
\r
118 if( !( type = X_NET_NAME_TO_ID[ opt[ 'type' ] ] ) ){
\r
120 alert( 'X.Net args error' );
\r
124 url = opt[ 'url' ];
\r
127 if( !X_Type_isString( url ) ){
\r
128 alert( 'X.Net args error' );
\r
133 if( X_Type_isString( urlOrObject ) ){
\r
136 if( X_Type_isObject( opt = opt_options ) ){
\r
137 type = opt[ 'type' ] || X_NET_TYPE_XHR;
\r
139 type = X_NET_TYPE_XHR;
\r
140 opt = { 'url' : url, 'method' : 'GET' };
\r
144 alert( 'X.Net args error' );
\r
150 if( auth = opt[ 'auth' ] ){
\r
151 delete opt[ 'auth' ];
\r
153 opt = X_Object_deepCopy( opt );
\r
155 opt[ 'auth' ] = auth; // auth は deep copy されるとまずい
\r
158 // params を url に追加
\r
159 if( opt[ 'params' ] ){
\r
160 url = X_URL_create( url, opt[ 'params' ] );
\r
161 delete opt[ 'params' ];
\r
164 if( type === X_NET_TYPE_XHR ){
\r
165 opt[ 'method' ] = opt[ 'method' ] || ( opt[ 'postdata' ] ? 'POST' : 'GET' );
\r
167 // XDomain 不可 -> Flash, Gears, Silverlight, canUseGadget なら gadget に切替?
\r
168 // PUT DELETE UPDATE 不可 -> Flash, Gears, Silverlight, canUseGadget なら gadget に切替?
\r
169 // xプロトコル な binary のロード -> gadget 内で proxyURL による XHR
\r
170 // or X_EVENT_ERROR
\r
172 opt[ 'dataType' ] = opt[ 'dataType' ] || X_URL_getEXT( url );
\r
175 opt.netType = type;
\r
176 opt[ 'url' ] = url;
\r
178 X_Pair_create( this, opt );
\r
180 this[ 'listen' ]( X_EVENT_KILL_INSTANCE, X_NET_proxyDispatch );
\r
182 X_NET_QUEUE_LIST[ X_NET_QUEUE_LIST.length ] = this;
\r
183 !X_NET_currentQueue && X_NET_shiftQueue();
\r
187 * 現在の状態。1:順番待ち, 2:通信中, 3:通信完了フェーズ
\r
188 * @alias Net.prototype.state
\r
190 'state' : function(){
\r
191 return this === X_NET_currentQueue ?
\r
192 ( X_NET_completePhase ? 3 : 2 ) :
\r
193 0 <= X_NET_QUEUE_LIST.indexOf( this ) ? 1 : 0;
\r
200 var X_NET_IWrapper = function(){};
\r
201 X_NET_IWrapper.prototype.load = function(){};
\r
202 X_NET_IWrapper.prototype.cancel = function(){};
\r
203 X_NET_IWrapper.prototype.reset = function(){};
\r
206 var X_NET_TYPE_XHR = 1,
\r
207 X_NET_TYPE_JSONP = 2,
\r
208 X_NET_TYPE_FORM = 3,
\r
209 X_NET_TYPE_IMAGE = 4,
\r
211 X_NET_NAME_TO_ID = {
\r
212 'xhr' : X_NET_TYPE_XHR,
\r
213 'jsonp' : X_NET_TYPE_JSONP,
\r
214 'form' : X_NET_TYPE_FORM,
\r
215 'img' : X_NET_TYPE_IMAGE,
\r
216 'image' : X_NET_TYPE_IMAGE
\r
219 X_NET_QUEUE_LIST = [],
\r
222 X_NET_JSONPWrapper,
\r
224 X_NET_ImageWrapper,
\r
227 X_NET_currentWrapper,
\r
228 X_NET_currentQueue,
\r
230 X_NET_completePhase;
\r
232 function X_NET_proxyDispatch( e ){
\r
236 case X_EVENT_KILL_INSTANCE :
\r
237 if( this === X_NET_currentQueue && X_NET_completePhase ){
\r
238 if( X_NET_completePhase === 1 )
\r
239 this[ 'unlisten' ]( X_EVENT_COMPLETE, X_NET_proxyDispatch )
\r
240 [ 'dispatch' ]( X_EVENT_COMPLETE );
\r
241 X_NET_shiftQueue();
\r
242 X_Pair_release( this );
\r
243 X_NET_completePhase = 0;
\r
245 if( this === X_NET_currentQueue ){
\r
246 X_NET_currentWrapper.cancel();
\r
247 X_NET_shiftQueue();
\r
250 if( ( i = X_NET_QUEUE_LIST.indexOf( this ) ) !== -1 ){
\r
251 X_NET_QUEUE_LIST.splice( i, 1 );
\r
255 if( flag ){ // flag が立つ場合、これは中断
\r
256 this[ 'dispatch' ]( X_EVENT_CANCELED );
\r
257 this[ 'dispatch' ]( { type : X_EVENT_COMPLETE, 'lastEventType' : X_EVENT_CANCELED } );
\r
258 X_Pair_release( this );
\r
261 case X_EVENT_PROGRESS :
\r
262 this[ 'dispatch' ]( e );
\r
265 case X_EVENT_ERROR :
\r
266 if( e.status === 401 ){
\r
267 if( auth = X_Pair_get( this )[ 'auth' ] ){
\r
268 X_Pair_get( auth ).onAuthError( auth, e );
\r
273 case X_EVENT_SUCCESS :
\r
274 X_NET_completePhase = 1;
\r
275 this[ 'listenOnce' ]( X_EVENT_COMPLETE, X_NET_proxyDispatch )
\r
276 [ 'asyncDispatch' ]( 32, { type : X_EVENT_COMPLETE, 'lastEventType' : e.type } );
\r
278 this[ 'asyncDispatch' ]( e );
\r
281 case X_EVENT_COMPLETE :
\r
282 X_NET_completePhase = 2;
\r
288 function X_NET_shiftQueue(){
\r
289 var auth, authSettings;
\r
291 if( X_NET_currentQueue ){
\r
292 if( X_NET_currentWrapper._busy ) return;
\r
294 X_NET_currentWrapper
\r
295 [ 'unlisten' ]( [ X_EVENT_PROGRESS, X_EVENT_SUCCESS, X_EVENT_ERROR ], X_NET_currentQueue, X_NET_proxyDispatch )
\r
298 X_NET_currentQueue = X_NET_currentWrapper = X_NET_currentData = null;
\r
301 if( !X_NET_QUEUE_LIST.length ) return;
\r
303 X_NET_currentQueue = X_NET_QUEUE_LIST.shift();
\r
304 X_NET_currentData = X_Pair_get( X_NET_currentQueue );
\r
306 switch( X_NET_currentData.netType ){
\r
307 case X_NET_TYPE_XHR :
\r
309 // TODO (xProtocol | method='update' | !cors) & canUse -> gadget.io.makeRequset, flash
\r
310 // force 'gadget', 'flash'
\r
311 switch( X_NET_currentData[ 'test' ] ){
\r
313 X_NET_currentWrapper = X_NET_GIMRWrapper || X_TEMP.X_Net_GIMR_init();
\r
319 X_NET_currentWrapper = X_NET_XHRWrapper || X_TEMP.X_Net_XHR_init();
\r
324 if( auth = X_NET_currentData[ 'auth' ] ){
\r
325 authSettings = X_Pair_get( auth );
\r
326 switch( auth[ 'state' ]() ){
\r
330 if( !( auth[ 'dispatch' ]( X_EVENT_NEED_AUTH ) & X_Callback_PREVENT_DEFAULT ) ){
\r
331 authSettings.lazyRequests = authSettings.lazyRequests || [];
\r
332 authSettings.lazyRequests.indexOf( X_NET_currentQueue ) === -1 && authSettings.lazyRequests.push( X_NET_currentQueue );
\r
334 X_NET_currentQueue = null;
\r
335 X_NET_shiftQueue();
\r
337 case 3 : // refresh token
\r
338 X_NET_QUEUE_LIST.push( X_NET_currentQueue );
\r
339 X_NET_currentQueue = null;
\r
340 X_NET_shiftQueue();
\r
343 authSettings.updateRequest( auth, X_NET_currentData );
\r
346 case X_NET_TYPE_JSONP :
\r
347 X_NET_currentWrapper = X_NET_JSONPWrapper || X_TEMP.X_NET_JSONP_init();
\r
349 case X_NET_TYPE_FORM :
\r
350 X_NET_currentWrapper = X_NET_FormWrapper || X_TEMP.X_NET_Form_init();
\r
352 case X_NET_TYPE_IMAGE :
\r
353 X_NET_currentWrapper = X_NET_ImageWrapper || X_TEMP.X_NET_Image_init();
\r
357 X_NET_currentWrapper[ 'listen' ]( [ X_EVENT_PROGRESS, X_EVENT_SUCCESS, X_EVENT_ERROR ], X_NET_currentQueue, X_NET_proxyDispatch );
\r
359 X_NET_currentWrapper.load( X_NET_currentData );
\r