| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  | import {ConnectionBackend, Connection} from '../interfaces'; | 
					
						
							| 
									
										
										
										
											2015-10-29 17:50:12 -07:00
										 |  |  | import {ReadyStates, RequestMethods, ResponseTypes} from '../enums'; | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  | import {Request} from '../static_request'; | 
					
						
							|  |  |  | import {Response} from '../static_response'; | 
					
						
							|  |  |  | import {ResponseOptions, BaseResponseOptions} from '../base_response_options'; | 
					
						
							| 
									
										
										
										
											2015-10-10 14:58:46 +02:00
										 |  |  | import {Injectable} from 'angular2/angular2'; | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  | import {BrowserJsonp} from './browser_jsonp'; | 
					
						
							| 
									
										
										
										
											2015-11-06 17:34:07 -08:00
										 |  |  | import {makeTypeError} from 'angular2/src/facade/exceptions'; | 
					
						
							|  |  |  | import {StringWrapper, isPresent} from 'angular2/src/facade/lang'; | 
					
						
							| 
									
										
										
										
											2015-10-30 23:52:37 -07:00
										 |  |  | import {Observable} from 'angular2/angular2'; | 
					
						
							| 
									
										
										
										
											2015-10-29 17:50:12 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | const JSONP_ERR_NO_CALLBACK = 'JSONP injected script did not invoke callback.'; | 
					
						
							|  |  |  | const JSONP_ERR_WRONG_METHOD = 'JSONP requests must use GET request method.'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-06 06:53:39 -07:00
										 |  |  | export abstract class JSONPConnection implements Connection { | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  |   readyState: ReadyStates; | 
					
						
							|  |  |  |   request: Request; | 
					
						
							| 
									
										
										
										
											2015-10-30 23:52:37 -07:00
										 |  |  |   response: Observable<Response>; | 
					
						
							| 
									
										
										
										
											2015-10-06 06:53:39 -07:00
										 |  |  |   abstract finished(data?: any): void; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export class JSONPConnection_ extends JSONPConnection { | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  |   private _id: string; | 
					
						
							|  |  |  |   private _script: Element; | 
					
						
							|  |  |  |   private _responseData: any; | 
					
						
							|  |  |  |   private _finished: boolean = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   constructor(req: Request, private _dom: BrowserJsonp, | 
					
						
							|  |  |  |               private baseResponseOptions?: ResponseOptions) { | 
					
						
							| 
									
										
										
										
											2015-10-06 06:53:39 -07:00
										 |  |  |     super(); | 
					
						
							| 
									
										
										
										
											2015-08-26 13:40:12 -07:00
										 |  |  |     if (req.method !== RequestMethods.Get) { | 
					
						
							| 
									
										
										
										
											2015-10-29 17:50:12 -07:00
										 |  |  |       throw makeTypeError(JSONP_ERR_WRONG_METHOD); | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     this.request = req; | 
					
						
							| 
									
										
										
										
											2015-09-25 18:53:32 -04:00
										 |  |  |     this.response = new Observable(responseObserver => { | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-25 18:53:32 -04:00
										 |  |  |       this.readyState = ReadyStates.Loading; | 
					
						
							|  |  |  |       let id = this._id = _dom.nextRequestID(); | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-25 18:53:32 -04:00
										 |  |  |       _dom.exposeConnection(id, this); | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-25 18:53:32 -04:00
										 |  |  |       // Workaround Dart
 | 
					
						
							|  |  |  |       // url = url.replace(/=JSONP_CALLBACK(&|$)/, `generated method`);
 | 
					
						
							|  |  |  |       let callback = _dom.requestCallback(this._id); | 
					
						
							|  |  |  |       let url: string = req.url; | 
					
						
							|  |  |  |       if (url.indexOf('=JSONP_CALLBACK&') > -1) { | 
					
						
							|  |  |  |         url = StringWrapper.replace(url, '=JSONP_CALLBACK&', `=${callback}&`); | 
					
						
							|  |  |  |       } else if (url.lastIndexOf('=JSONP_CALLBACK') === url.length - '=JSONP_CALLBACK'.length) { | 
					
						
							| 
									
										
										
										
											2015-10-31 13:04:26 -07:00
										 |  |  |         url = url.substring(0, url.length - '=JSONP_CALLBACK'.length) + `=${callback}`; | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-25 18:53:32 -04:00
										 |  |  |       let script = this._script = _dom.build(url); | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-25 18:53:32 -04:00
										 |  |  |       let onLoad = event => { | 
					
						
							|  |  |  |         if (this.readyState === ReadyStates.Cancelled) return; | 
					
						
							|  |  |  |         this.readyState = ReadyStates.Done; | 
					
						
							|  |  |  |         _dom.cleanup(script); | 
					
						
							|  |  |  |         if (!this._finished) { | 
					
						
							| 
									
										
										
										
											2015-10-29 17:50:12 -07:00
										 |  |  |           let responseOptions = | 
					
						
							| 
									
										
										
										
											2015-11-19 18:47:29 -08:00
										 |  |  |               new ResponseOptions({body: JSONP_ERR_NO_CALLBACK, type: ResponseTypes.Error, url}); | 
					
						
							| 
									
										
										
										
											2015-10-29 17:50:12 -07:00
										 |  |  |           if (isPresent(baseResponseOptions)) { | 
					
						
							|  |  |  |             responseOptions = baseResponseOptions.merge(responseOptions); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           responseObserver.error(new Response(responseOptions)); | 
					
						
							| 
									
										
										
										
											2015-09-25 18:53:32 -04:00
										 |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-19 18:47:29 -08:00
										 |  |  |         let responseOptions = new ResponseOptions({body: this._responseData, url}); | 
					
						
							| 
									
										
										
										
											2015-09-25 18:53:32 -04:00
										 |  |  |         if (isPresent(this.baseResponseOptions)) { | 
					
						
							|  |  |  |           responseOptions = this.baseResponseOptions.merge(responseOptions); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         responseObserver.next(new Response(responseOptions)); | 
					
						
							|  |  |  |         responseObserver.complete(); | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       let onError = error => { | 
					
						
							|  |  |  |         if (this.readyState === ReadyStates.Cancelled) return; | 
					
						
							|  |  |  |         this.readyState = ReadyStates.Done; | 
					
						
							|  |  |  |         _dom.cleanup(script); | 
					
						
							| 
									
										
										
										
											2015-10-29 17:50:12 -07:00
										 |  |  |         let responseOptions = new ResponseOptions({body: error.message, type: ResponseTypes.Error}); | 
					
						
							|  |  |  |         if (isPresent(baseResponseOptions)) { | 
					
						
							|  |  |  |           responseOptions = baseResponseOptions.merge(responseOptions); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         responseObserver.error(new Response(responseOptions)); | 
					
						
							| 
									
										
										
										
											2015-09-25 18:53:32 -04:00
										 |  |  |       }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       script.addEventListener('load', onLoad); | 
					
						
							|  |  |  |       script.addEventListener('error', onError); | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-25 18:53:32 -04:00
										 |  |  |       _dom.send(script); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return () => { | 
					
						
							|  |  |  |         this.readyState = ReadyStates.Cancelled; | 
					
						
							|  |  |  |         script.removeEventListener('load', onLoad); | 
					
						
							|  |  |  |         script.removeEventListener('error', onError); | 
					
						
							|  |  |  |         if (isPresent(script)) { | 
					
						
							|  |  |  |           this._dom.cleanup(script); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-03 10:04:17 -07:00
										 |  |  |       }; | 
					
						
							| 
									
										
										
										
											2015-09-25 18:53:32 -04:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   finished(data?: any) { | 
					
						
							|  |  |  |     // Don't leak connections
 | 
					
						
							|  |  |  |     this._finished = true; | 
					
						
							|  |  |  |     this._dom.removeConnection(this._id); | 
					
						
							| 
									
										
										
										
											2015-08-26 13:40:12 -07:00
										 |  |  |     if (this.readyState === ReadyStates.Cancelled) return; | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  |     this._responseData = data; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-06 06:53:39 -07:00
										 |  |  | export abstract class JSONPBackend extends ConnectionBackend {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  | @Injectable() | 
					
						
							| 
									
										
										
										
											2015-10-06 06:53:39 -07:00
										 |  |  | export class JSONPBackend_ extends JSONPBackend { | 
					
						
							|  |  |  |   constructor(private _browserJSONP: BrowserJsonp, private _baseResponseOptions: ResponseOptions) { | 
					
						
							|  |  |  |     super(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  |   createConnection(request: Request): JSONPConnection { | 
					
						
							| 
									
										
										
										
											2015-10-06 06:53:39 -07:00
										 |  |  |     return new JSONPConnection_(request, this._browserJSONP, this._baseResponseOptions); | 
					
						
							| 
									
										
										
										
											2015-07-14 19:53:04 -05:00
										 |  |  |   } | 
					
						
							|  |  |  | } |