forked from mirrors/gecko-dev
		
	 afb6727d56
			
		
	
	
		afb6727d56
		
	
	
	
	
		
			
			...r parsing. Hook up document loading to async networking events. Relies on https://github.com/servo/html5ever/pull/107, so we'll likely need to backport it rather than wait for the next rustc upgrade. Source-Repo: https://github.com/servo/servo Source-Revision: dd319c1a998bbd3eeb84fdc4ca8a41ee7877ca37
		
			
				
	
	
		
			110 lines
		
	
	
	
		
			3.6 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			110 lines
		
	
	
	
		
			3.6 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| /* This Source Code Form is subject to the terms of the Mozilla Public
 | |
|  * License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
|  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | |
| 
 | |
| //! Tracking of pending loads in a document.
 | |
| //! https://html.spec.whatwg.org/multipage/#the-end
 | |
| 
 | |
| use script_task::{ScriptMsg, ScriptChan};
 | |
| use msg::constellation_msg::{PipelineId};
 | |
| use net_traits::{Metadata, load_whole_resource, ResourceTask, PendingAsyncLoad};
 | |
| use net_traits::AsyncResponseTarget;
 | |
| use url::Url;
 | |
| 
 | |
| #[jstraceable]
 | |
| #[derive(PartialEq, Clone, Debug)]
 | |
| pub enum LoadType {
 | |
|     Image(Url),
 | |
|     Script(Url),
 | |
|     Subframe(Url),
 | |
|     Stylesheet(Url),
 | |
|     PageSource(Url),
 | |
| }
 | |
| 
 | |
| impl LoadType {
 | |
|     fn url(&self) -> &Url {
 | |
|         match *self {
 | |
|             LoadType::Image(ref url) |
 | |
|             LoadType::Script(ref url) |
 | |
|             LoadType::Subframe(ref url) |
 | |
|             LoadType::Stylesheet(ref url) |
 | |
|             LoadType::PageSource(ref url) => url,
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| #[jstraceable]
 | |
| pub struct DocumentLoader {
 | |
|     pub resource_task: ResourceTask,
 | |
|     notifier_data: Option<NotifierData>,
 | |
|     blocking_loads: Vec<LoadType>,
 | |
| }
 | |
| 
 | |
| #[jstraceable]
 | |
| pub struct NotifierData {
 | |
|     pub script_chan: Box<ScriptChan + Send>,
 | |
|     pub pipeline: PipelineId,
 | |
| }
 | |
| 
 | |
| impl DocumentLoader {
 | |
|     pub fn new(existing: &DocumentLoader) -> DocumentLoader {
 | |
|         DocumentLoader::new_with_task(existing.resource_task.clone(), None, None)
 | |
|     }
 | |
| 
 | |
|     pub fn new_with_task(resource_task: ResourceTask,
 | |
|                          data: Option<NotifierData>,
 | |
|                          initial_load: Option<Url>,)
 | |
|                          -> DocumentLoader {
 | |
|         let initial_loads = initial_load.into_iter().map(LoadType::PageSource).collect();
 | |
| 
 | |
|         DocumentLoader {
 | |
|             resource_task: resource_task,
 | |
|             notifier_data: data,
 | |
|             blocking_loads: initial_loads,
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /// Create a new pending network request, which can be initiated at some point in
 | |
|     /// the future.
 | |
|     pub fn prepare_async_load(&mut self, load: LoadType) -> PendingAsyncLoad {
 | |
|         let url = load.url().clone();
 | |
|         self.blocking_loads.push(load);
 | |
|         let pipeline = self.notifier_data.as_ref().map(|data| data.pipeline);
 | |
|         PendingAsyncLoad::new(self.resource_task.clone(), url, pipeline)
 | |
|     }
 | |
| 
 | |
|     /// Create and initiate a new network request.
 | |
|     pub fn load_async(&mut self, load: LoadType, listener: Box<AsyncResponseTarget + Send>) {
 | |
|         let pending = self.prepare_async_load(load);
 | |
|         pending.load_async(listener)
 | |
|     }
 | |
| 
 | |
|     /// Create, initiate, and await the response for a new network request.
 | |
|     pub fn load_sync(&mut self, load: LoadType) -> Result<(Metadata, Vec<u8>), String> {
 | |
|         self.blocking_loads.push(load.clone());
 | |
|         let result = load_whole_resource(&self.resource_task, load.url().clone());
 | |
|         self.finish_load(load);
 | |
|         result
 | |
|     }
 | |
| 
 | |
|     /// Mark an in-progress network request complete.
 | |
|     pub fn finish_load(&mut self, load: LoadType) {
 | |
|         let idx = self.blocking_loads.iter().position(|unfinished| *unfinished == load);
 | |
|         self.blocking_loads.remove(idx.expect(&format!("unknown completed load {:?}", load)));
 | |
| 
 | |
|         if let Some(NotifierData { ref script_chan, pipeline }) = self.notifier_data {
 | |
|             if !self.is_blocked() {
 | |
|                 script_chan.send(ScriptMsg::DocumentLoadsComplete(pipeline)).unwrap();
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     pub fn is_blocked(&self) -> bool {
 | |
|         //TODO: Ensure that we report blocked if parsing is still ongoing.
 | |
|         !self.blocking_loads.is_empty()
 | |
|     }
 | |
| 
 | |
|     pub fn inhibit_events(&mut self) {
 | |
|         self.notifier_data = None;
 | |
|     }
 | |
| }
 |